mirror of
				https://github.com/Rockbox/rockbox.git
				synced 2025-10-24 15:37:38 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			301 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			301 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2002 by Linus Nielsen Feltzing
 | |
|  *
 | |
|  * All files in this archive are subject to the GNU General Public License.
 | |
|  * See the file COPYING in the source tree root for full license agreement.
 | |
|  *
 | |
|  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| /*
 | |
| ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 | |
| 
 | |
|   Description:
 | |
|   
 | |
|   This file contains functions implementing a double linked list. 
 | |
|   The functions uses the types LIST and LISTNODE.
 | |
|   There is a trick with the three nodes in LIST. By placing the 
 | |
|   tail_pred field in middle, makes it possible to avoid special code
 | |
|   to handle nodes in the beginning or end of the list.
 | |
|   
 | |
|   The 'head' and 'tail' field in LIST are never NULL, even if the
 | |
|   list is empty.
 | |
|   
 | |
|   The 'succ' and 'pred' field in a LIST_NODE that is in the list,
 | |
|   are never NULL. The 'pred' field for the first LIST_NODE points
 | |
|   to the 'head' field in LIST. The 'succ' node for the last LIST_NODE
 | |
|   points the the tail_pred field in LIST.
 | |
|   
 | |
| ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 | |
| */
 | |
| #include "lists.h"
 | |
| 
 | |
| /****************************************************************************
 | |
| ** FUNCTION:    list_init
 | |
| **
 | |
| ** DESCRIPTION: Initiate a LIST structure. 
 | |
| **              If max_num_nodes is set to NO_SIZE_CHECK there 
 | |
| **              will be no check of the length of the list.
 | |
| **
 | |
| ** RETURN:      Nothing
 | |
| ******************************************************************************/
 | |
| void list_init(LIST *list,         /* The list to initiate */
 | |
| 	       int max_num_nodes)  /* Maximum number of nodes */
 | |
