*/
#include "tools.h"
+#include "format1.h"
+#include "format-text.h"
#include <sys/stat.h>
#include <signal.h>
int ret_max = ECMD_PROCESSED;
const char *vg_name;
- log_verbose("Using volume group(s) on command line.");
-
if (one_vgname) {
if (!str_list_add(cmd->mem, arg_vgnames,
dm_pool_strdup(cmd->mem, one_vgname))) {
int skip;
int notfound;
int process_all = 0;
+ int already_locked;
/*
* If no VG names or tags were supplied, then process all VGs.
continue;
}
+ already_locked = lvmcache_vgname_is_locked(vg_name);
+
vg = vg_read(cmd, vg_name, vg_uuid, read_flags, lockd_state);
if (_ignore_vg(vg, vg_name, arg_vgnames, read_flags, &skip, ¬found)) {
stack;
ret_max = ret;
}
- if (!vg_read_error(vg))
+ if (!vg_read_error(vg) && !already_locked)
unlock_vg(cmd, vg_name);
endvg:
release_vg(vg);
int ret_max = ECMD_PROCESSED;
int ret;
+ log_debug("Processing each VG");
+
/* Disable error in vg_read so we can print it from ignore_vg. */
cmd->vg_read_print_access_error = 0;
* . A VG name is specified which may refer to one
* of multiple VGs on the system with that name.
*/
- log_very_verbose("Get list of VGs on system");
+ log_debug("Get list of VGs on system");
if (!get_vgnameids(cmd, &vgnameids_on_system, NULL, 0)) {
ret_max = ECMD_FAILED;
const char *vgname_def;
unsigned dev_dir_found;
- log_verbose("Using logical volume(s) on command line.");
-
for (; opt < argc; opt++) {
lv_name = argv[opt];
dev_dir_found = 0;
int ret;
int skip;
int notfound;
+ int already_locked;
dm_list_iterate_items(vgnl, vgnameids_to_process) {
if (sigint_caught())
continue;
}
+ already_locked = lvmcache_vgname_is_locked(vg_name);
+
vg = vg_read(cmd, vg_name, vg_uuid, read_flags, lockd_state);
if (_ignore_vg(vg, vg_name, arg_vgnames, read_flags, &skip, ¬found)) {
stack;
if (ret > ret_max)
ret_max = ret;
- unlock_vg(cmd, vg_name);
+ if (!already_locked)
+ unlock_vg(cmd, vg_name);
endvg:
release_vg(vg);
if (!lockd_vg(cmd, vg_name, "un", 0, &lockd_state))
* . A VG name is specified which may refer to one
* of multiple VGs on the system with that name.
*/
- log_very_verbose("Get list of VGs on system");
+ log_debug("Get list of VGs on system");
if (!get_vgnameids(cmd, &vgnameids_on_system, NULL, 0)) {
ret_max = ECMD_FAILED;
char *arg_name;
int ret_max = ECMD_PROCESSED;
- log_verbose("Using physical volume(s) on command line.");
-
for (; opt < argc; opt++) {
arg_name = argv[opt];
struct device_id_list *dil;
int r = ECMD_FAILED;
+ log_debug("Getting list of all devices");
+
lvmcache_seed_infos_from_lvmetad(cmd);
if (!(iter = dev_iter_create(cmd->full_filter, 1))) {
int ret_max = ECMD_PROCESSED;
int ret = 0;
+ log_debug("Processing devices that are not PVs");
+
/*
* Pretend that each device is a PV with dummy values.
* FIXME Formalise this extension or find an alternative.
int ret;
int skip;
int notfound;
+ int already_locked;
dm_list_iterate_items(vgnl, all_vgnameids) {
if (sigint_caught())
continue;
}
+ log_debug("Processing PVs in VG %s", vg_name);
+
+ already_locked = lvmcache_vgname_is_locked(vg_name);
+
vg = vg_read(cmd, vg_name, vg_uuid, read_flags, lockd_state);
if (_ignore_vg(vg, vg_name, NULL, read_flags, &skip, ¬found)) {
stack;
if (ret > ret_max)
ret_max = ret;
- if (!skip)
+ if (!skip && !already_locked)
unlock_vg(cmd, vg->name);
endvg:
release_vg(vg);
}
int process_each_pv(struct cmd_context *cmd,
- int argc, char **argv,
- const char *only_this_vgname,
- uint32_t read_flags,
+ int argc, char **argv, const char *only_this_vgname,
+ int all_is_set, uint32_t read_flags,
struct processing_handle *handle,
process_single_pv_fn_t process_single_pv)
{
struct device_id_list *dil;
int process_all_pvs;
int process_all_devices;
+ int orphans_locked;
int ret_max = ECMD_PROCESSED;
int ret;
+ log_debug("Processing each PV");
+
/*
* When processing a specific VG name, warn if it's inconsistent and
* print an error if it's not found. Otherwise we're processing all
return ret;
}
+ orphans_locked = lvmcache_vgname_is_locked(VG_ORPHANS);
+
process_all_pvs = dm_list_empty(&arg_pvnames) && dm_list_empty(&arg_tags);
- process_all_devices = process_all_pvs && (cmd->command->flags & ENABLE_ALL_DEVS) &&
- arg_count(cmd, all_ARG);
+ process_all_devices = process_all_pvs && (cmd->command->flags & ENABLE_ALL_DEVS) && all_is_set;
/* Needed for a current listing of the global VG namespace. */
if (!only_this_vgname && !lockd_gl(cmd, "sh", 0))
* This full scan would be done by _get_all_devices() if
* it were not done here first. It's called here first
* so that get_vgnameids() will look at any new devices.
+ * When orphans is already locked, these steps are done
+ * before process_each_pv is called.
*/
- if (!trust_cache()) {
- dev_cache_full_scan(cmd->full_filter);
+ if (!trust_cache() && !orphans_locked) {
+ log_debug("Scanning for available devices");
lvmcache_destroy(cmd, 1, 0);
+ dev_cache_full_scan(cmd->full_filter);
}
/*
if (ret > ret_max)
ret_max = ret;
+ /*
+ * If the orphans lock was held, there shouldn't be missed devices. If
+ * there were, we cannot clear the cache while holding the orphans lock
+ * anyway.
+ */
+ if (orphans_locked)
+ goto skip_missed;
+
/*
* Some PVs may have been missed by the first search if another command
* moved them at the same time. Repeat the search for only the
}
}
+skip_missed:
dm_list_iterate_items(dil, &arg_devices) {
log_error("Failed to find physical volume \"%s\".", dev_name(dil->dev));
ret_max = ECMD_FAILED;
return ECMD_PROCESSED;
}
+
+void pvcreate_each_params_set_defaults(struct pvcreate_each_params *pp)
+{
+ memset(pp, 0, sizeof(*pp));
+
+ pp->zero = 1;
+ pp->size = 0;
+ pp->data_alignment = UINT64_C(0);
+ pp->data_alignment_offset = UINT64_C(0);
+ pp->pvmetadatacopies = DEFAULT_PVMETADATACOPIES;
+ pp->pvmetadatasize = DEFAULT_PVMETADATASIZE;
+ pp->labelsector = DEFAULT_LABELSECTOR;
+ pp->force = PROMPT;
+ pp->yes = 0;
+ pp->metadataignore = DEFAULT_PVMETADATAIGNORE;
+ pp->restorefile = NULL;
+ pp->uuid_str = NULL;
+ pp->ba_start = 0;
+ pp->ba_size = 0;
+ pp->pe_start = PV_PE_START_CALC;
+ pp->extent_count = 0;
+ pp->extent_size = 0;
+
+ dm_list_init(&pp->prompts);
+ dm_list_init(&pp->arg_devices);
+ dm_list_init(&pp->arg_process);
+ dm_list_init(&pp->arg_confirm);
+ dm_list_init(&pp->arg_create);
+ dm_list_init(&pp->arg_fail);
+ dm_list_init(&pp->pvs);
+}
+
+int pvcreate_each_params_from_args(struct cmd_context *cmd, struct pvcreate_each_params *pp)
+{
+ pp->yes = arg_count(cmd, yes_ARG);
+ pp->force = (force_t) arg_count(cmd, force_ARG);
+
+ if (arg_int_value(cmd, labelsector_ARG, 0) >= LABEL_SCAN_SECTORS) {
+ log_error("labelsector must be less than %lu.",
+ LABEL_SCAN_SECTORS);
+ return 0;
+ } else {
+ pp->labelsector = arg_int64_value(cmd, labelsector_ARG,
+ DEFAULT_LABELSECTOR);
+ }
+
+ if (!(cmd->fmt->features & FMT_MDAS) &&
+ (arg_count(cmd, pvmetadatacopies_ARG) ||
+ arg_count(cmd, metadatasize_ARG) ||
+ arg_count(cmd, dataalignment_ARG) ||
+ arg_count(cmd, dataalignmentoffset_ARG))) {
+ log_error("Metadata and data alignment parameters only "
+ "apply to text format.");
+ return 0;
+ }
+
+ if (!(cmd->fmt->features & FMT_BAS) &&
+ arg_count(cmd, bootloaderareasize_ARG)) {
+ log_error("Bootloader area parameters only "
+ "apply to text format.");
+ return 0;
+ }
+
+ if (arg_count(cmd, metadataignore_ARG))
+ pp->metadataignore = arg_int_value(cmd, metadataignore_ARG,
+ DEFAULT_PVMETADATAIGNORE);
+ else
+ pp->metadataignore = find_config_tree_bool(cmd, metadata_pvmetadataignore_CFG, NULL);
+
+ if (arg_count(cmd, pvmetadatacopies_ARG) &&
+ !arg_int_value(cmd, pvmetadatacopies_ARG, -1) &&
+ pp->metadataignore) {
+ log_error("metadataignore only applies to metadatacopies > 0");
+ return 0;
+ }
+
+ pp->zero = arg_int_value(cmd, zero_ARG, 1);
+
+ if (arg_sign_value(cmd, dataalignment_ARG, SIGN_NONE) == SIGN_MINUS) {
+ log_error("Physical volume data alignment may not be negative.");
+ return 0;
+ }
+ pp->data_alignment = arg_uint64_value(cmd, dataalignment_ARG, UINT64_C(0));
+
+ if (pp->data_alignment > UINT32_MAX) {
+ log_error("Physical volume data alignment is too big.");
+ return 0;
+ }
+
+ if (arg_sign_value(cmd, dataalignmentoffset_ARG, SIGN_NONE) == SIGN_MINUS) {
+ log_error("Physical volume data alignment offset may not be negative");
+ return 0;
+ }
+ pp->data_alignment_offset = arg_uint64_value(cmd, dataalignmentoffset_ARG, UINT64_C(0));
+
+ if (pp->data_alignment_offset > UINT32_MAX) {
+ log_error("Physical volume data alignment offset is too big.");
+ return 0;
+ }
+
+ if ((pp->data_alignment + pp->data_alignment_offset) &&
+ (pp->pe_start != PV_PE_START_CALC)) {
+ if ((pp->data_alignment ? pp->pe_start % pp->data_alignment : pp->pe_start) != pp->data_alignment_offset) {
+ log_warn("WARNING: Ignoring data alignment %s"
+ " incompatible with restored pe_start value %s)",
+ display_size(cmd, pp->data_alignment + pp->data_alignment_offset),
+ display_size(cmd, pp->pe_start));
+ pp->data_alignment = 0;
+ pp->data_alignment_offset = 0;
+ }
+ }
+
+ if (arg_sign_value(cmd, metadatasize_ARG, SIGN_NONE) == SIGN_MINUS) {
+ log_error("Metadata size may not be negative.");
+ return 0;
+ }
+
+ if (arg_sign_value(cmd, bootloaderareasize_ARG, SIGN_NONE) == SIGN_MINUS) {
+ log_error("Bootloader area size may not be negative.");
+ return 0;
+ }
+
+ pp->pvmetadatasize = arg_uint64_value(cmd, metadatasize_ARG, UINT64_C(0));
+ if (!pp->pvmetadatasize)
+ pp->pvmetadatasize = find_config_tree_int(cmd, metadata_pvmetadatasize_CFG, NULL);
+
+ pp->pvmetadatacopies = arg_int_value(cmd, pvmetadatacopies_ARG, -1);
+ if (pp->pvmetadatacopies < 0)
+ pp->pvmetadatacopies = find_config_tree_int(cmd, metadata_pvmetadatacopies_CFG, NULL);
+
+ if (pp->pvmetadatacopies > 2) {
+ log_error("Metadatacopies may only be 0, 1 or 2");
+ return 0;
+ }
+
+ pp->ba_size = arg_uint64_value(cmd, bootloaderareasize_ARG, pp->ba_size);
+
+ return 1;
+}
+
+enum {
+ PROMPT_PVCREATE_PV_IN_VG = 1,
+};
+
+enum {
+ PROMPT_ANSWER_NO = 1,
+ PROMPT_ANSWER_YES = 2
+};
+
+/*
+ * When a prompt entry is created, save any strings or info
+ * in this struct that are needed for the prompt messages.
+ * The VG/PV structs are not be available when the prompt
+ * is run.
+ */
+struct pvcreate_prompt {
+ struct dm_list list;
+ uint32_t type;
+ const char *pv_name;
+ const char *vg_name;
+ struct device *dev;
+ int answer;
+ unsigned abort_command : 1;
+ unsigned vg_name_unknown : 1;
+};
+
+struct pvcreate_device {
+ struct dm_list list;
+ const char *name;
+ struct device *dev;
+ char pvid[ID_LEN + 1];
+ const char *vg_name;
+ int wiped;
+ unsigned is_not_pv : 1; /* device is not a PV */
+ unsigned is_orphan_pv : 1; /* device is an orphan PV */
+ unsigned is_vg_pv : 1; /* device is a PV used in a VG */
+ unsigned is_used_unknown_pv : 1; /* device is a PV used in an unknown VG */
+};
+
+static void _check_pvcreate_prompt(struct cmd_context *cmd,
+ struct pvcreate_each_params *pp,
+ struct pvcreate_prompt *prompt,
+ int ask)
+{
+ if (prompt->type == PROMPT_PVCREATE_PV_IN_VG) {
+ if (pp->force != DONT_PROMPT_OVERRIDE) {
+ prompt->answer = PROMPT_ANSWER_NO;
+
+ /* FIXME: use similar error messages for these cases */
+
+ if (prompt->vg_name_unknown) {
+ log_error("PV '%s' is marked as belonging to a VG but its metadata is missing.",
+ prompt->pv_name);
+ log_error("Can't initialize PV '%s' without -ff.", prompt->pv_name);
+ } else if (!strcmp(command_name(cmd), "pvcreate")) {
+ log_error("Can't initialize physical volume \"%s\" of volume group \"%s\" without -ff",
+ prompt->pv_name, prompt->vg_name);
+ } else {
+ log_error("Physical volume '%s' is already in volume group '%s'",
+ prompt->pv_name, prompt->vg_name);
+ log_error("Unable to add physical volume '%s' to volume group '%s'",
+ prompt->pv_name, prompt->vg_name);
+ }
+ } else if (pp->yes) {
+ prompt->answer = PROMPT_ANSWER_YES;
+ } else if (ask) {
+ const char *vgname = prompt->vg_name ? prompt->vg_name : "<unknown>";
+
+ if (yes_no_prompt("Really INITIALIZE physical volume \"%s\" of volume group \"%s\" [y/n]? ",
+ prompt->pv_name, vgname) == 'n') {
+ prompt->answer = PROMPT_ANSWER_NO;
+ log_error("%s: physical volume not initialized", prompt->pv_name);
+ } else {
+ prompt->answer = PROMPT_ANSWER_YES;
+ log_warn("WARNING: Forcing physical volume creation on %s of volume group \"%s\"",
+ prompt->pv_name, vgname);
+ }
+ }
+ }
+}
+
+static struct pvcreate_device *_pvcreate_list_find_dev(struct dm_list *devices, struct device *dev)
+{
+ struct pvcreate_device *pd;
+
+ dm_list_iterate_items(pd, devices) {
+ if (pd->dev == dev)
+ return pd;
+ }
+
+ return NULL;
+}
+
+static struct pvcreate_device *_pvcreate_list_find_name(struct dm_list *devices, const char *name)
+{
+ struct pvcreate_device *pd;
+
+ dm_list_iterate_items(pd, devices) {
+ if (!strcmp(pd->name, name))
+ return pd;
+ }
+
+ return NULL;
+}
+
+/*
+ * If this function decides that a arg_devices entry cannot be used, but the
+ * command might be able to continue without it, then it moves that entry from
+ * arg_devices to arg_fail.
+ *
+ * If this function decides that an arg_devices entry could be used (possibly
+ * requiring a prompt), then it moves the entry from arg_devices to arg_process.
+ *
+ * Any arg_devices entries that are not moved to arg_fail or arg_process were
+ * not found. The caller will decide if the command can continue if any
+ * arg_devices entries were not found, or if any were moved to arg_fail.
+ *
+ * This check does not need to look at PVs in foreign, shared or clustered VGs.
+ * If pvcreate/vgcreate/vgextend specifies a device in a
+ * foreign/shared/clustered VG, that VG will not be processed by this function,
+ * and the arg will be reported as not found.
+ */
+
+static int _pvcreate_check1_single(struct cmd_context *cmd,
+ struct volume_group *vg,
+ struct physical_volume *pv,
+ struct processing_handle *handle)
+{
+ struct pvcreate_each_params *pp = (struct pvcreate_each_params *) handle->custom_handle;
+ struct pvcreate_device *pd;
+ struct pvcreate_prompt *prompt;
+ struct device *dev;
+ int found = 0;
+
+ if (!pv->dev)
+ return 1;
+
+ /*
+ * Check if one of the command args in arg_devices
+ * matches this device.
+ *
+ * (Possible optimization: the first time this _single
+ * function is called, we could iterate through all
+ * arg_devices entries, do the name to dev lookup
+ * with dev_cache_get() and set the pd->dev fields.
+ * Subsequent _single calls would just compare devs
+ * and not do any dev_cache_get(). This would avoid
+ * repeating dev_cache_get() for arg_devices entries.)
+ */
+ dm_list_iterate_items(pd, &pp->arg_devices) {
+ dev = dev_cache_get(pd->name, cmd->full_filter);
+ if (dev != pv->dev)
+ continue;
+
+ pd->dev = pv->dev;
+ if (pv->dev->pvid[0])
+ strncpy(pd->pvid, pv->dev->pvid, ID_LEN);
+ found = 1;
+ break;
+ }
+
+ /*
+ * Check if the uuid specified for the new PV is used by another PV.
+ */
+ if (!found && pv->dev && pp->uuid_str && id_equal(&pv->id, &pp->id)) {
+ log_error("uuid %s already in use on \"%s\"", pp->uuid_str, pv_dev_name(pv));
+ pp->check_failed = 1;
+ return 0;
+ }
+
+ if (!found)
+ return 1;
+
+ log_debug("Checking device %s for pvcreate %.32s",
+ pv_dev_name(pv), pv->dev->pvid[0] ? pv->dev->pvid : "");
+
+ /*
+ * This test will fail if the device belongs to an MD array.
+ */
+ if (!dev_test_excl(pv->dev)) {
+ /* FIXME Detect whether device-mapper itself is still using it */
+ log_error("Can't open %s exclusively. Mounted filesystem?",
+ pv_dev_name(pv));
+ dm_list_move(&pp->arg_fail, &pd->list);
+ return 1;
+ }
+
+ /*
+ * What kind of device is this: an orphan PV, an uninitialized/unused
+ * device, a PV used in a VG.
+ */
+
+ if (vg && !is_orphan_vg(vg->name)) {
+ /* Device is a PV used in a VG. */
+ log_debug("Found pvcreate arg %s: pv is used in %s", pd->name, vg->name);
+ pd->is_vg_pv = 1;
+ pd->vg_name = dm_pool_strdup(cmd->mem, vg->name);
+
+ } else if (vg && is_orphan_vg(vg->name)) {
+ if (is_used_pv(pv)) {
+ /* Device is used in an unknown VG. */
+ log_debug("Found pvcreate arg %s: pv is used in unknown VG", pd->name);
+ pd->is_used_unknown_pv = 1;
+ } else {
+ /* Device is an orphan PV. */
+ log_debug("Found pvcreate arg %s: pv is orphan in %s", pd->name, vg->name);
+ pd->is_orphan_pv = 1;
+ }
+
+ if (!strcmp(vg->name, FMT_LVM1_ORPHAN_VG_NAME))
+ pp->orphan_vg_name = FMT_LVM1_ORPHAN_VG_NAME;
+ else
+ pp->orphan_vg_name = FMT_TEXT_ORPHAN_VG_NAME;
+ } else {
+ log_debug("Found pvcreate arg %s: device is not a pv", pd->name);
+ /* Device is not a PV. */
+ pd->is_not_pv = 1;
+ }
+
+ /*
+ * pvcreate is being run on this device, and it's not a PV,
+ * or is an orphan PV. Neither case requires a prompt.
+ */
+ if (pd->is_orphan_pv || pd->is_not_pv) {
+ pd->dev = pv->dev;
+ dm_list_move(&pp->arg_process, &pd->list);
+ return 1;
+ }
+
+ /*
+ * pvcreate is being run on this device, but the device is already
+ * a PV in a VG. A prompt or force option is required to use it.
+ */
+
+ if (!(prompt = dm_pool_zalloc(cmd->mem, sizeof(*prompt)))) {
+ log_error("prompt alloc failed");
+ pp->check_failed = 1;
+ return 0;
+ }
+ prompt->dev = pd->dev;
+ prompt->type = PROMPT_PVCREATE_PV_IN_VG;
+ prompt->pv_name = dm_pool_strdup(cmd->mem, pd->name);
+ if (pd->is_used_unknown_pv)
+ prompt->vg_name_unknown = 1;
+ else
+ prompt->vg_name = dm_pool_strdup(cmd->mem, vg->name);
+ dm_list_add(&pp->prompts, &prompt->list);
+
+ pd->dev = pv->dev;
+ dm_list_move(&pp->arg_process, &pd->list);
+ return 1;
+}
+
+/*
+ * This repeats the first check -- devices should be found, and should not have
+ * changed since the first check. If they were changed/used while the orphans
+ * lock was not held (during prompting), then they can't be used any more and
+ * are moved to arg_fail. If they are not found by this loop, that also
+ * disqualifies them from being used. Each arg_confirm entry that's found and
+ * is ok, is moved to arg_process. Those not found will remain in arg_confirm.
+ *
+ * This check does not need to look in foreign/shared/clustered VGs. If a
+ * device from arg_confirm was used in a foreign/shared/clustered VG during the
+ * prompts, then it will not be found during this check.
+ */
+
+static int _pvcreate_check2_single(struct cmd_context *cmd,
+ struct volume_group *vg,
+ struct physical_volume *pv,
+ struct processing_handle *handle)
+{
+ struct pvcreate_each_params *pp = (struct pvcreate_each_params *) handle->custom_handle;
+ struct pvcreate_device *pd;
+ int found = 0;
+
+ dm_list_iterate_items(pd, &pp->arg_confirm) {
+ if (pd->dev != pv->dev)
+ continue;
+ found = 1;
+ break;
+ }
+
+ if (!found)
+ return 1;
+
+ /* Repeat the same from check1. */
+ if (!dev_test_excl(pv->dev)) {
+ /* FIXME Detect whether device-mapper itself is still using it */
+ log_error("Can't open %s exclusively. Mounted filesystem?",
+ pv_dev_name(pv));
+ goto fail;
+ }
+
+ /*
+ * What kind of device is this: an orphan PV, an uninitialized/unused
+ * device, a PV used in a VG.
+ */
+
+ if (vg && !is_orphan_vg(vg->name)) {
+ /* Device is a PV used in a VG. */
+
+ if (pd->is_orphan_pv || pd->is_not_pv || pd->is_used_unknown_pv) {
+ /* In check1 it was an orphan or unused. */
+ goto fail;
+ }
+
+ if (pd->is_vg_pv && pd->vg_name && strcmp(pd->vg_name, vg->name)) {
+ /* In check1 it was in a different VG. */
+ goto fail;
+ }
+
+ } else if (is_orphan(pv)) {
+ /* Device is an orphan PV. */
+
+ if (pd->is_not_pv) {
+ /* In check1 it was not a PV. */
+ goto fail;
+ }
+
+ if (pd->is_vg_pv) {
+ /* In check1 it was in a VG. */
+ goto fail;
+ }
+
+ if (is_used_pv(pv) != pd->is_used_unknown_pv) {
+ /* In check1 it was different. */
+ goto fail;
+ }
+
+ } else {
+ /* Device is not a PV. */
+
+ if (pd->is_orphan_pv || pd->is_used_unknown_pv) {
+ /* In check1 it was an orphan PV. */
+ goto fail;
+ }
+
+ if (pd->is_vg_pv) {
+ /* In check1 it was in a VG. */
+ goto fail;
+ }
+ }
+
+ /* Device is unchanged from check1. */
+ dm_list_move(&pp->arg_process, &pd->list);
+ return 1;
+
+fail:
+ log_error("Cannot use device %s: it changed during create.", pd->name);
+ dm_list_move(&pp->arg_fail, &pd->list);
+ return 1;
+}
+
+/*
+ * This can be used by pvcreate, vgcreate and vgextend to create PVs. The
+ * callers need to set up the pvcreate_each_params structure based on command
+ * line args. This includes the pv_names field which specifies the devices to
+ * create PVs on.
+ *
+ * This uses process_each_pv() and should be called from a high level in the
+ * command -- the same level at which other instances of process_each are
+ * called.
+ *
+ * This function returns 0 (failed) if the caller requires all specified
+ * devices to be created, and any of those devices are not found, or any of
+ * them cannot be created.
+ *
+ * This function returns 1 (success) if the caller requires all specified
+ * devices to be created, and all are created, or if the caller does not
+ * require all specified devices to be created and one or more were created.
+ *
+ * When this function returns 1 (success), it returns to the caller with the
+ * VG_ORPHANS write lock held.
+ */
+
+int pvcreate_each_device(struct cmd_context *cmd,
+ struct processing_handle *handle,
+ struct pvcreate_each_params *pp)
+{
+ struct pvcreate_restorable_params rp;
+ struct pvcreate_device *pd, *pd2;
+ struct pvcreate_prompt *prompt, *prompt2;
+ struct physical_volume *pv;
+ struct volume_group *orphan_vg;
+ struct dm_list arg_sort;
+ struct pv_list *pvl;
+ struct pv_list *vgpvl;
+ const char *pv_name;
+ int consistent = 0;
+ int must_use_all = (cmd->command->flags & MUST_USE_ALL_ARGS);
+ int found;
+ int i;
+
+ dm_list_init(&arg_sort);
+
+ handle->custom_handle = pp;
+
+ /*
+ * Create a list entry for each name arg.
+ */
+ for (i = 0; i < pp->pv_count; i++) {
+ dm_unescape_colons_and_at_signs(pp->pv_names[i], NULL, NULL);
+
+ pv_name = pp->pv_names[i];
+
+ if (!(pd = dm_pool_zalloc(cmd->mem, sizeof(*pd)))) {
+ log_error("alloc failed");
+ return 0;
+ }
+
+ if (!(pd->name = dm_pool_strdup(cmd->mem, pv_name))) {
+ log_error("strdup failed");
+ return 0;
+ }
+
+ dm_list_add(&pp->arg_devices, &pd->list);
+ }
+
+ /*
+ * This function holds the orphans lock while reading VGs to look for
+ * devices. This means the orphans lock is held while VG locks are
+ * acquired, which is against lvmcache lock ordering rules, so disable
+ * the lvmcache lock ordering checks.
+ */
+ lvmcache_lock_ordering(0);
+
+ /*
+ * Clear the cache before acquiring the orphan lock. (Clearing the
+ * cache with locks held is an error.) We want the orphan lock
+ * acquired before process_each_pv. If the orphan lock is not held
+ * when process_each_pv is called, then process_each_pv clears the
+ * cache.
+ */
+ lvmcache_destroy(cmd, 1, 0);
+
+ /*
+ * If no prompts require a user response, this orphan lock is held
+ * throughout, and pvcreate_each_device() returns with it held so that
+ * vgcreate/vgextend use the PVs created here to add to a VG.
+ */
+ if (!lock_vol(cmd, VG_ORPHANS, LCK_VG_WRITE, NULL)) {
+ log_error("Can't get lock for orphan PVs");
+ return 0;
+ }
+
+ dev_cache_full_scan(cmd->full_filter);
+
+ /*
+ * Use process_each_pv to search all existing PVs and devices.
+ *
+ * This is a slightly different way to use process_each_pv, because the
+ * command args (arg_devices) are not being processed directly by
+ * process_each_pv (argc and argv are not passed). Instead,
+ * process_each_pv is processing all existing PVs and devices, and the
+ * single function is matching each of those against the command args
+ * (arg_devices).
+ *
+ * If an arg_devices entry is found during process_each_pv, it's moved
+ * to arg_process if it can be used, or arg_fail if it cannot be used.
+ * If it's added to arg_process but needs a prompt or force option, then
+ * a corresponding prompt entry is added to pp->prompts.
+ */
+ process_each_pv(cmd, 0, NULL, NULL, 1, 0, handle, _pvcreate_check1_single);
+
+ /*
+ * A fatal error was found while checking.
+ */
+ if (pp->check_failed)
+ goto_bad;
+
+ /*
+ * Check if all arg_devices were found by process_each_pv.
+ */
+ dm_list_iterate_items(pd, &pp->arg_devices)
+ log_error("Device %s not found (or ignored by filtering).", pd->name);
+
+ /*
+ * Can the command continue if some specified devices were not found?
+ */
+ if (!dm_list_empty(&pp->arg_devices) && must_use_all)
+ goto_bad;
+
+ /*
+ * Can the command continue if some specified devices cannot be used?
+ */
+ if (!dm_list_empty(&pp->arg_fail) && must_use_all)
+ goto_bad;
+
+ /*
+ * The command cannot continue if there are no devices to create.
+ */
+ if (dm_list_empty(&pp->arg_process)) {
+ log_error("No devices found.");
+ goto_bad;
+ }
+
+ /*
+ * Clear any prompts that have answers without asking the user.
+ */
+ dm_list_iterate_items_safe(prompt, prompt2, &pp->prompts) {
+ _check_pvcreate_prompt(cmd, pp, prompt, 0);
+
+ switch (prompt->answer) {
+ case PROMPT_ANSWER_YES:
+ /* The PV can be used, leave it on arg_process. */
+ dm_list_del(&prompt->list);
+ break;
+ case PROMPT_ANSWER_NO:
+ /* The PV cannot be used, remove it from arg_process. */
+ if ((pd = _pvcreate_list_find_dev(&pp->arg_process, prompt->dev)))
+ dm_list_move(&pp->arg_fail, &pd->list);
+ dm_list_del(&prompt->list);
+ break;
+ }
+ }
+
+ if (!dm_list_empty(&pp->arg_fail) && must_use_all)
+ goto_bad;
+
+ /*
+ * If no remaining prompts need a user response, then keep orphans
+ * locked and go directly to the create steps.
+ */
+ if (dm_list_empty(&pp->prompts))
+ goto do_command;
+
+ /*
+ * Prompts require asking the user, so release the orphans lock, ask
+ * the questions, reacquire the orphans lock, verify that the PVs were
+ * not used during the questions, then do the create steps.
+ */
+ unlock_vg(cmd, VG_ORPHANS);
+
+ /*
+ * Process prompts that require asking the user. The orphans lock is
+ * not held, so there's no harm in waiting for a user to respond.
+ */
+ dm_list_iterate_items_safe(prompt, prompt2, &pp->prompts) {
+ _check_pvcreate_prompt(cmd, pp, prompt, 1);
+
+ switch (prompt->answer) {
+ case PROMPT_ANSWER_YES:
+ /* The PV can be used, leave it on arg_process. */
+ dm_list_del(&prompt->list);
+ break;
+ case PROMPT_ANSWER_NO:
+ /* The PV cannot be used, remove it from arg_process. */
+ if ((pd = _pvcreate_list_find_dev(&pp->arg_process, prompt->dev)))
+ dm_list_move(&pp->arg_fail, &pd->list);
+ dm_list_del(&prompt->list);
+ break;
+ }
+
+ if (!dm_list_empty(&pp->arg_fail) && must_use_all)
+ goto_out;
+
+ if (sigint_caught())
+ goto_out;
+
+ if (prompt->abort_command)
+ goto_out;
+ }
+
+ /*
+ * Clear the cache, reacquire the orphans write lock, then check again
+ * that the devices can still be used. If the second loop finds them
+ * changed, or can't find them any more, then they aren't used.
+ * Clear the cache here before locking orphans, since it won't be
+ * done by process_each_pv with orphans already locked.
+ */
+
+ lvmcache_destroy(cmd, 1, 0);
+
+ if (!lock_vol(cmd, VG_ORPHANS, LCK_VG_WRITE, NULL)) {
+ log_error("Can't get lock for orphan PVs");
+ goto_out;
+ }
+
+ /*
+ * The device args began on the arg_devices list, then the first check
+ * loop moved those entries to arg_process as they were found. Devices
+ * not found during the first loop are not being used, and remain on
+ * arg_devices.
+ *
+ * Now, the arg_process entries are moved to arg_confirm, and the second
+ * check loop moves them back to arg_process as they are found and are
+ * unchanged. Like the first loop, the second loop moves an entry to
+ * arg_fail if it cannot be used. After the second loop, any devices
+ * remaining on arg_confirm were not found and are not used.
+ */
+
+ dm_list_splice(&pp->arg_confirm, &pp->arg_process);
+
+ process_each_pv(cmd, 0, NULL, NULL, 1, 0, handle, _pvcreate_check2_single);
+
+ dm_list_iterate_items(pd, &pp->arg_confirm)
+ log_error("Device %s not found (or ignored by filtering).", pd->name);
+
+ /* Some devices were not found during the second check. */
+ if (!dm_list_empty(&pp->arg_confirm) && must_use_all)
+ goto_bad;
+
+ /* Some devices changed during the second check. */
+ if (!dm_list_empty(&pp->arg_fail) && must_use_all)
+ goto_bad;
+
+ if (dm_list_empty(&pp->arg_process)) {
+ log_error("No devices found.");
+ goto_bad;
+ }
+
+do_command:
+
+ /*
+ * Reorder arg_process entries to match the original order of args.
+ */
+ dm_list_splice(&arg_sort, &pp->arg_process);
+ for (i = 0; i < pp->pv_count; i++) {
+ if ((pd = _pvcreate_list_find_name(&arg_sort, pp->pv_names[i])))
+ dm_list_move(&pp->arg_process, &pd->list);
+ }
+
+ dm_list_splice(&pp->arg_create, &pp->arg_process);
+
+ /*
+ * Wipe signatures on devices being created.
+ */
+ dm_list_iterate_items_safe(pd, pd2, &pp->arg_create) {
+ log_verbose("Wiping signatures on new PV %s", pd->name);
+
+ if (!wipe_known_signatures(cmd, pd->dev, pd->name, TYPE_LVM1_MEMBER | TYPE_LVM2_MEMBER,
+ 0, pp->yes, pp->force, &pd->wiped)) {
+ dm_list_move(&pp->arg_fail, &pd->list);
+ }
+
+ if (sigint_caught())
+ goto_bad;
+ }
+
+ if (!dm_list_empty(&pp->arg_fail) && must_use_all)
+ goto_bad;
+
+ /*
+ * Find existing orphan PVs that vgcreate or vgextend want to use.
+ * "preserve_existing" means that the command wants to use existing PVs
+ * and not recreate a new PV on top of an existing PV.
+ */
+ if (pp->preserve_existing && pp->orphan_vg_name) {
+ log_debug("Using existing orphan PVs in %s", pp->orphan_vg_name);
+
+ if (!(orphan_vg = vg_read_internal(cmd, pp->orphan_vg_name, NULL, 0, &consistent))) {
+ log_error("Cannot read orphans VG %s", pp->orphan_vg_name);
+ goto_bad;
+ }
+
+ dm_list_iterate_items_safe(pd, pd2, &pp->arg_create) {
+ if (!pd->is_orphan_pv)
+ continue;
+
+ if (!(pvl = dm_pool_alloc(cmd->mem, sizeof(*pvl)))) {
+ log_error("alloc pvl failed");
+ dm_list_move(&pp->arg_fail, &pd->list);
+ continue;
+ }
+
+ found = 0;
+ dm_list_iterate_items(vgpvl, &orphan_vg->pvs) {
+ if (vgpvl->pv->dev == pd->dev) {
+ found = 1;
+ break;
+ }
+ }
+
+ if (found) {
+ log_debug("Using existing orphan PV %s", pv_dev_name(vgpvl->pv));
+ pvl->pv = vgpvl->pv;
+ dm_list_add(&pp->pvs, &pvl->list);
+ } else {
+ log_error("Failed to find PV %s", pd->name);
+ dm_list_move(&pp->arg_fail, &pd->list);
+ }
+ }
+ }
+
+ /*
+ * Create PVs on devices. Either create a new PV on top of an existing
+ * one (e.g. for pvcreate), or create a new PV on a device that is not
+ * a PV.
+ */
+ dm_list_iterate_items_safe(pd, pd2, &pp->arg_create) {
+ /* Using existing orphan PVs is covered above. */
+ if (pp->preserve_existing && pd->is_orphan_pv)
+ continue;
+
+ if (!dm_list_empty(&pp->arg_fail) && must_use_all)
+ break;
+
+ if (!(pvl = dm_pool_alloc(cmd->mem, sizeof(*pvl)))) {
+ log_error("alloc pvl failed");
+ dm_list_move(&pp->arg_fail, &pd->list);
+ }
+
+ pv_name = pd->name;
+
+ log_debug("Creating a new PV on %s", pv_name);
+
+ /* FIXME: get rid of rp usage in pv_create to avoid this. */
+ memset(&rp, 0, sizeof(rp));
+ rp.restorefile = pp->restorefile;
+ if (pp->uuid_str) {
+ rp.id = pp->id;
+ rp.idp = &pp->id;
+ }
+ rp.ba_start = pp->ba_start;
+ rp.ba_size = pp->ba_size;
+ rp.pe_start = pp->pe_start;
+ rp.extent_count = pp->extent_count;
+ rp.extent_size = pp->extent_size;
+
+ if (!(pv = pv_create(cmd, pd->dev, pp->size, pp->data_alignment,
+ pp->data_alignment_offset, pp->labelsector,
+ pp->pvmetadatacopies, pp->pvmetadatasize,
+ pp->metadataignore, &rp))) {
+ log_error("Failed to setup physical volume \"%s\"", pv_name);
+ dm_list_move(&pp->arg_fail, &pd->list);
+ continue;
+ }
+
+ log_verbose("Set up physical volume for \"%s\" with %" PRIu64
+ " available sectors", pv_name, pv_size(pv));
+
+ if (!label_remove(pv->dev)) {
+ log_error("Failed to wipe existing label on %s", pv_name);
+ dm_list_move(&pp->arg_fail, &pd->list);
+ continue;
+ }
+
+ if (pp->zero) {
+ log_verbose("Zeroing start of device %s", pv_name);
+
+ if (!dev_open_quiet(pv->dev)) {
+ log_error("%s not opened: device not zeroed", pv_name);
+ dm_list_move(&pp->arg_fail, &pd->list);
+ continue;
+ }
+
+ if (!dev_set(pv->dev, UINT64_C(0), (size_t) 2048, 0)) {
+ log_error("%s not wiped: aborting", pv_name);
+ if (!dev_close(pv->dev))
+ stack;
+ dm_list_move(&pp->arg_fail, &pd->list);
+ continue;
+ }
+ if (!dev_close(pv->dev))
+ stack;
+ }
+
+ log_verbose("Writing physical volume data to disk \"%s\"", pv_name);
+
+ if (!pv_write(cmd, pv, 0)) {
+ log_error("Failed to write physical volume \"%s\"", pv_name);
+ dm_list_move(&pp->arg_fail, &pd->list);
+ }
+
+ log_print_unless_silent("Physical volume \"%s\" successfully created", pv_name);
+
+ pvl->pv = pv;
+ dm_list_add(&pp->pvs, &pvl->list);
+ }
+
+ dm_list_iterate_items(pvl, &pp->pvs)
+ log_debug("pv command succeeded for %s", pv_dev_name(pvl->pv));
+
+ dm_list_iterate_items(pd, &pp->arg_fail)
+ log_debug("pv command failed for %s", pd->name);
+
+ if (!dm_list_empty(&pp->arg_fail))
+ goto_bad;
+
+ /*
+ * Returns with VG_ORPHANS write lock held because vgcreate and
+ * vgextend want to use the newly created PVs.
+ */
+ return 1;
+
+bad:
+ unlock_vg(cmd, VG_ORPHANS);
+out:
+ return 0;
+}
+