1*7c604eeaShaad /* $NetBSD: pvremove.c,v 1.1.1.2 2009/12/02 00:25:54 haad Exp $ */
256a34939Shaad
356a34939Shaad /*
456a34939Shaad * Copyright (C) 2002-2004 Sistina Software, Inc. All rights reserved.
556a34939Shaad * Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
656a34939Shaad *
756a34939Shaad * This file is part of LVM2.
856a34939Shaad *
956a34939Shaad * This copyrighted material is made available to anyone wishing to use,
1056a34939Shaad * modify, copy, or redistribute it subject to the terms and conditions
1156a34939Shaad * of the GNU Lesser General Public License v.2.1.
1256a34939Shaad *
1356a34939Shaad * You should have received a copy of the GNU Lesser General Public License
1456a34939Shaad * along with this program; if not, write to the Free Software Foundation,
1556a34939Shaad * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
1656a34939Shaad */
1756a34939Shaad
1856a34939Shaad #include "tools.h"
1956a34939Shaad
2056a34939Shaad const char _really_wipe[] =
2156a34939Shaad "Really WIPE LABELS from physical volume \"%s\" of volume group \"%s\" [y/n]? ";
2256a34939Shaad
2356a34939Shaad /*
2456a34939Shaad * Decide whether it is "safe" to wipe the labels on this device.
2556a34939Shaad * 0 indicates we may not.
2656a34939Shaad */
pvremove_check(struct cmd_context * cmd,const char * name)2756a34939Shaad static int pvremove_check(struct cmd_context *cmd, const char *name)
2856a34939Shaad {
2956a34939Shaad struct physical_volume *pv;
3056a34939Shaad struct dm_list mdas;
3156a34939Shaad
3256a34939Shaad dm_list_init(&mdas);
3356a34939Shaad
3456a34939Shaad /* FIXME Check partition type is LVM unless --force is given */
3556a34939Shaad
3656a34939Shaad /* Is there a pv here already? */
3756a34939Shaad /* If not, this is an error unless you used -f. */
38*7c604eeaShaad if (!(pv = pv_read(cmd, name, &mdas, NULL, 1, 0))) {
3956a34939Shaad if (arg_count(cmd, force_ARG))
4056a34939Shaad return 1;
4156a34939Shaad log_error("Physical Volume %s not found", name);
4256a34939Shaad return 0;
4356a34939Shaad }
4456a34939Shaad
4556a34939Shaad /*
4656a34939Shaad * If a PV has no MDAs it may appear to be an
4756a34939Shaad * orphan until the metadata is read off
4856a34939Shaad * another PV in the same VG. Detecting this
4956a34939Shaad * means checking every VG by scanning every
5056a34939Shaad * PV on the system.
5156a34939Shaad */
5256a34939Shaad if (is_orphan(pv) && !dm_list_size(&mdas)) {
5356a34939Shaad if (!scan_vgs_for_pvs(cmd)) {
5456a34939Shaad log_error("Rescan for PVs without metadata areas "
5556a34939Shaad "failed.");
5656a34939Shaad return 0;
5756a34939Shaad }
58*7c604eeaShaad if (!(pv = pv_read(cmd, name, NULL, NULL, 1, 0))) {
5956a34939Shaad log_error("Failed to read physical volume %s", name);
6056a34939Shaad return 0;
6156a34939Shaad }
6256a34939Shaad }
6356a34939Shaad
6456a34939Shaad /* orphan ? */
6556a34939Shaad if (is_orphan(pv))
6656a34939Shaad return 1;
6756a34939Shaad
6856a34939Shaad /* Allow partial & exported VGs to be destroyed. */
6956a34939Shaad /* we must have -ff to overwrite a non orphan */
7056a34939Shaad if (arg_count(cmd, force_ARG) < 2) {
7156a34939Shaad log_error("Can't pvremove physical volume \"%s\" of "
7256a34939Shaad "volume group \"%s\" without -ff", name, pv_vg_name(pv));
7356a34939Shaad return 0;
7456a34939Shaad }
7556a34939Shaad
7656a34939Shaad /* prompt */
7756a34939Shaad if (!arg_count(cmd, yes_ARG) &&
7856a34939Shaad yes_no_prompt(_really_wipe, name, pv_vg_name(pv)) == 'n') {
7956a34939Shaad log_print("%s: physical volume label not removed", name);
8056a34939Shaad return 0;
8156a34939Shaad }
8256a34939Shaad
8356a34939Shaad if (arg_count(cmd, force_ARG)) {
8456a34939Shaad log_warn("WARNING: Wiping physical volume label from "
8556a34939Shaad "%s%s%s%s", name,
8656a34939Shaad !is_orphan(pv) ? " of volume group \"" : "",
8756a34939Shaad !is_orphan(pv) ? pv_vg_name(pv) : "",
8856a34939Shaad !is_orphan(pv) ? "\"" : "");
8956a34939Shaad }
9056a34939Shaad
9156a34939Shaad return 1;
9256a34939Shaad }
9356a34939Shaad
pvremove_single(struct cmd_context * cmd,const char * pv_name,void * handle __attribute ((unused)))9456a34939Shaad static int pvremove_single(struct cmd_context *cmd, const char *pv_name,
9556a34939Shaad void *handle __attribute((unused)))
9656a34939Shaad {
9756a34939Shaad struct device *dev;
9856a34939Shaad int ret = ECMD_FAILED;
9956a34939Shaad
10056a34939Shaad if (!lock_vol(cmd, VG_ORPHANS, LCK_VG_WRITE)) {
10156a34939Shaad log_error("Can't get lock for orphan PVs");
10256a34939Shaad return ECMD_FAILED;
10356a34939Shaad }
10456a34939Shaad
10556a34939Shaad if (!pvremove_check(cmd, pv_name))
10656a34939Shaad goto error;
10756a34939Shaad
10856a34939Shaad if (!(dev = dev_cache_get(pv_name, cmd->filter))) {
10956a34939Shaad log_error("%s: Couldn't find device. Check your filters?",
11056a34939Shaad pv_name);
11156a34939Shaad goto error;
11256a34939Shaad }
11356a34939Shaad
11456a34939Shaad if (!dev_test_excl(dev)) {
11556a34939Shaad /* FIXME Detect whether device-mapper is still using the device */
11656a34939Shaad log_error("Can't open %s exclusively - not removing. "
11756a34939Shaad "Mounted filesystem?", dev_name(dev));
11856a34939Shaad goto error;
11956a34939Shaad }
12056a34939Shaad
12156a34939Shaad /* Wipe existing label(s) */
12256a34939Shaad if (!label_remove(dev)) {
12356a34939Shaad log_error("Failed to wipe existing label(s) on %s", pv_name);
12456a34939Shaad goto error;
12556a34939Shaad }
12656a34939Shaad
12756a34939Shaad log_print("Labels on physical volume \"%s\" successfully wiped",
12856a34939Shaad pv_name);
12956a34939Shaad
13056a34939Shaad ret = ECMD_PROCESSED;
13156a34939Shaad
13256a34939Shaad error:
13356a34939Shaad unlock_vg(cmd, VG_ORPHANS);
13456a34939Shaad
13556a34939Shaad return ret;
13656a34939Shaad }
13756a34939Shaad
pvremove(struct cmd_context * cmd,int argc,char ** argv)13856a34939Shaad int pvremove(struct cmd_context *cmd, int argc, char **argv)
13956a34939Shaad {
14056a34939Shaad int i, r;
14156a34939Shaad int ret = ECMD_PROCESSED;
14256a34939Shaad
14356a34939Shaad if (!argc) {
14456a34939Shaad log_error("Please enter a physical volume path");
14556a34939Shaad return EINVALID_CMD_LINE;
14656a34939Shaad }
14756a34939Shaad
14856a34939Shaad if (arg_count(cmd, yes_ARG) && !arg_count(cmd, force_ARG)) {
14956a34939Shaad log_error("Option y can only be given with option f");
15056a34939Shaad return EINVALID_CMD_LINE;
15156a34939Shaad }
15256a34939Shaad
15356a34939Shaad for (i = 0; i < argc; i++) {
15456a34939Shaad r = pvremove_single(cmd, argv[i], NULL);
15556a34939Shaad if (r > ret)
15656a34939Shaad ret = r;
15756a34939Shaad }
15856a34939Shaad
15956a34939Shaad return ret;
16056a34939Shaad }
161