mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-18 10:04:20 +03:00
activation: add support for flagging an LV to skip udev scanning during activation
A common scenario is during new LV creation when we need to wipe the newly created LV and avoid any udev scanning before this stage otherwise it could cause the device (the LV) to be claimed by some other subsystem for which there were stale metadata within LV data. This patch adds possibility to mark the LV we're just about to wipe with a flag that gets passed to udev via DM_COOKIE as a subsystem specific flag - DM_SUBSYSTEM_UDEV_FLAG0 (in this case the subsystem is "LVM") so LVM udev rules will take care of handling that.
This commit is contained in:
parent
273373bf53
commit
ce7489ed22
@ -1,5 +1,6 @@
|
|||||||
Version 2.02.104
|
Version 2.02.104
|
||||||
===================================
|
===================================
|
||||||
|
Add support for flagging an LV to skip udev scanning during activation.
|
||||||
Improve message when unable to change discards setting on active thin pool.
|
Improve message when unable to change discards setting on active thin pool.
|
||||||
Run full scan before vgrename operation to avoid any cache name collision.
|
Run full scan before vgrename operation to avoid any cache name collision.
|
||||||
Fix lvconvert when converting to a thin pool and thin LV at once.
|
Fix lvconvert when converting to a thin pool and thin LV at once.
|
||||||
|
@ -401,7 +401,7 @@ static int do_activate_lv(char *resource, unsigned char command, unsigned char l
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Now activate it */
|
/* Now activate it */
|
||||||
if (!lv_activate(cmd, resource, exclusive, NULL))
|
if (!lv_activate(cmd, resource, exclusive, 0, NULL))
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -337,12 +337,13 @@ int lv_activation_filter(struct cmd_context *cmd, const char *lvid_s,
|
|||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive, struct logical_volume *lv)
|
int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive, int noscan,
|
||||||
|
struct logical_volume *lv)
|
||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
int lv_activate_with_filter(struct cmd_context *cmd, const char *lvid_s, int exclusive,
|
int lv_activate_with_filter(struct cmd_context *cmd, const char *lvid_s, int exclusive,
|
||||||
struct logical_volume *lv)
|
int noscan, struct logical_volume *lv)
|
||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@ -2027,9 +2028,10 @@ static int _lv_activate(struct cmd_context *cmd, const char *lvid_s,
|
|||||||
if (filter)
|
if (filter)
|
||||||
laopts->read_only = _passes_readonly_filter(cmd, lv);
|
laopts->read_only = _passes_readonly_filter(cmd, lv);
|
||||||
|
|
||||||
log_debug_activation("Activating %s/%s%s%s.", lv->vg->name, lv->name,
|
log_debug_activation("Activating %s/%s%s%s%s.", lv->vg->name, lv->name,
|
||||||
laopts->exclusive ? " exclusively" : "",
|
laopts->exclusive ? " exclusively" : "",
|
||||||
laopts->read_only ? " read-only" : "");
|
laopts->read_only ? " read-only" : "",
|
||||||
|
laopts->noscan ? " noscan" : "");
|
||||||
|
|
||||||
if (!lv_info(cmd, lv, 0, &info, 0, 0))
|
if (!lv_info(cmd, lv, 0, &info, 0, 0))
|
||||||
goto_out;
|
goto_out;
|
||||||
@ -2066,9 +2068,10 @@ out:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Activate LV */
|
/* Activate LV */
|
||||||
int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive, struct logical_volume *lv)
|
int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive,
|
||||||
|
int noscan, struct logical_volume *lv)
|
||||||
{
|
{
|
||||||
struct lv_activate_opts laopts = { .exclusive = exclusive };
|
struct lv_activate_opts laopts = { .exclusive = exclusive, .noscan = noscan };
|
||||||
|
|
||||||
if (!_lv_activate(cmd, lvid_s, &laopts, 0, lv))
|
if (!_lv_activate(cmd, lvid_s, &laopts, 0, lv))
|
||||||
return_0;
|
return_0;
|
||||||
@ -2077,9 +2080,10 @@ int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive, stru
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Activate LV only if it passes filter */
|
/* Activate LV only if it passes filter */
|
||||||
int lv_activate_with_filter(struct cmd_context *cmd, const char *lvid_s, int exclusive, struct logical_volume *lv)
|
int lv_activate_with_filter(struct cmd_context *cmd, const char *lvid_s, int exclusive,
|
||||||
|
int noscan, struct logical_volume *lv)
|
||||||
{
|
{
|
||||||
struct lv_activate_opts laopts = { .exclusive = exclusive };
|
struct lv_activate_opts laopts = { .exclusive = exclusive, .noscan = noscan };
|
||||||
|
|
||||||
if (!_lv_activate(cmd, lvid_s, &laopts, 1, lv))
|
if (!_lv_activate(cmd, lvid_s, &laopts, 1, lv))
|
||||||
return_0;
|
return_0;
|
||||||
|
@ -38,6 +38,7 @@ struct lv_activate_opts {
|
|||||||
int skip_in_use;
|
int skip_in_use;
|
||||||
unsigned revert;
|
unsigned revert;
|
||||||
unsigned read_only;
|
unsigned read_only;
|
||||||
|
unsigned noscan;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* target attribute flags */
|
/* target attribute flags */
|
||||||
@ -80,9 +81,9 @@ int lv_suspend_if_active(struct cmd_context *cmd, const char *lvid_s, unsigned o
|
|||||||
int lv_resume(struct cmd_context *cmd, const char *lvid_s, unsigned origin_only, struct logical_volume *lv);
|
int lv_resume(struct cmd_context *cmd, const char *lvid_s, unsigned origin_only, struct logical_volume *lv);
|
||||||
int lv_resume_if_active(struct cmd_context *cmd, const char *lvid_s,
|
int lv_resume_if_active(struct cmd_context *cmd, const char *lvid_s,
|
||||||
unsigned origin_only, unsigned exclusive, unsigned revert, struct logical_volume *lv);
|
unsigned origin_only, unsigned exclusive, unsigned revert, struct logical_volume *lv);
|
||||||
int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive, struct logical_volume *lv);
|
int lv_activate(struct cmd_context *cmd, const char *lvid_s, int exclusive, int noscan, struct logical_volume *lv);
|
||||||
int lv_activate_with_filter(struct cmd_context *cmd, const char *lvid_s,
|
int lv_activate_with_filter(struct cmd_context *cmd, const char *lvid_s,
|
||||||
int exclusive, struct logical_volume *lv);
|
int exclusive, int noscan, struct logical_volume *lv);
|
||||||
int lv_deactivate(struct cmd_context *cmd, const char *lvid_s, struct logical_volume *lv);
|
int lv_deactivate(struct cmd_context *cmd, const char *lvid_s, struct logical_volume *lv);
|
||||||
|
|
||||||
int lv_mknodes(struct cmd_context *cmd, const struct logical_volume *lv);
|
int lv_mknodes(struct cmd_context *cmd, const struct logical_volume *lv);
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
|
|
||||||
#define MAX_TARGET_PARAMSIZE 50000
|
#define MAX_TARGET_PARAMSIZE 50000
|
||||||
|
#define LVM_UDEV_NOSCAN_FLAG DM_SUBSYSTEM_UDEV_FLAG0
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
PRELOAD,
|
PRELOAD,
|
||||||
@ -1399,7 +1400,7 @@ static int _check_udev_fallback(struct cmd_context *cmd)
|
|||||||
#endif /* UDEV_SYNC_SUPPORT */
|
#endif /* UDEV_SYNC_SUPPORT */
|
||||||
|
|
||||||
static uint16_t _get_udev_flags(struct dev_manager *dm, struct logical_volume *lv,
|
static uint16_t _get_udev_flags(struct dev_manager *dm, struct logical_volume *lv,
|
||||||
const char *layer)
|
const char *layer, uint16_t flags)
|
||||||
{
|
{
|
||||||
uint16_t udev_flags = 0;
|
uint16_t udev_flags = 0;
|
||||||
|
|
||||||
@ -1447,6 +1448,11 @@ static uint16_t _get_udev_flags(struct dev_manager *dm, struct logical_volume *l
|
|||||||
udev_flags |= DM_UDEV_DISABLE_DM_RULES_FLAG |
|
udev_flags |= DM_UDEV_DISABLE_DM_RULES_FLAG |
|
||||||
DM_UDEV_DISABLE_SUBSYSTEM_RULES_FLAG;
|
DM_UDEV_DISABLE_SUBSYSTEM_RULES_FLAG;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Firmly set requested flags.
|
||||||
|
*/
|
||||||
|
udev_flags |= flags;
|
||||||
|
|
||||||
return udev_flags;
|
return udev_flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1493,7 +1499,7 @@ static int _add_dev_to_dtree(struct dev_manager *dm, struct dm_tree *dtree,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (info.exists && !dm_tree_add_dev_with_udev_flags(dtree, info.major, info.minor,
|
if (info.exists && !dm_tree_add_dev_with_udev_flags(dtree, info.major, info.minor,
|
||||||
_get_udev_flags(dm, lv, layer))) {
|
_get_udev_flags(dm, lv, layer, 0))) {
|
||||||
log_error("Failed to add device (%" PRIu32 ":%" PRIu32") to dtree",
|
log_error("Failed to add device (%" PRIu32 ":%" PRIu32") to dtree",
|
||||||
info.major, info.minor);
|
info.major, info.minor);
|
||||||
return 0;
|
return 0;
|
||||||
@ -2334,7 +2340,7 @@ static int _set_udev_flags_for_children(struct dev_manager *dm,
|
|||||||
}
|
}
|
||||||
|
|
||||||
dm_tree_node_set_udev_flags(child,
|
dm_tree_node_set_udev_flags(child,
|
||||||
_get_udev_flags(dm, lvl->lv, NULL));
|
_get_udev_flags(dm, lvl->lv, NULL, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
@ -2411,7 +2417,8 @@ static int _add_new_lv_to_dtree(struct dev_manager *dm, struct dm_tree *dtree,
|
|||||||
read_only_lv(lv, laopts),
|
read_only_lv(lv, laopts),
|
||||||
((lv->vg->status & PRECOMMITTED) | laopts->revert) ? 1 : 0,
|
((lv->vg->status & PRECOMMITTED) | laopts->revert) ? 1 : 0,
|
||||||
lvlayer,
|
lvlayer,
|
||||||
_get_udev_flags(dm, lv, layer))))
|
_get_udev_flags(dm, lv, layer,
|
||||||
|
laopts->noscan ? LVM_UDEV_NOSCAN_FLAG : 0))))
|
||||||
return_0;
|
return_0;
|
||||||
|
|
||||||
/* Store existing name so we can do rename later */
|
/* Store existing name so we can do rename later */
|
||||||
|
@ -305,7 +305,7 @@ static int _file_lock_resource(struct cmd_context *cmd, const char *resource,
|
|||||||
break;
|
break;
|
||||||
case LCK_READ:
|
case LCK_READ:
|
||||||
log_very_verbose("Locking LV %s (R)", resource);
|
log_very_verbose("Locking LV %s (R)", resource);
|
||||||
if (!lv_activate_with_filter(cmd, resource, 0, lv_ondisk(lv)))
|
if (!lv_activate_with_filter(cmd, resource, 0, lv->status & LV_NOSCAN ? 1 : 0, lv_ondisk(lv)))
|
||||||
return 0;
|
return 0;
|
||||||
break;
|
break;
|
||||||
case LCK_PREAD:
|
case LCK_PREAD:
|
||||||
@ -318,7 +318,7 @@ static int _file_lock_resource(struct cmd_context *cmd, const char *resource,
|
|||||||
break;
|
break;
|
||||||
case LCK_EXCL:
|
case LCK_EXCL:
|
||||||
log_very_verbose("Locking LV %s (EX)", resource);
|
log_very_verbose("Locking LV %s (EX)", resource);
|
||||||
if (!lv_activate_with_filter(cmd, resource, 1, lv_ondisk(lv)))
|
if (!lv_activate_with_filter(cmd, resource, 1, lv->status & LV_NOSCAN ? 1 : 0, lv_ondisk(lv)))
|
||||||
return 0;
|
return 0;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -48,11 +48,11 @@ static int _no_lock_resource(struct cmd_context *cmd, const char *resource,
|
|||||||
case LCK_UNLOCK:
|
case LCK_UNLOCK:
|
||||||
return lv_resume_if_active(cmd, resource, (flags & LCK_ORIGIN_ONLY) ? 1: 0, 0, (flags & LCK_REVERT) ? 1 : 0, lv_ondisk(lv));
|
return lv_resume_if_active(cmd, resource, (flags & LCK_ORIGIN_ONLY) ? 1: 0, 0, (flags & LCK_REVERT) ? 1 : 0, lv_ondisk(lv));
|
||||||
case LCK_READ:
|
case LCK_READ:
|
||||||
return lv_activate_with_filter(cmd, resource, 0, lv_ondisk(lv));
|
return lv_activate_with_filter(cmd, resource, 0, lv->status & LV_NOSCAN ? 1 : 0, lv_ondisk(lv));
|
||||||
case LCK_WRITE:
|
case LCK_WRITE:
|
||||||
return lv_suspend_if_active(cmd, resource, (flags & LCK_ORIGIN_ONLY) ? 1 : 0, 0, lv_ondisk(lv), lv);
|
return lv_suspend_if_active(cmd, resource, (flags & LCK_ORIGIN_ONLY) ? 1 : 0, 0, lv_ondisk(lv), lv);
|
||||||
case LCK_EXCL:
|
case LCK_EXCL:
|
||||||
return lv_activate_with_filter(cmd, resource, 1, lv_ondisk(lv));
|
return lv_activate_with_filter(cmd, resource, 1, lv->status & LV_NOSCAN ? 1 : 0, lv_ondisk(lv));
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -5421,6 +5421,8 @@ int set_lv(struct cmd_context *cmd, struct logical_volume *lv,
|
|||||||
if (!dev_close_immediate(dev))
|
if (!dev_close_immediate(dev))
|
||||||
stack;
|
stack;
|
||||||
|
|
||||||
|
lv->status &= ~LV_NOSCAN;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5977,6 +5979,10 @@ static struct logical_volume *_lv_create_an_lv(struct volume_group *vg,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Do not scan this LV until properly zeroed. */
|
||||||
|
if (lp->zero)
|
||||||
|
lv->status |= LV_NOSCAN;
|
||||||
|
|
||||||
if (lv_is_thin_pool(lv)) {
|
if (lv_is_thin_pool(lv)) {
|
||||||
if (is_change_activating(lp->activate)) {
|
if (is_change_activating(lp->activate)) {
|
||||||
if (vg_is_clustered(lv->vg)) {
|
if (vg_is_clustered(lv->vg)) {
|
||||||
|
@ -102,6 +102,7 @@
|
|||||||
#define LV_WRITEMOSTLY UINT64_C(0x0000020000000000) /* LV (RAID1) */
|
#define LV_WRITEMOSTLY UINT64_C(0x0000020000000000) /* LV (RAID1) */
|
||||||
|
|
||||||
#define LV_ACTIVATION_SKIP UINT64_C(0x0000040000000000) /* LV */
|
#define LV_ACTIVATION_SKIP UINT64_C(0x0000040000000000) /* LV */
|
||||||
|
#define LV_NOSCAN UINT64_C(0x0000080000000000) /* LV */
|
||||||
|
|
||||||
/* Format features flags */
|
/* Format features flags */
|
||||||
#define FMT_SEGMENTS 0x00000001U /* Arbitrary segment params? */
|
#define FMT_SEGMENTS 0x00000001U /* Arbitrary segment params? */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user