mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-21 13:34:40 +03:00
Add libdevmapper functions to support synchronisation with udev.
This commit is contained in:
parent
9607eba5c2
commit
4d8c4ea7e6
@ -1,5 +1,6 @@
|
|||||||
Version 2.02.51 -
|
Version 2.02.51 -
|
||||||
================================
|
================================
|
||||||
|
Add libdevmapper functions to support synchronisation with udev.
|
||||||
Added configure --enable-udev_rules --enable-udev_sync.
|
Added configure --enable-udev_rules --enable-udev_sync.
|
||||||
Added configure --with-udev-prefix --with-udevdir.
|
Added configure --with-udev-prefix --with-udevdir.
|
||||||
Added udev dir to hold udev rules.
|
Added udev dir to hold udev rules.
|
||||||
|
@ -24,6 +24,7 @@ dm_task_get_uuid
|
|||||||
dm_task_get_read_ahead
|
dm_task_get_read_ahead
|
||||||
dm_task_set_ro
|
dm_task_set_ro
|
||||||
dm_task_set_newname
|
dm_task_set_newname
|
||||||
|
dm_task_set_cookie
|
||||||
dm_task_set_event_nr
|
dm_task_set_event_nr
|
||||||
dm_task_set_major
|
dm_task_set_major
|
||||||
dm_task_set_minor
|
dm_task_set_minor
|
||||||
@ -154,3 +155,8 @@ dm_list_last
|
|||||||
dm_list_prev
|
dm_list_prev
|
||||||
dm_list_next
|
dm_list_next
|
||||||
dm_list_size
|
dm_list_size
|
||||||
|
dm_udev_set_sync_support
|
||||||
|
dm_udev_get_sync_support
|
||||||
|
dm_udev_notify
|
||||||
|
dm_udev_wait
|
||||||
|
dm_udev_cleanup
|
||||||
|
@ -864,6 +864,13 @@ int dm_check_version(void)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int dm_cookie_supported(void)
|
||||||
|
{
|
||||||
|
return (dm_check_version() &&
|
||||||
|
_dm_version >= 4 &&
|
||||||
|
_dm_version_minor >= 15);
|
||||||
|
}
|
||||||
|
|
||||||
void *dm_get_next_target(struct dm_task *dmt, void *next,
|
void *dm_get_next_target(struct dm_task *dmt, void *next,
|
||||||
uint64_t *start, uint64_t *length,
|
uint64_t *start, uint64_t *length,
|
||||||
char **target_type, char **params)
|
char **target_type, char **params)
|
||||||
|
@ -164,6 +164,7 @@ int dm_task_set_major_minor(struct dm_task *dmt, int major, int minor, int allow
|
|||||||
int dm_task_set_uid(struct dm_task *dmt, uid_t uid);
|
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_gid(struct dm_task *dmt, gid_t gid);
|
||||||
int dm_task_set_mode(struct dm_task *dmt, mode_t mode);
|
int dm_task_set_mode(struct dm_task *dmt, mode_t mode);
|
||||||
|
int dm_task_set_cookie(struct dm_task *dmt, uint32_t *cookie);
|
||||||
int dm_task_set_event_nr(struct dm_task *dmt, uint32_t event_nr);
|
int dm_task_set_event_nr(struct dm_task *dmt, uint32_t event_nr);
|
||||||
int dm_task_set_geometry(struct dm_task *dmt, const char *cylinders, const char *heads, const char *sectors, const char *start);
|
int dm_task_set_geometry(struct dm_task *dmt, const char *cylinders, const char *heads, const char *sectors, const char *start);
|
||||||
int dm_task_set_message(struct dm_task *dmt, const char *message);
|
int dm_task_set_message(struct dm_task *dmt, const char *message);
|
||||||
@ -1010,4 +1011,15 @@ int dm_report_field_uint64(struct dm_report *rh, struct dm_report_field *field,
|
|||||||
void dm_report_field_set_value(struct dm_report_field *field, const void *value,
|
void dm_report_field_set_value(struct dm_report_field *field, const void *value,
|
||||||
const void *sortvalue);
|
const void *sortvalue);
|
||||||
|
|
||||||
|
int dm_cookie_supported(void);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Udev notification functions.
|
||||||
|
*/
|
||||||
|
void dm_udev_set_sync_support(int sync_with_udev);
|
||||||
|
int dm_udev_get_sync_support(void);
|
||||||
|
int dm_udev_notify(uint32_t cookie);
|
||||||
|
int dm_udev_wait(uint32_t cookie);
|
||||||
|
int dm_udev_cleanup(uint32_t cookie);
|
||||||
|
|
||||||
#endif /* LIB_DEVICE_MAPPER_H */
|
#endif /* LIB_DEVICE_MAPPER_H */
|
||||||
|
@ -24,6 +24,12 @@
|
|||||||
#include <sys/ioctl.h>
|
#include <sys/ioctl.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
|
|
||||||
|
#ifdef UDEV_SYNC_SUPPORT
|
||||||
|
# include <sys/types.h>
|
||||||
|
# include <sys/ipc.h>
|
||||||
|
# include <sys/sem.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef linux
|
#ifdef linux
|
||||||
# include <linux/fs.h>
|
# include <linux/fs.h>
|
||||||
#endif
|
#endif
|
||||||
@ -33,11 +39,16 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define DEV_DIR "/dev/"
|
#define DEV_DIR "/dev/"
|
||||||
|
#define COOKIE_MAGIC 0x0D4D
|
||||||
|
|
||||||
static char _dm_dir[PATH_MAX] = DEV_DIR DM_DIR;
|
static char _dm_dir[PATH_MAX] = DEV_DIR DM_DIR;
|
||||||
|
|
||||||
static int _verbose = 0;
|
static int _verbose = 0;
|
||||||
|
|
||||||
|
#ifdef UDEV_SYNC_SUPPORT
|
||||||
|
static int _sync_with_udev = 1;
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Library users can provide their own logging
|
* Library users can provide their own logging
|
||||||
* function.
|
* function.
|
||||||
@ -761,3 +772,281 @@ out:
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef UDEV_SYNC_SUPPORT
|
||||||
|
void dm_udev_set_sync_support(int sync_with_udev)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_get_sync_support(void)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_task_set_cookie(struct dm_task *dmt, uint32_t *cookie)
|
||||||
|
{
|
||||||
|
*cookie = 0;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_notify(uint32_t cookie)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_wait(uint32_t cookie)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_cleanup(uint32_t cookie)
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* UDEV_SYNC_SUPPORT */
|
||||||
|
|
||||||
|
void dm_udev_set_sync_support(int sync_with_udev)
|
||||||
|
{
|
||||||
|
_sync_with_udev = sync_with_udev;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_get_sync_support(void)
|
||||||
|
{
|
||||||
|
return _sync_with_udev;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _get_cookie_sem(uint32_t cookie, int *semid)
|
||||||
|
{
|
||||||
|
if ((*semid = semget((key_t) cookie, 1, 0)) >= 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
switch (errno) {
|
||||||
|
case ENOENT:
|
||||||
|
log_error("Could not find notification "
|
||||||
|
"semaphore identified by cookie "
|
||||||
|
"value %" PRIu32 " (0x%x)",
|
||||||
|
cookie, cookie);
|
||||||
|
break;
|
||||||
|
case EACCES:
|
||||||
|
log_error("No permission to access "
|
||||||
|
"notificaton semaphore identified "
|
||||||
|
"by cookie value %" PRIu32 " (0x%x)",
|
||||||
|
cookie, cookie);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* FIXME errno use missing */
|
||||||
|
log_error("Failed to access notification "
|
||||||
|
"semaphore identified by cookie "
|
||||||
|
"value %" PRIu32 " (0x%x)",
|
||||||
|
cookie, cookie);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _udev_notify_sem_inc(int semid)
|
||||||
|
{
|
||||||
|
struct sembuf sb = {0, 1, 0};
|
||||||
|
|
||||||
|
/* FIXME errno use missing */
|
||||||
|
return semop(semid, &sb, 1) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _udev_notify_sem_dec(int semid)
|
||||||
|
{
|
||||||
|
/* FIXME Think we should have IPC_NOWAIT here in case something went wrong and it's already 0 */
|
||||||
|
struct sembuf sb = {0, -1, 0};
|
||||||
|
|
||||||
|
/* FIXME errno use missing */
|
||||||
|
return semop(semid, &sb, 1) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _udev_notify_sem_destroy(int semid, uint32_t cookie)
|
||||||
|
{
|
||||||
|
/* FIXME errno use missing */
|
||||||
|
if (semctl(semid, 0, IPC_RMID, 0) < 0) {
|
||||||
|
log_error("Could not cleanup notification semaphore "
|
||||||
|
"identified by cookie value %" PRIu32 " (0x%x)",
|
||||||
|
cookie, cookie);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _udev_notify_sem_create(uint32_t *cookie, int *semid)
|
||||||
|
{
|
||||||
|
int fd;
|
||||||
|
int gen_semid;
|
||||||
|
uint16_t base_cookie;
|
||||||
|
uint32_t gen_cookie;
|
||||||
|
|
||||||
|
if ((fd = open("/dev/urandom", O_RDONLY)) < 0) {
|
||||||
|
log_error("Failed to open /dev/urandom "
|
||||||
|
"to create random cookie value");
|
||||||
|
*cookie = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Generate random cookie value. Be sure it is unique and non-zero. */
|
||||||
|
do {
|
||||||
|
/* FIXME Handle non-error returns from read(). Move _io() into libdm? */
|
||||||
|
if (read(fd, &base_cookie, sizeof(base_cookie)) != sizeof(base_cookie)) {
|
||||||
|
log_error("Failed to initialize notification cookie");
|
||||||
|
goto bad;
|
||||||
|
}
|
||||||
|
|
||||||
|
gen_cookie = COOKIE_MAGIC << 16 | base_cookie;
|
||||||
|
|
||||||
|
if (base_cookie && (gen_semid = semget((key_t) gen_cookie,
|
||||||
|
1, 0600 | IPC_CREAT | IPC_EXCL)) < 0) {
|
||||||
|
switch (errno) {
|
||||||
|
case EEXIST:
|
||||||
|
/* if the semaphore key exists, we
|
||||||
|
* simply generate another random one */
|
||||||
|
base_cookie = 0;
|
||||||
|
break;
|
||||||
|
case ENOMEM:
|
||||||
|
log_error("Not enough memory to create "
|
||||||
|
"notification semaphore");
|
||||||
|
goto bad;
|
||||||
|
case ENOSPC:
|
||||||
|
/* FIXME Suggest what to check & do */
|
||||||
|
log_error("Limit for the maximum number "
|
||||||
|
"of semaphores reached");
|
||||||
|
goto bad;
|
||||||
|
default:
|
||||||
|
/* FIXME Use errno */
|
||||||
|
log_error("Failed to create "
|
||||||
|
"notification semaphore");
|
||||||
|
goto bad;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while (!base_cookie);
|
||||||
|
|
||||||
|
if (semctl(gen_semid, 0, SETVAL, 1) < 0) {
|
||||||
|
/* FIXME Use errno and give gen_semid */
|
||||||
|
log_error("Failed to initialize notification semaphore");
|
||||||
|
/* We have to destroy just created semaphore
|
||||||
|
* so it won't stay in the system. */
|
||||||
|
_udev_notify_sem_destroy(gen_semid, gen_cookie);
|
||||||
|
goto bad;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (close(fd))
|
||||||
|
stack;
|
||||||
|
|
||||||
|
*semid = gen_semid;
|
||||||
|
*cookie = gen_cookie;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
bad:
|
||||||
|
if (close(fd))
|
||||||
|
stack;
|
||||||
|
|
||||||
|
*cookie = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_task_set_cookie(struct dm_task *dmt, uint32_t *cookie)
|
||||||
|
{
|
||||||
|
int semid;
|
||||||
|
|
||||||
|
if (!dm_udev_get_sync_support() || !dm_cookie_supported()) {
|
||||||
|
dmt->event_nr = *cookie = 0;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*cookie) {
|
||||||
|
if (!_get_cookie_sem(*cookie, &semid))
|
||||||
|
goto_bad;
|
||||||
|
} else if (!_udev_notify_sem_create(cookie, &semid))
|
||||||
|
goto_bad;
|
||||||
|
|
||||||
|
if (!_udev_notify_sem_inc(semid)) {
|
||||||
|
log_error("Could not set notification semaphore "
|
||||||
|
"identified by cookie value %" PRIu32 " (0x%x)",
|
||||||
|
*cookie, *cookie);
|
||||||
|
goto bad;
|
||||||
|
}
|
||||||
|
|
||||||
|
dmt->event_nr = *cookie;
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
bad:
|
||||||
|
dmt->event_nr = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_notify(uint32_t cookie)
|
||||||
|
{
|
||||||
|
int semid;
|
||||||
|
|
||||||
|
if (!cookie || !dm_udev_get_sync_support() || !dm_cookie_supported())
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (!_get_cookie_sem(cookie, &semid))
|
||||||
|
return_0;
|
||||||
|
|
||||||
|
if (!_udev_notify_sem_dec(semid)) {
|
||||||
|
log_error("Could not signal waiting process using notification "
|
||||||
|
"semaphore identified by cookie value %" PRIu32 " (0x%x)",
|
||||||
|
cookie, cookie);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_wait(uint32_t cookie)
|
||||||
|
{
|
||||||
|
int semid;
|
||||||
|
struct sembuf sb = {0, 0, 0};
|
||||||
|
|
||||||
|
if (!cookie || !dm_udev_get_sync_support() || !dm_cookie_supported())
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (!_get_cookie_sem(cookie, &semid))
|
||||||
|
return_0;
|
||||||
|
|
||||||
|
if (!_udev_notify_sem_dec(semid)) {
|
||||||
|
log_error("Failed to set a proper state for notification "
|
||||||
|
"semaphore identified by cookie value %" PRIu32 " (0x%x) "
|
||||||
|
"to initialize waiting for incoming notifications.",
|
||||||
|
cookie, cookie);
|
||||||
|
_udev_notify_sem_destroy(semid, cookie);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
repeat_wait:
|
||||||
|
if (semop(semid, &sb, 1) < 0) {
|
||||||
|
if (errno == EINTR)
|
||||||
|
goto repeat_wait;
|
||||||
|
/* FIXME missing errno use */
|
||||||
|
log_error("Could not set wait state for notification semaphore "
|
||||||
|
"identified by cookie value %" PRIu32 " (0x%x)",
|
||||||
|
cookie, cookie);
|
||||||
|
_udev_notify_sem_destroy(semid, cookie);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return _udev_notify_sem_destroy(semid, cookie);
|
||||||
|
}
|
||||||
|
|
||||||
|
int dm_udev_cleanup(uint32_t cookie)
|
||||||
|
{
|
||||||
|
int semid;
|
||||||
|
|
||||||
|
if (!cookie || !dm_udev_get_sync_support() || !dm_cookie_supported())
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (!_get_cookie_sem(cookie, &semid))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return _udev_notify_sem_destroy(semid, cookie);
|
||||||
|
}
|
||||||
|
#endif /* UDEV_SYNC_SUPPORT */
|
||||||
|
Loading…
Reference in New Issue
Block a user