]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - sbin/gpt/create.c
style(9) sweep.
[FreeBSD/FreeBSD.git] / sbin / gpt / create.c
1 /*
2  * Copyright (c) 2002 Marcel Moolenaar
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  *
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
16  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
17  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
18  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
19  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
20  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
24  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25  *
26  * $FreeBSD$
27  */
28
29 #include <sys/types.h>
30 #include <sys/gpt.h>
31
32 #include <err.h>
33 #include <stddef.h>
34 #include <stdio.h>
35 #include <stdlib.h>
36 #include <string.h>
37 #include <unistd.h>
38 #include <uuid.h>
39
40 #include "map.h"
41 #include "gpt.h"
42
43 int primary_only;
44
45 static void
46 usage_create(void)
47 {
48
49         fprintf(stderr,
50             "usage: %s [-p] device ...\n", getprogname());
51         exit(1);
52 }
53
54 static void
55 create(int fd)
56 {
57         off_t blocks, last;
58         map_t *gpt, *tpg;
59         map_t *tbl, *lbt;
60         map_t *map;
61         struct mbr *mbr;
62         struct gpt_hdr *hdr;
63         struct gpt_ent *ent;
64         unsigned int i;
65
66         if (map_find(MAP_TYPE_PRI_GPT_HDR) != NULL ||
67             map_find(MAP_TYPE_SEC_GPT_HDR) != NULL) {
68                 warnx("%s: error: device already contains a GPT", device_name);
69                 return;
70         }
71         if (map_find(MAP_TYPE_MBR) != NULL) {
72                 warnx("%s: error: device contains a MBR", device_name);
73                 return;
74         }
75
76         /*
77          * Create PMBR.
78          */
79         if (map_find(MAP_TYPE_PMBR) == NULL) {
80                 if (map_free(0LL, 1LL) == 0) {
81                         warnx("%s: error: no room for the PMBR", device_name);
82                         return;
83                 }
84                 mbr = gpt_read(fd, 0LL, 1);
85                 bzero(mbr, sizeof(*mbr));
86                 mbr->mbr_sig = MBR_SIG;
87                 mbr->mbr_part[0].part_shd = 0xff;
88                 mbr->mbr_part[0].part_ssect = 0xff;
89                 mbr->mbr_part[0].part_scyl = 0xff;
90                 mbr->mbr_part[0].part_typ = 0xee;
91                 mbr->mbr_part[0].part_ehd = 0xff;
92                 mbr->mbr_part[0].part_esect = 0xff;
93                 mbr->mbr_part[0].part_ecyl = 0xff;
94                 mbr->mbr_part[0].part_start_lo = 1;
95                 if (mediasz > 0xffffffff) {
96                         mbr->mbr_part[0].part_size_lo = 0xffff;
97                         mbr->mbr_part[0].part_size_hi = 0xffff;
98                 } else {
99                         mbr->mbr_part[0].part_size_lo = mediasz & 0xffff;
100                         mbr->mbr_part[0].part_size_hi = mediasz >> 16;
101                 }
102                 map = map_add(0LL, 1LL, MAP_TYPE_PMBR, mbr);
103                 gpt_write(fd, map);
104         }
105
106         /* Get the amount of free space after the MBR */
107         blocks = map_free(1LL, 0LL);
108         if (blocks == 0LL) {
109                 warnx("%s: error: no room for the GPT header", device_name);
110                 return;
111         }
112
113         /* Don't create more than parts entries. */
114         if ((uint64_t)(blocks - 1) * secsz > parts * sizeof(struct gpt_ent)) {
115                 blocks = (parts * sizeof(struct gpt_ent)) / secsz;
116                 if ((parts * sizeof(struct gpt_ent)) % secsz)
117                         blocks++;
118                 blocks++;               /* Don't forget the header itself */
119         }
120
121         last = mediasz / secsz - 1LL;
122
123         /* Never cross the median of the device. */
124         if ((blocks + 1LL) > ((last + 1LL) >> 1))
125                 blocks = ((last + 1LL) >> 1) - 1LL;
126
127         /*
128          * Get the amount of free space at the end of the device and
129          * calculate the size for the GPT structures.
130          */
131         map = map_last();
132         if (map->map_type != MAP_TYPE_UNUSED) {
133                 warnx("%s: error: no room for the backup header", device_name);
134                 return;
135         }
136
137         if (map->map_size < blocks)
138                 blocks = map->map_size;
139         if (blocks == 1LL) {
140                 warnx("%s: error: no room for the GPT table", device_name);
141                 return;
142         }
143
144         blocks--;               /* Number of blocks in the GPT table. */
145         gpt = map_add(1LL, 1LL, MAP_TYPE_PRI_GPT_HDR, calloc(1, secsz));
146         tbl = map_add(2LL, blocks, MAP_TYPE_PRI_GPT_TBL,
147             calloc(blocks, secsz));
148         if (gpt == NULL || tbl == NULL)
149                 return;
150
151         hdr = gpt->map_data;
152         memcpy(hdr->hdr_sig, GPT_HDR_SIG, sizeof(hdr->hdr_sig));
153         hdr->hdr_revision = GPT_HDR_REVISION;
154         /*
155          * XXX struct gpt_hdr is not a multiple of 8 bytes in size and thus
156          * contains padding we must not include in the size.
157          */
158         hdr->hdr_size = offsetof(struct gpt_hdr, padding);
159         hdr->hdr_lba_self = gpt->map_start;
160         hdr->hdr_lba_alt = last;
161         hdr->hdr_lba_start = tbl->map_start + blocks;
162         hdr->hdr_lba_end = last - blocks - 1LL;
163         uuid_create(&hdr->hdr_uuid, NULL);
164         hdr->hdr_lba_table = tbl->map_start;
165         hdr->hdr_entries = (blocks * secsz) / sizeof(struct gpt_ent);
166         if (hdr->hdr_entries > parts)
167                 hdr->hdr_entries = parts;
168         hdr->hdr_entsz = sizeof(struct gpt_ent);
169
170         ent = tbl->map_data;
171         for (i = 0; i < hdr->hdr_entries; i++)
172                 uuid_create(&ent[i].ent_uuid, NULL);
173
174         hdr->hdr_crc_table = crc32(ent, hdr->hdr_entries * hdr->hdr_entsz);
175         hdr->hdr_crc_self = crc32(hdr, hdr->hdr_size);
176
177         gpt_write(fd, gpt);
178         gpt_write(fd, tbl);
179
180         /*
181          * Create backup GPT if the user didn't suppress it.
182          */
183         if (!primary_only) {
184                 tpg = map_add(last, 1LL, MAP_TYPE_SEC_GPT_HDR,
185                     calloc(1, secsz));
186                 lbt = map_add(last - blocks, blocks, MAP_TYPE_SEC_GPT_TBL,
187                     tbl->map_data);
188                 memcpy(tpg->map_data, gpt->map_data, secsz);
189                 hdr = tpg->map_data;
190                 hdr->hdr_lba_self = tpg->map_start;
191                 hdr->hdr_lba_alt = gpt->map_start;
192                 hdr->hdr_lba_table = lbt->map_start;
193                 hdr->hdr_crc_self = 0;          /* Don't ever forget this! */
194                 hdr->hdr_crc_self = crc32(hdr, hdr->hdr_size);
195                 gpt_write(fd, lbt);
196                 gpt_write(fd, tpg);
197         }
198 }
199
200 int
201 cmd_create(int argc, char *argv[])
202 {
203         int ch, fd;
204
205         while ((ch = getopt(argc, argv, "p")) != -1) {
206                 switch(ch) {
207                 case 'p':
208                         primary_only = 1;
209                         break;
210                 default:
211                         usage_create();
212                 }
213         }
214
215         if (argc == optind)
216                 usage_create();
217
218         while (optind < argc) {
219                 fd = gpt_open(argv[optind++]);
220                 if (fd == -1) {
221                         warn("unable to open device '%s'", device_name);
222                         continue;
223                 }
224
225                 create(fd);
226
227                 gpt_close(fd);
228         }
229
230         return (0);
231 }