1 /* $NetBSD: partutil.c,v 1.12 2013/04/13 22:08:57 jakllsch Exp $ */
4 * Copyright (c) 2006 The NetBSD Foundation, Inc.
7 * This code is derived from software contributed to The NetBSD Foundation
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
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.
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.
32 #include <sys/cdefs.h>
33 __RCSID("$NetBSD: partutil.c,v 1.12 2013/04/13 22:08:57 jakllsch Exp $");
35 #include <sys/types.h>
36 #include <sys/disklabel.h>
38 #include <sys/ioctl.h>
51 #include <prop/proplib.h>
56 * Convert disklabel geometry info to disk_geom.
59 label2geom(struct disk_geom
*geo
, const struct disklabel
*lp
)
61 geo
->dg_secperunit
= lp
->d_secperunit
;
62 geo
->dg_secsize
= lp
->d_secsize
;
63 geo
->dg_nsectors
= lp
->d_nsectors
;
64 geo
->dg_ntracks
= lp
->d_ntracks
;
65 geo
->dg_ncylinders
= lp
->d_ncylinders
;
66 geo
->dg_secpercyl
= lp
->d_secpercyl
;
67 geo
->dg_pcylinders
= lp
->d_ncylinders
;
68 geo
->dg_sparespertrack
= lp
->d_sparespertrack
;
69 geo
->dg_sparespercyl
= lp
->d_sparespercyl
;
70 geo
->dg_acylinders
= lp
->d_acylinders
;
74 * Set what we need to know about disk geometry.
77 dict2geom(struct disk_geom
*geo
, prop_dictionary_t dict
)
79 (void)memset(geo
, 0, sizeof(struct disk_geom
));
80 prop_dictionary_get_int64(dict
, "sectors-per-unit",
82 prop_dictionary_get_uint32(dict
, "sector-size", &geo
->dg_secsize
);
83 prop_dictionary_get_uint32(dict
, "sectors-per-track",
85 prop_dictionary_get_uint32(dict
, "tracks-per-cylinder",
87 prop_dictionary_get_uint32(dict
, "cylinders-per-unit",
93 getdiskinfo(const char *s
, int fd
, const char *dt
, struct disk_geom
*geo
,
94 struct dkwedge_info
*dkw
)
97 struct disklabel
*lp
= &lab
;
98 prop_dictionary_t disk_dict
, geom_dict
;
101 const struct partition
*pp
;
103 #endif /* defined(__minix) */
107 errx(1, "minix doesn't know about disk types (%s)", dt
);
109 lp
= getdiskbyname(dt
);
111 errx(1, "unknown disk type `%s'", dt
);
112 #endif /* defined(__minix) */
115 /* Get disk description dictionary */
116 #if !defined(__minix)
117 if (prop_dictionary_recv_ioctl(fd
, DIOCGDISKINFO
, &disk_dict
)) {
120 #endif /* !defined(__minix) */
122 * Ask for disklabel if DIOCGDISKINFO failed. This is
123 * compatibility call and can be removed when all devices
124 * will support DIOCGDISKINFO.
125 * cgd, ccd pseudo disk drives doesn't support DIOCGDDISKINFO
127 if (ioctl(fd
, DIOCGDINFO
, lp
) == -1) {
128 warn("DIOCGDINFO on %s failed", s
);
133 geom_dict
= prop_dictionary_get(disk_dict
, "geometry");
134 dict2geom(geo
, geom_dict
);
137 /* Get info about partition/wedge */
138 if (ioctl(fd
, DIOCGWEDGEINFO
, dkw
) != -1) {
139 /* DIOCGWEDGEINFO didn't fail, we're done */
143 if (ioctl(fd
, DIOCGDINFO
, lp
) == -1) {
144 err(1, "Please implement DIOCGWEDGEINFO or "
145 "DIOCGDINFO for disk device %s", s
);
148 #if !defined(__minix)
149 /* DIOCGDINFO didn't fail */
151 (void)memset(dkw
, 0, sizeof(*dkw
));
153 if (stat(s
, &sb
) == -1)
156 ptn
= strchr(s
, '\0')[-1] - 'a';
157 if ((unsigned)ptn
>= lp
->d_npartitions
||
158 (devminor_t
)ptn
!= DISKPART(sb
.st_rdev
))
161 pp
= &lp
->d_partitions
[ptn
];
162 if (ptn
!= getrawpartition()) {
163 dkw
->dkw_offset
= pp
->p_offset
;
164 dkw
->dkw_size
= pp
->p_size
;
167 dkw
->dkw_size
= geo
->dg_secperunit
;
169 dkw
->dkw_parent
[0] = '*';
170 strlcpy(dkw
->dkw_ptype
, getfstypename(pp
->p_fstype
),
171 sizeof(dkw
->dkw_ptype
));
172 #endif /* !defined(__minix) */