sbin/hammer: Cleanup blocks with a single statement
[dragonfly.git] / sbin / hammer / cmd_reblock.c
blob78ed812bedbea811ecc0135ddde78c3bc99d17b4
1 /*
2 * Copyright (c) 2008 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Matthew Dillon <dillon@backplane.com>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
16 * distribution.
17 * 3. Neither the name of The DragonFly Project nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific, prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 * SUCH DAMAGE.
34 * $DragonFly: src/sbin/hammer/cmd_reblock.c,v 1.11 2008/07/16 00:53:48 thomas Exp $
37 #include "hammer.h"
39 static void reblock_usage(int exit_code);
42 * reblock <filesystem> [fill_percentage] (default 100%)
44 void
45 hammer_cmd_reblock(char **av, int ac, int flags)
47 struct hammer_ioc_reblock reblock;
48 const char *filesystem;
49 int fd;
50 int perc;
52 if (TimeoutOpt > 0)
53 alarm(TimeoutOpt);
55 bzero(&reblock, sizeof(reblock));
57 reblock.key_beg.obj_id = HAMMER_MIN_OBJID;
58 reblock.key_end.obj_id = HAMMER_MAX_OBJID;
59 hammer_get_cycle(&reblock.key_beg, NULL);
61 reblock.head.flags = flags & HAMMER_IOC_DO_FLAGS;
62 reblock.allpfs = AllPFS;
63 reblock.vol_no = -1; /* Don't select a volume */
66 * Restrict the localization domain if asked to do inodes or data,
67 * but not both. Note that using INODE or MISC for localization
68 * adds a limitation to reblock range of B-Tree node, if BTREE is
69 * used with INODES or (DATA and/or DIRS) but not both.
71 switch(flags & (HAMMER_IOC_DO_INODES|HAMMER_IOC_DO_DATA|HAMMER_IOC_DO_DIRS)) {
72 case HAMMER_IOC_DO_INODES:
73 reblock.key_beg.localization = HAMMER_LOCALIZE_INODE;
74 reblock.key_end.localization = HAMMER_LOCALIZE_INODE;
75 break;
76 case HAMMER_IOC_DO_DATA:
77 case HAMMER_IOC_DO_DIRS:
78 case HAMMER_IOC_DO_DATA | HAMMER_IOC_DO_DIRS:
79 reblock.key_beg.localization = HAMMER_LOCALIZE_MISC;
80 reblock.key_end.localization = HAMMER_LOCALIZE_MISC;
81 break;
82 default:
83 reblock.key_beg.localization = HAMMER_MIN_LOCALIZATION;
84 reblock.key_end.localization = HAMMER_MAX_LOCALIZATION;
85 break;
88 if (ac == 0)
89 reblock_usage(1);
90 filesystem = av[0];
91 if (ac == 1) {
92 perc = 100;
93 } else {
94 perc = strtol(av[1], NULL, 0);
95 if (perc < 0 || perc > 100)
96 reblock_usage(1);
98 reblock.free_level = (int)((int64_t)perc *
99 HAMMER_BIGBLOCK_SIZE / 100);
100 reblock.free_level = HAMMER_BIGBLOCK_SIZE - reblock.free_level;
101 if (reblock.free_level < 0)
102 reblock.free_level = 0;
103 printf("reblock start %016jx:%04x\nfree level %d/%d\n",
104 (uintmax_t)reblock.key_beg.obj_id,
105 reblock.key_beg.localization,
106 reblock.free_level,
107 HAMMER_BIGBLOCK_SIZE);
109 fd = open(filesystem, O_RDONLY);
110 if (fd < 0)
111 err(1, "Unable to open %s", filesystem);
112 RunningIoctl = 1;
113 if (ioctl(fd, HAMMERIOC_REBLOCK, &reblock) < 0) {
114 printf("Reblock %s failed: %s\n", filesystem, strerror(errno));
115 } else if (reblock.head.flags & HAMMER_IOC_HEAD_INTR) {
116 printf("Reblock %s interrupted by timer at %016jx:%04x\n",
117 filesystem,
118 (uintmax_t)reblock.key_cur.obj_id,
119 reblock.key_cur.localization);
120 if (CyclePath)
121 hammer_set_cycle(&reblock.key_cur, 0);
122 } else {
123 if (CyclePath)
124 hammer_reset_cycle();
125 printf("Reblock %s succeeded\n", filesystem);
127 RunningIoctl = 0;
128 close(fd);
129 printf("Reblocked:\n"
130 " %jd/%jd B-Tree nodes\n"
131 " %jd/%jd B-Tree bytes\n"
132 " %jd/%jd data elements\n"
133 " %jd/%jd data bytes\n",
134 (intmax_t)reblock.btree_moves, (intmax_t)reblock.btree_count,
135 (intmax_t)(reblock.btree_moves *
136 sizeof(struct hammer_node_ondisk)),
137 (intmax_t)(reblock.btree_count *
138 sizeof(struct hammer_node_ondisk)),
139 (intmax_t)reblock.data_moves, (intmax_t)reblock.data_count,
140 (intmax_t)reblock.data_byte_moves,
141 (intmax_t)reblock.data_byte_count
145 static
146 void
147 reblock_usage(int exit_code)
149 fprintf(stderr, "hammer reblock <filesystem> [fill_percentage]\n");
150 fprintf(stderr, "hammer reblock-btree <filesystem> [fill_percentage]\n");
151 fprintf(stderr, "hammer reblock-inodes <filesystem> [fill_percentage]\n");
152 fprintf(stderr, "hammer reblock-dirs <filesystem> [fill_percentage]\n");
153 fprintf(stderr, "hammer reblock-data <filesystem> [fill_percentage]\n");
154 fprintf(stderr, "By default 100%% is used.\n");
155 exit(exit_code);