libneo/include/neo/list.h

177 lines
4.7 KiB
C
Raw Permalink Normal View History

2021-07-26 16:54:05 +02:00
/* See the end of this file for copyright and license terms. */
2021-07-18 21:10:43 +02:00
#pragma once
#ifdef __cplusplus
extern "C" {
#endif
#include "neo/_stddef.h"
#include "neo/_types.h"
struct _neo_list;
2021-07-26 22:40:50 +02:00
/** @private */
2021-07-18 21:10:43 +02:00
struct _neo_listnode {
struct _neo_listnode *_next;
struct _neo_listnode *_prev;
struct _neo_list *_list;
};
/**
2021-07-26 16:54:05 +02:00
* @brief List node anchor for embedding into your own structure.
*
* @ingroup list
2021-07-18 21:10:43 +02:00
*/
typedef struct _neo_listnode listnode_t;
2021-07-26 22:40:50 +02:00
/** @private */
2021-07-18 21:10:43 +02:00
struct _neo_list {
struct _neo_listnode _root;
NLEN_FIELD(_len);
};
/**
2021-07-26 16:54:05 +02:00
* @brief List data type
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @ingroup list
2021-07-18 21:10:43 +02:00
*/
2021-07-26 16:54:05 +02:00
typedef struct _neo_list list_t;
2021-07-18 21:10:43 +02:00
/**
2021-07-26 16:54:05 +02:00
* When casting out from a @a `listnode_t` to its supposed containing structure
* (in a loop), it is possible that it was in fact the @a `list_t` that contained
* the @a `listnode_t`. This macro checks whether that is the case.
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @param casted `struct *` that the @a `listnode_t *` was casted out to
* @param list @a `list_t *` storing the root @a `listnode_t`
* @param member Name of the @a `listnode_t` member embedded within the `struct *`
2021-07-26 22:40:50 +02:00
* @private
2021-07-18 21:10:43 +02:00
*/
#define _neo_list_is_root(casted, list, member) \
( &(casted)->member == &(list)->_root )
/*
* we use u8 * for pointer arithmetic here because this is code is compiled with
* external compiler settings which my complain about calculating with void *
*/
2021-07-26 22:40:50 +02:00
/** @private */
#define _neo_list_first(list, type, member) \
((type *)( (u8 *)((list)->_root._next) - offsetof(type, member) ))
2021-07-18 21:10:43 +02:00
2021-07-26 22:40:50 +02:00
/** @private */
#define _neo_list_last(list, type, member) \
((type *)( (u8 *)((list)->_root._prev) - offsetof(type, member) ))
2021-07-18 21:10:43 +02:00
2021-07-26 22:40:50 +02:00
/** @private */
#define _neo_list_next(current, member) \
((typeof(current))( (u8 *)((current)->member._next) - offsetof(typeof(*(current)), member) ))
2021-07-18 21:10:43 +02:00
2021-07-26 22:40:50 +02:00
/** @private */
#define _neo_list_prev(current, member) \
((typeof(current))( (u8 *)((current)->member._prev) - offsetof(typeof(*(current)), member) ))
2021-07-18 21:10:43 +02:00
/**
2021-07-26 16:54:05 +02:00
* @defgroup list List API
*
* @{
*/
/**
2021-07-26 22:40:50 +02:00
* @brief Initialize a list.
2021-07-26 16:54:05 +02:00
*
* @param list The list
*/
void list_init(list_t *list);
/**
* @brief Append a new node to the end of a list.
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @param list List to append to
* @param new_node New node to append
2021-07-18 21:10:43 +02:00
*/
void list_add(list_t *list, listnode_t *new_node);
2021-07-18 21:10:43 +02:00
/**
2021-07-26 16:54:05 +02:00
* @brief Remove a node from a list.
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @param node Node to remove
2021-07-18 21:10:43 +02:00
*/
void list_del(listnode_t *node);
/**
2021-07-26 16:54:05 +02:00
* @brief Insert a new node to the beginning of a list.
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @param list List to insert the node into
* @param new_node New node to insert
2021-07-18 21:10:43 +02:00
*/
void list_add_first(list_t *list, listnode_t *new_node);
2021-07-18 21:10:43 +02:00
/**
2021-07-26 22:40:50 +02:00
* @brief Insert a new node after the specified list node.
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @param pos List node to insert the new node after
* @param new_node Node to insert after the specified position
2021-07-18 21:10:43 +02:00
*/
void list_insert(listnode_t *pos, listnode_t *new_node);
2021-07-18 21:10:43 +02:00
/**
2021-07-26 22:40:50 +02:00
* @brief Insert a new node before the specified list node.
2021-07-18 21:10:43 +02:00
*
2021-07-26 16:54:05 +02:00
* @param pos List node to insert the new node before
* @param new_node Node to insert before the specified position
2021-07-18 21:10:43 +02:00
*/
void list_insert_before(listnode_t *pos, listnode_t *new_node);
2021-07-18 21:10:43 +02:00
/**
2021-07-26 16:54:05 +02:00
* @brief Iterate over each item in a list.
2021-07-18 21:10:43 +02:00
*
* The current entry can be safely removed from the list.
*
2021-07-26 16:54:05 +02:00
* @param cursor `type *` to use as a cursor
* @param list `list_t *` to iterate over
2021-07-26 22:40:50 +02:00
* @param member Name of the `listnode_t` member embedded within `cursor`
2021-07-18 21:10:43 +02:00
*/
#define list_foreach(cursor, list, member) \
for (typeof(cursor) __tmp = _neo_list_next( \
cursor = _neo_list_first( list, typeof(*(cursor)), member ), \
member \
); \
2021-07-18 21:10:43 +02:00
!_neo_list_is_root(cursor, list, member); \
cursor = __tmp, __tmp = _neo_list_next(__tmp, member))
/**
2021-07-26 16:54:05 +02:00
* @brief Iterate over each item in a list in reverse order.
2021-07-18 21:10:43 +02:00
*
* The current entry can be safely removed from the list.
*
2021-07-26 16:54:05 +02:00
* @param cursor `type *` to use as a cursor
* @param list `list_t *` to iterate over
2021-07-26 22:40:50 +02:00
* @param member Name of the `listnode_t` member embedded within `cursor`
2021-07-18 21:10:43 +02:00
*/
#define list_foreach_reverse(cursor, list, member) \
for (typeof(cursor) __tmp = _neo_list_prev( \
cursor = _neo_list_last( list, typeof(*(cursor)), member ), \
member \
); \
2021-07-18 21:10:43 +02:00
!_neo_list_is_root(cursor, list, member); \
cursor = __tmp, __tmp = _neo_list_prev(__tmp, member))
2021-07-26 16:54:05 +02:00
/** @} */
2021-07-18 21:10:43 +02:00
#ifdef __cplusplus
}; /* extern "C" */
#endif
/*
* This file is part of libneo.
* Copyright (c) 2021 Fefie <owo@fef.moe>.
*
* libneo is non-violent software: you may only use, redistribute,
* and/or modify it under the terms of the CNPLv6+ as found in
* the LICENSE file in the source code root directory or at
* <https://git.pixie.town/thufie/CNPL>.
*
* libneo comes with ABSOLUTELY NO WARRANTY, to the extent
* permitted by applicable law. See the CNPLv6+ for details.
*/