/* * (C)opyright MMIV-MMV Anselm R. Garbe * See LICENSE file for license details. */ #include #include "cext.h" static CItem zero_item = { 0 }; static int comp_ptr(void *p1, void *p2) { return p1 == p2; } static void detach_from_stack(CItem *i) { /* remove from stack */ if (i->up) i->up->down = i->down; if (i->down) i->down->up = i->up; } static void attach_to_stack(Container *c, CItem *i) { if (!c->stack) c->stack = i; else { i->down = c->stack; c->stack->up = i; c->stack = i; } } void cext_attach_item(Container *c, void *item) { CItem *i, *new = cext_emalloc(sizeof(CItem)); *new = zero_item; new->item = item; for (i = c->list; i && i->next; i = i->next); if (!c->list) c->list = new; else i->next = new; attach_to_stack(c, new); } void cext_detach_item(Container *c, void *item) { CItem *i = c->list; if (!i) return; /* remove from list */ if (i->item == item) c->list = c->list->next; else { for (; i->next && i->next->item != item; i = i->next); if (!i->next) return; i->next = i->next->next; } detach_from_stack(i); free(i); } void *cext_find_item(Container *c, void *pattern, int (*comp)(void *pattern, void *item)) { CItem *i; for (i = c->list; i && comp(pattern, i->item); i = i->next); return i ? i->item : nil; } void cext_iterate(Container *c, void *aux, void (*iter)(void *, void *aux)) { CItem *i; for (i = c->list; i; i = i->next) iter(i->item, aux); } void cext_top_item(Container *c, void *item) { CItem *i = cext_find_item(c, item, comp_ptr); if (!i) return; detach_from_stack(i); attach_to_stack(c, i); } void *cext_get_top_item(Container *c) { return c->stack ? c->stack->item : nil; } void *cext_get_down_item(Container *c, void *item) { CItem *i = cext_find_item(c, item, comp_ptr); if (!i) return nil; return i->down ? i->down->item : c->stack->item; } void *cext_get_up_item(Container *c, void *item) { CItem *i = cext_find_item(c, item, comp_ptr); CItem *bottom; if (!i) return nil; for (bottom = c->stack; bottom && bottom->down; bottom = bottom->down); return i->up ? i->up->item : bottom->item; } void *cext_get_item(Container *c, size_t index) { size_t idx = 0; CItem *i; for (i = c->list; i && index != idx; i = i->next) idx++; return i ? i->item : nil; } int cext_get_item_index(Container *c, void *item) { int idx = 0; CItem *i; for (i = c->list; i && i->item != item; i = i->next) idx++; return i ? idx : -1; } size_t cext_sizeof(Container *c) { size_t idx = 0; CItem *i; for (i = c->list; i; i = i->next) idx++; return idx; } void cext_swap_items(Container *c, void *item1, void *item2) { CItem *i1 = cext_find_item(c, item1, comp_ptr); CItem *i2 = cext_find_item(c, item2, comp_ptr); i1->item = item2; i2->item = item1; } /* old obsolete stuff follows */ void **attach_item_begin(void **old, void *item, size_t size_item) { int i, size_old; void **result = 0; for (size_old = 0; old && old[size_old]; size_old++); result = cext_emalloc(size_item * (size_old + 2)); result[0] = item; for (i = 0; old && old[i]; i++) result[i + 1] = old[i]; result[i + 1] = 0; if (old) free(old); return result; } void **attach_item_end(void **old, void *item, size_t size_item) { int i, size_old; void **result = 0; for (size_old = 0; old && old[size_old]; size_old++); result = cext_emalloc(size_item * (size_old + 2)); for (i = 0; old && old[i]; i++) result[i] = old[i]; result[i++] = item; result[i] = 0; if (old) free(old); return result; } void **detach_item(void **old, void *item, size_t size_item) { int size_old, i, j = 0; void **result = 0; for (size_old = 0; old && old[size_old]; size_old++); if (size_old != 1) { result = cext_emalloc(size_item * size_old); for (i = 0; old[i]; i++) if (old[i] != item) result[j++] = old[i]; result[j] = 0; } if (old) free(old); return result; } int index_item(void **items, void *item) { int i = 0; for (i = 0; items && items[i] && (items[i] != item); i++); return items[i] ? i : -1; } int count_items(void **items) { int i; for (i = 0; items && items[i]; i++); return i; } int index_next_item(void **items, void *item) { int idx = index_item(items, item); if (idx == -1) return idx; if (idx == count_items(items) - 1) return 0; else return idx + 1; } int index_prev_item(void **items, void *item) { int idx = index_item(items, item); if (idx == -1) return idx; if (idx == 0) return count_items(items) - 1; else return idx - 1; }