LL(3) LL(3) 16 June 1992 NAME LL - a double linked list library, ConsLL, ConsCopyLL, ConsPtrLL, DestLL , EmptyLL, IsEmptyLL, ApplyLL, SortLL, SizeLL, LookInLL, FprintLL, SprintLL, SscanLL, InsBefLL, InsAftLL, InsFirstLL, InsLastLL, InsBefLLf, InsAftLLf, InsFirstLLf, InsLastLLf, MoveListFirstLL, MoveListLastLL, MoveListAftLL, MoveListBefLL, MoveHeadFirstLL, MoveHeadLastLL, MoveHeadAftLL, MoveHeadBefLL, MoveTailFirstLL, MoveTailLastLL, MoveTailAftLL, MoveTailBefLL, FirstElmLL, LastElmLL, NthElmLL, NextElmLL, PrevElmLL, RelNthElmLL, DelElmLL, IsElmLL, IsFirstElmLL, IsLastElmLL, IndexElmLL, ForeachLL_M, ForeachDownLL_M, SafeForeachLL_M SYNOPSIS #include ''LL.h'' t_LL ConsLL (); t_LL ConsCopyLL(t_LL dest); t_LL ConsPtrLL(t_LL dest); void * DestLL(t_LL list); t_LL EmptyLL (t_LL list) int IsEmptyLL(t_LL list) void * ApplyLL (t_LL list, void * (*apply) (void *)); t_LL SortLL (t_LL list, int (*compar) (void *, void*)) long SizeLL (t_LL list); void * LookInLL(t_LL list); char * FprintLL(t_LL list, FILE * file, char *b, char *control, char *a) ; char * printLL(t_LL list, char * control); char * SprintLL(t_LL list, char * str , char *b, char *control, char *a) ; char * SscanLL(t_LL list, char *string, char * control, int termination); #define InsBefLL(p_el,data) InsBefLLf(p_el, sizeof(data), &data) #define InsAftLL(p_el,data) InsAftLLf(p_el, sizeof(data), &data) #define InsFirstLL(list,data) InsFirstLLf(list, sizeof(data), &data) #define InsLastLL(list,data) InsLastLLf(list, sizeof(data), &data) void * InsBefLLf (void * p_elm, size_t size, void * data); void * InsAftLLf (void * p_elm, size_t size, void * data); void * InsFirstLLf (t_LL list, size_t size, void * data); void * InsLastLLf (t_LL list, size_t size, void * data); void DelElmLL (void * p_elm); void * DelElmNeLL(void * p_elm); void * DelElmPrLL(void * p_elm); t_LL MoveListFirstLL(t_LL dest, t_LL src); t_LL MoveListLastLL(t_LL dest, t_LL src); void * MoveListAftLL(void *el, t_LL src); void * MoveListBefLL(void *el, t_LL src); t_LL MoveHeadFirstLL(t_LL dest, t_LL src, void *head); t_LL MoveHeadLastLL(t_LL dest, t_LL src, void *head); void * MoveHeadAftLL(void *el, t_LL src, void *head); void * MoveHeadBefLL(void *el, t_LL src, void *head); t_LL MoveTailFirstLL(t_LL dest, t_LL src, void *tail); t_LL MoveTailLastLL(t_LL dest, t_LL src, void *tail); void * MoveTailAftLL(void *el, t_LL src, void *tail); - 1 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 void * MoveTailBefLL(void *el, t_LL src, void *tail); void * FirstElmLL (t_LL list); void * LastElmLL (t_LL list); void * NthElmLL (t_LL list, long n); void * NextElmLL (void * p_elm); void * PrevElmLL (void * p_elm); void * RelNthElmLL (void * p_elm, long n); void * LinkAftLL(void * curr,void * new); void * LinkBefLL(void * curr,void * new); void * UnlinkLL(void * el); void * UnlinkNeLL(void * el); void * UnlinkPrLL(void * el); int IsElmLL (void * p_elm); int IsFirstElmLL (void * p_elm); int IsLastElmLL (void * p_elm); long IndexElmLL (t_LL list, void *ind_el); ForeachLL_M(list,p_elm) ForeachDownLL_M(list,p_elm) SafeForeachLL_M(list,p_elm,next_p_elm) DESCRIPTION List construction and destruction LL is double-linked list handler. Variables of any type can be stored in a LL list. Individual elements of a list may be of different types. Lists of lists of .. can be created. An instance of a list is created using either ConsLL, ConsCopyLL or ConsPtrLL functions. It is recommended to call one of these functions at the point of declaration of a list variable; result of one of the constructor functions must be assigned to list 'list' before 'list' is passed to any other function in the LL library. ConsLL() creates an empty list. ConsCopyLL(src) creates a new copy of an existing list. ConsPtrLL(src) creates a list of pointers to elements stored in list 'src'. DestLL(list) destroys a list, ie. deletes all elements and frees all memory allocated for 'list'. It should be called at the point where 'list' goes out of scope. Basic functions on lists EmptyLL(list) deletes all elements from list 'list'. EmptyLL returns its parameter - an empty list 'list'. IsEmptyLL(list) returns a non- zero value if 'list' is an empty list, 0 otherwise. ApplyLL(list,function) calls function 'function' in a loop. Pointers to elements in the list are one by one passed as an argument to 'function'. The loop is terminated when either all pointers to elements were passed to the function or the function 'function' returned a non-NULL value. ApplyLL returns NULL in the former case, an the return value of 'function' if it is non-NULL. If 'function' is to be applied to all elements the user-defined 'function' must therefore always return NULL. The latter mode can be used as 'FindElement' if function 'function' returns a pointer to an element satisfying a certain condition. SortLL(list,compare) sort elements in a list - 2 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 according to the compare function. The compare function has an identical format as the function required by qsort(3). SortLL builds an array of references and uses qsort(3). LookInLL(list) creates a look-up table for list 'list' that enables random access to elements. If the type of elements in the list is 'int' than the type of the look-up table is of type int **table. *table[i] returns the value of the i-th element of 'list'. The ordering reflects the position of an element at the time of LookInLL call. Sorting 'list' according to different criteria and creating a look-up table after every sort allows to order elements according to multiple criteria. It is an error to use the look-up table after any function that deletes elements from a list as any the pointer in 'table' can to reference a deallocated part of memory. It is possible to use the table after any other function that inserts, moves, changes value of list elements. *table[i] can be then interpreted as the value of the i-th element of list 'list' at the time of LookInLL call regardless of the fact that the element may be in a different position or a member of a different list. The memory allocated by LookInLL for the table should be freed by calling free(table). Input/Output functions FprintLL(list,file,before,control,after) first writes string 'before' to 'file'. Then contents of all elements is output according to the 'control' string. Finally, the 'after' string is written to 'file'. With two exceptions, the control string is identical to the control string of the printf(3) family of functions. Because Fprintf must distinguish between double and float parameters, %lf, %lg, %le conversion were introduced for fields of type double. The %f, %g, %e conversions are used for fields of type float. The %S conversion is used for a zero-terminated string field. The %s is used for char* field (as in printf). EXAMPLE2 gives an example usage. The format specification can contain printf(3) conversion flags, modifiers, etc. The field suppression character '*' may be used. FprintLL always returns NULL. printLL(list,control) is identical to FprintLL(list,stdout,"",control,"\n"). SprintLL(list,str,before,control,after) writes its output to string 'str', otherwise it is identical to FprintLL. User must make sure that 'str' is large enough to hold the output string. If NULL is passed instead of 'str', a large internal buffer is used instead. SprintLL returns 'str' or a pointer to the internal buffer if 'str' was NULL. The buffer contents is overwritten after a next call to SprintLL. SscanLL(list,string,control,termination) converts 'string' into structures defined by the control string and appends the structures to 'list'. The operation can be viewed as parsing of the string into a list of identical elements. The parsing is terminated either after converting a 'termination' number of elements (if 'termination' is positive) or after the end of 'string' was reached (if 'termination is 0). If 'termination' is equal to -1, then the first token of 'string' defines the number of tokens converted. To skip data in 'string', use a conversion with '*' suppressing the assigment. See EXAMPLE3 for - 3 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 details. Inserting and Deleting elements Macros InsBefLL, InsAftLL, InsFirstLL, InsLastLL and functions InsBefLLf, InsAftLLf, InsFirstLLf, InsLastLLf allow user to insert data of any type into a list. After InsFirstLL(f)/InsLastLL(f), the new element will become the first/last element of a list. Using InsAftLL(f)/InsBefLL(f), data can be inserted in an arbitrary place in a list after/before a given element. The functions (as opposed to macros) must be used when size of the of the inserted object is not known at compilation time (for instance in the case of a zero- terminated string). Insert functions creat a new element (element = stored object (data) + neccessary overhead (links etc.)) by allocating memory and copying the object in the element. DelElmLL, DelElmNeLL, DelElmPrLL delete an element from a list. DelElmNeLL returns a pointer to the next element, DelElmPrLL to the previous element. See EXAMPLE4. Moving Strings and Substrings The Move family of functions can be used to cut and paste parts of strings. All Move functions have the same structure: MoveSomethingSomewhereLL. The structure MoveSomethingSomewhereLL is similar to the naming system of insert functions (eg. InsLastElmLL). MoveList___LL moves the whole source list (ie. the src list becomes empty) somewhere (First,Last,Aft(er) element, Bef(ore)) into the destination list. The Head and Tail functions are analogous, but just Head (all elements up to (not including) the 'head' one) or Tail(all elements from 'tail'(including) to end) is moved. The including/not including choice was made to preserve Head+Tail=List. Getting a particular element FirstElmLL(list)/LastElmLL(list) return a pointer to the first element in 'list'. NextElmLL(p_elm)/PrevElmLL(p_elm) return a pointer to the next/previous element in 'list'. To access all elements in a list the following construct can be used: for(p_elm=FirstElmLL(list); IsElmLL(p_elm); p_elm=NextElmLL(p_elm)) The construct is used so often that the LL package provides it in a form of the ForeachLL_M macro. A similar macro, ForeachDownLL_M, can be used to scan all the elements starting with the last. The user must ensure that the current 'p_elm' is not deleted in the body of the for loop - the p_elm=NextElmLL(p_elm) would fail. Use SafeForeachLL_M in this case. NthElmLL(list,n) returns a pointer to the n-th element of the list. 'n' can be negative. Note that the cost of the operations is proportional to 'n'; this functions should be therefore used sparingly; if random access to elements is required, use LookInLL. RelNthElmLL(p_elm,n) returns a pointer to an element with a distance (relative position) 'n' from 'p_elm'. IndexElmLL(list,p_elm) returns the position (order) of 'p_elm' in 'list'. Miscellaneous function IsElmLL(p_elm) tests for the end of a list. IsFirstElmLL/IsLastElmLL - 4 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 test for the first and last element of the list. EXAMPLE1 - ConsLL, DestLL, ApplyLL, SortLL, LookInLL, InsLastLL usage /*----------------------------------------------------------------------*/ #include "LL.h" #include <stdio.h> int cmp(void *n1, void *n2) /* comparison function for SortLL */ { if ( *(int*)n1 > *(int*)n2) return 1; if ( *(int*)n1 < *(int*)n2) return -1; return 0; } void *negate(void *n) /* function called by ApplyLL */ { *(int*)n=-*(int*)n; return NULL; /* do it for all elements */ } int main(int argc, char ** argv) { t_LL num = ConsLL(); int **look_tab1, **look_tab_s, **look_tab_n; int some_num [] = { 7, 3 , 8, 1, 19}; { /* insert all numbers in some_num into the list */ int i; printf("list: "); for (i=0; i < sizeof(some_num)/sizeof(int); i++){ InsLastLL(num,some_num[i]); printf("%d ",some_num[i]); } printf("0); } look_tab1 = LookInLL(num); /* accessing some random elements */ printf("start: 3rd and 4th : %d %d0, *look_tab1[3],*look_tab1[4]); SortLL(num,cmp); look_tab_s = LookInLL(num); printf("original: 3rd and 4th : %d %d0, *look_tab1[3],*look_tab1[4]); printf("sorted: 3rd and 4th : %d %d0, *look_tab_s[3],*look_tab_s[4]); ApplyLL(num,negate); SortLL(num,cmp); look_tab_n = LookInLL(num); printf("original: 3rd and 4th : %d %d0, *look_tab1[3], *look_tab1[4]); printf("sorted: 3rd and 4th : %d %d0, *look_tab_s[3],*look_tab_s[4]); printf("neg-sort: 3rd and 4th : %d %d0, *look_tab_n[3],*look_tab_n[4]); - 5 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 DestLL(num); /* clean up, destroy list */ free(look_tab1); free(look_tab_s); free(look_tab_n); return 0; } /*--------------------------------------------------------------------*/ EXAMPLE2 - FprintLL, ConsLL, DestLL, InsLastLL usage /*--------------------------------------------------------------------*/ #include "LL.h" #include <stdio.h> int main(int argc, char ** argv) { t_LL num = ConsLL(); int some_num [] = { 7, 3 , 8, 1, 19}; { /* insert all numbers in some_num into the list */ int i; for (i=0; i < sizeof(some_num)/sizeof(int); i++) InsLastLL(num,some_num[i]); } /* print the list of integers in various formats */ FprintLL(num,stdout,"Plain : ","%d ","0); FprintLL(num,stdout,"2digits: ","%2d ","0); FprintLL(num,stdout,"Element_a_line :0,"%d0,"0); FprintLL(num,stdout,"Elm= - : ","%*d-","0); DestLL(num); /* clean up, destroy list */ return 0; } /*--------------------------------------------------------------------*/ EXAMPLE3 - SscanLL, FprintLL, ConsLL, DestLL, EmptyLL usage /*--------------------------------------------------------------------*/ #include "LL.h" #include <stdio.h> int main(int argc, char ** argv) { t_LL num = ConsLL(); char * some_num_str = " 7 3 8 1 21 25 19 10 10 10"; SscanLL(num,some_num_str,"%d",0); /* get all numbers */ FprintLL(num,stdout,"List: ","%d ","0); EmptyLL(num); /* remove old elms */ - 6 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 SscanLL(num,some_num_str,"%d %*d",0); /* get odd order numbers*/ FprintLL(num,stdout,"Odd Pos: ","%d ","0); EmptyLL(num); /* remove old elms */ SscanLL(num,some_num_str,"%d",5); /* get first 5 numbers */ FprintLL(num,stdout,"1-5: ","%d ","0); EmptyLL(num); /* remove old elms */ SscanLL(num,some_num_str,"%d",-1); /* convert according to 1st num*/ FprintLL(num,stdout,"List: ","%d ","0); DestLL(num); /* clean up, destroy list */ return 0; } /*--------------------------------------------------------------------*/ EXAMPLE4 - Inserting and Deleting Elms /*----------------------------------------------------------------------*/ #include "LL.h" #include <stdio.h> #include <string.h> /* for strlen */ int main(int argc, char ** argv) { t_LL list = ConsLL(); char * strings[] = {"S1","S2","S3","forth"}; float numbs[] = {1.2,3.4,5.6}; float * numb; numb = InsFirstLL(list,numbs[0]); /*remember ptr to the 1st insert. elm*/ InsLastLL(list,numbs[1]); InsLastLL(list,numbs[2]); InsAftLL(numb,numbs[2]); /* numb[2] is Last and second */ FprintLL(list,stdout,"List : ","%3.1f ","0); DelElmLL(numb); /* delete one (first) element */ FprintLL(list,stdout,"List : ","%3.1f ","0); for(numb=FirstElmLL(list); IsElmLL(numb); numb=DelElmNeLL(numb)) ; /* empty the list*/ InsFirstLLf(list,strlen(strings[1])+1,strings[1]); /* + 1 for term. zero */ InsFirstLLf(list,strlen(strings[2])+1,strings[2]); FprintLL(list,stdout,"List :0,"%S0,"0); DestLL(list); /* clean up, destroy list */ return 0; } /*----------------------------------------------------------------------*/ - 7 - Formatted: January 14, 2025 LL(3) LL(3) 16 June 1992 SEE ALSO qsort(3) BUGS The input/output family of functions has not been tested for all possible format strings. Moreover, these functions are not portable. No padding inside structures (for alignment) is assumed. I recommend to use these functions for debugging prints only. AUTHOR George Matas, University of Surrey; g.matas@ee.surrey.ac.uk. Thanks to Duane Morse for the idea of representing an empty list by a circular list with a dummy node and to Radek Marik and Radim Sara for numerous discussions on list implementation. - 8 - Formatted: January 14, 2025