mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-06 17:18:29 +03:00
606 lines
15 KiB
C
606 lines
15 KiB
C
/*
|
|
* Copyright (C) 2014-2015 Red Hat, Inc.
|
|
*
|
|
* 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
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*/
|
|
|
|
#ifndef _LVM_LVMLOCKD_INTERNAL_H
|
|
#define _LVM_LVMLOCKD_INTERNAL_H
|
|
|
|
#define MAX_NAME 64
|
|
#define MAX_ARGS 64
|
|
|
|
#define R_NAME_GL_DISABLED "_GLLK_disabled"
|
|
#define R_NAME_GL "GLLK"
|
|
#define R_NAME_VG "VGLK"
|
|
#define S_NAME_GL_DLM "lvm_global"
|
|
#define LVM_LS_PREFIX "lvm_" /* ls name is prefix + vg_name */
|
|
/* global lockspace name for sanlock is a vg name */
|
|
|
|
/* lock manager types */
|
|
enum {
|
|
LD_LM_NONE = 0,
|
|
LD_LM_UNUSED = 1, /* place holder so values match lib/locking/lvmlockd.h */
|
|
LD_LM_DLM = 2,
|
|
LD_LM_SANLOCK = 3,
|
|
};
|
|
|
|
/* operation types */
|
|
enum {
|
|
LD_OP_HELLO = 1,
|
|
LD_OP_QUIT,
|
|
LD_OP_INIT,
|
|
LD_OP_FREE,
|
|
LD_OP_START,
|
|
LD_OP_STOP,
|
|
LD_OP_LOCK,
|
|
LD_OP_UPDATE,
|
|
LD_OP_CLOSE,
|
|
LD_OP_ENABLE,
|
|
LD_OP_DISABLE,
|
|
LD_OP_START_WAIT,
|
|
LD_OP_STOP_ALL,
|
|
LD_OP_DUMP_INFO,
|
|
LD_OP_DUMP_LOG,
|
|
LD_OP_RENAME_BEFORE,
|
|
LD_OP_RENAME_FINAL,
|
|
LD_OP_RUNNING_LM,
|
|
LD_OP_FIND_FREE_LOCK,
|
|
LD_OP_KILL_VG,
|
|
LD_OP_DROP_VG,
|
|
LD_OP_BUSY,
|
|
};
|
|
|
|
/* resource types */
|
|
enum {
|
|
LD_RT_GL = 1,
|
|
LD_RT_VG,
|
|
LD_RT_LV,
|
|
};
|
|
|
|
/* lock modes, more restrictive must be larger value */
|
|
enum {
|
|
LD_LK_IV = -1,
|
|
LD_LK_UN = 0,
|
|
LD_LK_NL = 1,
|
|
LD_LK_SH = 2,
|
|
LD_LK_EX = 3,
|
|
};
|
|
|
|
struct list_head {
|
|
struct list_head *next, *prev;
|
|
};
|
|
|
|
struct client {
|
|
struct list_head list;
|
|
pthread_mutex_t mutex;
|
|
int pid;
|
|
int fd;
|
|
int pi;
|
|
uint32_t id;
|
|
unsigned int recv : 1;
|
|
unsigned int dead : 1;
|
|
unsigned int poll_ignore : 1;
|
|
unsigned int lock_ops : 1;
|
|
char name[MAX_NAME+1];
|
|
};
|
|
|
|
#define LD_AF_PERSISTENT 0x00000001
|
|
#define LD_AF_NO_CLIENT 0x00000002
|
|
#define LD_AF_UNLOCK_CANCEL 0x00000004
|
|
#define LD_AF_NEXT_VERSION 0x00000008
|
|
#define LD_AF_WAIT 0x00000010
|
|
#define LD_AF_FORCE 0x00000020
|
|
#define LD_AF_EX_DISABLE 0x00000040
|
|
#define LD_AF_ENABLE 0x00000080
|
|
#define LD_AF_DISABLE 0x00000100
|
|
#define LD_AF_SEARCH_LS 0x00000200
|
|
#define LD_AF_WAIT_STARTING 0x00001000
|
|
#define LD_AF_DUP_GL_LS 0x00002000
|
|
#define LD_AF_ADOPT 0x00010000
|
|
#define LD_AF_WARN_GL_REMOVED 0x00020000
|
|
#define LD_AF_LV_LOCK 0x00040000
|
|
#define LD_AF_LV_UNLOCK 0x00080000
|
|
|
|
/*
|
|
* Number of times to repeat a lock request after
|
|
* a lock conflict (-EAGAIN) if unspecified in the
|
|
* request.
|
|
*/
|
|
#define DEFAULT_MAX_RETRIES 4
|
|
|
|
struct action {
|
|
struct list_head list;
|
|
uint32_t client_id;
|
|
uint32_t flags; /* LD_AF_ */
|
|
uint32_t version;
|
|
uint64_t host_id;
|
|
int8_t op; /* operation type LD_OP_ */
|
|
int8_t rt; /* resource type LD_RT_ */
|
|
int8_t mode; /* lock mode LD_LK_ */
|
|
int8_t lm_type; /* lock manager: LM_DLM, LM_SANLOCK */
|
|
int retries;
|
|
int max_retries;
|
|
int result;
|
|
int lm_rv; /* return value from lm_ function */
|
|
char vg_uuid[64];
|
|
char vg_name[MAX_NAME+1];
|
|
char lv_name[MAX_NAME+1];
|
|
char lv_uuid[MAX_NAME+1];
|
|
char vg_args[MAX_ARGS+1];
|
|
char lv_args[MAX_ARGS+1];
|
|
char vg_sysid[MAX_NAME+1];
|
|
};
|
|
|
|
struct resource {
|
|
struct list_head list; /* lockspace.resources */
|
|
char name[MAX_NAME+1]; /* vg name or lv name */
|
|
int8_t type; /* resource type LD_RT_ */
|
|
int8_t mode;
|
|
unsigned int sh_count; /* number of sh locks on locks list */
|
|
uint32_t version;
|
|
uint32_t last_client_id; /* last client_id to lock or unlock resource */
|
|
unsigned int lm_init : 1; /* lm_data is initialized */
|
|
unsigned int adopt : 1; /* temp flag in remove_inactive_lvs */
|
|
unsigned int version_zero_valid : 1;
|
|
unsigned int use_vb : 1;
|
|
struct list_head locks;
|
|
struct list_head actions;
|
|
char lv_args[MAX_ARGS+1];
|
|
char lm_data[0]; /* lock manager specific data */
|
|
};
|
|
|
|
#define LD_LF_PERSISTENT 0x00000001
|
|
|
|
struct lock {
|
|
struct list_head list; /* resource.locks */
|
|
int8_t mode; /* lock mode LD_LK_ */
|
|
uint32_t version;
|
|
uint32_t flags; /* LD_LF_ */
|
|
uint32_t client_id; /* may be 0 for persistent or internal locks */
|
|
};
|
|
|
|
struct lockspace {
|
|
struct list_head list; /* lockspaces */
|
|
char name[MAX_NAME+1];
|
|
char vg_name[MAX_NAME+1];
|
|
char vg_uuid[64];
|
|
char vg_args[MAX_ARGS+1]; /* lock manager specific args */
|
|
char vg_sysid[MAX_NAME+1];
|
|
int8_t lm_type; /* lock manager: LM_DLM, LM_SANLOCK */
|
|
void *lm_data;
|
|
uint64_t host_id;
|
|
uint64_t free_lock_offset; /* start search for free lock here */
|
|
|
|
uint32_t start_client_id; /* client_id that started the lockspace */
|
|
pthread_t thread; /* makes synchronous lock requests */
|
|
pthread_cond_t cond;
|
|
pthread_mutex_t mutex;
|
|
unsigned int create_fail : 1;
|
|
unsigned int create_done : 1;
|
|
unsigned int thread_work : 1;
|
|
unsigned int thread_stop : 1;
|
|
unsigned int thread_done : 1;
|
|
unsigned int sanlock_gl_enabled: 1;
|
|
unsigned int sanlock_gl_dup: 1;
|
|
unsigned int free_vg: 1;
|
|
unsigned int kill_vg: 1;
|
|
unsigned int drop_vg: 1;
|
|
|
|
struct list_head actions; /* new client actions */
|
|
struct list_head resources; /* resource/lock state for gl/vg/lv */
|
|
};
|
|
|
|
/* val_blk version */
|
|
#define VAL_BLK_VERSION 0x0101
|
|
|
|
/* val_blk flags */
|
|
#define VBF_REMOVED 0x0001
|
|
|
|
struct val_blk {
|
|
uint16_t version;
|
|
uint16_t flags;
|
|
uint32_t r_version;
|
|
};
|
|
|
|
/* lm_unlock flags */
|
|
#define LMUF_FREE_VG 0x00000001
|
|
|
|
#define container_of(ptr, type, member) ({ \
|
|
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
|
|
(type *)( (char *)__mptr - offsetof(type,member) );})
|
|
|
|
static inline void INIT_LIST_HEAD(struct list_head *list)
|
|
{
|
|
list->next = list;
|
|
list->prev = list;
|
|
}
|
|
|
|
static inline void __list_add(struct list_head *new,
|
|
struct list_head *prev,
|
|
struct list_head *next)
|
|
{
|
|
next->prev = new;
|
|
new->next = next;
|
|
new->prev = prev;
|
|
prev->next = new;
|
|
}
|
|
|
|
static inline void __list_del(struct list_head *prev, struct list_head *next)
|
|
{
|
|
next->prev = prev;
|
|
prev->next = next;
|
|
}
|
|
|
|
static inline void list_add(struct list_head *new, struct list_head *head)
|
|
{
|
|
__list_add(new, head, head->next);
|
|
}
|
|
|
|
static inline void list_add_tail(struct list_head *new, struct list_head *head)
|
|
{
|
|
__list_add(new, head->prev, head);
|
|
}
|
|
|
|
static inline void list_del(struct list_head *entry)
|
|
{
|
|
__list_del(entry->prev, entry->next);
|
|
}
|
|
|
|
static inline int list_empty(const struct list_head *head)
|
|
{
|
|
return head->next == head;
|
|
}
|
|
|
|
#define list_entry(ptr, type, member) \
|
|
container_of(ptr, type, member)
|
|
|
|
#define list_first_entry(ptr, type, member) \
|
|
list_entry((ptr)->next, type, member)
|
|
|
|
#define list_for_each_entry(pos, head, member) \
|
|
for (pos = list_entry((head)->next, typeof(*pos), member); \
|
|
&pos->member != (head); \
|
|
pos = list_entry(pos->member.next, typeof(*pos), member))
|
|
|
|
#define list_for_each_entry_safe(pos, n, head, member) \
|
|
for (pos = list_entry((head)->next, typeof(*pos), member), \
|
|
n = list_entry(pos->member.next, typeof(*pos), member); \
|
|
&pos->member != (head); \
|
|
pos = n, n = list_entry(n->member.next, typeof(*n), member))
|
|
|
|
|
|
/* to improve readability */
|
|
#define WAIT 1
|
|
#define NO_WAIT 0
|
|
#define FORCE 1
|
|
#define NO_FORCE 0
|
|
|
|
/*
|
|
* global variables
|
|
*/
|
|
|
|
#ifndef EXTERN
|
|
#define EXTERN extern
|
|
#define INIT(X)
|
|
#else
|
|
#undef EXTERN
|
|
#define EXTERN
|
|
#define INIT(X) =X
|
|
#endif
|
|
|
|
/*
|
|
* gl_type_static and gl_use_ are set by command line or config file
|
|
* to specify whether the global lock comes from dlm or sanlock.
|
|
* Without a static setting, lvmlockd will figure out where the
|
|
* global lock should be (but it could get mixed up in cases where
|
|
* both sanlock and dlm vgs exist.)
|
|
*
|
|
* gl_use_dlm means that the gl should come from lockspace gl_lsname_dlm
|
|
* gl_use_sanlock means that the gl should come from lockspace gl_lsname_sanlock
|
|
*
|
|
* gl_use_dlm has precedence over gl_use_sanlock, so if a node sees both
|
|
* dlm and sanlock vgs, it will use the dlm gl.
|
|
*
|
|
* gl_use_ is set when the first evidence of that lm_type is seen
|
|
* in any command.
|
|
*
|
|
* gl_lsname_sanlock is set when the first vg is seen in which an
|
|
* enabled gl is exists, or when init_vg creates a vg with gl enabled,
|
|
* or when enable_gl is used.
|
|
*
|
|
* gl_lsname_sanlock is cleared when free_vg deletes a vg with gl enabled
|
|
* or when disable_gl matches.
|
|
*/
|
|
|
|
EXTERN int gl_type_static;
|
|
EXTERN int gl_use_dlm;
|
|
EXTERN int gl_use_sanlock;
|
|
EXTERN int gl_vg_removed;
|
|
EXTERN char gl_lsname_dlm[MAX_NAME+1];
|
|
EXTERN char gl_lsname_sanlock[MAX_NAME+1];
|
|
EXTERN int global_dlm_lockspace_exists;
|
|
|
|
EXTERN int daemon_test; /* run as much as possible without a live lock manager */
|
|
EXTERN int daemon_debug;
|
|
EXTERN int daemon_host_id;
|
|
EXTERN const char *daemon_host_id_file;
|
|
EXTERN int sanlock_io_timeout;
|
|
|
|
/*
|
|
* This flag is set to 1 if we see multiple vgs with the global
|
|
* lock enabled. While this is set, we return a special flag
|
|
* with the vg lock result indicating to the lvm command that
|
|
* there is a duplicate gl in the vg which should be resolved.
|
|
* While this is set, find_lockspace_name has the side job of
|
|
* counting the number of lockspaces with enabled gl's so that
|
|
* this can be set back to zero when the duplicates are disabled.
|
|
*/
|
|
EXTERN int sanlock_gl_dup;
|
|
|
|
void log_level(int level, const char *fmt, ...) __attribute__((format(printf, 2, 3)));
|
|
#define log_debug(fmt, args...) log_level(LOG_DEBUG, fmt, ##args)
|
|
#define log_error(fmt, args...) log_level(LOG_ERR, fmt, ##args)
|
|
#define log_warn(fmt, args...) log_level(LOG_WARNING, fmt, ##args)
|
|
|
|
struct lockspace *alloc_lockspace(void);
|
|
int lockspaces_empty(void);
|
|
int last_string_from_args(char *args_in, char *last);
|
|
int version_from_args(char *args, unsigned int *major, unsigned int *minor, unsigned int *patch);
|
|
|
|
static inline const char *mode_str(int x)
|
|
{
|
|
switch (x) {
|
|
case LD_LK_IV:
|
|
return "iv";
|
|
case LD_LK_UN:
|
|
return "un";
|
|
case LD_LK_NL:
|
|
return "nl";
|
|
case LD_LK_SH:
|
|
return "sh";
|
|
case LD_LK_EX:
|
|
return "ex";
|
|
default:
|
|
return ".";
|
|
};
|
|
}
|
|
|
|
#ifdef LOCKDDLM_SUPPORT
|
|
|
|
int lm_init_vg_dlm(char *ls_name, char *vg_name, uint32_t flags, char *vg_args);
|
|
int lm_prepare_lockspace_dlm(struct lockspace *ls);
|
|
int lm_add_lockspace_dlm(struct lockspace *ls, int adopt);
|
|
int lm_rem_lockspace_dlm(struct lockspace *ls, int free_vg);
|
|
int lm_lock_dlm(struct lockspace *ls, struct resource *r, int ld_mode,
|
|
struct val_blk *vb_out, int adopt);
|
|
int lm_convert_dlm(struct lockspace *ls, struct resource *r,
|
|
int ld_mode, uint32_t r_version);
|
|
int lm_unlock_dlm(struct lockspace *ls, struct resource *r,
|
|
uint32_t r_version, uint32_t lmu_flags);
|
|
int lm_rem_resource_dlm(struct lockspace *ls, struct resource *r);
|
|
int lm_get_lockspaces_dlm(struct list_head *ls_rejoin);
|
|
int lm_data_size_dlm(void);
|
|
int lm_is_running_dlm(void);
|
|
int lm_hosts_dlm(struct lockspace *ls, int notify);
|
|
|
|
static inline int lm_support_dlm(void)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
#else
|
|
|
|
static inline int lm_init_vg_dlm(char *ls_name, char *vg_name, uint32_t flags, char *vg_args)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_prepare_lockspace_dlm(struct lockspace *ls)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_add_lockspace_dlm(struct lockspace *ls, int adopt)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_rem_lockspace_dlm(struct lockspace *ls, int free_vg)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_lock_dlm(struct lockspace *ls, struct resource *r, int ld_mode,
|
|
struct val_blk *vb_out, int adopt)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_convert_dlm(struct lockspace *ls, struct resource *r,
|
|
int ld_mode, uint32_t r_version)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_unlock_dlm(struct lockspace *ls, struct resource *r,
|
|
uint32_t r_version, uint32_t lmu_flags)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_rem_resource_dlm(struct lockspace *ls, struct resource *r)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_get_lockspaces_dlm(struct list_head *ls_rejoin)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_data_size_dlm(void)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_is_running_dlm(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static inline int lm_support_dlm(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static inline int lm_hosts_dlm(struct lockspace *ls, int notify)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
#endif /* dlm support */
|
|
|
|
#ifdef LOCKDSANLOCK_SUPPORT
|
|
|
|
int lm_init_vg_sanlock(char *ls_name, char *vg_name, uint32_t flags, char *vg_args);
|
|
int lm_init_lv_sanlock(char *ls_name, char *vg_name, char *lv_name, char *vg_args, char *lv_args, uint64_t free_offset);
|
|
int lm_free_lv_sanlock(struct lockspace *ls, struct resource *r);
|
|
int lm_rename_vg_sanlock(char *ls_name, char *vg_name, uint32_t flags, char *vg_args);
|
|
int lm_prepare_lockspace_sanlock(struct lockspace *ls);
|
|
int lm_add_lockspace_sanlock(struct lockspace *ls, int adopt);
|
|
int lm_rem_lockspace_sanlock(struct lockspace *ls, int free_vg);
|
|
int lm_lock_sanlock(struct lockspace *ls, struct resource *r, int ld_mode,
|
|
struct val_blk *vb_out, int *retry, int adopt);
|
|
int lm_convert_sanlock(struct lockspace *ls, struct resource *r,
|
|
int ld_mode, uint32_t r_version);
|
|
int lm_unlock_sanlock(struct lockspace *ls, struct resource *r,
|
|
uint32_t r_version, uint32_t lmu_flags);
|
|
int lm_able_gl_sanlock(struct lockspace *ls, int enable);
|
|
int lm_ex_disable_gl_sanlock(struct lockspace *ls);
|
|
int lm_hosts_sanlock(struct lockspace *ls, int notify);
|
|
int lm_rem_resource_sanlock(struct lockspace *ls, struct resource *r);
|
|
int lm_gl_is_enabled(struct lockspace *ls);
|
|
int lm_get_lockspaces_sanlock(struct list_head *ls_rejoin);
|
|
int lm_data_size_sanlock(void);
|
|
int lm_is_running_sanlock(void);
|
|
int lm_find_free_lock_sanlock(struct lockspace *ls, uint64_t *free_offset);
|
|
|
|
static inline int lm_support_sanlock(void)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
#else
|
|
|
|
static inline int lm_init_vg_sanlock(char *ls_name, char *vg_name, uint32_t flags, char *vg_args)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_init_lv_sanlock(char *ls_name, char *vg_name, char *lv_name, char *vg_args, char *lv_args, int sector_size, int align_size, uint64_t free_offset)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_free_lv_sanlock(struct lockspace *ls, struct resource *r)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_rename_vg_sanlock(char *ls_name, char *vg_name, uint32_t flags, char *vg_args)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_prepare_lockspace_sanlock(struct lockspace *ls)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_add_lockspace_sanlock(struct lockspace *ls, int adopt)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_rem_lockspace_sanlock(struct lockspace *ls, int free_vg)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_lock_sanlock(struct lockspace *ls, struct resource *r, int ld_mode,
|
|
struct val_blk *vb_out, int *retry, int adopt)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_convert_sanlock(struct lockspace *ls, struct resource *r,
|
|
int ld_mode, uint32_t r_version)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_unlock_sanlock(struct lockspace *ls, struct resource *r,
|
|
uint32_t r_version, uint32_t lmu_flags)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_able_gl_sanlock(struct lockspace *ls, int enable)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_ex_disable_gl_sanlock(struct lockspace *ls)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_hosts_sanlock(struct lockspace *ls, int notify)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_rem_resource_sanlock(struct lockspace *ls, struct resource *r)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_gl_is_enabled(struct lockspace *ls)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_get_lockspaces_sanlock(struct list_head *ls_rejoin)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_data_size_sanlock(void)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_is_running_sanlock(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static inline int lm_find_free_lock_sanlock(struct lockspace *ls, uint64_t *free_offset, int *sector_size, int *align_size)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
static inline int lm_support_sanlock(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
#endif /* sanlock support */
|
|
|
|
#endif /* _LVM_LVMLOCKD_INTERNAL_H */
|