/* Fast atomical linklist class * (c) Copyright 2001 Denis Roio aka jaromil * * This source code is free software; you can redistribute it and/or * modify it under the terms of the GNU Public License as published * by the Free Software Foundation; either version 2 of the License, * or (at your option) any later version. * * This source code is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * Please refer to the GNU Public License for more details. * * You should have received a copy of the GNU Public License along with * this source code; if not, write to: * Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * * "$Id: linklist.cpp,v 1.3 2004/02/13 16:58:16 jaromil Exp $" * ------------------------------------------------------------------------- linked list container class NOTE: add and rem don't take care of deleting pointers that has to be done by the process that creates them and knows which inheriting class they are (delete is done in main) */ #include #include #include #include Linklist::Linklist() { length = 0; first = NULL; last = NULL; if(pthread_mutex_init (&_mutex,NULL)) error("%i:%s error initializing POSIX thread mutex", __LINE__,__FILE__); } Linklist::~Linklist() { clear(); if(pthread_mutex_destroy(&_mutex)) error("error destroying POSIX thread mutex", __LINE__,__FILE__); } void Linklist::lock() { pthread_mutex_lock(&_mutex); } void Linklist::unlock() { pthread_mutex_unlock(&_mutex); } /* adds one element at the end of the list */ void Linklist::append(Entry *addr) { Entry *ptr = NULL; if(addr->list) addr->rem(); lock(); if(!last) { /* that's the first entry */ last = addr; last->next = NULL; last->prev = NULL; first = last; } else { /* add the entry to the end */ ptr = last; ptr->next = addr; addr->next = NULL; addr->prev = ptr; last = addr; } /* save the pointer to this list */ addr->list = this; length++; unlock(); } void Linklist::prepend(Entry *addr) { Entry *ptr = NULL; if(addr->list) addr->rem(); lock(); if(!first) { /* that's the first entry */ first = addr; first->next = NULL; first->prev = NULL; last = first; } else { /* add an entry to the beginning */ ptr = first; ptr->prev = addr; addr->next = ptr; addr->prev = NULL; first = addr; } addr->list = this; length++; unlock(); } /* adds an element at the position specified if pos is out of bounds adds it at the beginning or the end the element occupying allready the position slides down THIS FUNCTION IS NOT YET RELIABLE */ void Linklist::insert(Entry *addr, int pos) { if(length<=pos) { /* adds it at the end */ append(addr); return; } else if(pos<=1) { prepend(addr); return; } if(addr->list) addr->rem(); Entry *ptr = pick(pos); lock(); ptr->prev->next = addr; addr->prev = ptr->prev; ptr->prev = addr; addr->next = ptr; length++; addr->list = this; unlock(); } /* clears the list i don't delete filters here because they have to be deleted from the procedure creating them. so this call simply discards the pointers stored into the linked list. OBJECTS ARE NOT FREED */ bool Linklist::clear() { // func("you are using Linklist::clear()"); // func("you have to free all the elements of the list by hand"); // func("it's use is highly error prone, be warned!"); lock(); sel(0); length = 0; first = NULL; last = NULL; unlock(); return(true); } /* takes one element from the list === STARTING FROM 1 === returns NULL if called with pos=0 or pos>length returns Entry pointer otherwise this function is then overloading the operator[] */ Entry *Linklist::pick(int pos) { if((lengthnext; return(ptr); } /* this function is a wrapper around Entry::up() better to use that if you have a pointer to your Entry */ bool Linklist::moveup(int pos) { Entry *p = pick(pos); if(!p) return(false); return( p->up() ); } bool Linklist::movedown(int pos) { Entry *p = pick(pos); if(!p) return(false); return( p->down() ); } bool Linklist::moveto(int num, int pos) { Entry *p = pick(num); if(!p) return(false); return( p->move(pos) ); } /* removes one element from the list */ bool Linklist::rem(int pos) { Entry *ptr = pick(pos); if(ptr==NULL) return(false); ptr->rem(); return(true); } /* selects ONLY ONE, deselects the others use Entry::sel() if you want to do multiple selects */ bool Linklist::sel(int pos) { int c; bool res = false; Entry *ptr = last; if(pos>length) return(res); for(c=length;c>0;c--) { if(c==pos) { if(ptr->select) res = false; // was allready selected else { ptr->sel(true); res = true; // a new one is selected } } else ptr->sel(false); ptr = ptr->prev; } return(res); } /* returns the last one selected this is supposed to be used with single selections */ Entry *Linklist::selected() { int c; Entry *ptr = last; for(c=length;c>0;c--) { if(ptr->select) return ptr; ptr = ptr->prev; } return NULL; } Entry::Entry() { next = NULL; prev = NULL; list = NULL; select = false; } Entry::~Entry() { rem(); } bool Entry::up() { if(!prev || !list) return(false); list->lock(); Entry *tprev = prev, *tnext = next, *pp = prev->prev; if(!next) list->last = prev; if(tnext) tnext->prev = tprev; next = tprev; prev = pp; tprev->next = tnext; tprev->prev = this; if(pp) pp->next = this; if(!prev) list->first = this; list->unlock(); return(true); } bool Entry::down() { if(!next || !list) return(false); list->lock(); Entry *tprev = prev, *tnext = next, *nn = next->next; if(!prev) list->first = next; if(tprev) tprev->next = tnext; prev = tnext; next = nn; tnext->prev = tprev; tnext->next = this; if(nn) nn->prev = this; if(!next) list->last = this; list->unlock(); return(true); } bool Entry::move(int pos) { func("Entry::move(%i) - NEW LINKLIST MOVE"); if(!list) return(false); list->lock(); Entry *tn, *tp; Entry *swapping = list->pick(pos); if(swapping == this) return(true); if(!swapping) return(false); tn = swapping->next; tp = swapping->prev; swapping->next = next; swapping->prev = prev; if(next) next->prev = swapping; else list->last = swapping; if(prev) prev->next = swapping; else list->first = swapping; next = tn; prev = tp; if(next) next->prev = this; else list->last = this; if(prev) prev->next = this; else list->first = this; list->unlock(); func("LINKLIST MOVE RETURNS SUCCESS"); return(true); } void Entry::rem() { if(!list) return; list->lock(); if(prev) prev->next = next; else list->first = next; if(next) next->prev = prev; else list->last = prev; list->length--; list->unlock(); list = NULL; } void Entry::sel(bool on) { select = on; } /* MUSE specific, deprecated use of ID and position */ Entry *Linklist::pick_id(int id) { Entry *d = begin(); while(d) { if(d->id == id) break; d = d->next; } return d; } int Linklist::selected_pos() { int c; Entry *ptr = last; for(c=length;c>0;c--) { if(ptr->select) return c; ptr = ptr->prev; } return 0; }