xref: /netbsd-src/sbin/apmlabel/apmlabel.c (revision 3e4993b3960c8a6cbda5929ab7f95887d20ce18c)
1*3e4993b3Schristos /*	$NetBSD: apmlabel.c,v 1.3 2013/10/19 01:09:58 christos Exp $	*/
2eaf2e0beSdillo 
3eaf2e0beSdillo /*
4eaf2e0beSdillo  * Copyright (C) 1998 Wolfgang Solfrank.
5eaf2e0beSdillo  * Copyright (C) 1998 TooLs GmbH.
6eaf2e0beSdillo  * All rights reserved.
7eaf2e0beSdillo  *
8eaf2e0beSdillo  * Redistribution and use in source and binary forms, with or without
9eaf2e0beSdillo  * modification, are permitted provided that the following conditions
10eaf2e0beSdillo  * are met:
11eaf2e0beSdillo  * 1. Redistributions of source code must retain the above copyright
12eaf2e0beSdillo  *    notice, this list of conditions and the following disclaimer.
13eaf2e0beSdillo  * 2. Redistributions in binary form must reproduce the above copyright
14eaf2e0beSdillo  *    notice, this list of conditions and the following disclaimer in the
15eaf2e0beSdillo  *    documentation and/or other materials provided with the distribution.
16eaf2e0beSdillo  * 3. All advertising materials mentioning features or use of this software
17eaf2e0beSdillo  *    must display the following acknowledgement:
18eaf2e0beSdillo  *	This product includes software developed by TooLs GmbH.
19eaf2e0beSdillo  * 4. The name of TooLs GmbH may not be used to endorse or promote products
20eaf2e0beSdillo  *    derived from this software without specific prior written permission.
21eaf2e0beSdillo  *
22eaf2e0beSdillo  * THIS SOFTWARE IS PROVIDED BY TOOLS GMBH ``AS IS'' AND ANY EXPRESS OR
23eaf2e0beSdillo  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
24eaf2e0beSdillo  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
25eaf2e0beSdillo  * IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26eaf2e0beSdillo  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
27eaf2e0beSdillo  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
28eaf2e0beSdillo  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
29eaf2e0beSdillo  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
30eaf2e0beSdillo  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
31eaf2e0beSdillo  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32eaf2e0beSdillo  */
33eaf2e0beSdillo 
34eaf2e0beSdillo #include <sys/cdefs.h>
35eaf2e0beSdillo #ifndef lint
36*3e4993b3Schristos __RCSID("$NetBSD: apmlabel.c,v 1.3 2013/10/19 01:09:58 christos Exp $");
37eaf2e0beSdillo #endif /* not lint */
38eaf2e0beSdillo 
39eaf2e0beSdillo #include <stdio.h>
40eaf2e0beSdillo #include <err.h>
41eaf2e0beSdillo #include <errno.h>
42eaf2e0beSdillo #include <fcntl.h>
43eaf2e0beSdillo #include <limits.h>
44eaf2e0beSdillo #include <stdlib.h>
45eaf2e0beSdillo #include <string.h>
46eaf2e0beSdillo #include <unistd.h>
47eaf2e0beSdillo #include <util.h>
48eaf2e0beSdillo 
49eaf2e0beSdillo #include <sys/param.h>
50eaf2e0beSdillo #define FSTYPENAMES
51eaf2e0beSdillo #include <sys/disklabel.h>
52eaf2e0beSdillo #include <sys/bootblock.h>
53eaf2e0beSdillo #include <sys/ioctl.h>
54eaf2e0beSdillo 
55eaf2e0beSdillo #include "dkcksum.h"
56eaf2e0beSdillo #include "extern.h"
57eaf2e0beSdillo 
581cd05199Sjoerg __dead static void	usage(void);
591cd05199Sjoerg static void	getlabel(int);
601cd05199Sjoerg static void	setlabel(int, int);
611cd05199Sjoerg static int	getparts(int, int);
621cd05199Sjoerg static struct apple_drvr_map *convert_drvr_map(unsigned char *);
631cd05199Sjoerg static struct apple_part_map_entry *convert_part_map_entry(unsigned char *);
64eaf2e0beSdillo 
651cd05199Sjoerg static struct disklabel label;
66eaf2e0beSdillo 
671cd05199Sjoerg static void
getlabel(int sd)68eaf2e0beSdillo getlabel(int sd)
69eaf2e0beSdillo {
70eaf2e0beSdillo 
71eaf2e0beSdillo 	if (ioctl(sd, DIOCGDINFO, &label) < 0) {
72eaf2e0beSdillo 		perror("get label");
73eaf2e0beSdillo 		exit(1);
74eaf2e0beSdillo 	}
75eaf2e0beSdillo 	/*
76eaf2e0beSdillo 	 * Some ports seem to not set the number of partitions
77eaf2e0beSdillo 	 * correctly, albeit they seem to set the raw partition ok!
78eaf2e0beSdillo 	 */
79eaf2e0beSdillo 	if (label.d_npartitions <= getrawpartition())
80eaf2e0beSdillo 		label.d_npartitions = getrawpartition() + 1;
81eaf2e0beSdillo }
82eaf2e0beSdillo 
831cd05199Sjoerg static void
setlabel(int sd,int doraw)84eaf2e0beSdillo setlabel(int sd, int doraw)
85eaf2e0beSdillo {
86eaf2e0beSdillo 	int one = 1;
87eaf2e0beSdillo 
88eaf2e0beSdillo 	label.d_checksum = 0;
89eaf2e0beSdillo 	label.d_checksum = dkcksum(&label);
90eaf2e0beSdillo 	if (ioctl(sd, doraw ? DIOCWDINFO : DIOCSDINFO, &label) < 0) {
91eaf2e0beSdillo 		perror("set label");
92eaf2e0beSdillo 		exit(1);
93eaf2e0beSdillo 	}
94eaf2e0beSdillo 	if (!doraw)
95eaf2e0beSdillo 		/* If we haven't written to the disk, don't discard on close */
96eaf2e0beSdillo 		ioctl(sd, DIOCKLABEL, &one);
97eaf2e0beSdillo 
98eaf2e0beSdillo }
99eaf2e0beSdillo 
1001cd05199Sjoerg static int
getparts(int sd,int verbose)101eaf2e0beSdillo getparts(int sd, int verbose)
102eaf2e0beSdillo {
103eaf2e0beSdillo 	unsigned char		buf[DEV_BSIZE];
104eaf2e0beSdillo 	struct apple_drvr_map	*drvr;
105eaf2e0beSdillo 	struct apple_part_map_entry *part;
106eaf2e0beSdillo 	struct partition	npe;
107*3e4993b3Schristos 	uint16_t		blksize, partcnt;
108eaf2e0beSdillo 	int			i, j, unused, changed;
109eaf2e0beSdillo 	uint64_t		temp;
110eaf2e0beSdillo 
111eaf2e0beSdillo 	changed = 0;
112eaf2e0beSdillo 
113eaf2e0beSdillo 	if (lseek(sd, 0, SEEK_SET) == -1) {
114eaf2e0beSdillo 		perror("seek drvr map");
115eaf2e0beSdillo 		exit(1);
116eaf2e0beSdillo 	}
117eaf2e0beSdillo 	if ((i=read(sd, buf, sizeof(buf))) != DEV_BSIZE) {
118eaf2e0beSdillo 		perror("read drvr map");
119eaf2e0beSdillo 		exit(1);
120eaf2e0beSdillo 	}
121eaf2e0beSdillo 	drvr = convert_drvr_map(buf);
122eaf2e0beSdillo 
123eaf2e0beSdillo 	if (drvr->sbSig != APPLE_DRVR_MAP_MAGIC)
124eaf2e0beSdillo 		return (changed);
125eaf2e0beSdillo 	blksize = drvr->sbBlockSize;
126eaf2e0beSdillo 
127eaf2e0beSdillo 	partcnt = 1;
128eaf2e0beSdillo 
129eaf2e0beSdillo 	for (i = 0; i < partcnt; i++) {
130eaf2e0beSdillo 		if (lseek(sd, (i+1)*blksize, SEEK_SET) == -1) {
131eaf2e0beSdillo 			perror("seek part");
132eaf2e0beSdillo 			exit(1);
133eaf2e0beSdillo 		}
134eaf2e0beSdillo 		if (read(sd, buf, sizeof(buf)) != DEV_BSIZE) {
135eaf2e0beSdillo 			perror("read part");
136eaf2e0beSdillo 			exit(1);
137eaf2e0beSdillo 		}
138eaf2e0beSdillo 
139eaf2e0beSdillo 		part = convert_part_map_entry(buf);
140eaf2e0beSdillo 
141eaf2e0beSdillo 		if (part->pmSig != APPLE_PART_MAP_ENTRY_MAGIC)
142eaf2e0beSdillo 			return (changed);
143eaf2e0beSdillo 		if (i == 0)
144eaf2e0beSdillo 			partcnt = part->pmMapBlkCnt;
145eaf2e0beSdillo 		/* XXX: consistency checks? */
146eaf2e0beSdillo 
147eaf2e0beSdillo 		memset((void *)&npe, 0, sizeof(npe));
148eaf2e0beSdillo 
149eaf2e0beSdillo 		if (strcasecmp((char *)part->pmPartType,
150eaf2e0beSdillo 			APPLE_PART_TYPE_MAC) == 0
151eaf2e0beSdillo 		    || strcasecmp((char *)part->pmPartType, "Apple_HFSX") == 0)
152eaf2e0beSdillo 			npe.p_fstype = FS_HFS;
153eaf2e0beSdillo 		else if (strcasecmp((char *)part->pmPartType,
154eaf2e0beSdillo 			     "Apple_UFS") == 0) {
155eaf2e0beSdillo 			npe.p_fstype = FS_APPLEUFS;
156eaf2e0beSdillo 			npe.p_size = 16384;	/* XXX */
157eaf2e0beSdillo 			npe.p_fsize = 1024;
158eaf2e0beSdillo 			npe.p_frag = 8;
159eaf2e0beSdillo 			npe.p_cpg = 16;
160eaf2e0beSdillo 		}
161eaf2e0beSdillo 		else
162eaf2e0beSdillo 			continue;
163eaf2e0beSdillo 
164eaf2e0beSdillo 		temp = (uint64_t)part->pmDataCnt * (uint64_t)blksize;
165eaf2e0beSdillo 		if (temp % label.d_secsize != 0) {
166eaf2e0beSdillo 		    warnx("partition size not multiple of sector size"
167eaf2e0beSdillo 			  ", skipping");
168eaf2e0beSdillo 		    continue;
169eaf2e0beSdillo 		}
170eaf2e0beSdillo 		npe.p_size = temp / label.d_secsize;
171eaf2e0beSdillo 		temp = (uint64_t)(part->pmPyPartStart + part->pmLgDataStart)
172eaf2e0beSdillo 		    * (uint64_t)blksize;
173eaf2e0beSdillo 		if (temp % label.d_secsize != 0) {
174eaf2e0beSdillo 		    warnx("partition offset not multiple of sector size"
175eaf2e0beSdillo 			  ", skipping");
176eaf2e0beSdillo 		    continue;
177eaf2e0beSdillo 		}
178eaf2e0beSdillo 		npe.p_offset = temp / label.d_secsize;
179eaf2e0beSdillo 
180eaf2e0beSdillo 				/* find existing entry, or first free slot */
181eaf2e0beSdillo 		unused = -1;	/* flag as no free slot */
182eaf2e0beSdillo 		if (verbose)
183eaf2e0beSdillo 			printf(
184eaf2e0beSdillo 			    "Found %s partition; size %u (%u MB), offset %u\n",
185eaf2e0beSdillo 			    fstypenames[npe.p_fstype],
186eaf2e0beSdillo 			    npe.p_size, npe.p_size / 2048, npe.p_offset);
187eaf2e0beSdillo 		for (j = 0; j < label.d_npartitions; j++) {
188eaf2e0beSdillo 			struct partition *lpe;
189eaf2e0beSdillo 
190eaf2e0beSdillo 			if (j == RAW_PART)
191eaf2e0beSdillo 				continue;
192eaf2e0beSdillo 			lpe = &label.d_partitions[j];
193eaf2e0beSdillo 			if (lpe->p_size == npe.p_size &&
194eaf2e0beSdillo 			    lpe->p_offset == npe.p_offset
195eaf2e0beSdillo #ifdef notyet
196eaf2e0beSdillo 			    && (lpe->p_fstype == npe.p_fstype ||
197eaf2e0beSdillo 			     lpe->p_fstype == FS_UNUSED) */
198eaf2e0beSdillo #endif
199eaf2e0beSdillo 			     ) {
200eaf2e0beSdillo 				if (verbose)
201eaf2e0beSdillo 					printf(
202eaf2e0beSdillo 			    "  skipping existing %s partition at slot %c.\n",
203eaf2e0beSdillo 					    fstypenames[lpe->p_fstype],
204eaf2e0beSdillo 					    j + 'a');
205eaf2e0beSdillo 				unused = -2;	/* flag as existing */
206eaf2e0beSdillo 				break;
207eaf2e0beSdillo 			}
208eaf2e0beSdillo 			if (unused == -1 && lpe->p_size == 0 &&
209eaf2e0beSdillo 			    lpe->p_fstype == FS_UNUSED)
210eaf2e0beSdillo 				unused = j;
211eaf2e0beSdillo 		}
212eaf2e0beSdillo 		if (unused == -2)
213eaf2e0beSdillo 			continue;	/* entry exists, skip... */
214eaf2e0beSdillo 		if (unused == -1) {
215eaf2e0beSdillo 			if (label.d_npartitions < MAXPARTITIONS) {
216eaf2e0beSdillo 				unused = label.d_npartitions;
217eaf2e0beSdillo 				label.d_npartitions++;
218eaf2e0beSdillo 			} else {
219eaf2e0beSdillo 				printf(
220eaf2e0beSdillo 				"  WARNING: no slots free for %s partition.\n",
221eaf2e0beSdillo 				    fstypenames[npe.p_fstype]);
222eaf2e0beSdillo 				continue;
223eaf2e0beSdillo 			}
224eaf2e0beSdillo 		}
225eaf2e0beSdillo 
226eaf2e0beSdillo 		if (verbose)
227eaf2e0beSdillo 			printf("  adding %s partition to slot %c.\n",
228eaf2e0beSdillo 			    fstypenames[npe.p_fstype], unused + 'a');
229eaf2e0beSdillo 		changed++;
230eaf2e0beSdillo 		label.d_partitions[unused] = npe;
231eaf2e0beSdillo 	}
232eaf2e0beSdillo 
233eaf2e0beSdillo 	return (changed);
234eaf2e0beSdillo }
235eaf2e0beSdillo 
2361cd05199Sjoerg static struct apple_drvr_map *
convert_drvr_map(unsigned char * buf)237eaf2e0beSdillo convert_drvr_map(unsigned char *buf)
238eaf2e0beSdillo {
239eaf2e0beSdillo 	struct apple_drvr_map *drvr;
240eaf2e0beSdillo 	int i;
241eaf2e0beSdillo 
242eaf2e0beSdillo 	drvr = (struct apple_drvr_map *)buf;
243eaf2e0beSdillo 
244eaf2e0beSdillo 	BE16TOH(drvr->sbSig);
245eaf2e0beSdillo 	BE16TOH(drvr->sbBlockSize);
246eaf2e0beSdillo 	BE32TOH(drvr->sbBlkCount);
247eaf2e0beSdillo 	BE16TOH(drvr->sbDevType);
248eaf2e0beSdillo 	BE16TOH(drvr->sbDevID);
249eaf2e0beSdillo 	BE32TOH(drvr->sbData);
250eaf2e0beSdillo 	BE16TOH(drvr->sbDrvrCount);
251eaf2e0beSdillo 	for (i=0; i<APPLE_DRVR_MAP_MAX_DESCRIPTORS; i++) {
252eaf2e0beSdillo 		BE32TOH(drvr->sb_dd[i].descBlock);
253eaf2e0beSdillo 		BE16TOH(drvr->sb_dd[i].descSize);
254eaf2e0beSdillo 		BE16TOH(drvr->sb_dd[i].descType);
255eaf2e0beSdillo 	}
256eaf2e0beSdillo 
257eaf2e0beSdillo 	return drvr;
258eaf2e0beSdillo }
259eaf2e0beSdillo 
2601cd05199Sjoerg static struct apple_part_map_entry *
convert_part_map_entry(unsigned char * buf)261eaf2e0beSdillo convert_part_map_entry(unsigned char *buf)
262eaf2e0beSdillo {
263eaf2e0beSdillo 	struct apple_part_map_entry *part;
264eaf2e0beSdillo 
265eaf2e0beSdillo 	part = (struct apple_part_map_entry *)buf;
266eaf2e0beSdillo 
267eaf2e0beSdillo 	BE16TOH(part->pmSig);
268eaf2e0beSdillo 	BE16TOH(part->pmSigPad);
269eaf2e0beSdillo 	BE32TOH(part->pmMapBlkCnt);
270eaf2e0beSdillo 	BE32TOH(part->pmPyPartStart);
271eaf2e0beSdillo 	BE32TOH(part->pmPartBlkCnt);
272eaf2e0beSdillo 	BE32TOH(part->pmLgDataStart);
273eaf2e0beSdillo 	BE32TOH(part->pmDataCnt);
274eaf2e0beSdillo 	BE32TOH(part->pmPartStatus);
275eaf2e0beSdillo 	BE32TOH(part->pmLgBootStart);
276eaf2e0beSdillo 	BE32TOH(part->pmBootSize);
277eaf2e0beSdillo 	BE32TOH(part->pmBootLoad);
278eaf2e0beSdillo 	BE32TOH(part->pmBootLoad2);
279eaf2e0beSdillo 	BE32TOH(part->pmBootEntry);
280eaf2e0beSdillo 	BE32TOH(part->pmBootEntry2);
281eaf2e0beSdillo 	BE32TOH(part->pmBootCksum);
282eaf2e0beSdillo 
283eaf2e0beSdillo 	return part;
284eaf2e0beSdillo }
285eaf2e0beSdillo 
2861cd05199Sjoerg static void
usage(void)287eaf2e0beSdillo usage(void)
288eaf2e0beSdillo {
289eaf2e0beSdillo 	fprintf(stderr, "usage: %s [-fqrw] rawdisk\n",
290eaf2e0beSdillo 	    getprogname());
291eaf2e0beSdillo 	exit(1);
292eaf2e0beSdillo }
293eaf2e0beSdillo 
294eaf2e0beSdillo 
295eaf2e0beSdillo int
main(int argc,char ** argv)296eaf2e0beSdillo main(int argc, char **argv)
297eaf2e0beSdillo {
298eaf2e0beSdillo 	int	sd, ch, changed;
299eaf2e0beSdillo 	char	name[MAXPATHLEN];
300eaf2e0beSdillo 	int	force;			/* force label update */
301eaf2e0beSdillo 	int	raw;			/* update on-disk label as well */
302eaf2e0beSdillo 	int	verbose;		/* verbose output */
303eaf2e0beSdillo 	int	write_it;		/* update in-core label if changed */
304eaf2e0beSdillo 
305eaf2e0beSdillo 	force = 0;
306eaf2e0beSdillo 	raw = 0;
307eaf2e0beSdillo 	verbose = 1;
308eaf2e0beSdillo 	write_it = 0;
309eaf2e0beSdillo 	while ((ch = getopt(argc, argv, "fqrw")) != -1) {
310eaf2e0beSdillo 		switch (ch) {
311eaf2e0beSdillo 		case 'f':
312eaf2e0beSdillo 			force = 1;
313eaf2e0beSdillo 			break;
314eaf2e0beSdillo 		case 'q':
315eaf2e0beSdillo 			verbose = 0;
316eaf2e0beSdillo 			break;
317eaf2e0beSdillo 		case 'r':
318eaf2e0beSdillo 			raw = 1;
319eaf2e0beSdillo 			break;
320eaf2e0beSdillo 		case 'w':
321eaf2e0beSdillo 			write_it = 1;
322eaf2e0beSdillo 			break;
323eaf2e0beSdillo 		default:
324eaf2e0beSdillo 			usage();
325eaf2e0beSdillo 		}
326eaf2e0beSdillo 	}
327eaf2e0beSdillo 	argc -= optind;
328eaf2e0beSdillo 	argv += optind;
329eaf2e0beSdillo 	if (argc != 1)
330eaf2e0beSdillo 		usage();
331eaf2e0beSdillo 
332eaf2e0beSdillo 	if ((sd = opendisk(argv[0], write_it ? O_RDWR : O_RDONLY, name,
333eaf2e0beSdillo 	    (size_t)MAXPATHLEN, 1)) < 0) {
334eaf2e0beSdillo 		perror(argv[0]);
335eaf2e0beSdillo 		exit(1);
336eaf2e0beSdillo 	}
337eaf2e0beSdillo 	getlabel(sd);
338eaf2e0beSdillo 	changed = getparts(sd, verbose);
339eaf2e0beSdillo 
340eaf2e0beSdillo 	if (verbose) {
341eaf2e0beSdillo 		putchar('\n');
342eaf2e0beSdillo 		showpartitions(stdout, &label, 0);
343eaf2e0beSdillo 		putchar('\n');
344eaf2e0beSdillo 	}
345eaf2e0beSdillo 	if (write_it) {
346eaf2e0beSdillo 		if (! changed && ! force)
347eaf2e0beSdillo 			printf("No change; not updating disk label.\n");
348eaf2e0beSdillo 		else {
349eaf2e0beSdillo 			if (verbose)
350eaf2e0beSdillo 				printf("Updating in-core %sdisk label.\n",
351eaf2e0beSdillo 				    raw ? "and on-disk " : "");
352eaf2e0beSdillo 			setlabel(sd, raw);
353eaf2e0beSdillo 		}
354eaf2e0beSdillo 	} else {
355eaf2e0beSdillo 		printf("Not updating disk label.\n");
356eaf2e0beSdillo 	}
357eaf2e0beSdillo 	close(sd);
358eaf2e0beSdillo 	return (0);
359eaf2e0beSdillo }
360