xref: /netbsd-src/sbin/fsck/partutil.c (revision 93bf6008f8b7982c1d1a9486e4a4a0e687fe36eb)
1 /*	$NetBSD: partutil.c,v 1.4 2009/04/11 06:48:36 lukem Exp $	*/
2 
3 /*-
4  * Copyright (c) 2006 The NetBSD Foundation, Inc.
5  * All rights reserved.
6  *
7  * This code is derived from software contributed to The NetBSD Foundation
8  * by Christos Zoulas.
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29  * POSSIBILITY OF SUCH DAMAGE.
30  */
31 
32 #include <sys/cdefs.h>
33 __RCSID("$NetBSD: partutil.c,v 1.4 2009/04/11 06:48:36 lukem Exp $");
34 
35 #include <sys/types.h>
36 #include <sys/disklabel.h>
37 #include <sys/disk.h>
38 #include <sys/ioctl.h>
39 #include <sys/stat.h>
40 
41 #include <disktab.h>
42 #include <fcntl.h>
43 #include <util.h>
44 #include <unistd.h>
45 #include <err.h>
46 #include <string.h>
47 #include <errno.h>
48 
49 #include "partutil.h"
50 
51 static void
52 label2geom(struct disk_geom *geo, const struct disklabel *lp)
53 {
54 	geo->dg_secperunit = lp->d_secperunit;
55 	geo->dg_secsize = lp->d_secsize;
56 	geo->dg_nsectors = lp->d_nsectors;
57 	geo->dg_ntracks = lp->d_ntracks;
58 	geo->dg_ncylinders = lp->d_ncylinders;
59 	geo->dg_secpercyl = lp->d_secpercyl;
60 	geo->dg_pcylinders = lp->d_ncylinders;
61 	geo->dg_sparespertrack = lp->d_sparespertrack;
62 	geo->dg_sparespercyl = lp->d_sparespercyl;
63 	geo->dg_acylinders = lp->d_acylinders;
64 }
65 
66 static void
67 part2wedge(struct dkwedge_info *dkw, const struct disklabel *lp, const char *s)
68 {
69 	struct stat sb;
70 	const struct partition *pp;
71 	int ptn;
72 
73 	(void)memset(dkw, 0, sizeof(*dkw));
74 	if (stat(s, &sb) == -1)
75 		return;
76 
77 	ptn = strchr(s, '\0')[-1] - 'a';
78 	if ((unsigned)ptn >= lp->d_npartitions ||
79 	    (devminor_t)ptn != DISKPART(sb.st_rdev))
80 		return;
81 
82 	pp = &lp->d_partitions[ptn];
83 	dkw->dkw_offset = pp->p_offset;
84 	dkw->dkw_size = pp->p_size;
85 	dkw->dkw_parent[0] = '*';
86 	switch (pp->p_fstype) {
87 	default:
88 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_UNKNOWN);
89 		break;
90 	case FS_UNUSED:
91 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_UNUSED);
92 		break;
93 	case FS_SWAP:
94 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_SWAP);
95 		break;
96 	case FS_BSDFFS:
97 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_FFS);
98 		break;
99 	case FS_BSDLFS:
100 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_LFS);
101 		break;
102 	case FS_EX2FS:
103 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_EXT2FS);
104 		break;
105 	case FS_ISO9660:
106 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_ISO9660);
107 		break;
108 	case FS_ADOS:
109 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_AMIGADOS);
110 		break;
111 	case FS_HFS:
112 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_APPLEHFS);
113 		break;
114 	case FS_MSDOS:
115 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_FAT);
116 		break;
117 	case FS_FILECORE:
118 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_FILECORE);
119 		break;
120 	case FS_APPLEUFS:
121 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_APPLEUFS);
122 		break;
123 	case FS_NTFS:
124 		(void)strcpy(dkw->dkw_ptype, DKW_PTYPE_NTFS);
125 		break;
126 	}
127 }
128 
129 int
130 getdiskinfo(const char *s, int fd, const char *dt, struct disk_geom *geo,
131     struct dkwedge_info *dkw)
132 {
133 	struct disklabel lab;
134 	struct disklabel *lp = &lab;
135 	char parent[1024];
136 
137 	if (dt) {
138 		lp = getdiskbyname(dt);
139 		if (lp == NULL)
140 			errx(1, "%s: unknown disk type", dt);
141 		goto part;
142 	}
143 
144 	if (ioctl(fd, DIOCGDINFO, lp) == -1) {
145 		if (errno == ENOTTY) {
146 			int pfd;
147 			if (ioctl(fd, DIOCGWEDGEINFO, dkw) == -1) {
148 				warn("ioctl (DIOCGWEDGEINFO)");
149 				goto bad;
150 			}
151 			pfd = opendisk(dkw->dkw_parent, O_RDONLY,
152 			    parent, sizeof(parent), 0);
153 			if (pfd == -1) {
154 				warn("Cannot open `%s'", dkw->dkw_parent);
155 				goto bad;
156 			}
157 			if (ioctl(pfd, DIOCGDINFO, lp) != -1) {
158 				(void)close(pfd);
159 				goto label;
160 			} else {
161 				int serrno = errno;
162 				(void)close(pfd);
163 				errno = serrno;
164 			}
165 		}
166 		warn("ioctl (DIOCGDINFO)");
167 		goto bad;
168 	}
169 part:
170 	part2wedge(dkw, lp, s);
171 label:
172 	label2geom(geo, lp);
173 	return 0;
174 bad:
175 	return -1;
176 }
177