mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-07 21:18:59 +03:00
dmsetup: Support remove --deferred.
This patch adds a new flag --deferred to dmsetup remove. If this flag is specified and the device is open, it is scheduled to be deleted on close. struct dm_info is extended. The existing dm_task_get_info() is converted into a wrapper around the new version dm_task_get_info_with_deferred_remove() so existing binaries can still use the old smaller structure. Recompiled code will pick up the new larger structure. From: Mikulas Patocka <mpatocka@redhat.com>
This commit is contained in:
parent
ec41bd1920
commit
42e07d2bce
@ -1,5 +1,6 @@
|
|||||||
Version 1.02.89 -
|
Version 1.02.89 -
|
||||||
=================================
|
=================================
|
||||||
|
Support --deferred with dmsetup remove to defer removal of open devices.
|
||||||
Update dm-ioctl.h to include DM_DEFERRED_REMOVE flag.
|
Update dm-ioctl.h to include DM_DEFERRED_REMOVE flag.
|
||||||
Add support for selection to match string list subset, recognize { } operator.
|
Add support for selection to match string list subset, recognize { } operator.
|
||||||
Fix string list selection with '[value]' to not match list that's superset.
|
Fix string list selection with '[value]' to not match list that's superset.
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
dm_log
|
dm_log
|
||||||
dm_log_with_errno
|
dm_log_with_errno
|
||||||
|
dm_task_get_info
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
#include "libdm-targets.h"
|
#include "libdm-targets.h"
|
||||||
#include "libdm-common.h"
|
#include "libdm-common.h"
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
#include <sys/ioctl.h>
|
#include <sys/ioctl.h>
|
||||||
@ -654,6 +655,7 @@ int dm_task_get_info(struct dm_task *dmt, struct dm_info *info)
|
|||||||
info->live_table = dmt->dmi.v4->flags & DM_ACTIVE_PRESENT_FLAG ? 1 : 0;
|
info->live_table = dmt->dmi.v4->flags & DM_ACTIVE_PRESENT_FLAG ? 1 : 0;
|
||||||
info->inactive_table = dmt->dmi.v4->flags & DM_INACTIVE_PRESENT_FLAG ?
|
info->inactive_table = dmt->dmi.v4->flags & DM_INACTIVE_PRESENT_FLAG ?
|
||||||
1 : 0;
|
1 : 0;
|
||||||
|
info->deferred_remove = dmt->dmi.v4->flags & DM_DEFERRED_REMOVE;
|
||||||
info->target_count = dmt->dmi.v4->target_count;
|
info->target_count = dmt->dmi.v4->target_count;
|
||||||
info->open_count = dmt->dmi.v4->open_count;
|
info->open_count = dmt->dmi.v4->open_count;
|
||||||
info->event_nr = dmt->dmi.v4->event_nr;
|
info->event_nr = dmt->dmi.v4->event_nr;
|
||||||
@ -862,6 +864,13 @@ int dm_task_retry_remove(struct dm_task *dmt)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int dm_task_deferred_remove(struct dm_task *dmt)
|
||||||
|
{
|
||||||
|
dmt->deferred_remove = 1;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
int dm_task_query_inactive_table(struct dm_task *dmt)
|
int dm_task_query_inactive_table(struct dm_task *dmt)
|
||||||
{
|
{
|
||||||
dmt->query_inactive_table = 1;
|
dmt->query_inactive_table = 1;
|
||||||
@ -1137,6 +1146,9 @@ static struct dm_ioctl *_flatten(struct dm_task *dmt, unsigned repeat_count)
|
|||||||
dmi->flags |= DM_READONLY_FLAG;
|
dmi->flags |= DM_READONLY_FLAG;
|
||||||
if (dmt->skip_lockfs)
|
if (dmt->skip_lockfs)
|
||||||
dmi->flags |= DM_SKIP_LOCKFS_FLAG;
|
dmi->flags |= DM_SKIP_LOCKFS_FLAG;
|
||||||
|
if (dmt->deferred_remove && (dmt->type == DM_DEVICE_REMOVE || dmt->type == DM_DEVICE_REMOVE_ALL))
|
||||||
|
dmi->flags |= DM_DEFERRED_REMOVE;
|
||||||
|
|
||||||
if (dmt->secure_data) {
|
if (dmt->secure_data) {
|
||||||
if (_dm_version_minor < 20)
|
if (_dm_version_minor < 20)
|
||||||
log_verbose("Secure data flag unsupported by kernel. "
|
log_verbose("Secure data flag unsupported by kernel. "
|
||||||
@ -1732,7 +1744,7 @@ static struct dm_ioctl *_do_dm_ioctl(struct dm_task *dmt, unsigned command,
|
|||||||
}
|
}
|
||||||
|
|
||||||
log_debug_activation("dm %s %s%s %s%s%s %s%.0d%s%.0d%s"
|
log_debug_activation("dm %s %s%s %s%s%s %s%.0d%s%.0d%s"
|
||||||
"%s%c%c%s%s%s%s%s%s %.0" PRIu64 " %s [%u] (*%u)",
|
"%s%c%c%s%s%s%s%s%s%s %.0" PRIu64 " %s [%u] (*%u)",
|
||||||
_cmd_data_v4[dmt->type].name,
|
_cmd_data_v4[dmt->type].name,
|
||||||
dmt->new_uuid ? "UUID " : "",
|
dmt->new_uuid ? "UUID " : "",
|
||||||
dmi->name, dmi->uuid, dmt->newname ? " " : "",
|
dmi->name, dmi->uuid, dmt->newname ? " " : "",
|
||||||
@ -1748,6 +1760,7 @@ static struct dm_ioctl *_do_dm_ioctl(struct dm_task *dmt, unsigned command,
|
|||||||
dmt->read_only ? "R" : "",
|
dmt->read_only ? "R" : "",
|
||||||
dmt->skip_lockfs ? "S " : "",
|
dmt->skip_lockfs ? "S " : "",
|
||||||
dmt->retry_remove ? "T " : "",
|
dmt->retry_remove ? "T " : "",
|
||||||
|
dmt->deferred_remove ? "D " : "",
|
||||||
dmt->secure_data ? "W " : "",
|
dmt->secure_data ? "W " : "",
|
||||||
dmt->query_inactive_table ? "I " : "",
|
dmt->query_inactive_table ? "I " : "",
|
||||||
dmt->enable_checks ? "C" : "",
|
dmt->enable_checks ? "C" : "",
|
||||||
@ -2006,3 +2019,32 @@ void dm_lib_exit(void)
|
|||||||
_version_ok = 1;
|
_version_ok = 1;
|
||||||
_version_checked = 0;
|
_version_checked = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This following code is here to retain ABI compatibility after adding
|
||||||
|
* the field deferred_remove to struct dm_info in version 1.02.89.
|
||||||
|
*
|
||||||
|
* Binaries linked against version 1.02.88 of libdevmapper or earlier
|
||||||
|
* will use this function that returns dm_info without the
|
||||||
|
* deferred_remove field.
|
||||||
|
*
|
||||||
|
* Binaries compiled against version 1.02.89 onwards will use
|
||||||
|
* the new function dm_task_get_info_with_deferred_remove due to the
|
||||||
|
* #define.
|
||||||
|
*
|
||||||
|
* N.B. Keep this function at the end of the file to make sure that
|
||||||
|
* no code in this file accidentally calls it.
|
||||||
|
*/
|
||||||
|
#undef dm_task_get_info
|
||||||
|
int dm_task_get_info(struct dm_task *dmt, struct dm_info *info);
|
||||||
|
int dm_task_get_info(struct dm_task *dmt, struct dm_info *info)
|
||||||
|
{
|
||||||
|
struct dm_info new_info;
|
||||||
|
|
||||||
|
if (!dm_task_get_info_with_deferred_remove(dmt, &new_info))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
memcpy(info, &new_info, offsetof(struct dm_info, deferred_remove));
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
@ -65,6 +65,7 @@ struct dm_task {
|
|||||||
int new_uuid;
|
int new_uuid;
|
||||||
int secure_data;
|
int secure_data;
|
||||||
int retry_remove;
|
int retry_remove;
|
||||||
|
int deferred_remove;
|
||||||
int enable_checks;
|
int enable_checks;
|
||||||
int expected_errno;
|
int expected_errno;
|
||||||
|
|
||||||
|
@ -149,6 +149,8 @@ struct dm_info {
|
|||||||
int read_only; /* 0:read-write; 1:read-only */
|
int read_only; /* 0:read-write; 1:read-only */
|
||||||
|
|
||||||
int32_t target_count;
|
int32_t target_count;
|
||||||
|
|
||||||
|
int deferred_remove;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct dm_deps {
|
struct dm_deps {
|
||||||
@ -172,6 +174,8 @@ struct dm_versions {
|
|||||||
|
|
||||||
int dm_get_library_version(char *version, size_t size);
|
int dm_get_library_version(char *version, size_t size);
|
||||||
int dm_task_get_driver_version(struct dm_task *dmt, char *version, size_t size);
|
int dm_task_get_driver_version(struct dm_task *dmt, char *version, size_t size);
|
||||||
|
|
||||||
|
#define dm_task_get_info dm_task_get_info_with_deferred_remove
|
||||||
int dm_task_get_info(struct dm_task *dmt, struct dm_info *dmi);
|
int dm_task_get_info(struct dm_task *dmt, struct dm_info *dmi);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -222,6 +226,7 @@ int dm_task_query_inactive_table(struct dm_task *dmt);
|
|||||||
int dm_task_suppress_identical_reload(struct dm_task *dmt);
|
int dm_task_suppress_identical_reload(struct dm_task *dmt);
|
||||||
int dm_task_secure_data(struct dm_task *dmt);
|
int dm_task_secure_data(struct dm_task *dmt);
|
||||||
int dm_task_retry_remove(struct dm_task *dmt);
|
int dm_task_retry_remove(struct dm_task *dmt);
|
||||||
|
int dm_task_deferred_remove(struct dm_task *dmt);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Enable checks for common mistakes such as issuing ioctls in an unsafe order.
|
* Enable checks for common mistakes such as issuing ioctls in an unsafe order.
|
||||||
|
@ -81,10 +81,12 @@ dmsetup \(em low level logical volume management
|
|||||||
.B dmsetup remove
|
.B dmsetup remove
|
||||||
.RB [ \-f | \-\-force ]
|
.RB [ \-f | \-\-force ]
|
||||||
.RB [ \-\-retry ]
|
.RB [ \-\-retry ]
|
||||||
|
.RB [ \-\-deferred ]
|
||||||
.I device_name
|
.I device_name
|
||||||
.br
|
.br
|
||||||
.B dmsetup remove_all
|
.B dmsetup remove_all
|
||||||
.RB [ \-f | \-\-force ]
|
.RB [ \-f | \-\-force ]
|
||||||
|
.RB [ \-\-deferred ]
|
||||||
.br
|
.br
|
||||||
.B dmsetup rename
|
.B dmsetup rename
|
||||||
.I device_name new_name
|
.I device_name new_name
|
||||||
@ -436,34 +438,35 @@ reactivating it with proper mangling mode used (see also \fB\-\-manglename\fP).
|
|||||||
.B remove
|
.B remove
|
||||||
.RB [ \-f | \-\-force ]
|
.RB [ \-f | \-\-force ]
|
||||||
.RB [ \-\-retry ]
|
.RB [ \-\-retry ]
|
||||||
|
.RB [ \-\-deferred ]
|
||||||
.I device_name
|
.I device_name
|
||||||
.br
|
.br
|
||||||
Removes a device. It will no longer be visible to dmsetup.
|
Removes a device. It will no longer be visible to dmsetup. Open devices
|
||||||
Open devices cannot be removed except with older kernels
|
cannot be removed, but adding \fB\-\-force\fP will replace the table with one
|
||||||
that contain a version of device-mapper prior to 4.8.0.
|
that fails all I/O. \fB\-\-deferred\fP will enable deferred removal of open
|
||||||
In this case the device will be deleted when its open_count
|
devices - the device will be removed when the last user closes it. The deferred
|
||||||
drops to zero. From version 4.8.0 onwards, if a device can't
|
removal feature is supported since version 4.27.0 of the device-mapper
|
||||||
be removed because an uninterruptible process is waiting for
|
driver available in upstream kernel version 3.13. (Use \fBdmsetup version\fP
|
||||||
I/O to return from it, adding \fB\-\-force\fP will replace the table
|
to check this.) If an attempt to remove a device fails, perhaps because a process run
|
||||||
with one that fails all I/O, which might allow the
|
from a quick udev rule temporarily opened the device, the \fB\-\-retry\fP
|
||||||
process to be killed. If an attempt to remove a device fails,
|
option will cause the operation to be retried for a few seconds before failing.
|
||||||
perhaps because a process run from a quick udev rule
|
Do NOT combine
|
||||||
temporarily opened the device, the \fB\-\-retry\fP option will cause
|
\fB\-\-force\fP and \fB\-\-udevcookie\fP, as udev may start to process udev
|
||||||
the operation to be retried for a few seconds before failing.
|
rules in the middle of error target replacement and result in nondeterministic
|
||||||
Do NOT combine \fB\-\-force\fP and \fB\-\-udevcookie\fP,
|
result.
|
||||||
as udev may start to process udev rules in the middle of error target
|
|
||||||
replacement and result in nondeterministic result.
|
|
||||||
.br
|
.br
|
||||||
.HP
|
.HP
|
||||||
.B remove_all
|
.B remove_all
|
||||||
.RB [ \-f | \-\-force ]
|
.RB [ \-f | \-\-force ]
|
||||||
|
.RB [ \-\-deferred ]
|
||||||
.br
|
.br
|
||||||
Attempts to remove all device definitions i.e. reset the driver.
|
Attempts to remove all device definitions i.e. reset the driver. This also runs
|
||||||
Use with care! From version 4.8.0 onwards, if devices can't
|
\fBmknodes\fP afterwards. Use with care! Open devices cannot be removed, but
|
||||||
be removed because uninterruptible processes are waiting for
|
adding \fB\-\-force\fP will replace the table with one that fails all I/O.
|
||||||
I/O to return from them, adding \fB\-\-force\fP will replace the table
|
\fB\-\-deferred\fP will enable deferred removal of open devices - the device
|
||||||
with one that fails all I/O, which might allow the
|
will be removed when the last user closes it. The deferred removal feature is
|
||||||
process to be killed. This also runs \fBmknodes\fP afterwards.
|
supported since version 4.27.0 of the device-mapper driver available in
|
||||||
|
upstream kernel version 3.13.
|
||||||
.br
|
.br
|
||||||
.HP
|
.HP
|
||||||
.B rename
|
.B rename
|
||||||
|
@ -120,6 +120,7 @@ enum {
|
|||||||
ADD_NODE_ON_RESUME_ARG,
|
ADD_NODE_ON_RESUME_ARG,
|
||||||
CHECKS_ARG,
|
CHECKS_ARG,
|
||||||
COLS_ARG,
|
COLS_ARG,
|
||||||
|
DEFERRED_ARG,
|
||||||
SELECT_ARG,
|
SELECT_ARG,
|
||||||
EXEC_ARG,
|
EXEC_ARG,
|
||||||
FORCE_ARG,
|
FORCE_ARG,
|
||||||
@ -468,9 +469,10 @@ static void _display_info_long(struct dm_task *dmt, struct dm_info *info)
|
|||||||
|
|
||||||
printf("Name: %s\n", dm_task_get_name(dmt));
|
printf("Name: %s\n", dm_task_get_name(dmt));
|
||||||
|
|
||||||
printf("State: %s%s\n",
|
printf("State: %s%s%s\n",
|
||||||
info->suspended ? "SUSPENDED" : "ACTIVE",
|
info->suspended ? "SUSPENDED" : "ACTIVE",
|
||||||
info->read_only ? " (READ-ONLY)" : "");
|
info->read_only ? " (READ-ONLY)" : "",
|
||||||
|
info->deferred_remove ? " (DEFERRED REMOVE)" : "");
|
||||||
|
|
||||||
/* FIXME Old value is being printed when it's being changed. */
|
/* FIXME Old value is being printed when it's being changed. */
|
||||||
if (dm_task_get_read_ahead(dmt, &read_ahead))
|
if (dm_task_get_read_ahead(dmt, &read_ahead))
|
||||||
@ -1321,6 +1323,9 @@ static int _simple(int task, const char *name, uint32_t event_nr, int display)
|
|||||||
if (_switches[RETRY_ARG] && task == DM_DEVICE_REMOVE)
|
if (_switches[RETRY_ARG] && task == DM_DEVICE_REMOVE)
|
||||||
dm_task_retry_remove(dmt);
|
dm_task_retry_remove(dmt);
|
||||||
|
|
||||||
|
if (_switches[DEFERRED_ARG] && (task == DM_DEVICE_REMOVE || task == DM_DEVICE_REMOVE_ALL))
|
||||||
|
dm_task_deferred_remove(dmt);
|
||||||
|
|
||||||
r = dm_task_run(dmt);
|
r = dm_task_run(dmt);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
@ -3071,7 +3076,7 @@ static struct command _commands[] = {
|
|||||||
"\t [-u|uuid <uuid>] [{--addnodeonresume|--addnodeoncreate}]\n"
|
"\t [-u|uuid <uuid>] [{--addnodeonresume|--addnodeoncreate}]\n"
|
||||||
"\t [--notable | --table <table> | <table_file>]",
|
"\t [--notable | --table <table> | <table_file>]",
|
||||||
1, 2,0, _create},
|
1, 2,0, _create},
|
||||||
{"remove", "[-f|--force] <device>", 0, -1, 1, _remove},
|
{"remove", "[-f|--force] [--deferred] <device>", 0, -1, 1, _remove},
|
||||||
{"remove_all", "[-f|--force]", 0, 0, 0, _remove_all},
|
{"remove_all", "[-f|--force]", 0, 0, 0, _remove_all},
|
||||||
{"suspend", "[--noflush] <device>", 0, -1, 1, _suspend},
|
{"suspend", "[--noflush] <device>", 0, -1, 1, _suspend},
|
||||||
{"resume", "<device> [{--addnodeonresume|--addnodeoncreate}]", 0, -1, 1, _resume},
|
{"resume", "<device> [{--addnodeonresume|--addnodeoncreate}]", 0, -1, 1, _resume},
|
||||||
@ -3521,6 +3526,7 @@ static int _process_switches(int *argc, char ***argv, const char *dev_dir)
|
|||||||
{"readonly", 0, &ind, READ_ONLY},
|
{"readonly", 0, &ind, READ_ONLY},
|
||||||
{"checks", 0, &ind, CHECKS_ARG},
|
{"checks", 0, &ind, CHECKS_ARG},
|
||||||
{"columns", 0, &ind, COLS_ARG},
|
{"columns", 0, &ind, COLS_ARG},
|
||||||
|
{"deferred", 0, &ind, DEFERRED_ARG},
|
||||||
{"select", 1, &ind, SELECT_ARG},
|
{"select", 1, &ind, SELECT_ARG},
|
||||||
{"exec", 1, &ind, EXEC_ARG},
|
{"exec", 1, &ind, EXEC_ARG},
|
||||||
{"force", 0, &ind, FORCE_ARG},
|
{"force", 0, &ind, FORCE_ARG},
|
||||||
@ -3694,6 +3700,8 @@ static int _process_switches(int *argc, char ***argv, const char *dev_dir)
|
|||||||
/* FIXME Accept modes as per chmod */
|
/* FIXME Accept modes as per chmod */
|
||||||
_int_args[MODE_ARG] = (int) strtol(optarg, NULL, 8);
|
_int_args[MODE_ARG] = (int) strtol(optarg, NULL, 8);
|
||||||
}
|
}
|
||||||
|
if (ind == DEFERRED_ARG)
|
||||||
|
_switches[DEFERRED_ARG]++;
|
||||||
if (ind == EXEC_ARG) {
|
if (ind == EXEC_ARG) {
|
||||||
_switches[EXEC_ARG]++;
|
_switches[EXEC_ARG]++;
|
||||||
_command = optarg;
|
_command = optarg;
|
||||||
|
Loading…
Reference in New Issue
Block a user