| {
 | |
|    list->first = (LIST_NODE *)&list->last_prev;
 | |
|    list->last_prev = NULL;
 | |
|    list->last = (LIST_NODE *)&list->first;
 | |
|    list->num_nodes = 0;
 | |
|    list->max_num_nodes = max_num_nodes;
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
| ** FUNCTION:    list_insert_before
 | |
| **
 | |
| ** DESCRIPTION: Insert a LIST_NODE in a list before another node in a LIST.
 | |
| **
 | |
| ** RETURN:      Nothing
 | |
| ******************************************************************************/
 | |
| void list_insert_before(LIST_NODE *ref,    /* The reference node */
 | |
| 			     LIST_NODE *ln)     /* The node to insert */
 | |
| {
 | |
|    ln->next = ref;               /* ref is after us */
 | |
|    ln->prev = ref->prev;         /* ref's prev is before us */
 | |
|    ref->prev = ln;               /* we are before ref */
 | |
|    ln->prev->next = ln;          /* we are after ref's prev */
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
| ** FUNCTION:    list_insert_after
 | |
| **
 | |
| ** DESCRIPTION: Insert a LIST_NODE in a list after another node in a LIST.
 | |
| **
 | |
| ** RETURN:      Nothing
 | |
| *****************************************************************************/
 | |
| void list_insert_after(LIST_NODE *ref,     /* The reference node */
 | |
| 			    LIST_NODE *ln)      /* The node to insert */
 | |
| {
 | |
|    ln->prev = ref;               /* ref is before us */
 | |
|    ln->next = ref->next;         /* ref's next is after us */
 | |
|    ref->next = ln;               /* we are after ref */
 | |
|    ln->next->prev = ln;          /* we are before ref's next */
 | |
| }
 | |
| 
 | |
| /****************************************************************************
 | |
| ** FUNCTION:    list_extract_node
 | |
| **
 | |
| ** DESCRIPTION: Extract a LIST_NODE from a list.
 | |
| **
 | |
| ** RETURN:      The same LIST_NODE pointer that was passed as a parameter.
 | |
| *****************************************************************************/
 | |
| LIST_NODE *list_extract_node(LIST_NODE *ln)     /* The node to extract */
 | |
| {
 | |
|   ln->prev->next = ln->next;      /* Our prev's next points to our next */
 | |
|   ln->next->prev = ln->prev;      /* Our next's prev points to our prev */
 | |
|   return ln;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_add_first
 | |
| **
 | |
| ** DESCRIPTION: Add a LIST_NODE at the beginning of a LIST.
 | |
| **
 | |
| ** RETURN:      1 if OK
 | |
| **              0 if list was full
 | |
| ******************************************************************************/
 | |
| int list_add_first(LIST *list,             /* The list to add to */
 | |
|                  LIST_NODE *ln)         /* The node to add */
 | |
| {
 | |
|    if (NO_SIZE_CHECK != list->max_num_nodes)
 | |
|    {
 | |
|       if(list->num_nodes >= list->max_num_nodes)  /* List full? */
 | |
|       {
 | |
| 	 return 0;
 | |
|       }
 | |
|    }
 | |
|    list_insert_after((LIST_NODE *)list, ln);
 | |
|    list->num_nodes++;            /* Increment node counter */
 | |
|    return 1;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_extract_first
 | |
| **
 | |
| ** DESCRIPTION: Extract a LIST_NODE from the beginning of a LIST.
 | |
| **
 | |
| ** RETURN:      The extracted LIST_NODE or NULL if the list is empty
 | |
| ******************************************************************************/
 | |
| LIST_NODE *list_extract_first(LIST *list)       /* The list to extract from */
 | |
| {
 | |
|    LIST_NODE *ln;
 | |
|   
 | |
|    if(list_empty(list))             /* Return NULL if the list is empty */
 | |
|    {
 | |
|       return NULL;
 | |
|    }
 | |
|   
 | |
|    ln = list_extract_node((LIST_NODE *)list->first);  /* Get first node */
 | |
|   
 | |
|    list->num_nodes--;                    /* Decrement node counter */
 | |
|   
 | |
|    return ln;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_add_last
 | |
| **
 | |
| ** DESCRIPTION: Add a LIST_NODE at the end of a LIST.
 | |
| **
 | |
| ** RETURN:      NULL if OK
 | |
| **              0 if list was full
 | |
| ******************************************************************************/
 | |
| int list_add_last(LIST *list,              /* The list to add to */
 | |
|                 LIST_NODE *ln)          /* The node to add */
 | |
| {
 | |
|    if (NO_SIZE_CHECK != list->max_num_nodes)
 | |
|    {
 | |
|       if(list->num_nodes >= list->max_num_nodes)  /* List full? */
 | |
|       {
 | |
| 	 return 0;      
 | |
|       }
 | |
|    }
 | |
|    list_insert_before((LIST_NODE *)&list->last_prev, ln);
 | |
|    list->num_nodes++;                    /* Increment node counter */
 | |
|    return 1;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_extract_last
 | |
| **
 | |
| ** DESCRIPTION: Extract a LIST_NODE from the end of a LIST.
 | |
| **
 | |
| ** RETURN:      The extracted LIST_NODE or NULL if the list is empty
 | |
| ******************************************************************************/
 | |
| LIST_NODE *list_extract_last(LIST *list)        /* The list to extract from */
 | |
| {
 | |
|    LIST_NODE *ln;
 | |
|   
 | |
|    if(list_empty(list))             /* Return NULL if the list is empty */
 | |
|    {
 | |
|       return NULL;
 | |
|    }
 | |
|   
 | |
|    ln = list_extract_node((LIST_NODE *)list->last);
 | |
|   
 | |
|    list->num_nodes--;    /* Decrement node counter */
 | |
|   
 | |
|    return ln;           /* Is NULL if the list is empty */
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_last_in_list
 | |
| **
 | |
| ** DESCRIPTION: Check if a LIST_NODE is last in the list
 | |
| **
 | |
| ** RETURN:      1 if last in list
 | |
| ******************************************************************************/
 | |
| int list_last_in_list(LIST_NODE *ln)    /* The node to check */
 | |
| {
 | |
|    return ln->next->next == NULL;
 | |
| }
 | |
| 
 | |
| /*****************************************************************************
 | |
| ** FUNCTION:    list_first_in_list
 | |
| **
 | |
| ** DESCRIPTION: Check if a LIST_NODE is first in the list
 | |
| **
 | |
| ** RETURN:      1 if first in list
 | |
| ******************************************************************************/
 | |
| int list_first_in_list(LIST_NODE *ln)   /* The node to check */
 | |
| {
 | |
|    return ln->prev->prev == NULL;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_empty
 | |
| **
 | |
| ** DESCRIPTION: Check if a LIST is empty
 | |
| **
 | |
| ** RETURN:      1 if list is empty
 | |
| ******************************************************************************/
 | |
| int list_empty(LIST *list)         /* The list to check */
 | |
| {
 | |
|    return list->first == (LIST_NODE *)&list->last_prev;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_get_first
 | |
| **
 | |
| ** DESCRIPTION: Return a LIST_NODE from the beginning of a LIST.
 | |
| **
 | |
| ** RETURN:      The first LIST_NODE or NULL if the list is empty
 | |
| ******************************************************************************/
 | |
| LIST_NODE *list_get_first(LIST *lh)   /* The list to read from */
 | |
| {
 | |
|    if(list_empty(lh))       /* Return NULL if the list is empty */
 | |
|    {
 | |
|       return NULL;
 | |
|    }
 | |
|   
 | |
|    return lh->first;     /* Get first node */
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_get_last
 | |
| **
 | |
| ** DESCRIPTION: Return a LIST_NODE from the end of a LIST.
 | |
| **
 | |
| ** RETURN:      The last LIST_NODE or NULL if the list is empty
 | |
| ******************************************************************************/
 | |
| LIST_NODE *list_get_last(LIST *lh)    /* The list to read from */
 | |
| {
 | |
|    if(list_empty(lh))               /* Return NULL if the list is empty */
 | |
|    {
 | |
|       return NULL;
 | |
|    }
 | |
|   
 | |
|    return lh->last;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_get_next
 | |
| **
 | |
| ** DESCRIPTION: Return the LIST_NODE following the specified one.
 | |
| **
 | |
| ** RETURN:      Next LIST_NODE or NULL if the list ends here
 | |
| *******************************************************************************/
 | |
| LIST_NODE *list_get_next(LIST_NODE *ln) /* The list node to get next from */
 | |
| {
 | |
|    if(list_last_in_list(ln))       /* Return NULL if this is the end of list */
 | |
|    {
 | |
|       return NULL;
 | |
|    }
 | |
|   
 | |
|    return ln->next;
 | |
| }
 | |
| 
 | |
| /******************************************************************************
 | |
| ** FUNCTION:    list_get_prev
 | |
| **
 | |
| ** DESCRIPTION: Return the LIST_NODE preceding the specified one.
 | |
| **
 | |
| ** RETURN:      Previous LIST_NODE or NULL if the list ends here
 | |
| *******************************************************************************/
 | |
| LIST_NODE *list_get_prev(LIST_NODE *ln) /* The list node to get previous from */
 | |
| {
 | |
|    if(list_first_in_list(ln))            /* Return NULL if this is the start of list */
 | |
|    {
 | |
|       return NULL;
 | |
|    }
 | |
|   
 | |
|    return ln->prev;
 | |
| }
 |