2001-11-21 12:47:42 +00:00
/*
2004-03-30 19:08:57 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2011-03-10 12:48:40 +00:00
* Copyright ( C ) 2004 - 2011 Red Hat , Inc . All rights reserved .
2001-11-21 12:47:42 +00:00
*
2004-03-30 19:08:57 +00:00
* This file is part of the device - mapper userspace tools .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2001-11-21 12:47:42 +00:00
*/
# ifndef LIB_DEVICE_MAPPER_H
# define LIB_DEVICE_MAPPER_H
2001-12-05 16:41:52 +00:00
# include <inttypes.h>
2006-08-21 12:52:39 +00:00
# include <stdarg.h>
2002-01-15 15:21:57 +00:00
# include <sys/types.h>
2011-08-30 14:55:15 +00:00
# include <sys/stat.h>
2003-11-12 17:30:32 +00:00
# ifdef linux
# include <linux / types.h>
# endif
2001-12-05 16:41:52 +00:00
2005-10-16 22:57:20 +00:00
# include <limits.h>
# include <string.h>
# include <stdlib.h>
2007-07-24 14:15:45 +00:00
# include <stdio.h>
2005-10-16 22:57:20 +00:00
2010-06-23 17:03:14 +00:00
# ifndef __GNUC__
# define __typeof__ typeof
# endif
2010-06-16 13:01:25 +00:00
# ifdef __cplusplus
extern " C " {
# endif
2005-11-09 14:10:50 +00:00
/*****************************************************************
2009-07-10 09:59:37 +00:00
* The first section of this file provides direct access to the
* individual device - mapper ioctls . Since it is quite laborious to
* build the ioctl arguments for the device - mapper , people are
* encouraged to use this library .
2005-11-09 14:10:50 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-11-21 12:47:42 +00:00
/*
2009-07-10 09:59:37 +00:00
* The library user may wish to register their own
* logging function . By default errors go to stderr .
* Use dm_log_with_errno_init ( NULL ) to restore the default log fn .
2001-11-21 12:47:42 +00:00
*/
2009-07-10 09:59:37 +00:00
typedef void ( * dm_log_with_errno_fn ) ( int level , const char * file , int line ,
int dm_errno , const char * f , . . . )
__attribute__ ( ( format ( printf , 5 , 6 ) ) ) ;
void dm_log_with_errno_init ( dm_log_with_errno_fn fn ) ;
void dm_log_init_verbose ( int level ) ;
/*
* Original version of this function .
* dm_errno is set to 0.
*
* Deprecated : Use the _with_errno_ versions above instead .
*/
2002-03-07 20:56:10 +00:00
typedef void ( * dm_log_fn ) ( int level , const char * file , int line ,
2006-01-31 14:50:38 +00:00
const char * f , . . . )
__attribute__ ( ( format ( printf , 4 , 5 ) ) ) ;
2010-07-02 21:16:50 +00:00
2009-07-10 09:59:37 +00:00
void dm_log_init ( dm_log_fn fn ) ;
2001-11-21 12:47:42 +00:00
/*
2009-07-10 09:59:37 +00:00
* For backward - compatibility , indicate that dm_log_init ( ) was used
* to set a non - default value of dm_log ( ) .
2001-11-21 12:47:42 +00:00
*/
2009-07-10 09:59:37 +00:00
int dm_log_is_non_default ( void ) ;
2001-11-21 12:47:42 +00:00
2011-06-13 03:32:45 +00:00
/*
* Number of devices currently in suspended state ( via the library ) .
*/
int dm_get_suspended_counter ( void ) ;
2001-11-21 12:47:42 +00:00
enum {
DM_DEVICE_CREATE ,
DM_DEVICE_RELOAD ,
DM_DEVICE_REMOVE ,
2002-03-07 20:56:10 +00:00
DM_DEVICE_REMOVE_ALL ,
2001-11-21 12:47:42 +00:00
DM_DEVICE_SUSPEND ,
DM_DEVICE_RESUME ,
DM_DEVICE_INFO ,
2002-03-06 14:38:25 +00:00
DM_DEVICE_DEPS ,
2002-01-10 23:29:16 +00:00
DM_DEVICE_RENAME ,
2002-01-15 15:21:57 +00:00
DM_DEVICE_VERSION ,
2002-05-03 11:55:58 +00:00
DM_DEVICE_STATUS ,
DM_DEVICE_TABLE ,
2003-07-01 21:20:58 +00:00
DM_DEVICE_WAITEVENT ,
DM_DEVICE_LIST ,
2003-11-13 13:14:28 +00:00
DM_DEVICE_CLEAR ,
2004-01-23 14:37:47 +00:00
DM_DEVICE_MKNODES ,
2004-06-08 20:34:40 +00:00
DM_DEVICE_LIST_VERSIONS ,
2006-02-20 23:55:58 +00:00
DM_DEVICE_TARGET_MSG ,
DM_DEVICE_SET_GEOMETRY
2001-11-21 12:47:42 +00:00
} ;
2009-07-10 09:59:37 +00:00
/*
* You will need to build a struct dm_task for
* each ioctl command you want to execute .
*/
2001-11-21 12:47:42 +00:00
struct dm_task ;
struct dm_task * dm_task_create ( int type ) ;
void dm_task_destroy ( struct dm_task * dmt ) ;
int dm_task_set_name ( struct dm_task * dmt , const char * name ) ;
2002-03-11 22:44:36 +00:00
int dm_task_set_uuid ( struct dm_task * dmt , const char * uuid ) ;
2001-11-21 12:47:42 +00:00
/*
* Retrieve attributes after an info .
*/
struct dm_info {
int exists ;
int suspended ;
2003-07-01 21:20:58 +00:00
int live_table ;
int inactive_table ;
2003-03-28 18:58:59 +00:00
int32_t open_count ;
2003-04-29 11:34:23 +00:00
uint32_t event_nr ;
2003-03-28 18:58:59 +00:00
uint32_t major ;
uint32_t minor ; /* minor device number */
2002-01-03 15:12:02 +00:00
int read_only ; /* 0:read-write; 1:read-only */
2002-01-03 10:39:21 +00:00
2003-03-28 18:58:59 +00:00
int32_t target_count ;
2001-11-21 12:47:42 +00:00
} ;
2002-03-06 14:38:25 +00:00
struct dm_deps {
2003-03-28 18:58:59 +00:00
uint32_t count ;
uint32_t filler ;
uint64_t device [ 0 ] ;
2002-03-06 14:38:25 +00:00
} ;
2003-07-01 21:20:58 +00:00
struct dm_names {
uint64_t dev ;
uint32_t next ; /* Offset to next struct from start of this struct */
char name [ 0 ] ;
} ;
2004-01-23 14:37:47 +00:00
struct dm_versions {
2007-01-16 18:04:15 +00:00
uint32_t next ; /* Offset to next struct from start of this struct */
uint32_t version [ 3 ] ;
2004-01-23 14:37:47 +00:00
2007-01-16 18:04:15 +00:00
char name [ 0 ] ;
2004-01-23 14:37:47 +00:00
} ;
2002-01-17 14:13:25 +00:00
int dm_get_library_version ( char * version , size_t size ) ;
2002-03-07 20:56:10 +00:00
int dm_task_get_driver_version ( struct dm_task * dmt , char * version , size_t size ) ;
2001-11-21 12:47:42 +00:00
int dm_task_get_info ( struct dm_task * dmt , struct dm_info * dmi ) ;
2007-04-27 14:52:41 +00:00
const char * dm_task_get_name ( const struct dm_task * dmt ) ;
const char * dm_task_get_uuid ( const struct dm_task * dmt ) ;
2001-11-21 12:47:42 +00:00
2002-03-06 14:38:25 +00:00
struct dm_deps * dm_task_get_deps ( struct dm_task * dmt ) ;
2003-07-01 21:20:58 +00:00
struct dm_names * dm_task_get_names ( struct dm_task * dmt ) ;
2004-01-23 14:37:47 +00:00
struct dm_versions * dm_task_get_versions ( struct dm_task * dmt ) ;
2002-03-06 14:38:25 +00:00
2002-01-03 10:39:21 +00:00
int dm_task_set_ro ( struct dm_task * dmt ) ;
2002-01-10 23:29:16 +00:00
int dm_task_set_newname ( struct dm_task * dmt , const char * newname ) ;
2010-10-15 01:10:27 +00:00
int dm_task_set_newuuid ( struct dm_task * dmt , const char * newuuid ) ;
2002-01-11 12:12:46 +00:00
int dm_task_set_minor ( struct dm_task * dmt , int minor ) ;
2003-04-02 19:03:00 +00:00
int dm_task_set_major ( struct dm_task * dmt , int major ) ;
2009-06-17 20:55:24 +00:00
int dm_task_set_major_minor ( struct dm_task * dmt , int major , int minor , int allow_default_major_fallback ) ;
2006-02-03 14:23:22 +00:00
int dm_task_set_uid ( struct dm_task * dmt , uid_t uid ) ;
int dm_task_set_gid ( struct dm_task * dmt , gid_t gid ) ;
int dm_task_set_mode ( struct dm_task * dmt , mode_t mode ) ;
2009-10-22 12:55:47 +00:00
int dm_task_set_cookie ( struct dm_task * dmt , uint32_t * cookie , uint16_t flags ) ;
2003-04-29 11:34:23 +00:00
int dm_task_set_event_nr ( struct dm_task * dmt , uint32_t event_nr ) ;
2006-02-20 23:55:58 +00:00
int dm_task_set_geometry ( struct dm_task * dmt , const char * cylinders , const char * heads , const char * sectors , const char * start ) ;
2004-06-08 20:34:40 +00:00
int dm_task_set_message ( struct dm_task * dmt , const char * message ) ;
int dm_task_set_sector ( struct dm_task * dmt , uint64_t sector ) ;
2006-10-12 15:42:25 +00:00
int dm_task_no_flush ( struct dm_task * dmt ) ;
2005-01-12 22:10:14 +00:00
int dm_task_no_open_count ( struct dm_task * dmt ) ;
2005-10-04 20:12:32 +00:00
int dm_task_skip_lockfs ( struct dm_task * dmt ) ;
2009-11-06 00:43:08 +00:00
int dm_task_query_inactive_table ( struct dm_task * dmt ) ;
2005-11-22 18:43:12 +00:00
int dm_task_suppress_identical_reload ( struct dm_task * dmt ) ;
2011-02-04 16:08:11 +00:00
int dm_task_secure_data ( struct dm_task * dmt ) ;
2011-09-22 17:09:48 +00:00
int dm_task_retry_remove ( struct dm_task * dmt ) ;
2011-07-01 14:09:19 +00:00
/*
* Enable checks for common mistakes such as issuing ioctls in an unsafe order .
*/
int dm_task_enable_checks ( struct dm_task * dmt ) ;
2011-02-04 19:33:53 +00:00
typedef enum {
DM_ADD_NODE_ON_RESUME , /* add /dev/mapper node with dmsetup resume */
DM_ADD_NODE_ON_CREATE /* add /dev/mapper node with dmsetup create */
} dm_add_node_t ;
int dm_task_set_add_node ( struct dm_task * dmt , dm_add_node_t add_node ) ;
2002-01-10 23:29:16 +00:00
2007-11-27 20:57:05 +00:00
/*
* Control read_ahead .
*/
# define DM_READ_AHEAD_AUTO UINT32_MAX /* Use kernel default readahead */
# define DM_READ_AHEAD_NONE 0 /* Disable readahead */
# define DM_READ_AHEAD_MINIMUM_FLAG 0x1 /* Value supplied is minimum */
2007-12-05 16:28:19 +00:00
/*
* Read ahead is set with DM_DEVICE_CREATE with a table or DM_DEVICE_RESUME .
*/
2007-11-27 20:57:05 +00:00
int dm_task_set_read_ahead ( struct dm_task * dmt , uint32_t read_ahead ,
uint32_t read_ahead_flags ) ;
2007-11-30 14:59:57 +00:00
uint32_t dm_task_get_read_ahead ( const struct dm_task * dmt ,
uint32_t * read_ahead ) ;
2007-11-27 20:57:05 +00:00
2001-11-21 12:47:42 +00:00
/*
* Use these to prepare for a create or reload .
*/
int dm_task_add_target ( struct dm_task * dmt ,
2001-12-05 16:41:52 +00:00
uint64_t start ,
2002-03-07 20:56:10 +00:00
uint64_t size , const char * ttype , const char * params ) ;
2001-11-21 12:47:42 +00:00
2003-07-01 21:20:58 +00:00
/*
2008-09-02 12:16:07 +00:00
* Format major / minor numbers correctly for input to driver .
2003-07-01 21:20:58 +00:00
*/
2008-09-02 12:16:07 +00:00
# define DM_FORMAT_DEV_BUFSIZE 13 /* Minimum bufsize to handle worst case. */
2003-07-01 21:20:58 +00:00
int dm_format_dev ( char * buf , int bufsize , uint32_t dev_major , uint32_t dev_minor ) ;
2002-05-03 11:55:58 +00:00
/* Use this to retrive target information returned from a STATUS call */
void * dm_get_next_target ( struct dm_task * dmt ,
2002-05-10 15:25:38 +00:00
void * next , uint64_t * start , uint64_t * length ,
char * * target_type , char * * params ) ;
2002-05-03 11:55:58 +00:00
2001-11-21 12:47:42 +00:00
/*
* Call this to actually run the ioctl .
*/
int dm_task_run ( struct dm_task * dmt ) ;
2005-12-01 23:11:41 +00:00
/*
* Call this to make or remove the device nodes associated with previously
* issued commands .
*/
void dm_task_update_nodes ( void ) ;
2001-11-21 12:47:42 +00:00
/*
2001-11-21 17:08:37 +00:00
* Configure the device - mapper directory
2001-11-21 12:47:42 +00:00
*/
2001-11-21 17:08:37 +00:00
int dm_set_dev_dir ( const char * dir ) ;
2001-11-21 12:47:42 +00:00
const char * dm_dir ( void ) ;
2011-09-22 17:17:07 +00:00
/*
* Configure sysfs directory , / sys by default
*/
int dm_set_sysfs_dir ( const char * dir ) ;
const char * dm_sysfs_dir ( void ) ;
2005-10-16 14:33:22 +00:00
/*
* Determine whether a major number belongs to device - mapper or not .
*/
int dm_is_dm_major ( uint32_t major ) ;
2011-09-22 17:23:35 +00:00
/*
* Determine whether a device has any holders ( devices
* using this device ) . If sysfs is not used ( or configured
* incorrectly ) , returns 0.
*/
int dm_device_has_holders ( uint32_t major , uint32_t minor ) ;
/*
* Determine whether a device contains mounted filesystem .
* If sysfs is not used ( or configured incorrectly ) , returns 0.
*/
int dm_device_has_mounted_fs ( uint32_t major , uint32_t minor ) ;
2005-10-16 14:33:22 +00:00
/*
* Release library resources
*/
2003-07-01 21:20:58 +00:00
void dm_lib_release ( void ) ;
2010-07-09 15:34:40 +00:00
void dm_lib_exit ( void ) __attribute__ ( ( destructor ) ) ;
2003-07-01 21:20:58 +00:00
2005-10-16 22:57:20 +00:00
/*
* Use NULL for all devices .
*/
int dm_mknodes ( const char * name ) ;
2005-10-17 18:05:39 +00:00
int dm_driver_version ( char * version , size_t size ) ;
2005-10-16 22:57:20 +00:00
2005-11-09 14:10:50 +00:00
/******************************************************
* Functions to build and manipulate trees of devices *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct dm_tree ;
struct dm_tree_node ;
2005-10-16 14:33:22 +00:00
/*
* Initialise an empty dependency tree .
*
* The tree consists of a root node together with one node for each mapped
* device which has child nodes for each device referenced in its table .
*
* Every node in the tree has one or more children and one or more parents .
*
* The root node is the parent / child of every node that doesn ' t have other
* parents / children .
*/
2005-11-09 14:10:50 +00:00
struct dm_tree * dm_tree_create ( void ) ;
void dm_tree_free ( struct dm_tree * tree ) ;
2005-10-16 14:33:22 +00:00
/*
* Add nodes to the tree for a given device and all the devices it uses .
*/
2005-11-09 14:10:50 +00:00
int dm_tree_add_dev ( struct dm_tree * tree , uint32_t major , uint32_t minor ) ;
2010-01-07 19:54:21 +00:00
int dm_tree_add_dev_with_udev_flags ( struct dm_tree * tree , uint32_t major ,
uint32_t minor , uint16_t udev_flags ) ;
2005-10-16 14:33:22 +00:00
2005-11-08 22:50:11 +00:00
/*
* Add a new node to the tree if it doesn ' t already exist .
*/
2005-11-09 14:10:50 +00:00
struct dm_tree_node * dm_tree_add_new_dev ( struct dm_tree * tree ,
2007-01-16 18:04:15 +00:00
const char * name ,
const char * uuid ,
uint32_t major , uint32_t minor ,
int read_only ,
int clear_inactive ,
void * context ) ;
2009-10-22 13:00:07 +00:00
struct dm_tree_node * dm_tree_add_new_dev_with_udev_flags ( struct dm_tree * tree ,
const char * name ,
const char * uuid ,
uint32_t major ,
uint32_t minor ,
int read_only ,
int clear_inactive ,
void * context ,
uint16_t udev_flags ) ;
2005-11-08 22:50:11 +00:00
2005-10-16 14:33:22 +00:00
/*
* Search for a node in the tree .
2005-11-08 22:50:11 +00:00
* Set major and minor to 0 or uuid to NULL to get the root node .
2005-10-16 14:33:22 +00:00
*/
2005-11-09 14:10:50 +00:00
struct dm_tree_node * dm_tree_find_node ( struct dm_tree * tree ,
2005-10-16 14:33:22 +00:00
uint32_t major ,
uint32_t minor ) ;
2005-11-09 14:10:50 +00:00
struct dm_tree_node * dm_tree_find_node_by_uuid ( struct dm_tree * tree ,
2005-11-08 22:50:11 +00:00
const char * uuid ) ;
2005-10-16 14:33:22 +00:00
/*
* Use this to walk through all children of a given node .
* Set handle to NULL in first call .
* Returns NULL after the last child .
* Set inverted to use inverted tree .
*/
2005-11-09 14:10:50 +00:00
struct dm_tree_node * dm_tree_next_child ( void * * handle ,
2010-01-14 10:15:23 +00:00
const struct dm_tree_node * parent ,
uint32_t inverted ) ;
2005-10-16 14:33:22 +00:00
/*
* Get properties of a node .
*/
2010-01-14 10:15:23 +00:00
const char * dm_tree_node_get_name ( const struct dm_tree_node * node ) ;
const char * dm_tree_node_get_uuid ( const struct dm_tree_node * node ) ;
const struct dm_info * dm_tree_node_get_info ( const struct dm_tree_node * node ) ;
void * dm_tree_node_get_context ( const struct dm_tree_node * node ) ;
int dm_tree_node_size_changed ( const struct dm_tree_node * dnode ) ;
2005-10-16 14:33:22 +00:00
/*
* Returns the number of children of the given node ( excluding the root node ) .
* Set inverted for the number of parents .
*/
2010-01-14 10:15:23 +00:00
int dm_tree_node_num_children ( const struct dm_tree_node * node , uint32_t inverted ) ;
2005-10-16 14:33:22 +00:00
2005-10-18 12:37:53 +00:00
/*
2005-10-18 13:57:11 +00:00
* Deactivate a device plus all dependencies .
* Ignores devices that don ' t have a uuid starting with uuid_prefix .
2005-10-18 12:37:53 +00:00
*/
2005-11-09 14:10:50 +00:00
int dm_tree_deactivate_children ( struct dm_tree_node * dnode ,
2005-10-18 12:37:53 +00:00
const char * uuid_prefix ,
size_t uuid_prefix_len ) ;
2005-11-08 22:50:11 +00:00
/*
* Preload / create a device plus all dependencies .
* Ignores devices that don ' t have a uuid starting with uuid_prefix .
*/
2005-11-09 14:10:50 +00:00
int dm_tree_preload_children ( struct dm_tree_node * dnode ,
2007-01-16 18:04:15 +00:00
const char * uuid_prefix ,
size_t uuid_prefix_len ) ;
2005-11-08 22:50:11 +00:00
/*
* Resume a device plus all dependencies .
* Ignores devices that don ' t have a uuid starting with uuid_prefix .
*/
2005-11-09 14:10:50 +00:00
int dm_tree_activate_children ( struct dm_tree_node * dnode ,
2007-01-16 18:04:15 +00:00
const char * uuid_prefix ,
size_t uuid_prefix_len ) ;
2005-10-18 12:37:53 +00:00
2005-10-25 19:09:41 +00:00
/*
* Suspend a device plus all dependencies .
* Ignores devices that don ' t have a uuid starting with uuid_prefix .
*/
2005-11-09 14:10:50 +00:00
int dm_tree_suspend_children ( struct dm_tree_node * dnode ,
2005-10-25 19:09:41 +00:00
const char * uuid_prefix ,
size_t uuid_prefix_len ) ;
2006-01-30 23:36:04 +00:00
/*
* Skip the filesystem sync when suspending .
* Does nothing with other functions .
* Use this when no snapshots are involved .
*/
void dm_tree_skip_lockfs ( struct dm_tree_node * dnode ) ;
2007-01-09 19:44:07 +00:00
/*
* Set the ' noflush ' flag when suspending devices .
* If the kernel supports it , instead of erroring outstanding I / O that
* cannot be completed , the I / O is queued and resubmitted when the
* device is resumed . This affects multipath devices when all paths
* have failed and queue_if_no_path is set , and mirror devices when
* block_on_error is set and the mirror log has failed .
*/
void dm_tree_use_no_flush_suspend ( struct dm_tree_node * dnode ) ;
2011-09-22 17:36:50 +00:00
/*
* Retry removal of each device if not successful .
*/
void dm_tree_retry_remove ( struct dm_tree_node * dnode ) ;
2005-10-25 19:09:41 +00:00
/*
* Is the uuid prefix present in the tree ?
* Only returns 0 if every node was checked successfully .
* Returns 1 if the tree walk has to be aborted .
*/
2005-11-09 14:10:50 +00:00
int dm_tree_children_use_uuid ( struct dm_tree_node * dnode ,
2005-10-25 19:09:41 +00:00
const char * uuid_prefix ,
size_t uuid_prefix_len ) ;
2005-11-08 22:50:11 +00:00
/*
2005-11-09 14:10:50 +00:00
* Construct tables for new nodes before activating them .
2005-11-08 22:50:11 +00:00
*/
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_snapshot_origin_target ( struct dm_tree_node * dnode ,
2005-11-08 22:50:11 +00:00
uint64_t size ,
const char * origin_uuid ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_snapshot_target ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint64_t size ,
const char * origin_uuid ,
const char * cow_uuid ,
int persistent ,
uint32_t chunk_size ) ;
2010-01-13 01:39:44 +00:00
int dm_tree_node_add_snapshot_merge_target ( struct dm_tree_node * node ,
uint64_t size ,
const char * origin_uuid ,
const char * cow_uuid ,
const char * merge_uuid ,
uint32_t chunk_size ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_error_target ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint64_t size ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_zero_target ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint64_t size ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_linear_target ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint64_t size ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_striped_target ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint64_t size ,
uint32_t stripe_size ) ;
2009-06-09 16:10:20 +00:00
# define DM_CRYPT_IV_DEFAULT UINT64_C(-1) /* iv_offset == seg offset */
/*
* Function accepts one string in cipher specification
* ( chainmode and iv should be NULL because included in cipher string )
* or
* separate arguments which will be joined to " cipher-chainmode-iv "
*/
int dm_tree_node_add_crypt_target ( struct dm_tree_node * node ,
uint64_t size ,
const char * cipher ,
const char * chainmode ,
const char * iv ,
uint64_t iv_offset ,
const char * key ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_mirror_target ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint64_t size ) ;
2005-12-13 15:49:27 +00:00
/* Mirror log flags */
# define DM_NOSYNC 0x00000001 /* Known already in sync */
# define DM_FORCESYNC 0x00000002 /* Force resync */
# define DM_BLOCK_ON_ERROR 0x00000004 /* On error, suspend I/O */
2006-05-11 19:10:55 +00:00
# define DM_CORELOG 0x00000008 /* In-memory log */
2005-12-13 15:49:27 +00:00
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_mirror_target_log ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
uint32_t region_size ,
unsigned clustered ,
const char * log_uuid ,
2005-12-19 21:03:17 +00:00
unsigned area_count ,
uint32_t flags ) ;
2010-05-21 12:24:15 +00:00
2011-08-02 22:07:20 +00:00
int dm_tree_node_add_raid_target ( struct dm_tree_node * node ,
uint64_t size ,
const char * raid_type ,
uint32_t region_size ,
uint32_t stripe_size ,
2011-08-18 19:41:21 +00:00
uint64_t rebuilds ,
2011-08-02 22:07:20 +00:00
uint64_t reserved2 ) ;
2010-05-21 12:24:15 +00:00
/*
* Replicator operation mode
* Note : API for Replicator is not yet stable
*/
typedef enum {
DM_REPLICATOR_SYNC , /* Synchronous replication */
DM_REPLICATOR_ASYNC_WARN , /* Warn if async replicator is slow */
DM_REPLICATOR_ASYNC_STALL , /* Stall replicator if not fast enough */
DM_REPLICATOR_ASYNC_DROP , /* Drop sites out of sync */
DM_REPLICATOR_ASYNC_FAIL , /* Fail replicator if slow */
NUM_DM_REPLICATOR_MODES
} dm_replicator_mode_t ;
int dm_tree_node_add_replicator_target ( struct dm_tree_node * node ,
uint64_t size ,
const char * rlog_uuid ,
const char * rlog_type ,
unsigned rsite_index ,
dm_replicator_mode_t mode ,
uint32_t async_timeout ,
uint64_t fall_behind_data ,
uint32_t fall_behind_ios ) ;
int dm_tree_node_add_replicator_dev_target ( struct dm_tree_node * node ,
uint64_t size ,
const char * replicator_uuid , /* Replicator control device */
uint64_t rdevice_index ,
const char * rdev_uuid , /* Rimage device name/uuid */
unsigned rsite_index ,
const char * slog_uuid ,
uint32_t slog_flags , /* Mirror log flags */
uint32_t slog_region_size ) ;
/* End of Replicator API */
2010-05-21 12:27:02 +00:00
void dm_tree_node_set_presuspend_node ( struct dm_tree_node * node ,
struct dm_tree_node * presuspend_node ) ;
2005-11-09 14:10:50 +00:00
int dm_tree_node_add_target_area ( struct dm_tree_node * node ,
2005-11-08 22:50:11 +00:00
const char * dev_name ,
const char * dlid ,
uint64_t offset ) ;
2011-08-19 16:26:02 +00:00
/*
* Only for temporarily - missing raid devices where changes are tracked .
*/
2011-08-18 19:38:26 +00:00
int dm_tree_node_add_null_area ( struct dm_tree_node * node , uint64_t offset ) ;
2005-11-08 22:50:11 +00:00
2007-11-27 20:57:05 +00:00
/*
* Set readahead ( in sectors ) after loading the node .
*/
void dm_tree_node_set_read_ahead ( struct dm_tree_node * dnode ,
uint32_t read_ahead ,
uint32_t read_ahead_flags ) ;
2009-07-31 18:30:31 +00:00
void dm_tree_set_cookie ( struct dm_tree_node * node , uint32_t cookie ) ;
uint32_t dm_tree_get_cookie ( struct dm_tree_node * node ) ;
2005-10-16 22:57:20 +00:00
/*****************************************************************************
* Library functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-11-09 14:10:50 +00:00
/*******************
* Memory management
* * * * * * * * * * * * * * * * * * */
2005-11-10 16:33:04 +00:00
void * dm_malloc_aux ( size_t s , const char * file , int line ) ;
void * dm_malloc_aux_debug ( size_t s , const char * file , int line ) ;
2010-09-30 21:06:50 +00:00
void * dm_zalloc_aux ( size_t s , const char * file , int line ) ;
void * dm_zalloc_aux_debug ( size_t s , const char * file , int line ) ;
2006-02-23 19:11:51 +00:00
char * dm_strdup_aux ( const char * str , const char * file , int line ) ;
2005-10-16 22:57:20 +00:00
void dm_free_aux ( void * p ) ;
void * dm_realloc_aux ( void * p , unsigned int s , const char * file , int line ) ;
2005-11-10 16:33:04 +00:00
int dm_dump_memory_debug ( void ) ;
void dm_bounds_check_debug ( void ) ;
# ifdef DEBUG_MEM
2005-10-16 22:57:20 +00:00
2005-11-10 16:33:04 +00:00
# define dm_malloc(s) dm_malloc_aux_debug((s), __FILE__, __LINE__)
2010-09-30 21:06:50 +00:00
# define dm_zalloc(s) dm_zalloc_aux_debug((s), __FILE__, __LINE__)
2006-02-23 19:11:51 +00:00
# define dm_strdup(s) dm_strdup_aux((s), __FILE__, __LINE__)
2005-10-16 22:57:20 +00:00
# define dm_free(p) dm_free_aux(p)
# define dm_realloc(p, s) dm_realloc_aux(p, s, __FILE__, __LINE__)
2005-11-11 16:16:37 +00:00
# define dm_dump_memory() dm_dump_memory_debug()
# define dm_bounds_check() dm_bounds_check_debug()
2005-10-16 22:57:20 +00:00
# else
2005-11-10 16:33:04 +00:00
# define dm_malloc(s) dm_malloc_aux((s), __FILE__, __LINE__)
2010-09-30 21:06:50 +00:00
# define dm_zalloc(s) dm_zalloc_aux((s), __FILE__, __LINE__)
2006-01-10 22:19:41 +00:00
# define dm_strdup(s) strdup(s)
2005-10-16 22:57:20 +00:00
# define dm_free(p) free(p)
# define dm_realloc(p, s) realloc(p, s)
2005-11-11 16:16:37 +00:00
# define dm_dump_memory() {}
# define dm_bounds_check() {}
2005-10-16 22:57:20 +00:00
# endif
2005-11-10 16:33:04 +00:00
2005-10-16 22:57:20 +00:00
/*
* The pool allocator is useful when you are going to allocate
* lots of memory , use the memory for a bit , and then free the
* memory in one go . A surprising amount of code has this usage
* profile .
*
* You should think of the pool as an infinite , contiguous chunk
* of memory . The front of this chunk of memory contains
* allocated objects , the second half is free . dm_pool_alloc grabs
* the next ' size ' bytes from the free half , in effect moving it
* into the allocated half . This operation is very efficient .
*
* dm_pool_free frees the allocated object * and * all objects
* allocated after it . It is important to note this semantic
* difference from malloc / free . This is also extremely
* efficient , since a single dm_pool_free can dispose of a large
* complex object .
*
* dm_pool_destroy frees all allocated memory .
*
* eg , If you are building a binary tree in your program , and
* know that you are only ever going to insert into your tree ,
* and not delete ( eg , maintaining a symbol table for a
* compiler ) . You can create yourself a pool , allocate the nodes
* from it , and when the tree becomes redundant call dm_pool_destroy
* ( no nasty iterating through the tree to free nodes ) .
*
* eg , On the other hand if you wanted to repeatedly insert and
* remove objects into the tree , you would be better off
* allocating the nodes from a free list ; you cannot free a
* single arbitrary node with pool .
*/
struct dm_pool ;
/* constructor and destructor */
struct dm_pool * dm_pool_create ( const char * name , size_t chunk_hint ) ;
void dm_pool_destroy ( struct dm_pool * p ) ;
/* simple allocation/free routines */
void * dm_pool_alloc ( struct dm_pool * p , size_t s ) ;
void * dm_pool_alloc_aligned ( struct dm_pool * p , size_t s , unsigned alignment ) ;
void dm_pool_empty ( struct dm_pool * p ) ;
void dm_pool_free ( struct dm_pool * p , void * ptr ) ;
2011-08-11 17:29:04 +00:00
/*
* To aid debugging , a pool can be locked . Any modifications made
* to the content of the pool while it is locked can be detected .
* Default compilation is using a crc checksum to notice modifications .
* The pool locking is using the mprotect with the compilation flag
* DEBUG_ENFORCE_POOL_LOCKING to enforce the memory protection .
*/
/* query pool lock status */
int dm_pool_locked ( struct dm_pool * p ) ;
/* mark pool as locked */
int dm_pool_lock ( struct dm_pool * p , int crc )
__attribute__ ( ( __warn_unused_result__ ) ) ;
/* mark pool as unlocked */
int dm_pool_unlock ( struct dm_pool * p , int crc )
__attribute__ ( ( __warn_unused_result__ ) ) ;
2005-10-16 22:57:20 +00:00
/*
* Object building routines :
*
* These allow you to ' grow ' an object , useful for
* building strings , or filling in dynamic
* arrays .
*
* It ' s probably best explained with an example :
*
* char * build_string ( struct dm_pool * mem )
* {
* int i ;
* char buffer [ 16 ] ;
*
* if ( ! dm_pool_begin_object ( mem , 128 ) )
* return NULL ;
*
* for ( i = 0 ; i < 50 ; i + + ) {
* snprintf ( buffer , sizeof ( buffer ) , " %d, " , i ) ;
2008-04-19 15:50:18 +00:00
* if ( ! dm_pool_grow_object ( mem , buffer , 0 ) )
2005-10-16 22:57:20 +00:00
* goto bad ;
* }
*
* // add null
* if ( ! dm_pool_grow_object ( mem , " \0 " , 1 ) )
* goto bad ;
*
* return dm_pool_end_object ( mem ) ;
*
* bad :
*
* dm_pool_abandon_object ( mem ) ;
* return NULL ;
* }
*
* So start an object by calling dm_pool_begin_object
* with a guess at the final object size - if in
* doubt make the guess too small .
*
* Then append chunks of data to your object with
* dm_pool_grow_object . Finally get your object with
* a call to dm_pool_end_object .
*
2008-04-19 15:50:18 +00:00
* Setting delta to 0 means it will use strlen ( extra ) .
2005-10-16 22:57:20 +00:00
*/
int dm_pool_begin_object ( struct dm_pool * p , size_t hint ) ;
int dm_pool_grow_object ( struct dm_pool * p , const void * extra , size_t delta ) ;
void * dm_pool_end_object ( struct dm_pool * p ) ;
void dm_pool_abandon_object ( struct dm_pool * p ) ;
/* utilities */
char * dm_pool_strdup ( struct dm_pool * p , const char * str ) ;
char * dm_pool_strndup ( struct dm_pool * p , const char * str , size_t n ) ;
void * dm_pool_zalloc ( struct dm_pool * p , size_t s ) ;
/******************
* bitset functions
* * * * * * * * * * * * * * * * * */
typedef uint32_t * dm_bitset_t ;
dm_bitset_t dm_bitset_create ( struct dm_pool * mem , unsigned num_bits ) ;
void dm_bitset_destroy ( dm_bitset_t bs ) ;
2010-04-20 13:58:22 +00:00
int dm_bitset_equal ( dm_bitset_t in1 , dm_bitset_t in2 ) ;
2010-04-20 10:58:18 +00:00
void dm_bit_and ( dm_bitset_t out , dm_bitset_t in1 , dm_bitset_t in2 ) ;
2005-10-16 22:57:20 +00:00
void dm_bit_union ( dm_bitset_t out , dm_bitset_t in1 , dm_bitset_t in2 ) ;
int dm_bit_get_first ( dm_bitset_t bs ) ;
int dm_bit_get_next ( dm_bitset_t bs , int last_bit ) ;
# define DM_BITS_PER_INT (sizeof(int) * CHAR_BIT)
# define dm_bit(bs, i) \
2010-07-05 22:22:43 +00:00
( ( bs ) [ ( ( i ) / DM_BITS_PER_INT ) + 1 ] & ( 0x1 < < ( ( i ) & ( DM_BITS_PER_INT - 1 ) ) ) )
2005-10-16 22:57:20 +00:00
# define dm_bit_set(bs, i) \
2010-07-05 22:22:43 +00:00
( ( bs ) [ ( ( i ) / DM_BITS_PER_INT ) + 1 ] | = ( 0x1 < < ( ( i ) & ( DM_BITS_PER_INT - 1 ) ) ) )
2005-10-16 22:57:20 +00:00
# define dm_bit_clear(bs, i) \
2010-07-05 22:22:43 +00:00
( ( bs ) [ ( ( i ) / DM_BITS_PER_INT ) + 1 ] & = ~ ( 0x1 < < ( ( i ) & ( DM_BITS_PER_INT - 1 ) ) ) )
2005-10-16 22:57:20 +00:00
# define dm_bit_set_all(bs) \
2010-07-05 22:22:43 +00:00
memset ( ( bs ) + 1 , - 1 , ( ( * ( bs ) / DM_BITS_PER_INT ) + 1 ) * sizeof ( int ) )
2005-10-16 22:57:20 +00:00
# define dm_bit_clear_all(bs) \
2010-07-05 22:22:43 +00:00
memset ( ( bs ) + 1 , 0 , ( ( * ( bs ) / DM_BITS_PER_INT ) + 1 ) * sizeof ( int ) )
2005-10-16 22:57:20 +00:00
# define dm_bit_copy(bs1, bs2) \
2010-07-05 22:22:43 +00:00
memcpy ( ( bs1 ) + 1 , ( bs2 ) + 1 , ( ( * ( bs1 ) / DM_BITS_PER_INT ) + 1 ) * sizeof ( int ) )
2005-10-16 22:57:20 +00:00
2005-12-13 15:49:27 +00:00
/* Returns number of set bits */
static inline unsigned hweight32 ( uint32_t i )
{
unsigned r = ( i & 0x55555555 ) + ( ( i > > 1 ) & 0x55555555 ) ;
r = ( r & 0x33333333 ) + ( ( r > > 2 ) & 0x33333333 ) ;
r = ( r & 0x0F0F0F0F ) + ( ( r > > 4 ) & 0x0F0F0F0F ) ;
r = ( r & 0x00FF00FF ) + ( ( r > > 8 ) & 0x00FF00FF ) ;
return ( r & 0x0000FFFF ) + ( ( r > > 16 ) & 0x0000FFFF ) ;
}
2005-10-16 22:57:20 +00:00
/****************
* hash functions
* * * * * * * * * * * * * * * */
struct dm_hash_table ;
struct dm_hash_node ;
typedef void ( * dm_hash_iterate_fn ) ( void * data ) ;
struct dm_hash_table * dm_hash_create ( unsigned size_hint ) ;
void dm_hash_destroy ( struct dm_hash_table * t ) ;
void dm_hash_wipe ( struct dm_hash_table * t ) ;
void * dm_hash_lookup ( struct dm_hash_table * t , const char * key ) ;
int dm_hash_insert ( struct dm_hash_table * t , const char * key , void * data ) ;
void dm_hash_remove ( struct dm_hash_table * t , const char * key ) ;
2011-03-10 12:48:40 +00:00
void * dm_hash_lookup_binary ( struct dm_hash_table * t , const void * key , uint32_t len ) ;
int dm_hash_insert_binary ( struct dm_hash_table * t , const void * key , uint32_t len ,
2005-10-16 22:57:20 +00:00
void * data ) ;
2011-03-10 12:48:40 +00:00
void dm_hash_remove_binary ( struct dm_hash_table * t , const void * key , uint32_t len ) ;
2005-10-16 22:57:20 +00:00
unsigned dm_hash_get_num_entries ( struct dm_hash_table * t ) ;
void dm_hash_iter ( struct dm_hash_table * t , dm_hash_iterate_fn f ) ;
char * dm_hash_get_key ( struct dm_hash_table * t , struct dm_hash_node * n ) ;
void * dm_hash_get_data ( struct dm_hash_table * t , struct dm_hash_node * n ) ;
struct dm_hash_node * dm_hash_get_first ( struct dm_hash_table * t ) ;
struct dm_hash_node * dm_hash_get_next ( struct dm_hash_table * t , struct dm_hash_node * n ) ;
# define dm_hash_iterate(v, h) \
2010-07-05 22:22:43 +00:00
for ( v = dm_hash_get_first ( ( h ) ) ; v ; \
v = dm_hash_get_next ( ( h ) , v ) )
2005-10-16 22:57:20 +00:00
2008-11-04 14:57:06 +00:00
/****************
* list functions
* * * * * * * * * * * * * * * */
/*
* 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 .
*/
struct dm_list {
struct dm_list * n , * p ;
} ;
/*
* Initialise a list before use .
* The list head ' s next and previous pointers point back to itself .
*/
# define DM_LIST_INIT(name) struct dm_list name = { &(name), &(name) }
void dm_list_init ( struct dm_list * head ) ;
/*
* Insert an element before ' head ' .
* If ' head ' is the list head , this adds an element to the end of the list .
*/
void dm_list_add ( struct dm_list * head , struct dm_list * elem ) ;
/*
* Insert an element after ' head ' .
* If ' head ' is the list head , this adds an element to the front of the list .
*/
void dm_list_add_h ( struct dm_list * head , struct dm_list * elem ) ;
/*
* Delete an element from its list .
* Note that this doesn ' t change the element itself - it may still be safe
* to follow its pointers .
*/
void dm_list_del ( struct dm_list * elem ) ;
/*
* Remove an element from existing list and insert before ' head ' .
*/
void dm_list_move ( struct dm_list * head , struct dm_list * elem ) ;
2010-05-06 10:10:15 +00:00
/*
* Join ' head1 ' to the of ' head ' .
*/
void dm_list_splice ( struct dm_list * head , struct dm_list * head1 ) ;
2008-11-04 14:57:06 +00:00
/*
* Is the list empty ?
*/
int dm_list_empty ( const struct dm_list * head ) ;
/*
* Is this the first element of the list ?
*/
int dm_list_start ( const struct dm_list * head , const struct dm_list * elem ) ;
/*
* Is this the last element of the list ?
*/
int dm_list_end ( const struct dm_list * head , const struct dm_list * elem ) ;
/*
* Return first element of the list or NULL if empty
*/
struct dm_list * dm_list_first ( const struct dm_list * head ) ;
/*
* Return last element of the list or NULL if empty
*/
struct dm_list * dm_list_last ( const struct dm_list * head ) ;
/*
* Return the previous element of the list , or NULL if we ' ve reached the start .
*/
struct dm_list * dm_list_prev ( const struct dm_list * head , const struct dm_list * elem ) ;
/*
* Return the next element of the list , or NULL if we ' ve reached the end .
*/
struct dm_list * dm_list_next ( const struct dm_list * head , const struct dm_list * elem ) ;
/*
* Given the address v of an instance of ' struct dm_list ' called ' head '
* contained in a structure of type t , return the containing structure .
*/
# define dm_list_struct_base(v, t, head) \
2010-12-20 13:39:12 +00:00
( ( t * ) ( ( const char * ) ( v ) - ( const char * ) & ( ( t * ) 0 ) - > head ) )
2008-11-04 14:57:06 +00:00
/*
* Given the address v of an instance of ' struct dm_list list ' contained in
* a structure of type t , return the containing structure .
*/
# define dm_list_item(v, t) dm_list_struct_base((v), t, list)
/*
* Given the address v of one known element e in a known structure of type t ,
* return another element f .
*/
# define dm_struct_field(v, t, e, f) \
( ( ( t * ) ( ( uintptr_t ) ( v ) - ( uintptr_t ) & ( ( t * ) 0 ) - > e ) ) - > f )
/*
* Given the address v of a known element e in a known structure of type t ,
* return the list head ' list '
*/
# define dm_list_head(v, t, e) dm_struct_field(v, t, e, list)
/*
* Set v to each element of a list in turn .
*/
# define dm_list_iterate(v, head) \
for ( v = ( head ) - > n ; v ! = head ; v = v - > n )
/*
* 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 .
*/
# define dm_list_uniterate(v, head, start) \
for ( v = ( start ) - > p ; v ! = head ; v = v - > p )
/*
* 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 .
*/
# define dm_list_iterate_safe(v, t, head) \
for ( v = ( head ) - > n , t = v - > n ; v ! = head ; v = t , t = v - > n )
/*
* Walk a list , setting ' v ' in turn to the containing structure of each item .
* The containing structure should be the same type as ' v ' .
* The ' struct dm_list ' variable within the containing structure is ' field ' .
*/
# define dm_list_iterate_items_gen(v, head, field) \
2010-06-23 17:03:14 +00:00
for ( v = dm_list_struct_base ( ( head ) - > n , __typeof__ ( * v ) , field ) ; \
2008-11-04 14:57:06 +00:00
& v - > field ! = ( head ) ; \
2010-06-23 17:03:14 +00:00
v = dm_list_struct_base ( v - > field . n , __typeof__ ( * v ) , field ) )
2008-11-04 14:57:06 +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 ' .
* The list should be ' struct dm_list list ' within the containing structure .
*/
# define dm_list_iterate_items(v, head) dm_list_iterate_items_gen(v, (head), list)
/*
* Walk a list , setting ' v ' in turn to the containing structure of each item .
* The containing structure should be the same type as ' v ' .
* The ' struct dm_list ' variable within the containing structure is ' field ' .
* t must be defined as a temporary variable of the same type as v .
*/
# define dm_list_iterate_items_gen_safe(v, t, head, field) \
2010-06-23 17:03:14 +00:00
for ( v = dm_list_struct_base ( ( head ) - > n , __typeof__ ( * v ) , field ) , \
t = dm_list_struct_base ( v - > field . n , __typeof__ ( * v ) , field ) ; \
2008-11-04 14:57:06 +00:00
& v - > field ! = ( head ) ; \
2010-06-23 17:03:14 +00:00
v = t , t = dm_list_struct_base ( v - > field . n , __typeof__ ( * v ) , field ) )
2008-11-04 14:57:06 +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 ' .
* The list should be ' struct dm_list list ' within the containing structure .
* t must be defined as a temporary variable of the same type as v .
*/
# define dm_list_iterate_items_safe(v, t, head) \
dm_list_iterate_items_gen_safe ( v , t , ( head ) , list )
/*
* 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 ' .
* The ' struct dm_list ' variable within the containing structure is ' field ' .
*/
# define dm_list_iterate_back_items_gen(v, head, field) \
2010-06-23 17:03:14 +00:00
for ( v = dm_list_struct_base ( ( head ) - > p , __typeof__ ( * v ) , field ) ; \
2008-11-04 14:57:06 +00:00
& v - > field ! = ( head ) ; \
2010-06-23 17:03:14 +00:00
v = dm_list_struct_base ( v - > field . p , __typeof__ ( * v ) , field ) )
2008-11-04 14:57:06 +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 ' .
* The list should be ' struct dm_list list ' within the containing structure .
*/
2008-11-04 15:07:45 +00:00
# define dm_list_iterate_back_items(v, head) dm_list_iterate_back_items_gen(v, (head), list)
2008-11-04 14:57:06 +00:00
/*
* Return the number of elements in a list by walking it .
*/
unsigned int dm_list_size ( const struct dm_list * head ) ;
2005-11-09 14:10:50 +00:00
/*********
2005-10-25 17:30:00 +00:00
* selinux
2005-11-09 14:10:50 +00:00
* * * * * * * * */
2010-12-13 10:43:56 +00:00
/*
* Obtain SELinux security context assigned for the path and set this
* context for creating a new file system object . This security context
* is global and it is used until reset to default policy behaviour
* by calling ' dm_prepare_selinux_context ( NULL , 0 ) ' .
*/
int dm_prepare_selinux_context ( const char * path , mode_t mode ) ;
/*
* Set SELinux context for existing file system object .
*/
2005-10-25 17:30:00 +00:00
int dm_set_selinux_context ( const char * path , mode_t mode ) ;
2006-01-04 16:05:44 +00:00
2006-08-21 12:07:03 +00:00
/*********************
* string manipulation
* * * * * * * * * * * * * * * * * * * * */
/*
* Break up the name of a mapped device into its constituent
* Volume Group , Logical Volume and Layer ( if present ) .
2009-06-03 11:40:23 +00:00
* If mem is supplied , the result is allocated from the mempool .
* Otherwise the strings are changed in situ .
2006-08-21 12:07:03 +00:00
*/
int dm_split_lvm_name ( struct dm_pool * mem , const char * dmname ,
char * * vgname , char * * lvname , char * * layer ) ;
/*
* Destructively split buffer into NULL - separated words in argv .
* Returns number of words .
*/
int dm_split_words ( char * buffer , unsigned max ,
unsigned ignore_comments , /* Not implemented */
char * * argv ) ;
2006-08-21 12:52:39 +00:00
/*
* Returns - 1 if buffer too small
*/
2010-07-02 21:16:50 +00:00
int dm_snprintf ( char * buf , size_t bufsize , const char * format , . . . )
__attribute__ ( ( format ( printf , 3 , 4 ) ) ) ;
2006-08-21 12:52:39 +00:00
2007-01-08 15:18:52 +00:00
/*
* Returns pointer to the last component of the path .
*/
2010-10-25 13:13:53 +00:00
const char * dm_basename ( const char * path ) ;
2007-01-08 15:18:52 +00:00
2011-08-30 14:55:15 +00:00
/*
2011-09-01 21:04:14 +00:00
* Returns number of occurrences of ' c ' in ' str ' of length ' size ' .
2011-08-30 14:55:15 +00:00
*/
unsigned dm_count_chars ( const char * str , size_t len , const int c ) ;
/*
* Length of string after escaping double quotes and backslashes .
*/
size_t dm_escaped_len ( const char * str ) ;
/*
* < vg > - < lv > - < layer > or if ! layer just < vg > - < lv > .
*/
char * dm_build_dm_name ( struct dm_pool * mem , const char * vgname ,
const char * lvname , const char * layer ) ;
char * dm_build_dm_uuid ( struct dm_pool * mem , const char * prefix , const char * lvid , const char * layer ) ;
/*
* Copies a string , quoting double quotes with backslashes .
*/
char * dm_escape_double_quotes ( char * out , const char * src ) ;
/*
* Undo quoting in situ .
*/
void dm_unescape_double_quotes ( char * src ) ;
/*
* Unescape colons and " at " signs in situ and save the substrings
* starting at the position of the first unescaped colon and the
* first unescaped " at " sign . This is normally used to unescape
* device names used as PVs .
*/
void dm_unescape_colons_and_at_signs ( char * src ,
char * * substr_first_unquoted_colon ,
char * * substr_first_unquoted_at_sign ) ;
2007-07-24 14:15:45 +00:00
/**************************
* file / stream manipulation
* * * * * * * * * * * * * * * * * * * * * * * * * */
2007-07-28 10:48:36 +00:00
/*
* Create a directory ( with parent directories if necessary ) .
* Returns 1 on success , 0 on failure .
*/
int dm_create_dir ( const char * dir ) ;
2011-09-22 17:23:35 +00:00
int dm_is_empty_dir ( const char * dir ) ;
2007-07-24 14:15:45 +00:00
/*
* Close a stream , with nicer error checking than fclose ' s .
* Derived from gnulib ' s close - stream . c .
*
* Close " stream " . Return 0 if successful , and EOF ( setting errno )
* otherwise . Upon failure , set errno to 0 if the error number
* cannot be determined . Useful mainly for writable streams .
*/
int dm_fclose ( FILE * stream ) ;
2007-01-11 21:54:53 +00:00
/*
* Returns size of a buffer which is allocated with dm_malloc .
* Pointer to the buffer is stored in * buf .
2007-01-15 18:21:01 +00:00
* Returns - 1 on failure leaving buf undefined .
2007-01-11 21:54:53 +00:00
*/
2010-07-02 21:16:50 +00:00
int dm_asprintf ( char * * buf , const char * format , . . . )
__attribute__ ( ( format ( printf , 2 , 3 ) ) ) ;
2007-01-11 21:54:53 +00:00
2010-07-13 13:51:01 +00:00
/*
* create lockfile ( pidfile ) - create and lock a lock file
* @ lockfile : location of lock file
*
* Returns : 1 on success , 0 otherwise , errno is handled internally
*/
int dm_create_lockfile ( const char * lockfile ) ;
2010-07-21 13:40:21 +00:00
/*
* Query whether a daemon is running based on its lockfile
*
* Returns : 1 if running , 0 if not
*/
int dm_daemon_is_running ( const char * lockfile ) ;
2007-04-27 18:40:23 +00:00
/*********************
* regular expressions
* * * * * * * * * * * * * * * * * * * * */
struct dm_regex ;
/*
* Initialise an array of num patterns for matching .
* Uses memory from mem .
*/
2010-12-20 13:23:11 +00:00
struct dm_regex * dm_regex_create ( struct dm_pool * mem , const char * const * patterns ,
2007-04-27 18:40:23 +00:00
unsigned num_patterns ) ;
/*
* Match string s against the patterns .
* Returns the index of the highest pattern in the array that matches ,
* or - 1 if none match .
*/
int dm_regex_match ( struct dm_regex * regex , const char * s ) ;
2010-07-20 15:32:07 +00:00
/*
* This is useful for regression testing only . The idea is if two
* fingerprints are different , then the two dfas are certainly not
* isomorphic . If two fingerprints _are_ the same then it ' s very likely
* that the dfas are isomorphic .
2010-07-21 12:09:12 +00:00
*
* This function must be called before any matching is done .
2010-07-20 15:32:07 +00:00
*/
uint32_t dm_regex_fingerprint ( struct dm_regex * regex ) ;
2007-01-16 18:04:15 +00:00
/*********************
* reporting functions
* * * * * * * * * * * * * * * * * * * * */
struct dm_report_object_type {
uint32_t id ; /* Powers of 2 */
const char * desc ;
const char * prefix ; /* field id string prefix (optional) */
void * ( * data_fn ) ( void * object ) ; /* callback from report_object() */
} ;
struct dm_report_field ;
/*
* dm_report_field_type flags
*/
2007-01-18 17:47:58 +00:00
# define DM_REPORT_FIELD_MASK 0x000000FF
# define DM_REPORT_FIELD_ALIGN_MASK 0x0000000F
2007-01-16 18:04:15 +00:00
# define DM_REPORT_FIELD_ALIGN_LEFT 0x00000001
# define DM_REPORT_FIELD_ALIGN_RIGHT 0x00000002
2007-01-18 17:47:58 +00:00
# define DM_REPORT_FIELD_TYPE_MASK 0x000000F0
# define DM_REPORT_FIELD_TYPE_STRING 0x00000010
# define DM_REPORT_FIELD_TYPE_NUMBER 0x00000020
2007-01-16 18:04:15 +00:00
2010-08-20 12:44:30 +00:00
# define DM_REPORT_FIELD_TYPE_ID_LEN 32
# define DM_REPORT_FIELD_TYPE_HEADING_LEN 32
2007-01-16 18:04:15 +00:00
struct dm_report ;
struct dm_report_field_type {
uint32_t type ; /* object type id */
2007-01-29 17:23:54 +00:00
uint32_t flags ; /* DM_REPORT_FIELD_* */
uint32_t offset ; /* byte offset in the object */
int32_t width ; /* default width */
2010-08-20 12:44:30 +00:00
/* string used to specify the field */
const char id [ DM_REPORT_FIELD_TYPE_ID_LEN ] ;
/* string printed in header */
const char heading [ DM_REPORT_FIELD_TYPE_HEADING_LEN ] ;
2007-01-16 18:04:15 +00:00
int ( * report_fn ) ( struct dm_report * rh , struct dm_pool * mem ,
struct dm_report_field * field , const void * data ,
2010-06-16 13:01:25 +00:00
void * private_data ) ;
2007-01-23 19:18:52 +00:00
const char * desc ; /* description of the field */
2007-01-16 18:04:15 +00:00
} ;
/*
* dm_report_init output_flags
*/
2008-04-20 00:11:08 +00:00
# define DM_REPORT_OUTPUT_MASK 0x000000FF
# define DM_REPORT_OUTPUT_ALIGNED 0x00000001
# define DM_REPORT_OUTPUT_BUFFERED 0x00000002
# define DM_REPORT_OUTPUT_HEADINGS 0x00000004
# define DM_REPORT_OUTPUT_FIELD_NAME_PREFIX 0x00000008
2008-06-24 20:16:47 +00:00
# define DM_REPORT_OUTPUT_FIELD_UNQUOTED 0x00000010
2008-06-24 22:53:48 +00:00
# define DM_REPORT_OUTPUT_COLUMNS_AS_ROWS 0x00000020
2007-01-16 18:04:15 +00:00
struct dm_report * dm_report_init ( uint32_t * report_types ,
const struct dm_report_object_type * types ,
const struct dm_report_field_type * fields ,
const char * output_fields ,
const char * output_separator ,
uint32_t output_flags ,
const char * sort_keys ,
2010-06-16 13:01:25 +00:00
void * private_data ) ;
2007-01-16 18:04:15 +00:00
int dm_report_object ( struct dm_report * rh , void * object ) ;
int dm_report_output ( struct dm_report * rh ) ;
void dm_report_free ( struct dm_report * rh ) ;
2008-04-20 00:11:08 +00:00
/*
* Prefix added to each field name with DM_REPORT_OUTPUT_FIELD_NAME_PREFIX
*/
int dm_report_set_output_field_name_prefix ( struct dm_report * rh ,
const char * report_prefix ) ;
2007-01-16 18:04:15 +00:00
/*
2007-01-22 15:03:57 +00:00
* Report functions are provided for simple data types .
* They take care of allocating copies of the data .
2007-01-16 18:04:15 +00:00
*/
2007-01-22 15:03:57 +00:00
int dm_report_field_string ( struct dm_report * rh , struct dm_report_field * field ,
2011-02-18 14:38:47 +00:00
const char * const * data ) ;
2007-01-22 15:03:57 +00:00
int dm_report_field_int32 ( struct dm_report * rh , struct dm_report_field * field ,
const int32_t * data ) ;
int dm_report_field_uint32 ( struct dm_report * rh , struct dm_report_field * field ,
const uint32_t * data ) ;
int dm_report_field_int ( struct dm_report * rh , struct dm_report_field * field ,
const int * data ) ;
int dm_report_field_uint64 ( struct dm_report * rh , struct dm_report_field * field ,
const uint64_t * data ) ;
2007-01-16 18:04:15 +00:00
/*
2007-01-22 15:03:57 +00:00
* For custom fields , allocate the data in ' mem ' and use
* dm_report_field_set_value ( ) .
* ' sortvalue ' may be NULL if it matches ' value '
2007-01-16 18:04:15 +00:00
*/
2007-01-22 15:03:57 +00:00
void dm_report_field_set_value ( struct dm_report_field * field , const void * value ,
const void * sortvalue ) ;
2007-01-16 18:04:15 +00:00
2011-08-30 14:55:15 +00:00
2011-09-01 21:04:14 +00:00
/*************************
* config file parse / print
* * * * * * * * * * * * * * * * * * * * * * * * */
// FIXME AGK Review this interface before inclusion in a release.
2011-08-30 14:55:15 +00:00
enum {
DM_CFG_STRING ,
DM_CFG_FLOAT ,
DM_CFG_INT ,
DM_CFG_EMPTY_ARRAY
} ;
struct dm_config_value {
int type ;
union {
int64_t i ;
2011-09-01 21:04:14 +00:00
float f ;
double d ; /* For completeness. (Unused.) */
2011-08-30 14:55:15 +00:00
const char * str ;
} v ;
2011-09-01 21:04:14 +00:00
struct dm_config_value * next ; /* For arrays */
2011-08-30 14:55:15 +00:00
} ;
struct dm_config_node {
const char * key ;
struct dm_config_node * parent , * sib , * child ;
struct dm_config_value * v ;
} ;
2011-09-02 01:32:08 +00:00
/* FIXME Move cascade to dm_config_node and remove this struct */
2011-08-30 14:55:15 +00:00
struct dm_config_tree {
struct dm_config_node * root ;
struct dm_config_tree * cascade ;
} ;
struct dm_config_tree * dm_config_create ( const char * filename , int keep_open ) ;
struct dm_config_tree * dm_config_from_string ( const char * config_settings ) ;
int dm_config_parse ( struct dm_config_tree * cft , const char * start , const char * end ) ;
void * dm_config_get_custom ( struct dm_config_tree * cft ) ;
int dm_config_check_file ( struct dm_config_tree * cft , const char * * filename , struct stat * info ) ;
int dm_config_keep_open ( struct dm_config_tree * ctf ) ;
void dm_config_set_custom ( struct dm_config_tree * cft , void * custom ) ;
2011-09-02 01:32:08 +00:00
/*
* If there ' s a cascaded dm_config_tree , remove the top layer
* and return the layer below . Otherwise return NULL .
*/
struct dm_config_tree * dm_config_remove_cascaded_tree ( struct dm_config_tree * cft ) ;
/*
* When searching , first_cft is checked before second_cft .
*/
struct dm_config_tree * dm_config_insert_cascaded_tree ( struct dm_config_tree * first_cft , struct dm_config_tree * second_cft ) ;
2011-08-30 14:55:15 +00:00
void dm_config_destroy ( struct dm_config_tree * cft ) ;
int dm_config_write ( struct dm_config_tree * cft , const char * file ,
int argc , char * * argv ) ;
typedef int ( * dm_putline_fn ) ( const char * line , void * baton ) ;
int dm_config_write_node ( const struct dm_config_node * cn , dm_putline_fn putline , void * baton ) ;
time_t dm_config_timestamp ( struct dm_config_tree * cft ) ;
int dm_config_changed ( struct dm_config_tree * cft ) ;
2011-08-31 15:19:19 +00:00
struct dm_config_node * dm_config_find_node ( struct dm_config_node * cn , const char * path ) ;
int dm_config_has_node ( const struct dm_config_node * cn , const char * path ) ;
2011-08-30 14:55:15 +00:00
const char * dm_config_find_str ( const struct dm_config_node * cn , const char * path , const char * fail ) ;
int dm_config_find_int ( const struct dm_config_node * cn , const char * path , int fail ) ;
float dm_config_find_float ( const struct dm_config_node * cn , const char * path , float fail ) ;
2011-09-01 21:04:14 +00:00
const struct dm_config_node * dm_config_tree_find_node ( const struct dm_config_tree * cft , const char * path ) ;
2011-08-30 14:55:15 +00:00
const char * dm_config_tree_find_str ( const struct dm_config_tree * cft ,
const char * path , const char * fail ) ;
int dm_config_tree_find_int ( const struct dm_config_tree * cft ,
const char * path , int fail ) ;
int64_t dm_config_tree_find_int64 ( const struct dm_config_tree * cft ,
const char * path , int64_t fail ) ;
float dm_config_tree_find_float ( const struct dm_config_tree * cft ,
const char * path , float fail ) ;
int dm_config_tree_find_bool ( const struct dm_config_tree * cft ,
const char * path , int fail ) ;
/*
* Understands ( 0 , ~ 0 ) , ( y , n ) , ( yes , no ) , ( on ,
* off ) , ( true , false ) .
*/
int dm_config_find_bool ( const struct dm_config_node * cn , const char * path , int fail ) ;
int dm_config_get_uint32 ( const struct dm_config_node * cn , const char * path ,
uint32_t * result ) ;
int dm_config_get_uint64 ( const struct dm_config_node * cn , const char * path ,
uint64_t * result ) ;
int dm_config_get_str ( const struct dm_config_node * cn , const char * path ,
const char * * result ) ;
2011-08-31 15:19:19 +00:00
int dm_config_get_list ( const struct dm_config_node * cn , const char * path ,
const struct dm_config_value * * result ) ;
int dm_config_get_section ( const struct dm_config_node * cn , const char * path ,
const struct dm_config_node * * result ) ;
2011-08-30 14:55:15 +00:00
unsigned dm_config_maybe_section ( const char * str , unsigned len ) ;
const char * dm_config_parent_name ( const struct dm_config_node * n ) ;
struct dm_config_node * dm_config_clone_node_with_mem ( struct dm_pool * mem ,
const struct dm_config_node * node ,
int siblings ) ;
struct dm_config_node * dm_config_create_node ( struct dm_config_tree * cft , const char * key ) ;
struct dm_config_value * dm_config_create_value ( struct dm_config_tree * cft ) ;
struct dm_config_node * dm_config_clone_node ( struct dm_config_tree * cft ,
const struct dm_config_node * cn ,
int siblings ) ;
struct dm_pool * dm_config_memory ( struct dm_config_tree * cft ) ;
2009-10-22 12:55:47 +00:00
/* Cookie prefixes.
* The cookie value consists of a prefix ( 16 bits ) and a base ( 16 bits ) .
* We can use the prefix to store the flags . These flags are sent to
* kernel within given dm task . When returned back to userspace in
* DM_COOKIE udev environment variable , we can control several aspects
* of udev rules we use by decoding the cookie prefix . When doing the
* notification , we replace the cookie prefix with DM_COOKIE_MAGIC ,
* so we notify the right semaphore .
2009-11-13 12:43:21 +00:00
* It is still possible to use cookies for passing the flags to udev
* rules even when udev_sync is disabled . The base part of the cookie
* will be zero ( there ' s no notification semaphore ) and prefix will be
* set then . However , having udev_sync enabled is highly recommended .
2009-10-22 12:55:47 +00:00
*/
2009-08-06 15:04:30 +00:00
# define DM_COOKIE_MAGIC 0x0D4D
2009-10-22 12:55:47 +00:00
# define DM_UDEV_FLAGS_MASK 0xFFFF0000
# define DM_UDEV_FLAGS_SHIFT 16
2009-10-26 14:29:33 +00:00
/*
* DM_UDEV_DISABLE_DM_RULES_FLAG is set in case we need to disable
* basic device - mapper udev rules that create symlinks in / dev / < DM_DIR >
* directory . However , we can ' t reliably prevent creating default
* nodes by udev ( commonly / dev / dm - X , where X is a number ) .
*/
# define DM_UDEV_DISABLE_DM_RULES_FLAG 0x0001
2009-10-22 12:55:47 +00:00
/*
* DM_UDEV_DISABLE_SUBSYTEM_RULES_FLAG is set in case we need to disable
* subsystem udev rules , but still we need the general DM udev rules to
* be applied ( to create the nodes and symlinks under / dev and / dev / disk ) .
*/
2009-10-26 14:29:33 +00:00
# define DM_UDEV_DISABLE_SUBSYSTEM_RULES_FLAG 0x0002
2009-10-22 12:55:47 +00:00
/*
* DM_UDEV_DISABLE_DISK_RULES_FLAG is set in case we need to disable
* general DM rules that set symlinks in / dev / disk directory .
*/
2009-10-26 14:29:33 +00:00
# define DM_UDEV_DISABLE_DISK_RULES_FLAG 0x0004
/*
* DM_UDEV_DISABLE_OTHER_RULES_FLAG is set in case we need to disable
* all the other rules that are not general device - mapper nor subsystem
2009-11-13 12:33:27 +00:00
* related ( the rules belong to other software or packages ) . All foreign
* rules should check this flag directly and they should ignore further
* rule processing for such event .
2009-10-26 14:29:33 +00:00
*/
# define DM_UDEV_DISABLE_OTHER_RULES_FLAG 0x0008
2009-10-22 12:55:47 +00:00
/*
* DM_UDEV_LOW_PRIORITY_FLAG is set in case we need to instruct the
* udev rules to give low priority to the device that is currently
* processed . For example , this provides a way to select which symlinks
* could be overwritten by high priority ones if their names are equal .
* Common situation is a name based on FS UUID while using origin and
* snapshot devices .
*/
2009-10-26 14:29:33 +00:00
# define DM_UDEV_LOW_PRIORITY_FLAG 0x0010
2010-02-15 16:21:33 +00:00
/*
* DM_UDEV_DISABLE_LIBRARY_FALLBACK is set in case we need to disable
* libdevmapper ' s node management . We will rely on udev completely
* and there will be no fallback action provided by libdevmapper if
* udev does something improperly .
*/
# define DM_UDEV_DISABLE_LIBRARY_FALLBACK 0x0020
2010-04-28 13:37:36 +00:00
/*
* DM_UDEV_PRIMARY_SOURCE_FLAG is automatically appended by
* libdevmapper for all ioctls generating udev uevents . Once used in
* udev rules , we know if this is a real " primary sourced " event or not .
* We need to distinguish real events originated in libdevmapper from
* any spurious events to gather all missing information ( e . g . events
* generated as a result of " udevadm trigger " command or as a result
* of the " watch " udev rule ) .
*/
# define DM_UDEV_PRIMARY_SOURCE_FLAG 0x0040
2009-08-06 15:04:30 +00:00
2009-07-31 15:53:11 +00:00
int dm_cookie_supported ( void ) ;
/*
2009-11-13 12:43:21 +00:00
* Udev synchronisation functions .
2009-07-31 15:53:11 +00:00
*/
void dm_udev_set_sync_support ( int sync_with_udev ) ;
int dm_udev_get_sync_support ( void ) ;
2010-01-11 15:36:24 +00:00
void dm_udev_set_checking ( int checking ) ;
int dm_udev_get_checking ( void ) ;
2011-01-10 13:42:31 +00:00
/*
* Default value to get new auto generated cookie created
*/
# define DM_COOKIE_AUTO_CREATE 0
2010-02-15 16:21:33 +00:00
int dm_udev_create_cookie ( uint32_t * cookie ) ;
2009-07-31 17:51:45 +00:00
int dm_udev_complete ( uint32_t cookie ) ;
2009-07-31 15:53:11 +00:00
int dm_udev_wait ( uint32_t cookie ) ;
2009-09-25 11:58:00 +00:00
# define DM_DEV_DIR_UMASK 0022
2011-01-04 14:43:53 +00:00
# define DM_CONTROL_NODE_UMASK 0177
2009-09-25 11:58:00 +00:00
2010-06-16 13:01:25 +00:00
# ifdef __cplusplus
}
# endif
2006-01-04 16:05:44 +00:00
# endif /* LIB_DEVICE_MAPPER_H */