/* * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved. * Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved. * * 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. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include "tools.h" struct lvrename_params { int historical; const char *lv_name_old; const char *lv_name_new; }; /* * Dummy LV to represent historical LV. */ static struct logical_volume _historical_lv = { .name = "", .major = -1, .minor = -1, .snapshot_segs = DM_LIST_HEAD_INIT(_historical_lv.snapshot_segs), .segments = DM_LIST_HEAD_INIT(_historical_lv.segments), .tags = DM_LIST_HEAD_INIT(_historical_lv.tags), .segs_using_this_lv = DM_LIST_HEAD_INIT(_historical_lv.segs_using_this_lv), .indirect_glvs = DM_LIST_HEAD_INIT(_historical_lv.indirect_glvs), .hostname = "", }; static int _lvrename_single(struct cmd_context *cmd, const char *vg_name, struct volume_group *vg, struct processing_handle *handle) { struct lvrename_params *lp = (struct lvrename_params *) handle->custom_handle; struct generic_logical_volume *glv; struct logical_volume *lv; int ret = ECMD_FAILED; if (!lp->historical) { if (!(lv = find_lv(vg, lp->lv_name_old))) { log_error("Existing logical volume \"%s\" not found in " "volume group \"%s\"", lp->lv_name_old, vg_name); goto bad; } if (lv_is_raid_image(lv) || lv_is_raid_metadata(lv)) { log_error("Cannot rename a RAID %s directly", lv_is_raid_image(lv) ? "image" : "metadata area"); goto bad; } if (lv_is_raid_with_tracking(lv)) { log_error("Cannot rename %s while it is tracking a split image", lv->name); goto bad; } } else { if (!(glv = find_historical_glv(vg, lp->lv_name_old, 0, NULL))) { log_error("Existing historical logical volume \"%s\" not found in " "volume group \"%s\"", lp->lv_name_old, vg_name); goto bad; } _historical_lv.vg = vg; _historical_lv.name = lp->lv_name_old; _historical_lv.this_glv = glv; lv = &_historical_lv; } /* * The lvmlockd LV lock is only acquired here to ensure the LV is not * active on another host. This requests a transient LV lock. * If the LV is active, a persistent LV lock already exists in * lvmlockd, and the transient lock request does nothing. * If the LV is not active, then no LV lock exists and the transient * lock request acquires the LV lock (or fails). The transient lock * is automatically released when the command exits. */ if (!lockd_lv(cmd, lv, "ex", 0)) goto_bad; if (!lv_rename(cmd, lv, lp->lv_name_new)) goto_bad; log_print_unless_silent("Renamed \"%s%s\" to \"%s%s\" in volume group \"%s\"", lp->historical ? HISTORICAL_LV_PREFIX : "", lp->lv_name_old, lp->historical ? HISTORICAL_LV_PREFIX : "", lp->lv_name_new, vg_name); ret = ECMD_PROCESSED; bad: return ret; } /* * lvrename command implementation. * Check arguments and call lv_rename() to execute the request. */ int lvrename(struct cmd_context *cmd, int argc, char **argv) { struct processing_handle *handle = NULL; struct lvrename_params lp = { 0 }; size_t maxlen; char *lv_name_old, *lv_name_new; const char *vg_name, *vg_name_new, *vg_name_old; int historical = 0; char *st; int ret; cmd->include_historical_lvs = 1; if (argc == 3) { vg_name = skip_dev_dir(cmd, argv[0], NULL); lv_name_old = argv[1]; lv_name_new = argv[2]; if (strchr(lv_name_old, '/') && (vg_name_old = extract_vgname(cmd, lv_name_old)) && strcmp(vg_name_old, vg_name)) { log_error("Please use a single volume group name " "(\"%s\" or \"%s\")", vg_name, vg_name_old); return EINVALID_CMD_LINE; } } else if (argc == 2) { lv_name_old = argv[0]; lv_name_new = argv[1]; vg_name = extract_vgname(cmd, lv_name_old); } else { log_error("Old and new logical volume names required"); return EINVALID_CMD_LINE; } if (!validate_name(vg_name)) { log_error("Please provide a valid volume group name"); return EINVALID_CMD_LINE; } if (strchr(lv_name_new, '/') && (vg_name_new = extract_vgname(cmd, lv_name_new)) && strcmp(vg_name, vg_name_new)) { log_error("Logical volume names must " "have the same volume group (\"%s\" or \"%s\")", vg_name, vg_name_new); return EINVALID_CMD_LINE; } if ((st = strrchr(lv_name_old, '/'))) lv_name_old = st + 1; if ((st = strrchr(lv_name_new, '/'))) lv_name_new = st + 1; if (!strncmp(lv_name_old, HISTORICAL_LV_PREFIX, strlen(HISTORICAL_LV_PREFIX))) { lv_name_old = lv_name_old + strlen(HISTORICAL_LV_PREFIX); historical = 1; } if (!strncmp(lv_name_new, HISTORICAL_LV_PREFIX, strlen(HISTORICAL_LV_PREFIX))) { if (historical) lv_name_new = lv_name_old + strlen(HISTORICAL_LV_PREFIX); else { log_error("Old name references live LV while " "new name is for historical LV."); return EINVALID_CMD_LINE; } } /* Check sanity of new name */ maxlen = NAME_LEN - strlen(vg_name) - 3; if (strlen(lv_name_new) > maxlen) { log_error("New logical volume name \"%s\" may not exceed %" PRIsize_t " characters.", lv_name_new, maxlen); return EINVALID_CMD_LINE; } if (!*lv_name_new) { log_error("New logical volume name may not be blank"); return EINVALID_CMD_LINE; } if (!apply_lvname_restrictions(lv_name_new)) { stack; return EINVALID_CMD_LINE; } if (!validate_name(lv_name_new)) { log_error("New logical volume name \"%s\" is invalid", lv_name_new); return EINVALID_CMD_LINE; } if (!strcmp(lv_name_old, lv_name_new)) { log_error("Old and new logical volume names must differ"); return EINVALID_CMD_LINE; } lp.historical = historical; if (!(lp.lv_name_old = dm_pool_strdup(cmd->mem, lv_name_old))) return ECMD_FAILED; if (!(lp.lv_name_new = dm_pool_strdup(cmd->mem, lv_name_new))) return ECMD_FAILED; if (!(handle = init_processing_handle(cmd))) { log_error("Failed to initialize processing handle."); return ECMD_FAILED; } handle->custom_handle = &lp; ret = process_each_vg(cmd, 0, NULL, vg_name, NULL, READ_FOR_UPDATE, handle, _lvrename_single); destroy_processing_handle(cmd, handle); return ret; }