dkctl.c revision 1.14 1 1.14 dsl /* $NetBSD: dkctl.c,v 1.14 2006/03/17 19:18:33 dsl Exp $ */
2 1.1 thorpej
3 1.1 thorpej /*
4 1.1 thorpej * Copyright 2001 Wasabi Systems, Inc.
5 1.1 thorpej * All rights reserved.
6 1.1 thorpej *
7 1.1 thorpej * Written by Jason R. Thorpe for Wasabi Systems, Inc.
8 1.1 thorpej *
9 1.1 thorpej * Redistribution and use in source and binary forms, with or without
10 1.1 thorpej * modification, are permitted provided that the following conditions
11 1.1 thorpej * are met:
12 1.1 thorpej * 1. Redistributions of source code must retain the above copyright
13 1.1 thorpej * notice, this list of conditions and the following disclaimer.
14 1.1 thorpej * 2. Redistributions in binary form must reproduce the above copyright
15 1.1 thorpej * notice, this list of conditions and the following disclaimer in the
16 1.1 thorpej * documentation and/or other materials provided with the distribution.
17 1.1 thorpej * 3. All advertising materials mentioning features or use of this software
18 1.1 thorpej * must display the following acknowledgement:
19 1.1 thorpej * This product includes software developed for the NetBSD Project by
20 1.1 thorpej * Wasabi Systems, Inc.
21 1.1 thorpej * 4. The name of Wasabi Systems, Inc. may not be used to endorse
22 1.1 thorpej * or promote products derived from this software without specific prior
23 1.1 thorpej * written permission.
24 1.1 thorpej *
25 1.1 thorpej * THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
26 1.1 thorpej * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
27 1.1 thorpej * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
28 1.1 thorpej * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC
29 1.1 thorpej * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
30 1.1 thorpej * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
31 1.1 thorpej * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
32 1.1 thorpej * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
33 1.1 thorpej * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
34 1.1 thorpej * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
35 1.1 thorpej * POSSIBILITY OF SUCH DAMAGE.
36 1.1 thorpej */
37 1.1 thorpej
38 1.1 thorpej /*
39 1.1 thorpej * dkctl(8) -- a program to manipulate disks.
40 1.1 thorpej */
41 1.7 agc #include <sys/cdefs.h>
42 1.7 agc
43 1.7 agc #ifndef lint
44 1.14 dsl __RCSID("$NetBSD: dkctl.c,v 1.14 2006/03/17 19:18:33 dsl Exp $");
45 1.7 agc #endif
46 1.7 agc
47 1.1 thorpej
48 1.1 thorpej #include <sys/param.h>
49 1.1 thorpej #include <sys/ioctl.h>
50 1.1 thorpej #include <sys/dkio.h>
51 1.3 darrenr #include <sys/disk.h>
52 1.3 darrenr #include <sys/queue.h>
53 1.1 thorpej #include <err.h>
54 1.1 thorpej #include <errno.h>
55 1.1 thorpej #include <fcntl.h>
56 1.1 thorpej #include <stdio.h>
57 1.1 thorpej #include <stdlib.h>
58 1.1 thorpej #include <string.h>
59 1.1 thorpej #include <unistd.h>
60 1.1 thorpej #include <util.h>
61 1.1 thorpej
62 1.2 yamt #define YES 1
63 1.2 yamt #define NO 0
64 1.2 yamt
65 1.2 yamt /* I don't think nl_langinfo is suitable in this case */
66 1.2 yamt #define YES_STR "yes"
67 1.2 yamt #define NO_STR "no"
68 1.2 yamt #define YESNO_ARG YES_STR " | " NO_STR
69 1.2 yamt
70 1.3 darrenr #ifndef PRIdaddr
71 1.3 darrenr #define PRIdaddr PRId64
72 1.3 darrenr #endif
73 1.3 darrenr
74 1.1 thorpej struct command {
75 1.1 thorpej const char *cmd_name;
76 1.1 thorpej const char *arg_names;
77 1.1 thorpej void (*cmd_func)(int, char *[]);
78 1.1 thorpej int open_flags;
79 1.1 thorpej };
80 1.1 thorpej
81 1.1 thorpej void usage(void);
82 1.1 thorpej
83 1.1 thorpej int fd; /* file descriptor for device */
84 1.1 thorpej const char *dvname; /* device name */
85 1.1 thorpej char dvname_store[MAXPATHLEN]; /* for opendisk(3) */
86 1.1 thorpej const char *cmdname; /* command user issued */
87 1.1 thorpej const char *argnames; /* helpstring; expected arguments */
88 1.1 thorpej
89 1.2 yamt int yesno(const char *);
90 1.2 yamt
91 1.1 thorpej void disk_getcache(int, char *[]);
92 1.1 thorpej void disk_setcache(int, char *[]);
93 1.1 thorpej void disk_synccache(int, char *[]);
94 1.2 yamt void disk_keeplabel(int, char *[]);
95 1.3 darrenr void disk_badsectors(int, char *[]);
96 1.1 thorpej
97 1.9 thorpej void disk_addwedge(int, char *[]);
98 1.9 thorpej void disk_delwedge(int, char *[]);
99 1.9 thorpej void disk_getwedgeinfo(int, char *[]);
100 1.9 thorpej void disk_listwedges(int, char *[]);
101 1.12 yamt void disk_strategy(int, char *[]);
102 1.9 thorpej
103 1.1 thorpej struct command commands[] = {
104 1.1 thorpej { "getcache",
105 1.1 thorpej "",
106 1.1 thorpej disk_getcache,
107 1.1 thorpej O_RDONLY },
108 1.1 thorpej
109 1.1 thorpej { "setcache",
110 1.1 thorpej "none | r | w | rw [save]",
111 1.1 thorpej disk_setcache,
112 1.1 thorpej O_RDWR },
113 1.1 thorpej
114 1.1 thorpej { "synccache",
115 1.1 thorpej "[force]",
116 1.1 thorpej disk_synccache,
117 1.1 thorpej O_RDWR },
118 1.1 thorpej
119 1.2 yamt { "keeplabel",
120 1.2 yamt YESNO_ARG,
121 1.2 yamt disk_keeplabel,
122 1.2 yamt O_RDWR },
123 1.2 yamt
124 1.3 darrenr { "badsector",
125 1.3 darrenr "flush | list | retry",
126 1.3 darrenr disk_badsectors,
127 1.3 darrenr O_RDWR },
128 1.3 darrenr
129 1.9 thorpej { "addwedge",
130 1.9 thorpej "name startblk blkcnt ptype",
131 1.9 thorpej disk_addwedge,
132 1.9 thorpej O_RDWR },
133 1.9 thorpej
134 1.9 thorpej { "delwedge",
135 1.9 thorpej "dk",
136 1.9 thorpej disk_delwedge,
137 1.9 thorpej O_RDWR },
138 1.9 thorpej
139 1.9 thorpej { "getwedgeinfo",
140 1.9 thorpej "",
141 1.9 thorpej disk_getwedgeinfo,
142 1.9 thorpej O_RDONLY },
143 1.9 thorpej
144 1.9 thorpej { "listwedges",
145 1.9 thorpej "",
146 1.9 thorpej disk_listwedges,
147 1.9 thorpej O_RDONLY },
148 1.9 thorpej
149 1.12 yamt { "strategy",
150 1.12 yamt "[name]",
151 1.12 yamt disk_strategy,
152 1.12 yamt O_RDWR },
153 1.12 yamt
154 1.1 thorpej { NULL,
155 1.1 thorpej NULL,
156 1.1 thorpej NULL,
157 1.1 thorpej 0 },
158 1.1 thorpej };
159 1.1 thorpej
160 1.1 thorpej int
161 1.1 thorpej main(int argc, char *argv[])
162 1.1 thorpej {
163 1.1 thorpej int i;
164 1.1 thorpej
165 1.1 thorpej /* Must have at least: device command */
166 1.1 thorpej if (argc < 3)
167 1.1 thorpej usage();
168 1.1 thorpej
169 1.1 thorpej /* Skip program name, get and skip device name and command. */
170 1.1 thorpej dvname = argv[1];
171 1.1 thorpej cmdname = argv[2];
172 1.1 thorpej argv += 3;
173 1.1 thorpej argc -= 3;
174 1.1 thorpej
175 1.1 thorpej /* Look up and call the command. */
176 1.1 thorpej for (i = 0; commands[i].cmd_name != NULL; i++)
177 1.1 thorpej if (strcmp(cmdname, commands[i].cmd_name) == 0)
178 1.1 thorpej break;
179 1.1 thorpej if (commands[i].cmd_name == NULL)
180 1.1 thorpej errx(1, "unknown command: %s", cmdname);
181 1.1 thorpej
182 1.1 thorpej argnames = commands[i].arg_names;
183 1.1 thorpej
184 1.1 thorpej /* Open the device. */
185 1.1 thorpej fd = opendisk(dvname, commands[i].open_flags, dvname_store,
186 1.1 thorpej sizeof(dvname_store), 0);
187 1.1 thorpej if (fd == -1)
188 1.1 thorpej err(1, "%s", dvname);
189 1.1 thorpej
190 1.1 thorpej dvname = dvname_store;
191 1.1 thorpej
192 1.1 thorpej (*commands[i].cmd_func)(argc, argv);
193 1.1 thorpej exit(0);
194 1.1 thorpej }
195 1.1 thorpej
196 1.1 thorpej void
197 1.11 xtraeme usage(void)
198 1.1 thorpej {
199 1.1 thorpej int i;
200 1.1 thorpej
201 1.8 jmmv fprintf(stderr, "usage: %s device command [arg [...]]\n",
202 1.1 thorpej getprogname());
203 1.1 thorpej
204 1.1 thorpej fprintf(stderr, " Available commands:\n");
205 1.1 thorpej for (i = 0; commands[i].cmd_name != NULL; i++)
206 1.1 thorpej fprintf(stderr, "\t%s %s\n", commands[i].cmd_name,
207 1.1 thorpej commands[i].arg_names);
208 1.1 thorpej
209 1.1 thorpej exit(1);
210 1.1 thorpej }
211 1.1 thorpej
212 1.1 thorpej void
213 1.12 yamt disk_strategy(int argc, char *argv[])
214 1.12 yamt {
215 1.12 yamt struct disk_strategy odks;
216 1.12 yamt struct disk_strategy dks;
217 1.12 yamt
218 1.12 yamt memset(&dks, 0, sizeof(dks));
219 1.12 yamt if (ioctl(fd, DIOCGSTRATEGY, &odks) == -1) {
220 1.12 yamt err(EXIT_FAILURE, "%s: DIOCGSTRATEGY", dvname);
221 1.12 yamt }
222 1.12 yamt
223 1.12 yamt memset(&dks, 0, sizeof(dks));
224 1.12 yamt switch (argc) {
225 1.12 yamt case 0:
226 1.12 yamt /* show the buffer queue strategy used */
227 1.12 yamt printf("%s: %s\n", dvname, odks.dks_name);
228 1.12 yamt return;
229 1.12 yamt case 1:
230 1.12 yamt /* set the buffer queue strategy */
231 1.12 yamt strlcpy(dks.dks_name, argv[0], sizeof(dks.dks_name));
232 1.12 yamt if (ioctl(fd, DIOCSSTRATEGY, &dks) == -1) {
233 1.12 yamt err(EXIT_FAILURE, "%s: DIOCSSTRATEGY", dvname);
234 1.12 yamt }
235 1.12 yamt printf("%s: %s -> %s\n", dvname, odks.dks_name, argv[0]);
236 1.12 yamt break;
237 1.12 yamt default:
238 1.12 yamt usage();
239 1.12 yamt /* NOTREACHED */
240 1.12 yamt }
241 1.12 yamt }
242 1.12 yamt
243 1.12 yamt void
244 1.1 thorpej disk_getcache(int argc, char *argv[])
245 1.1 thorpej {
246 1.1 thorpej int bits;
247 1.1 thorpej
248 1.1 thorpej if (ioctl(fd, DIOCGCACHE, &bits) == -1)
249 1.1 thorpej err(1, "%s: getcache", dvname);
250 1.1 thorpej
251 1.1 thorpej if ((bits & (DKCACHE_READ|DKCACHE_WRITE)) == 0)
252 1.1 thorpej printf("%s: No caches enabled\n", dvname);
253 1.1 thorpej else {
254 1.1 thorpej if (bits & DKCACHE_READ)
255 1.1 thorpej printf("%s: read cache enabled\n", dvname);
256 1.1 thorpej if (bits & DKCACHE_WRITE)
257 1.1 thorpej printf("%s: write-back cache enabled\n", dvname);
258 1.1 thorpej }
259 1.1 thorpej
260 1.1 thorpej printf("%s: read cache enable is %schangeable\n", dvname,
261 1.1 thorpej (bits & DKCACHE_RCHANGE) ? "" : "not ");
262 1.1 thorpej printf("%s: write cache enable is %schangeable\n", dvname,
263 1.1 thorpej (bits & DKCACHE_WCHANGE) ? "" : "not ");
264 1.1 thorpej
265 1.1 thorpej printf("%s: cache parameters are %ssavable\n", dvname,
266 1.1 thorpej (bits & DKCACHE_SAVE) ? "" : "not ");
267 1.1 thorpej }
268 1.1 thorpej
269 1.1 thorpej void
270 1.1 thorpej disk_setcache(int argc, char *argv[])
271 1.1 thorpej {
272 1.1 thorpej int bits;
273 1.1 thorpej
274 1.1 thorpej if (argc > 2 || argc == 0)
275 1.1 thorpej usage();
276 1.1 thorpej
277 1.1 thorpej if (strcmp(argv[0], "none") == 0)
278 1.1 thorpej bits = 0;
279 1.1 thorpej else if (strcmp(argv[0], "r") == 0)
280 1.1 thorpej bits = DKCACHE_READ;
281 1.1 thorpej else if (strcmp(argv[0], "w") == 0)
282 1.1 thorpej bits = DKCACHE_WRITE;
283 1.1 thorpej else if (strcmp(argv[0], "rw") == 0)
284 1.1 thorpej bits = DKCACHE_READ|DKCACHE_WRITE;
285 1.1 thorpej else
286 1.1 thorpej usage();
287 1.1 thorpej
288 1.1 thorpej if (argc == 2) {
289 1.1 thorpej if (strcmp(argv[1], "save") == 0)
290 1.1 thorpej bits |= DKCACHE_SAVE;
291 1.1 thorpej else
292 1.1 thorpej usage();
293 1.1 thorpej }
294 1.1 thorpej
295 1.1 thorpej if (ioctl(fd, DIOCSCACHE, &bits) == -1)
296 1.1 thorpej err(1, "%s: setcache", dvname);
297 1.1 thorpej }
298 1.1 thorpej
299 1.1 thorpej void
300 1.1 thorpej disk_synccache(int argc, char *argv[])
301 1.1 thorpej {
302 1.1 thorpej int force;
303 1.1 thorpej
304 1.1 thorpej switch (argc) {
305 1.1 thorpej case 0:
306 1.1 thorpej force = 0;
307 1.1 thorpej break;
308 1.1 thorpej
309 1.1 thorpej case 1:
310 1.1 thorpej if (strcmp(argv[0], "force") == 0)
311 1.1 thorpej force = 1;
312 1.1 thorpej else
313 1.1 thorpej usage();
314 1.1 thorpej break;
315 1.1 thorpej
316 1.1 thorpej default:
317 1.1 thorpej usage();
318 1.1 thorpej }
319 1.1 thorpej
320 1.1 thorpej if (ioctl(fd, DIOCCACHESYNC, &force) == -1)
321 1.1 thorpej err(1, "%s: sync cache", dvname);
322 1.2 yamt }
323 1.2 yamt
324 1.2 yamt void
325 1.2 yamt disk_keeplabel(int argc, char *argv[])
326 1.2 yamt {
327 1.2 yamt int keep;
328 1.2 yamt int yn;
329 1.2 yamt
330 1.2 yamt if (argc != 1)
331 1.2 yamt usage();
332 1.2 yamt
333 1.2 yamt yn = yesno(argv[0]);
334 1.2 yamt if (yn < 0)
335 1.2 yamt usage();
336 1.2 yamt
337 1.2 yamt keep = yn == YES;
338 1.2 yamt
339 1.2 yamt if (ioctl(fd, DIOCKLABEL, &keep) == -1)
340 1.2 yamt err(1, "%s: keep label", dvname);
341 1.3 darrenr }
342 1.3 darrenr
343 1.3 darrenr
344 1.3 darrenr void
345 1.3 darrenr disk_badsectors(int argc, char *argv[])
346 1.3 darrenr {
347 1.3 darrenr struct disk_badsectors *dbs, *dbs2, buffer[200];
348 1.3 darrenr SLIST_HEAD(, disk_badsectors) dbstop;
349 1.3 darrenr struct disk_badsecinfo dbsi;
350 1.3 darrenr daddr_t blk, totbad, bad;
351 1.5 dsl u_int32_t count;
352 1.3 darrenr struct stat sb;
353 1.3 darrenr u_char *block;
354 1.6 martin time_t tm;
355 1.3 darrenr
356 1.3 darrenr if (argc != 1)
357 1.3 darrenr usage();
358 1.3 darrenr
359 1.3 darrenr if (strcmp(argv[0], "list") == 0) {
360 1.3 darrenr /*
361 1.3 darrenr * Copy the list of kernel bad sectors out in chunks that fit
362 1.3 darrenr * into buffer[]. Updating dbsi_skip means we don't sit here
363 1.3 darrenr * forever only getting the first chunk that fit in buffer[].
364 1.3 darrenr */
365 1.3 darrenr dbsi.dbsi_buffer = (caddr_t)buffer;
366 1.3 darrenr dbsi.dbsi_bufsize = sizeof(buffer);
367 1.3 darrenr dbsi.dbsi_skip = 0;
368 1.3 darrenr dbsi.dbsi_copied = 0;
369 1.3 darrenr dbsi.dbsi_left = 0;
370 1.3 darrenr
371 1.3 darrenr do {
372 1.3 darrenr if (ioctl(fd, DIOCBSLIST, (caddr_t)&dbsi) == -1)
373 1.3 darrenr err(1, "%s: badsectors list", dvname);
374 1.3 darrenr
375 1.3 darrenr dbs = (struct disk_badsectors *)dbsi.dbsi_buffer;
376 1.3 darrenr for (count = dbsi.dbsi_copied; count > 0; count--) {
377 1.6 martin tm = dbs->dbs_failedat.tv_sec;
378 1.5 dsl printf("%s: blocks %" PRIdaddr " - %" PRIdaddr " failed at %s",
379 1.3 darrenr dvname, dbs->dbs_min, dbs->dbs_max,
380 1.6 martin ctime(&tm));
381 1.4 darrenr dbs++;
382 1.3 darrenr }
383 1.3 darrenr dbsi.dbsi_skip += dbsi.dbsi_copied;
384 1.3 darrenr } while (dbsi.dbsi_left != 0);
385 1.3 darrenr
386 1.3 darrenr } else if (strcmp(argv[0], "flush") == 0) {
387 1.3 darrenr if (ioctl(fd, DIOCBSFLUSH) == -1)
388 1.3 darrenr err(1, "%s: badsectors flush", dvname);
389 1.3 darrenr
390 1.3 darrenr } else if (strcmp(argv[0], "retry") == 0) {
391 1.3 darrenr /*
392 1.3 darrenr * Enforce use of raw device here because the block device
393 1.3 darrenr * causes access to blocks to be clustered in a larger group,
394 1.3 darrenr * making it impossible to determine which individual sectors
395 1.3 darrenr * are the cause of a problem.
396 1.3 darrenr */
397 1.3 darrenr if (fstat(fd, &sb) == -1)
398 1.3 darrenr err(1, "fstat");
399 1.3 darrenr
400 1.3 darrenr if (!S_ISCHR(sb.st_mode)) {
401 1.3 darrenr fprintf(stderr, "'badsector retry' must be used %s\n",
402 1.3 darrenr "with character device");
403 1.3 darrenr exit(1);
404 1.3 darrenr }
405 1.3 darrenr
406 1.3 darrenr SLIST_INIT(&dbstop);
407 1.3 darrenr
408 1.3 darrenr /*
409 1.3 darrenr * Build up a copy of the in-kernel list in a number of stages.
410 1.3 darrenr * That the list we build up here is in the reverse order to
411 1.3 darrenr * the kernel's is of no concern.
412 1.3 darrenr */
413 1.3 darrenr dbsi.dbsi_buffer = (caddr_t)buffer;
414 1.3 darrenr dbsi.dbsi_bufsize = sizeof(buffer);
415 1.3 darrenr dbsi.dbsi_skip = 0;
416 1.3 darrenr dbsi.dbsi_copied = 0;
417 1.3 darrenr dbsi.dbsi_left = 0;
418 1.3 darrenr
419 1.3 darrenr do {
420 1.3 darrenr if (ioctl(fd, DIOCBSLIST, (caddr_t)&dbsi) == -1)
421 1.3 darrenr err(1, "%s: badsectors list", dvname);
422 1.3 darrenr
423 1.3 darrenr dbs = (struct disk_badsectors *)dbsi.dbsi_buffer;
424 1.3 darrenr for (count = dbsi.dbsi_copied; count > 0; count--) {
425 1.14 dsl dbs2 = malloc(sizeof *dbs2);
426 1.14 dsl if (dbs2 == NULL)
427 1.13 rumble err(1, NULL);
428 1.3 darrenr *dbs2 = *dbs;
429 1.3 darrenr SLIST_INSERT_HEAD(&dbstop, dbs2, dbs_next);
430 1.4 darrenr dbs++;
431 1.3 darrenr }
432 1.3 darrenr dbsi.dbsi_skip += dbsi.dbsi_copied;
433 1.3 darrenr } while (dbsi.dbsi_left != 0);
434 1.3 darrenr
435 1.3 darrenr /*
436 1.3 darrenr * Just calculate and print out something that will hopefully
437 1.3 darrenr * provide some useful information about what's going to take
438 1.3 darrenr * place next (if anything.)
439 1.3 darrenr */
440 1.3 darrenr bad = 0;
441 1.3 darrenr totbad = 0;
442 1.13 rumble if ((block = calloc(1, DEV_BSIZE)) == NULL)
443 1.13 rumble err(1, NULL);
444 1.3 darrenr SLIST_FOREACH(dbs, &dbstop, dbs_next) {
445 1.3 darrenr bad++;
446 1.3 darrenr totbad += dbs->dbs_max - dbs->dbs_min + 1;
447 1.3 darrenr }
448 1.3 darrenr
449 1.3 darrenr printf("%s: bad sector clusters %"PRIdaddr
450 1.3 darrenr " total sectors %"PRIdaddr"\n", dvname, bad, totbad);
451 1.3 darrenr
452 1.3 darrenr /*
453 1.3 darrenr * Clear out the kernel's list of bad sectors, ready for us
454 1.3 darrenr * to test all those it thought were bad.
455 1.3 darrenr */
456 1.3 darrenr if (ioctl(fd, DIOCBSFLUSH) == -1)
457 1.3 darrenr err(1, "%s: badsectors flush", dvname);
458 1.3 darrenr
459 1.3 darrenr printf("%s: bad sectors flushed\n", dvname);
460 1.3 darrenr
461 1.3 darrenr /*
462 1.3 darrenr * For each entry we obtained from the kernel, retry each
463 1.3 darrenr * individual sector recorded as bad by seeking to it and
464 1.3 darrenr * attempting to read it in. Print out a line item for each
465 1.3 darrenr * bad block we verify.
466 1.3 darrenr *
467 1.3 darrenr * PRIdaddr is used here because the type of dbs_max is daddr_t
468 1.3 darrenr * and that may be either a 32bit or 64bit number(!)
469 1.3 darrenr */
470 1.3 darrenr SLIST_FOREACH(dbs, &dbstop, dbs_next) {
471 1.3 darrenr printf("%s: Retrying %"PRIdaddr" - %"
472 1.3 darrenr PRIdaddr"\n", dvname, dbs->dbs_min, dbs->dbs_max);
473 1.3 darrenr
474 1.3 darrenr for (blk = dbs->dbs_min; blk <= dbs->dbs_max; blk++) {
475 1.3 darrenr if (lseek(fd, (off_t)blk * DEV_BSIZE,
476 1.3 darrenr SEEK_SET) == -1) {
477 1.5 dsl warn("%s: lseek block %" PRIdaddr "",
478 1.5 dsl dvname, blk);
479 1.3 darrenr continue;
480 1.3 darrenr }
481 1.3 darrenr printf("%s: block %"PRIdaddr" - ", dvname, blk);
482 1.3 darrenr if (read(fd, block, DEV_BSIZE) != DEV_BSIZE)
483 1.3 darrenr printf("failed\n");
484 1.3 darrenr else
485 1.3 darrenr printf("ok\n");
486 1.3 darrenr fflush(stdout);
487 1.3 darrenr }
488 1.3 darrenr }
489 1.3 darrenr }
490 1.2 yamt }
491 1.2 yamt
492 1.9 thorpej void
493 1.9 thorpej disk_addwedge(int argc, char *argv[])
494 1.9 thorpej {
495 1.9 thorpej struct dkwedge_info dkw;
496 1.9 thorpej char *cp;
497 1.9 thorpej daddr_t start;
498 1.9 thorpej uint64_t size;
499 1.9 thorpej
500 1.9 thorpej if (argc != 4)
501 1.9 thorpej usage();
502 1.9 thorpej
503 1.9 thorpej /* XXX Unicode. */
504 1.9 thorpej if (strlen(argv[0]) > sizeof(dkw.dkw_wname) - 1)
505 1.9 thorpej errx(1, "Wedge name too long; max %zd characters",
506 1.9 thorpej sizeof(dkw.dkw_wname) - 1);
507 1.9 thorpej strcpy(dkw.dkw_wname, argv[0]);
508 1.9 thorpej
509 1.9 thorpej if (strlen(argv[3]) > sizeof(dkw.dkw_ptype) - 1)
510 1.9 thorpej errx(1, "Wedge partition type too long; max %zd characters",
511 1.9 thorpej sizeof(dkw.dkw_ptype) - 1);
512 1.9 thorpej strcpy(dkw.dkw_ptype, argv[3]);
513 1.9 thorpej
514 1.9 thorpej errno = 0;
515 1.9 thorpej start = strtoll(argv[1], &cp, 0);
516 1.9 thorpej if (*cp != '\0')
517 1.9 thorpej errx(1, "Invalid start block: %s", argv[1]);
518 1.9 thorpej if (errno == ERANGE && (start == LLONG_MAX ||
519 1.9 thorpej start == LLONG_MIN))
520 1.9 thorpej errx(1, "Start block out of range.");
521 1.9 thorpej if (start < 0)
522 1.9 thorpej errx(1, "Start block must be >= 0.");
523 1.9 thorpej
524 1.9 thorpej errno = 0;
525 1.9 thorpej size = strtoull(argv[2], &cp, 0);
526 1.9 thorpej if (*cp != '\0')
527 1.9 thorpej errx(1, "Invalid block count: %s", argv[2]);
528 1.9 thorpej if (errno == ERANGE && (size == ULLONG_MAX))
529 1.9 thorpej errx(1, "Block count out of range.");
530 1.9 thorpej
531 1.9 thorpej dkw.dkw_offset = start;
532 1.9 thorpej dkw.dkw_size = size;
533 1.9 thorpej
534 1.9 thorpej if (ioctl(fd, DIOCAWEDGE, &dkw) == -1)
535 1.9 thorpej err(1, "%s: addwedge", dvname);
536 1.9 thorpej }
537 1.9 thorpej
538 1.9 thorpej void
539 1.9 thorpej disk_delwedge(int argc, char *argv[])
540 1.9 thorpej {
541 1.9 thorpej struct dkwedge_info dkw;
542 1.9 thorpej
543 1.9 thorpej if (argc != 1)
544 1.9 thorpej usage();
545 1.9 thorpej
546 1.9 thorpej if (strlen(argv[0]) > sizeof(dkw.dkw_devname) - 1)
547 1.9 thorpej errx(1, "Wedge dk name too long; max %zd characters",
548 1.9 thorpej sizeof(dkw.dkw_devname) - 1);
549 1.9 thorpej strcpy(dkw.dkw_devname, argv[0]);
550 1.9 thorpej
551 1.9 thorpej if (ioctl(fd, DIOCDWEDGE, &dkw) == -1)
552 1.9 thorpej err(1, "%s: delwedge", dvname);
553 1.9 thorpej }
554 1.9 thorpej
555 1.9 thorpej void
556 1.9 thorpej disk_getwedgeinfo(int argc, char *argv[])
557 1.9 thorpej {
558 1.9 thorpej struct dkwedge_info dkw;
559 1.9 thorpej
560 1.9 thorpej if (argc != 0)
561 1.9 thorpej usage();
562 1.9 thorpej
563 1.9 thorpej if (ioctl(fd, DIOCGWEDGEINFO, &dkw) == -1)
564 1.9 thorpej err(1, "%s: getwedgeinfo", dvname);
565 1.9 thorpej
566 1.9 thorpej printf("%s at %s: %s\n", dkw.dkw_devname, dkw.dkw_parent,
567 1.9 thorpej dkw.dkw_wname); /* XXX Unicode */
568 1.10 martin printf("%s: %"PRIu64" blocks at %"PRId64", type: %s\n",
569 1.9 thorpej dkw.dkw_devname, dkw.dkw_size, dkw.dkw_offset, dkw.dkw_ptype);
570 1.9 thorpej }
571 1.9 thorpej
572 1.9 thorpej void
573 1.9 thorpej disk_listwedges(int argc, char *argv[])
574 1.9 thorpej {
575 1.9 thorpej struct dkwedge_info *dkw;
576 1.9 thorpej struct dkwedge_list dkwl;
577 1.9 thorpej size_t bufsize;
578 1.9 thorpej u_int i;
579 1.9 thorpej
580 1.9 thorpej if (argc != 0)
581 1.9 thorpej usage();
582 1.9 thorpej
583 1.9 thorpej dkw = NULL;
584 1.9 thorpej dkwl.dkwl_buf = dkw;
585 1.9 thorpej dkwl.dkwl_bufsize = 0;
586 1.9 thorpej
587 1.9 thorpej for (;;) {
588 1.9 thorpej if (ioctl(fd, DIOCLWEDGES, &dkwl) == -1)
589 1.9 thorpej err(1, "%s: listwedges", dvname);
590 1.9 thorpej if (dkwl.dkwl_nwedges == dkwl.dkwl_ncopied)
591 1.9 thorpej break;
592 1.9 thorpej bufsize = dkwl.dkwl_nwedges * sizeof(*dkw);
593 1.9 thorpej if (dkwl.dkwl_bufsize < bufsize) {
594 1.9 thorpej dkw = realloc(dkwl.dkwl_buf, bufsize);
595 1.9 thorpej if (dkw == NULL)
596 1.9 thorpej errx(1, "%s: listwedges: unable to "
597 1.9 thorpej "allocate wedge info buffer", dvname);
598 1.9 thorpej dkwl.dkwl_buf = dkw;
599 1.9 thorpej dkwl.dkwl_bufsize = bufsize;
600 1.9 thorpej }
601 1.9 thorpej }
602 1.9 thorpej
603 1.9 thorpej if (dkwl.dkwl_nwedges == 0) {
604 1.9 thorpej printf("%s: no wedges configured\n", dvname);
605 1.9 thorpej return;
606 1.9 thorpej }
607 1.9 thorpej
608 1.9 thorpej printf("%s: %u wedge%s:\n", dvname, dkwl.dkwl_nwedges,
609 1.9 thorpej dkwl.dkwl_nwedges == 1 ? "" : "s");
610 1.9 thorpej for (i = 0; i < dkwl.dkwl_nwedges; i++) {
611 1.10 martin printf("%s: %s, %"PRIu64" blocks at %"PRId64", type: %s\n",
612 1.9 thorpej dkw[i].dkw_devname,
613 1.9 thorpej dkw[i].dkw_wname, /* XXX Unicode */
614 1.9 thorpej dkw[i].dkw_size, dkw[i].dkw_offset, dkw[i].dkw_ptype);
615 1.9 thorpej }
616 1.9 thorpej }
617 1.9 thorpej
618 1.2 yamt /*
619 1.2 yamt * return YES, NO or -1.
620 1.2 yamt */
621 1.2 yamt int
622 1.2 yamt yesno(const char *p)
623 1.2 yamt {
624 1.2 yamt
625 1.2 yamt if (!strcmp(p, YES_STR))
626 1.2 yamt return YES;
627 1.2 yamt if (!strcmp(p, NO_STR))
628 1.2 yamt return NO;
629 1.2 yamt return -1;
630 1.1 thorpej }
631