1
0
mirror of https://github.com/lvgl/lvgl.git synced 2025-01-21 06:53:01 +08:00
lvgl/lv_misc/lv_ll.c

392 lines
10 KiB
C
Raw Normal View History

2017-11-23 20:42:14 +01:00
/**
2017-11-24 17:48:47 +01:00
* @file lv_ll.c
2017-11-23 20:42:14 +01:00
* Handle linked lists.
2017-11-24 17:48:47 +01:00
* The nodes are dynamically allocated by the 'lv_mem' module,
2017-11-23 20:42:14 +01:00
*/
/*********************
* INCLUDES
*********************/
#include <stdint.h>
#include <string.h>
#include "lv_ll.h"
#include "lv_mem.h"
/*********************
* DEFINES
*********************/
2017-11-24 17:48:47 +01:00
#define LL_NODE_META_SIZE (sizeof(lv_ll_node_t*) + sizeof(lv_ll_node_t*))
2017-11-23 20:42:14 +01:00
#define LL_PREV_P_OFFSET(ll_p) (ll_p->n_size)
2017-11-24 17:48:47 +01:00
#define LL_NEXT_P_OFFSET(ll_p) (ll_p->n_size + sizeof(lv_ll_node_t*))
2017-11-23 20:42:14 +01:00
/**********************
* TYPEDEFS
**********************/
/**********************
* STATIC PROTOTYPES
**********************/
2018-06-19 09:49:58 +02:00
static void node_set_prev(lv_ll_t * ll_p, lv_ll_node_t * act, lv_ll_node_t * prev);
static void node_set_next(lv_ll_t * ll_p, lv_ll_node_t * act, lv_ll_node_t * next);
2017-11-23 20:42:14 +01:00
/**********************
* STATIC VARIABLES
**********************/
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
/**
* Initialize linked list
* @param ll_dsc pointer to ll_dsc variable
* @param node_size the size of 1 node in bytes
2017-11-23 20:42:14 +01:00
*/
void lv_ll_init(lv_ll_t * ll_p, uint32_t node_size)
2017-11-23 20:42:14 +01:00
{
ll_p->head = NULL;
ll_p->tail = NULL;
2018-09-27 15:03:20 +02:00
#ifdef LV_MEM_ENV64
/*Round the size up to 8*/
if(node_size & 0x7) {
node_size = node_size & (~0x7);
node_size += 8;
2018-09-27 15:03:20 +02:00
}
#else
/*Round the size up to 4*/
if(node_size & 0x3) {
node_size = node_size & (~0x3);
node_size += 4;
2017-11-23 20:42:14 +01:00
}
2018-09-27 15:03:20 +02:00
#endif
2017-11-23 20:42:14 +01:00
ll_p->n_size = node_size;
2017-11-23 20:42:14 +01:00
}
/**
* Add a new head to a linked list
* @param ll_p pointer to linked list
* @return pointer to the new head
*/
2017-11-24 17:48:47 +01:00
void * lv_ll_ins_head(lv_ll_t * ll_p)
2017-11-23 20:42:14 +01:00
{
2018-06-19 09:49:58 +02:00
lv_ll_node_t * n_new;
2017-11-23 21:28:36 +01:00
n_new = lv_mem_alloc(ll_p->n_size + LL_NODE_META_SIZE);
2018-06-19 09:49:58 +02:00
if(n_new != NULL) {
2017-11-23 20:42:14 +01:00
node_set_prev(ll_p, n_new, NULL); /*No prev. before the new head*/
node_set_next(ll_p, n_new, ll_p->head); /*After new comes the old head*/
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(ll_p->head != NULL) { /*If there is old head then before it goes the new*/
node_set_prev(ll_p, ll_p->head, n_new);
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
ll_p->head = n_new; /*Set the new head in the dsc.*/
if(ll_p->tail == NULL) {/*If there is no tail (1. node) set the tail too*/
ll_p->tail = n_new;
}
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
return n_new;
}
2018-02-15 13:24:56 +01:00
/**
* Insert a new node in front of the n_act node
* @param ll_p pointer to linked list
* @param n_act pointer a node
* @return pointer to the new head
*/
void * lv_ll_ins_prev(lv_ll_t * ll_p, void * n_act)
{
2018-06-19 09:49:58 +02:00
lv_ll_node_t * n_new;
lv_ll_node_t * n_prev;
2018-02-15 13:24:56 +01:00
if(NULL == ll_p || NULL == n_act) return NULL;
if(lv_ll_get_head(ll_p) == n_act) {
n_new = lv_ll_ins_head(ll_p);
if(n_new == NULL) return NULL;
2018-06-19 09:49:58 +02:00
} else {
2018-02-15 13:24:56 +01:00
n_new = lv_mem_alloc(ll_p->n_size + LL_NODE_META_SIZE);
if(n_new == NULL) return NULL;
2018-02-15 13:24:56 +01:00
n_prev = lv_ll_get_prev(ll_p, n_act);
node_set_next(ll_p, n_prev, n_new);
node_set_prev(ll_p, n_new, n_prev);
node_set_prev(ll_p, n_act, n_new);
node_set_next(ll_p, n_new, n_act);
}
return n_new;
}
2017-11-23 20:42:14 +01:00
/**
* Add a new tail to a linked list
* @param ll_p pointer to linked list
* @return pointer to the new tail
*/
2017-11-24 17:48:47 +01:00
void * lv_ll_ins_tail(lv_ll_t * ll_p)
2018-06-19 09:49:58 +02:00
{
lv_ll_node_t * n_new;
2017-11-23 21:28:36 +01:00
n_new = lv_mem_alloc(ll_p->n_size + LL_NODE_META_SIZE);
if(n_new == NULL) return NULL;
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(n_new != NULL) {
node_set_next(ll_p, n_new, NULL); /*No next after the new tail*/
node_set_prev(ll_p, n_new, ll_p->tail); /*The prev. before new is tho old tail*/
if(ll_p->tail != NULL) { /*If there is old tail then the new comes after it*/
node_set_next(ll_p, ll_p->tail, n_new);
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
ll_p->tail = n_new; /*Set the new tail in the dsc.*/
if(ll_p->head == NULL) { /*If there is no head (1. node) set the head too*/
ll_p->head = n_new;
}
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
return n_new;
}
/**
2018-06-19 09:49:58 +02:00
* Remove the node 'node_p' from 'll_p' linked list.
2018-10-30 09:18:45 +01:00
* It does not free the the memory of node.
2017-11-23 20:42:14 +01:00
* @param ll_p pointer to the linked list of 'node_p'
* @param node_p pointer to node in 'll_p' linked list
*/
2017-11-24 17:48:47 +01:00
void lv_ll_rem(lv_ll_t * ll_p, void * node_p)
2017-11-23 20:42:14 +01:00
{
2017-11-24 17:48:47 +01:00
if(lv_ll_get_head(ll_p) == node_p) {
2017-11-23 20:42:14 +01:00
/*The new head will be the node after 'n_act'*/
2017-11-24 17:48:47 +01:00
ll_p->head = lv_ll_get_next(ll_p, node_p);
2017-11-23 20:42:14 +01:00
if(ll_p->head == NULL) {
ll_p->tail = NULL;
2018-06-19 09:49:58 +02:00
} else {
2017-11-23 20:42:14 +01:00
node_set_prev(ll_p, ll_p->head, NULL);
2018-06-19 09:49:58 +02:00
}
} else if(lv_ll_get_tail(ll_p) == node_p) {
2017-11-23 20:42:14 +01:00
/*The new tail will be the node before 'n_act'*/
2017-11-24 17:48:47 +01:00
ll_p->tail = lv_ll_get_prev(ll_p, node_p);
2017-11-23 20:42:14 +01:00
if(ll_p->tail == NULL) {
ll_p->head = NULL;
2018-06-19 09:49:58 +02:00
} else {
2017-11-23 20:42:14 +01:00
node_set_next(ll_p, ll_p->tail, NULL);
}
2018-06-19 09:49:58 +02:00
} else {
lv_ll_node_t * n_prev = lv_ll_get_prev(ll_p, node_p);
lv_ll_node_t * n_next = lv_ll_get_next(ll_p, node_p);
2017-11-23 20:42:14 +01:00
node_set_next(ll_p, n_prev, n_next);
node_set_prev(ll_p, n_next, n_prev);
}
}
/**
2017-11-24 17:48:47 +01:00
* Remove and free all elements from a linked list. The list remain valid but become empty.
2017-11-23 20:42:14 +01:00
* @param ll_p pointer to linked list
*/
2017-11-24 17:48:47 +01:00
void lv_ll_clear(lv_ll_t * ll_p)
2017-11-23 20:42:14 +01:00
{
2018-06-19 09:49:58 +02:00
void * i;
void * i_next;
2017-11-23 20:42:14 +01:00
2018-06-19 09:49:58 +02:00
i = lv_ll_get_head(ll_p);
i_next = NULL;
2017-11-23 20:42:14 +01:00
2018-06-19 09:49:58 +02:00
while(i != NULL) {
i_next = lv_ll_get_next(ll_p, i);
2017-11-23 20:42:14 +01:00
2018-06-19 09:49:58 +02:00
lv_ll_rem(ll_p, i);
lv_mem_free(i);
2017-11-23 20:42:14 +01:00
2018-06-19 09:49:58 +02:00
i = i_next;
}
2017-11-23 20:42:14 +01:00
}
/**
* Move a node to a new linked list
* @param ll_ori_p pointer to the original (old) linked list
* @param ll_new_p pointer to the new linked list
* @param node pointer to a node
*/
2017-11-24 17:48:47 +01:00
void lv_ll_chg_list(lv_ll_t * ll_ori_p, lv_ll_t * ll_new_p, void * node)
2017-11-23 20:42:14 +01:00
{
2017-11-24 17:48:47 +01:00
lv_ll_rem(ll_ori_p, node);
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
/*Set node as head*/
2018-06-19 09:49:58 +02:00
node_set_prev(ll_new_p, node, NULL);
2017-11-23 20:42:14 +01:00
node_set_next(ll_new_p, node, ll_new_p->head);
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(ll_new_p->head != NULL) { /*If there is old head then before it goes the new*/
node_set_prev(ll_new_p, ll_new_p->head, node);
}
ll_new_p->head = node; /*Set the new head in the dsc.*/
if(ll_new_p->tail == NULL) { /*If there is no tail (first node) set the tail too*/
ll_new_p->tail = node;
}
}
/**
* Return with head node of the linked list
* @param ll_p pointer to linked list
2018-06-19 09:49:58 +02:00
* @return pointer to the head of 'll_p'
2017-11-23 20:42:14 +01:00
*/
void * lv_ll_get_head(const lv_ll_t * ll_p)
2018-06-19 09:49:58 +02:00
{
2017-11-23 20:42:14 +01:00
void * head = NULL;
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(ll_p != NULL) {
head = ll_p->head;
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
return head;
}
/**
* Return with tail node of the linked list
* @param ll_p pointer to linked list
* @return pointer to the head of 'll_p'
*/
void * lv_ll_get_tail(const lv_ll_t * ll_p)
2017-11-23 20:42:14 +01:00
{
void * tail = NULL;
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(ll_p != NULL) {
tail = ll_p->tail;
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
return tail;
}
/**
2018-06-19 09:49:58 +02:00
* Return with the pointer of the next node after 'n_act'
2017-11-23 20:42:14 +01:00
* @param ll_p pointer to linked list
2018-06-19 09:49:58 +02:00
* @param n_act pointer a node
* @return pointer to the next node
2017-11-23 20:42:14 +01:00
*/
2018-10-05 17:22:49 +02:00
void * lv_ll_get_next(const lv_ll_t * ll_p, const void * n_act)
2017-11-23 20:42:14 +01:00
{
void * next = NULL;
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(ll_p != NULL) {
2018-10-05 17:22:49 +02:00
const lv_ll_node_t * n_act_d = n_act;
2017-11-23 20:42:14 +01:00
memcpy(&next, n_act_d + LL_NEXT_P_OFFSET(ll_p), sizeof(void *));
2018-06-19 09:49:58 +02:00
}
2017-11-23 20:42:14 +01:00
return next;
}
/**
2018-06-19 09:49:58 +02:00
* Return with the pointer of the previous node after 'n_act'
2017-11-23 20:42:14 +01:00
* @param ll_p pointer to linked list
2018-06-19 09:49:58 +02:00
* @param n_act pointer a node
* @return pointer to the previous node
2017-11-23 20:42:14 +01:00
*/
2018-10-05 17:22:49 +02:00
void * lv_ll_get_prev(const lv_ll_t * ll_p, const void * n_act)
2017-11-23 20:42:14 +01:00
{
void * prev = NULL;
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
if(ll_p != NULL) {
2018-10-05 17:22:49 +02:00
const lv_ll_node_t * n_act_d = n_act;
2017-11-23 20:42:14 +01:00
memcpy(&prev, n_act_d + LL_PREV_P_OFFSET(ll_p), sizeof(void *));
}
2018-06-19 09:49:58 +02:00
2017-11-23 20:42:14 +01:00
return prev;
}
2017-11-24 17:48:47 +01:00
void lv_ll_swap(lv_ll_t * ll_p, void * n1_p, void * n2_p)
2017-11-23 20:42:14 +01:00
{
2017-12-02 20:43:50 +01:00
(void)(ll_p);
(void)(n1_p);
(void)(n2_p);
2017-11-24 17:48:47 +01:00
/*TODO*/
2017-11-23 20:42:14 +01:00
}
2018-04-18 13:23:19 +02:00
/**
* Move a nodw before an other node in the same linked list
* @param ll_p pointer to a linked list
* @param n_act pointer to node to move
* @param n_after pointer to a node which should be after `n_act`
*/
void lv_ll_move_before(lv_ll_t * ll_p, void * n_act, void * n_after)
{
if(n_act == n_after) return; /*Can't move before itself*/
2018-09-27 15:03:20 +02:00
void * n_before;
if(n_after != NULL) n_before = lv_ll_get_prev(ll_p, n_after);
else n_before = lv_ll_get_tail(ll_p); /*if `n_after` is NULL `n_act` should be the new tail*/
2018-10-05 17:22:49 +02:00
if(n_act == n_before) return; /*Already before `n_after`*/
2018-09-27 15:03:20 +02:00
/*It's much easier to remove from the list and add again*/
lv_ll_rem(ll_p, n_act);
/*Add again by setting the prev. and next nodes*/
2018-10-05 17:22:49 +02:00
node_set_next(ll_p, n_before, n_act);
node_set_prev(ll_p, n_act, n_before);
node_set_prev(ll_p, n_after, n_act);
node_set_next(ll_p, n_act, n_after);
2018-09-27 15:03:20 +02:00
2018-10-05 17:22:49 +02:00
/*If `n_act` was moved before NULL then it become the new tail*/
if(n_after == NULL) ll_p->tail = n_act;
2018-04-18 13:23:19 +02:00
}
/**
* Check if a linked list is empty
* @param ll_p pointer to a linked list
* @return true: the linked list is empty; false: not empty
*/
bool lv_ll_is_empty(lv_ll_t * ll_p)
{
if(ll_p == NULL) return true;
if(ll_p->head == NULL && ll_p->tail == NULL) return true;
return false;
}
2017-11-23 20:42:14 +01:00
/**********************
* STATIC FUNCTIONS
**********************/
/**
2018-06-19 09:49:58 +02:00
* Set the 'pervious node pointer' of a node
2017-11-23 20:42:14 +01:00
* @param ll_p pointer to linked list
* @param act pointer to a node which prev. node pointer should be set
* @param prev pointer to a node which should be the previous node before 'act'
*/
2018-06-19 09:49:58 +02:00
static void node_set_prev(lv_ll_t * ll_p, lv_ll_node_t * act, lv_ll_node_t * prev)
2017-11-23 20:42:14 +01:00
{
2018-10-05 17:22:49 +02:00
if(act == NULL) return; /*Can't set the prev node of `NULL`*/
2018-09-27 15:03:20 +02:00
2018-10-05 17:22:49 +02:00
uint32_t node_p_size = sizeof(lv_ll_node_t *);
if(prev) memcpy(act + LL_PREV_P_OFFSET(ll_p), &prev, node_p_size);
else memset(act + LL_PREV_P_OFFSET(ll_p), 0, node_p_size);
2017-11-23 20:42:14 +01:00
}
/**
2018-06-19 09:49:58 +02:00
* Set the 'next node pointer' of a node
2017-11-23 20:42:14 +01:00
* @param ll_p pointer to linked list
* @param act pointer to a node which next node pointer should be set
* @param next pointer to a node which should be the next node before 'act'
*/
2018-06-19 09:49:58 +02:00
static void node_set_next(lv_ll_t * ll_p, lv_ll_node_t * act, lv_ll_node_t * next)
2017-11-23 20:42:14 +01:00
{
2018-10-05 17:22:49 +02:00
if(act == NULL) return; /*Can't set the next node of `NULL`*/
2018-09-27 15:03:20 +02:00
2018-10-05 17:22:49 +02:00
uint32_t node_p_size = sizeof(lv_ll_node_t *);
if(next) memcpy(act + LL_NEXT_P_OFFSET(ll_p), &next, node_p_size);
2018-09-27 15:03:20 +02:00
else memset(act + LL_NEXT_P_OFFSET(ll_p), 0, node_p_size);
2017-11-23 20:42:14 +01:00
}