2001-10-31 12:47:01 +00:00
/*
2004-03-30 19:35:44 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2007-08-20 20:55:30 +00:00
* Copyright ( C ) 2004 - 2007 Red Hat , Inc . All rights reserved .
2001-10-31 12:47:01 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-20 20:55:30 +00:00
* of the GNU Lesser General Public License v .2 .1 .
2004-03-30 19:35:44 +00:00
*
2007-08-20 20:55:30 +00:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 19:35:44 +00:00
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2001-10-31 12:47:01 +00:00
*/
2001-09-28 13:19:17 +00:00
2001-10-01 15:14:39 +00:00
# ifndef _LVM_LIST_H
# define _LVM_LIST_H
2001-09-28 13:19:17 +00:00
2003-03-24 18:08:53 +00:00
# include <assert.h>
2005-04-06 14:50:37 +00:00
/*
* A list consists of a list head plus elements .
* Each element has ' next ' and ' previous ' pointers .
* The list head ' s pointers point to the first and the last element .
*/
2008-11-03 22:14:30 +00:00
struct dm_list {
struct dm_list * n , * p ;
2001-09-28 13:19:17 +00:00
} ;
2005-04-06 14:50:37 +00:00
/*
* Initialise a list before use .
* The list head ' s next and previous pointers point back to itself .
*/
2008-11-03 22:14:30 +00:00
# define DM_LIST_INIT(name) struct dm_list name = { &(name), &(name) }
void dm_list_init ( struct dm_list * head ) ;
2001-09-28 13:19:17 +00:00
2005-04-06 14:50:37 +00:00
/*
* Insert an element before ' head ' .
* If ' head ' is the list head , this adds an element to the end of the list .
*/
2008-11-03 22:14:30 +00:00
void dm_list_add ( struct dm_list * head , struct dm_list * elem ) ;
2001-09-28 13:19:17 +00:00
2005-04-06 14:50:37 +00:00
/*
* Insert an element after ' head ' .
* If ' head ' is the list head , this adds an element to the front of the list .
*/
2008-11-03 22:14:30 +00:00
void dm_list_add_h ( struct dm_list * head , struct dm_list * elem ) ;
2001-09-28 13:19:17 +00:00
2005-04-06 14:50:37 +00:00
/*
* Delete an element from its list .
* Note that this doesn ' t change the element itself - it may still be safe
* to follow its pointers .
*/
2008-11-03 22:14:30 +00:00
void dm_list_del ( struct dm_list * elem ) ;
2001-09-28 13:19:17 +00:00
2008-03-26 16:20:54 +00:00
/*
2008-04-10 19:14:27 +00:00
* Remove an element from existing list and insert before ' head ' .
2008-03-26 16:20:54 +00:00
*/
2008-11-03 22:14:30 +00:00
void dm_list_move ( struct dm_list * head , struct dm_list * elem ) ;
2008-03-26 16:20:54 +00:00
2005-04-06 14:50:37 +00:00
/*
* Is the list empty ?
*/
2008-11-03 22:14:30 +00:00
int dm_list_empty ( const struct dm_list * head ) ;
2001-09-28 13:19:17 +00:00
2005-04-06 14:50:37 +00:00
/*
* Is this the first element of the list ?
*/
2008-11-03 22:14:30 +00:00
int dm_list_start ( const struct dm_list * head , const struct dm_list * elem ) ;
2005-04-06 14:50:37 +00:00
/*
* Is this the last element of the list ?
*/
2008-11-03 22:14:30 +00:00
int dm_list_end ( const struct dm_list * head , const struct dm_list * elem ) ;
2002-12-12 20:55:49 +00:00
2005-06-01 16:51:55 +00:00
/*
* Return first element of the list or NULL if empty
*/
2008-11-03 22:14:30 +00:00
struct dm_list * dm_list_first ( const struct dm_list * head ) ;
2005-06-01 16:51:55 +00:00
2005-05-11 16:46:59 +00:00
/*
* Return last element of the list or NULL if empty
*/
2008-11-03 22:14:30 +00:00
struct dm_list * dm_list_last ( const struct dm_list * head ) ;
2005-05-11 16:46:59 +00:00
2005-04-06 14:50:37 +00:00
/*
* Return the previous element of the list , or NULL if we ' ve reached the start .
*/
2008-11-03 22:14:30 +00:00
struct dm_list * dm_list_prev ( const struct dm_list * head , const struct dm_list * elem ) ;
2005-04-06 14:50:37 +00:00
/*
* Return the next element of the list , or NULL if we ' ve reached the end .
*/
2008-11-03 22:14:30 +00:00
struct dm_list * dm_list_next ( const struct dm_list * head , const struct dm_list * elem ) ;
2003-05-06 12:01:13 +00:00
2005-04-06 14:50:37 +00:00
/*
2008-11-03 22:14:30 +00:00
* Given the address v of an instance of ' struct dm_list ' called ' head '
2005-04-06 15:21:28 +00:00
* contained in a structure of type t , return the containing structure .
2005-04-06 14:50:37 +00:00
*/
2008-11-03 22:14:30 +00:00
# define dm_list_struct_base(v, t, head) \
2005-04-06 15:21:28 +00:00
( ( t * ) ( ( uintptr_t ) ( v ) - ( uintptr_t ) & ( ( t * ) 0 ) - > head ) )
2003-09-15 18:22:50 +00:00
2005-04-06 14:50:37 +00:00
/*
2008-11-03 22:14:30 +00:00
* Given the address v of an instance of ' struct dm_list list ' contained in
2005-04-06 14:50:37 +00:00
* a structure of type t , return the containing structure .
*/
2008-11-03 22:14:30 +00:00
# define dm_list_item(v, t) dm_list_struct_base((v), t, list)
2005-04-06 14:50:37 +00:00
/*
* Given the address v of one known element e in a known structure of type t ,
* return another element f .
*/
2008-11-03 22:14:30 +00:00
# define dm_struct_field(v, t, e, f) \
2003-09-15 18:22:50 +00:00
( ( ( t * ) ( ( uintptr_t ) ( v ) - ( uintptr_t ) & ( ( t * ) 0 ) - > e ) ) - > f )
2005-04-06 14:50:37 +00:00
/*
* Given the address v of a known element e in a known structure of type t ,
* return the list head ' list '
*/
2008-11-03 22:14:30 +00:00
# define dm_list_head(v, t, e) dm_struct_field(v, t, e, list)
2003-09-15 18:22:50 +00:00
2005-04-06 14:50:37 +00:00
/*
* Set v to each element of a list in turn .
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate(v, head) \
2002-01-29 16:28:52 +00:00
for ( v = ( head ) - > n ; v ! = head ; v = v - > n )
2005-04-06 14:50:37 +00:00
/*
* Set v to each element in a list in turn , starting from the element
* in front of ' start ' .
* You can use this to ' unwind ' a list_iterate and back out actions on
* already - processed elements .
* If ' start ' is ' head ' it walks the list backwards .
*/
2008-11-03 22:14:30 +00:00
# define dm_list_uniterate(v, head, start) \
2003-07-04 22:34:56 +00:00
for ( v = ( start ) - > p ; v ! = head ; v = v - > p )
2005-04-06 14:50:37 +00:00
/*
* A safe way to walk a list and delete and free some elements along
* the way .
* t must be defined as a temporary variable of the same type as v .
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_safe(v, t, head) \
2002-01-29 16:28:52 +00:00
for ( v = ( head ) - > n , t = v - > n ; v ! = head ; v = t , t = v - > n )
2001-09-28 13:19:17 +00:00
2005-04-06 14:50:37 +00:00
/*
* Walk a list , setting ' v ' in turn to the containing structure of each item .
* The containing structure should be the same type as ' v ' .
2008-11-03 22:14:30 +00:00
* The ' struct dm_list ' variable within the containing structure is ' field ' .
2005-04-06 14:50:37 +00:00
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_items_gen(v, head, field) \
for ( v = dm_list_struct_base ( ( head ) - > n , typeof ( * v ) , field ) ; \
2005-04-06 15:21:28 +00:00
& v - > field ! = ( head ) ; \
2008-11-03 22:14:30 +00:00
v = dm_list_struct_base ( v - > field . n , typeof ( * v ) , field ) )
2005-04-06 14:50:37 +00:00
/*
* Walk a list , setting ' v ' in turn to the containing structure of each item .
* The containing structure should be the same type as ' v ' .
2008-11-03 22:14:30 +00:00
* The list should be ' struct dm_list list ' within the containing structure .
2005-04-06 14:50:37 +00:00
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_items(v, head) dm_list_iterate_items_gen(v, (head), list)
2003-09-15 18:22:50 +00:00
2005-10-16 23:03:59 +00:00
/*
* Walk a list , setting ' v ' in turn to the containing structure of each item .
* The containing structure should be the same type as ' v ' .
2008-11-03 22:14:30 +00:00
* The ' struct dm_list ' variable within the containing structure is ' field ' .
2005-10-16 23:03:59 +00:00
* t must be defined as a temporary variable of the same type as v .
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_items_gen_safe(v, t, head, field) \
for ( v = dm_list_struct_base ( ( head ) - > n , typeof ( * v ) , field ) , \
t = dm_list_struct_base ( v - > field . n , typeof ( * v ) , field ) ; \
2005-10-16 23:03:59 +00:00
& v - > field ! = ( head ) ; \
2008-11-03 22:14:30 +00:00
v = t , t = dm_list_struct_base ( v - > field . n , typeof ( * v ) , field ) )
2005-10-16 23:03:59 +00:00
/*
* Walk a list , setting ' v ' in turn to the containing structure of each item .
* The containing structure should be the same type as ' v ' .
2008-11-03 22:14:30 +00:00
* The list should be ' struct dm_list list ' within the containing structure .
2005-10-16 23:03:59 +00:00
* t must be defined as a temporary variable of the same type as v .
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_items_safe(v, t, head) \
dm_list_iterate_items_gen_safe ( v , t , ( head ) , list )
2005-10-16 23:03:59 +00:00
2005-06-01 16:51:55 +00:00
/*
* Walk a list backwards , setting ' v ' in turn to the containing structure
* of each item .
* The containing structure should be the same type as ' v ' .
2008-11-03 22:14:30 +00:00
* The ' struct dm_list ' variable within the containing structure is ' field ' .
2005-06-01 16:51:55 +00:00
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_back_items_gen(v, head, field) \
for ( v = dm_list_struct_base ( ( head ) - > p , typeof ( * v ) , field ) ; \
2005-06-01 16:51:55 +00:00
& v - > field ! = ( head ) ; \
2008-11-03 22:14:30 +00:00
v = dm_list_struct_base ( v - > field . p , typeof ( * v ) , field ) )
2005-06-01 16:51:55 +00:00
/*
* Walk a list backwards , setting ' v ' in turn to the containing structure
* of each item .
* The containing structure should be the same type as ' v ' .
2008-11-03 22:14:30 +00:00
* The list should be ' struct dm_list list ' within the containing structure .
2005-06-01 16:51:55 +00:00
*/
2008-11-03 22:14:30 +00:00
# define dm_list_iterate_back_items(v, head) dm_list_iterate_back_items_gen(v, (head), list)
2005-06-01 16:51:55 +00:00
2005-04-06 14:50:37 +00:00
/*
* Return the number of elements in a list by walking it .
*/
2008-11-03 22:14:30 +00:00
unsigned int dm_list_size ( const struct dm_list * head ) ;
2001-11-28 13:45:50 +00:00
2001-09-28 13:19:17 +00:00
# endif