vgmerge.c revision 1.1.1.2 1 1.1 haad /* $NetBSD: vgmerge.c,v 1.1.1.2 2009/12/02 00:25:57 haad Exp $ */
2 1.1 haad
3 1.1 haad /*
4 1.1 haad * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
5 1.1.1.2 haad * Copyright (C) 2004-2009 Red Hat, Inc. All rights reserved.
6 1.1 haad *
7 1.1 haad * This file is part of LVM2.
8 1.1 haad *
9 1.1 haad * This copyrighted material is made available to anyone wishing to use,
10 1.1 haad * modify, copy, or redistribute it subject to the terms and conditions
11 1.1 haad * of the GNU Lesser General Public License v.2.1.
12 1.1 haad *
13 1.1 haad * You should have received a copy of the GNU Lesser General Public License
14 1.1 haad * along with this program; if not, write to the Free Software Foundation,
15 1.1 haad * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
16 1.1 haad */
17 1.1 haad
18 1.1 haad #include "tools.h"
19 1.1 haad
20 1.1.1.2 haad static struct volume_group *_vgmerge_vg_read(struct cmd_context *cmd,
21 1.1.1.2 haad const char *vg_name)
22 1.1.1.2 haad {
23 1.1.1.2 haad struct volume_group *vg;
24 1.1.1.2 haad log_verbose("Checking for volume group \"%s\"", vg_name);
25 1.1.1.2 haad vg = vg_read_for_update(cmd, vg_name, NULL, 0);
26 1.1.1.2 haad if (vg_read_error(vg)) {
27 1.1.1.2 haad vg_release(vg);
28 1.1.1.2 haad return NULL;
29 1.1.1.2 haad }
30 1.1.1.2 haad return vg;
31 1.1.1.2 haad }
32 1.1.1.2 haad
33 1.1 haad static int _vgmerge_single(struct cmd_context *cmd, const char *vg_name_to,
34 1.1 haad const char *vg_name_from)
35 1.1 haad {
36 1.1 haad struct volume_group *vg_to, *vg_from;
37 1.1 haad struct lv_list *lvl1, *lvl2;
38 1.1.1.2 haad int r = ECMD_FAILED;
39 1.1.1.2 haad int lock_vg_from_first = 0;
40 1.1 haad
41 1.1 haad if (!strcmp(vg_name_to, vg_name_from)) {
42 1.1 haad log_error("Duplicate volume group name \"%s\"", vg_name_from);
43 1.1 haad return ECMD_FAILED;
44 1.1 haad }
45 1.1 haad
46 1.1.1.2 haad if (strcmp(vg_name_to, vg_name_from) > 0)
47 1.1.1.2 haad lock_vg_from_first = 1;
48 1.1.1.2 haad
49 1.1.1.2 haad if (lock_vg_from_first) {
50 1.1.1.2 haad vg_from = _vgmerge_vg_read(cmd, vg_name_from);
51 1.1.1.2 haad if (!vg_from) {
52 1.1.1.2 haad stack;
53 1.1.1.2 haad return ECMD_FAILED;
54 1.1.1.2 haad }
55 1.1.1.2 haad vg_to = _vgmerge_vg_read(cmd, vg_name_to);
56 1.1.1.2 haad if (!vg_to) {
57 1.1.1.2 haad stack;
58 1.1.1.2 haad unlock_and_release_vg(cmd, vg_from, vg_name_from);
59 1.1.1.2 haad return ECMD_FAILED;
60 1.1.1.2 haad }
61 1.1.1.2 haad } else {
62 1.1.1.2 haad vg_to = _vgmerge_vg_read(cmd, vg_name_to);
63 1.1.1.2 haad if (!vg_to) {
64 1.1.1.2 haad stack;
65 1.1.1.2 haad return ECMD_FAILED;
66 1.1.1.2 haad }
67 1.1.1.2 haad
68 1.1.1.2 haad vg_from = _vgmerge_vg_read(cmd, vg_name_from);
69 1.1.1.2 haad if (!vg_from) {
70 1.1.1.2 haad stack;
71 1.1.1.2 haad unlock_and_release_vg(cmd, vg_to, vg_name_to);
72 1.1.1.2 haad return ECMD_FAILED;
73 1.1.1.2 haad }
74 1.1 haad }
75 1.1 haad
76 1.1 haad if (!vgs_are_compatible(cmd, vg_from, vg_to))
77 1.1 haad goto_bad;
78 1.1 haad
79 1.1 haad /* FIXME List arg: vg_show_with_pv_and_lv(vg_to); */
80 1.1 haad
81 1.1 haad if (!archive(vg_from) || !archive(vg_to))
82 1.1 haad goto_bad;
83 1.1 haad
84 1.1 haad drop_cached_metadata(vg_from);
85 1.1 haad
86 1.1 haad /* Merge volume groups */
87 1.1 haad while (!dm_list_empty(&vg_from->pvs)) {
88 1.1 haad struct dm_list *pvh = vg_from->pvs.n;
89 1.1 haad struct physical_volume *pv;
90 1.1 haad
91 1.1 haad dm_list_move(&vg_to->pvs, pvh);
92 1.1 haad
93 1.1 haad pv = dm_list_item(pvh, struct pv_list)->pv;
94 1.1 haad pv->vg_name = dm_pool_strdup(cmd->mem, vg_to->name);
95 1.1 haad }
96 1.1 haad vg_to->pv_count += vg_from->pv_count;
97 1.1 haad
98 1.1 haad /* Fix up LVIDs */
99 1.1 haad dm_list_iterate_items(lvl1, &vg_to->lvs) {
100 1.1 haad union lvid *lvid1 = &lvl1->lv->lvid;
101 1.1 haad char uuid[64] __attribute((aligned(8)));
102 1.1 haad
103 1.1 haad dm_list_iterate_items(lvl2, &vg_from->lvs) {
104 1.1 haad union lvid *lvid2 = &lvl2->lv->lvid;
105 1.1 haad
106 1.1 haad if (id_equal(&lvid1->id[1], &lvid2->id[1])) {
107 1.1 haad if (!id_create(&lvid2->id[1])) {
108 1.1 haad log_error("Failed to generate new "
109 1.1 haad "random LVID for %s",
110 1.1 haad lvl2->lv->name);
111 1.1 haad goto bad;
112 1.1 haad }
113 1.1 haad if (!id_write_format(&lvid2->id[1], uuid,
114 1.1 haad sizeof(uuid)))
115 1.1 haad goto_bad;
116 1.1 haad
117 1.1 haad log_verbose("Changed LVID for %s to %s",
118 1.1 haad lvl2->lv->name, uuid);
119 1.1 haad }
120 1.1 haad }
121 1.1 haad }
122 1.1 haad
123 1.1 haad while (!dm_list_empty(&vg_from->lvs)) {
124 1.1 haad struct dm_list *lvh = vg_from->lvs.n;
125 1.1 haad
126 1.1 haad dm_list_move(&vg_to->lvs, lvh);
127 1.1 haad }
128 1.1 haad
129 1.1 haad while (!dm_list_empty(&vg_from->fid->metadata_areas)) {
130 1.1 haad struct dm_list *mdah = vg_from->fid->metadata_areas.n;
131 1.1 haad
132 1.1 haad dm_list_move(&vg_to->fid->metadata_areas, mdah);
133 1.1 haad }
134 1.1 haad
135 1.1 haad vg_to->extent_count += vg_from->extent_count;
136 1.1 haad vg_to->free_count += vg_from->free_count;
137 1.1 haad
138 1.1 haad /* store it on disks */
139 1.1 haad log_verbose("Writing out updated volume group");
140 1.1 haad if (!vg_write(vg_to) || !vg_commit(vg_to))
141 1.1 haad goto_bad;
142 1.1 haad
143 1.1 haad /* FIXME Remove /dev/vgfrom */
144 1.1 haad
145 1.1 haad backup(vg_to);
146 1.1 haad log_print("Volume group \"%s\" successfully merged into \"%s\"",
147 1.1 haad vg_from->name, vg_to->name);
148 1.1.1.2 haad r = ECMD_PROCESSED;
149 1.1.1.2 haad bad:
150 1.1.1.2 haad if (lock_vg_from_first) {
151 1.1.1.2 haad unlock_and_release_vg(cmd, vg_to, vg_name_to);
152 1.1.1.2 haad unlock_and_release_vg(cmd, vg_from, vg_name_from);
153 1.1.1.2 haad } else {
154 1.1.1.2 haad unlock_and_release_vg(cmd, vg_from, vg_name_from);
155 1.1.1.2 haad unlock_and_release_vg(cmd, vg_to, vg_name_to);
156 1.1.1.2 haad }
157 1.1.1.2 haad return r;
158 1.1 haad }
159 1.1 haad
160 1.1 haad int vgmerge(struct cmd_context *cmd, int argc, char **argv)
161 1.1 haad {
162 1.1 haad char *vg_name_to, *vg_name_from;
163 1.1 haad int opt = 0;
164 1.1 haad int ret = 0, ret_max = 0;
165 1.1 haad
166 1.1 haad if (argc < 2) {
167 1.1 haad log_error("Please enter 2 or more volume groups to merge");
168 1.1 haad return EINVALID_CMD_LINE;
169 1.1 haad }
170 1.1 haad
171 1.1 haad vg_name_to = skip_dev_dir(cmd, argv[0], NULL);
172 1.1 haad argc--;
173 1.1 haad argv++;
174 1.1 haad
175 1.1 haad for (; opt < argc; opt++) {
176 1.1 haad vg_name_from = skip_dev_dir(cmd, argv[opt], NULL);
177 1.1 haad
178 1.1 haad ret = _vgmerge_single(cmd, vg_name_to, vg_name_from);
179 1.1 haad if (ret > ret_max)
180 1.1 haad ret_max = ret;
181 1.1 haad }
182 1.1 haad
183 1.1 haad return ret_max;
184 1.1 haad }
185