48266eaa0881095936c1c0ca75459483a9b5e3d0
[dragonfly.git] / sbin / hammer / cmd_volume.c
1 /*
2  * Copyright (c) 2009 The DragonFly Project.  All rights reserved.
3  *
4  * This code is derived from software contributed to The DragonFly Project
5  * by Matthew Dillon <dillon@backplane.com> and
6  * Michael Neumann <mneumann@ntecs.de>
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  *
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in
16  *    the documentation and/or other materials provided with the
17  *    distribution.
18  * 3. Neither the name of The DragonFly Project nor the names of its
19  *    contributors may be used to endorse or promote products derived
20  *    from this software without specific, prior written permission.
21  *
22  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
23  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
24  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
25  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
26  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
27  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
28  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
30  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
31  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
32  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33  * SUCH DAMAGE.
34  *
35  */
36 /*
37  * Volume operations:
38  *
39  *   - volume-add: Add new volume to HAMMER filesystem
40  *   - volume-del: Remove volume from HAMMER filesystem
41  *   - volume-list: List volumes making up a HAMMER filesystem
42  *   - volume-blkdevs: List volumes making up a HAMMER filesystem
43  *     in blkdevs format
44  */
45
46 #include "hammer.h"
47
48 /*
49  * volume-add <device> <filesystem>
50  */
51 void
52 hammer_cmd_volume_add(char **av, int ac)
53 {
54         struct hammer_ioc_volume ioc;
55         struct volume_info *vol;
56         int fd;
57         const char *device, *filesystem;
58
59         if (ac != 2) {
60                 fprintf(stderr, "hammer volume-add <device> <filesystem>\n");
61                 exit(1);
62         }
63
64         device = av[0];
65         filesystem = av[1];
66
67         fd = open(filesystem, O_RDONLY);
68         if (fd < 0) {
69                 fprintf(stderr, "hammer volume-add: unable to access %s: %s\n",
70                         filesystem, strerror(errno));
71                 exit(1);
72         }
73
74         /*
75          * Initialize and check the device
76          */
77         vol = init_volume(-1, device, O_RDONLY);
78         assert(vol->vol_no == -1);
79         check_volume(vol);
80         if (strcmp(vol->type, "DEVICE")) {
81                 fprintf(stderr, "Not a block device: %s\n", device);
82                 exit(1);
83         }
84         close(vol->fd);
85
86         /*
87          * volume-add ioctl
88          */
89         bzero(&ioc, sizeof(ioc));
90         strncpy(ioc.device_name, device, MAXPATHLEN);
91         ioc.vol_size = vol->size;
92         ioc.boot_area_size = init_boot_area_size(0, ioc.vol_size);
93         ioc.mem_area_size = init_mem_area_size(0, ioc.vol_size);
94
95         if (ioctl(fd, HAMMERIOC_ADD_VOLUME, &ioc) < 0) {
96                 fprintf(stderr, "hammer volume-add ioctl: %s\n",
97                         strerror(errno));
98                 exit(1);
99         }
100
101         close(fd);
102 }
103
104 /*
105  * volume-del <device> <filesystem>
106  */
107 void
108 hammer_cmd_volume_del(char **av, int ac)
109 {
110         struct hammer_ioc_volume ioc;
111         int fd;
112         const char *device, *filesystem;
113
114         if (ac != 2) {
115                 fprintf(stderr, "hammer volume-del <device> <filesystem>\n");
116                 exit(1);
117         }
118
119         device = av[0];
120         filesystem = av[1];
121
122         fd = open(filesystem, O_RDONLY);
123         if (fd < 0) {
124                 fprintf(stderr, "hammer volume-del: unable to access %s: %s\n",
125                         filesystem, strerror(errno));
126                 exit(1);
127         }
128
129         /*
130          * volume-del ioctl
131          */
132         bzero(&ioc, sizeof(ioc));
133         strncpy(ioc.device_name, device, MAXPATHLEN);
134
135         if (ioctl(fd, HAMMERIOC_DEL_VOLUME, &ioc) < 0) {
136                 fprintf(stderr, "hammer volume-del ioctl: %s\n",
137                         strerror(errno));
138                 exit(1);
139         }
140
141         close(fd);
142 }
143
144 static void
145 hammer_print_volumes(char **av, int ac, const char *cmd, const char sep)
146 {
147         struct hammer_ioc_volume_list ioc;
148         int fd;
149         int i;
150         const char *filesystem;
151
152         if (ac != 1) {
153                 fprintf(stderr, "hammer %s <filesystem>\n", cmd);
154                 exit(1);
155         }
156
157         filesystem = av[0];
158
159         fd = open(filesystem, O_RDONLY);
160         if (fd < 0) {
161                 fprintf(stderr,
162                     "hammer %s: unable to access %s: %s\n",
163                     cmd, filesystem, strerror(errno));
164                 exit(1);
165         }
166
167         bzero(&ioc, sizeof(ioc));
168         ioc.vols = malloc(HAMMER_MAX_VOLUMES *
169                           sizeof(struct hammer_ioc_volume));
170         if (ioc.vols == NULL) {
171                 fprintf(stderr,
172                     "hammer %s: unable to allocate memory: %s\n",
173                     cmd, strerror(errno));
174                 exit(1);
175         }
176         ioc.nvols = HAMMER_MAX_VOLUMES;
177
178         if (ioctl(fd, HAMMERIOC_LIST_VOLUMES, &ioc) < 0) {
179                 fprintf(stderr, "hammer %s ioctl: %s\n",
180                         cmd, strerror(errno));
181                 free(ioc.vols);
182                 exit(1);
183         }
184
185         for (i = 0; i < ioc.nvols; i++) {
186                 printf("%s", ioc.vols[i].device_name);
187                 if (i != ioc.nvols - 1)
188                         printf("%c", sep);
189         }
190         printf("\n");
191
192         free(ioc.vols);
193         close(fd);
194 }
195
196 /*
197  * volume-list <filesystem>
198  */
199 void
200 hammer_cmd_volume_list(char **av, int ac, const char *cmd)
201 {
202         hammer_print_volumes(av, ac, cmd, '\n');
203 }
204
205 /*
206  * volume-blkdevs <filesystem>
207  */
208 void
209 hammer_cmd_volume_blkdevs(char **av, int ac, const char *cmd)
210 {
211         hammer_print_volumes(av, ac, cmd, ':');
212 }