Line data Source code
1 : /** 2 : * @file sclist.c 3 : * @brief Basic single chained generic list 4 : * 5 : * @author François Cerbelle (Fanfan), francois@cerbelle.net 6 : * 7 : * @internal 8 : * Created: 15/11/2024 9 : * Revision: none 10 : * Last modified: 2024-11-29 17:58 11 : * Compiler: gcc 12 : * Organization: Cerbelle.net 13 : * Copyright: Copyright (c) 2024, François Cerbelle 14 : * 15 : * This source code is released for free distribution under the terms of the 16 : * GNU General Public License as published by the Free Software Foundation. 17 : */ 18 : 19 : #ifdef HAVE_CONFIG_H 20 : #include "config.h" 21 : #endif 22 : 23 : #include "sclist.h" 24 : 25 : #include <stdlib.h> 26 : #include <stdio.h> 27 : #include <assert.h> 28 : 29 : /** Private list record structure 30 : */ 31 : typedef struct sclistrecord_s { 32 : void* value; /**< Pointer to value */ 33 : struct sclistrecord_s* next; /**< Next record in the list */ 34 : } sclistrecord_t; 35 : 36 : /** Opaque sclist structure 37 : */ 38 : typedef struct sclist_s { 39 : sclistrecord_t* first; /**< Pointer to the first record */ 40 : sclistrecord_t* last; /**< Pointer to the last record */ 41 : } sclist_t; 42 : 43 221 : sclist_t* sclist_new() { 44 : sclist_t* sclist; 45 221 : if (NULL==(sclist=malloc(sizeof(struct sclist_s)))) { 46 0 : perror("sclist_new OOM"); 47 0 : abort(); 48 : } 49 221 : sclist->first=NULL; 50 221 : sclist->last=NULL; 51 : 52 221 : return sclist; 53 : } 54 : 55 152 : void sclist_del(sclist_t* sclist) { 56 152 : assert(sclist); 57 : 58 161 : while (sclist->first!=NULL) { 59 9 : sclistrecord_t* first = sclist->first; 60 9 : sclist->first = sclist->first->next; 61 9 : free(first); 62 : } 63 152 : free(sclist); 64 152 : } 65 : 66 343 : sclistrecord_t* sclist_addrecord(sclist_t* sclist, void* value) { 67 : sclistrecord_t* record; 68 : 69 343 : assert(sclist); 70 : 71 : /* Create the record record */ 72 343 : if (NULL==(record=malloc(sizeof(struct sclistrecord_s)))) { 73 0 : perror("sclist_addrecord OOM"); 74 0 : abort(); 75 : } 76 343 : record->value = value; 77 343 : record->next = NULL; 78 : 79 : /* Add to the list */ 80 343 : if (NULL==sclist->first) { 81 : /* First record to be added in an empty list */ 82 : /* A sentinel record would avoid this test and optimize performances 83 : * at the cost of sentinel size RAM consumption */ 84 152 : sclist->first = record; 85 152 : sclist->last = record; 86 : } else { 87 : /* Add to the end of a non-empty list */ 88 191 : sclist->last->next = record; 89 191 : sclist->last = record; 90 : } 91 343 : return record; 92 : } 93 : 94 214 : sclist_t* sclist_remrecord(sclist_t* sclist, sclistrecord_t* record) { 95 : sclistrecord_t* cur; 96 : sclistrecord_t* prev; 97 : 98 214 : assert(sclist); 99 214 : assert(record); 100 : 101 247 : for (cur=sclist->first, prev=NULL; (cur)&&(cur!=record); prev=cur,cur=cur->next); 102 : 103 : /* If found */ 104 214 : if (NULL!=cur) { 105 : /* Remove from the chain */ 106 206 : if (sclist->first==cur) 107 191 : sclist->first = cur->next; 108 : else 109 15 : prev->next = cur->next; 110 : /* Update the last pointer if needed */ 111 206 : if (sclist->last==cur) 112 119 : sclist->last = prev; 113 206 : free(cur); 114 206 : return sclist; 115 : } else { 116 8 : fprintf(stderr,"ERROR: sclist_remrecord record not found.\n"); 117 8 : return NULL; 118 : } 119 : } 120 : 121 354 : sclistrecord_t* sclist_firstrecord(const sclist_t* sclist) { 122 354 : assert(sclist); 123 : 124 354 : return sclist->first; 125 : } 126 : 127 438 : sclistrecord_t* sclist_nextrecord(const sclistrecord_t* record) { 128 438 : assert(record); 129 : 130 438 : return record->next; 131 : } 132 : 133 571 : void* sclist_getvalue(sclistrecord_t* record) { 134 571 : assert(record); 135 : 136 571 : return record->value; 137 : } 138 : /* vim: set tw=80: */