This repository has been archived on 2021-11-20. You can view files and clone it, but cannot push or open issues or pull requests.
llist/llist.c

216 lines
4.4 KiB
C

/*
* This file is part of corelibs. (https://git.redxen.eu/corelibs)
* Copyright (c) 2021 Alex-David Denes
*
* corelibs is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* any later version.
*
* corelibs 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. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with corelibs. If not, see <https://www.gnu.org/licenses/>.
*/
#include "llist.h"
#include <stdint.h> // uintmax_t
#include <stdlib.h> // malloc() free()
static llist_err corelibs_llist_create(void*, void (*)(void*), llist_t**);
static llist_err corelibs_llist_link(llist_t*, llist_t*);
static llist_err corelibs_llist_free(llist_t*);
static llist_err corelibs_llist_set_cont(llist_t*, void*);
static llist_err corelibs_llist_set_free(llist_t*, void (*)(void*));
static llist_err corelibs_llist_get_next(const llist_t*, llist_t**);
static llist_err corelibs_llist_get_prev(const llist_t*, llist_t**);
static llist_err corelibs_llist_get_cont(const llist_t*, void**);
static void corelibs_llist_remove(llist_t*);
struct llist_t {
struct llist_t *next, *prev;
void* cont;
void (*free)(void*);
};
enum { // Avoid collision in error numbers
CORELIBS_LLIST_ERR_UNDEF = -3,
CORELIBS_LLIST_ERR_NULL,
CORELIBS_LLIST_ERR_ALLOC,
CORELIBS_LLIST_ERR_UNK = 0,
CORELIBS_LLIST_ERR_OK = 1,
};
struct corelibs_llist_interface const llist = {
.create = corelibs_llist_create,
.link = corelibs_llist_link,
.free = corelibs_llist_free,
.set = {
.cont = corelibs_llist_set_cont,
.free = corelibs_llist_set_free,
},
.get = {
.prev = corelibs_llist_get_prev,
.cont = corelibs_llist_get_cont,
.next = corelibs_llist_get_next,
},
};
static llist_err
corelibs_llist_create(void* c, void (*const f)(void*), llist_t** s)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
llist_t* cur = malloc(sizeof(*cur));
if (cur == NULL) {
err = CORELIBS_LLIST_ERR_ALLOC;
goto ret;
}
cur->prev = NULL;
cur->next = NULL;
cur->cont = c;
cur->free = f;
*s = cur;
ret:
return err;
}
static llist_err
corelibs_llist_free(llist_t* l)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (l == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
// Free previous elements if we have any
if (l->prev != NULL) {
for (llist_t* p = l->prev; p != NULL;) {
llist_t* c = p;
p = p->prev;
corelibs_llist_remove(c);
}
}
// Free current and next elements
for (llist_t* p = l; p != NULL;) {
llist_t* c = p;
p = p->next;
corelibs_llist_remove(c);
}
ret:
return err;
}
static llist_err
corelibs_llist_link(llist_t* a, llist_t* b)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (a == NULL && b == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
if (a != NULL) a->next = b;
if (b != NULL) b->prev = a;
ret:
return err;
}
static llist_err
corelibs_llist_get_next(const llist_t* e, llist_t** save)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (e == NULL || save == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
*save = e->next;
ret:
return err;
}
static llist_err
corelibs_llist_get_prev(const llist_t* e, llist_t** save)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (e == NULL || save == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
*save = e->prev;
ret:
return err;
}
static llist_err
corelibs_llist_get_cont(const llist_t* e, void** save)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (e == NULL || save == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
*save = e->cont;
ret:
return err;
}
static llist_err
corelibs_llist_set_cont(llist_t* e, void* i)
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (e == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
e->cont = i;
ret:
return err;
}
static llist_err
corelibs_llist_set_free(llist_t* e, void (*const f)(void*))
{
llist_err err = CORELIBS_LLIST_ERR_OK;
if (e == NULL) {
err = CORELIBS_LLIST_ERR_NULL;
goto ret;
}
e->free = f;
ret:
return err;
}
// Private function
static void
corelibs_llist_remove(llist_t* e)
{
corelibs_llist_link(e->prev, e->next);
if (e->free != NULL) e->free(e->cont);
free(e);
}