xref: /freebsd/lib/libgeom/geom_util.c (revision a2f733abcff64628b7771a47089628b7327a88bd)
1fbda685dSPawel Jakub Dawidek /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
35e53a4f9SPedro F. Giffuni  *
4fbda685dSPawel Jakub Dawidek  * Copyright (c) 2007 Pawel Jakub Dawidek <pjd@FreeBSD.org>
5fbda685dSPawel Jakub Dawidek  * All rights reserved.
6fbda685dSPawel Jakub Dawidek  *
7fbda685dSPawel Jakub Dawidek  * Redistribution and use in source and binary forms, with or without
8fbda685dSPawel Jakub Dawidek  * modification, are permitted provided that the following conditions
9fbda685dSPawel Jakub Dawidek  * are met:
10fbda685dSPawel Jakub Dawidek  * 1. Redistributions of source code must retain the above copyright
11fbda685dSPawel Jakub Dawidek  *    notice, this list of conditions and the following disclaimer.
12fbda685dSPawel Jakub Dawidek  * 2. Redistributions in binary form must reproduce the above copyright
13fbda685dSPawel Jakub Dawidek  *    notice, this list of conditions and the following disclaimer in the
14fbda685dSPawel Jakub Dawidek  *    documentation and/or other materials provided with the distribution.
15fbda685dSPawel Jakub Dawidek  *
16fbda685dSPawel Jakub Dawidek  * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
17fbda685dSPawel Jakub Dawidek  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18fbda685dSPawel Jakub Dawidek  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19fbda685dSPawel Jakub Dawidek  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
20fbda685dSPawel Jakub Dawidek  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21fbda685dSPawel Jakub Dawidek  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22fbda685dSPawel Jakub Dawidek  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23fbda685dSPawel Jakub Dawidek  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24fbda685dSPawel Jakub Dawidek  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25fbda685dSPawel Jakub Dawidek  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26fbda685dSPawel Jakub Dawidek  * SUCH DAMAGE.
27fbda685dSPawel Jakub Dawidek  */
28fbda685dSPawel Jakub Dawidek 
29fbda685dSPawel Jakub Dawidek #include <sys/param.h>
30fbda685dSPawel Jakub Dawidek #include <sys/disk.h>
31fbda685dSPawel Jakub Dawidek #include <sys/stat.h>
32fbda685dSPawel Jakub Dawidek 
33fbda685dSPawel Jakub Dawidek #include <stdio.h>
34fbda685dSPawel Jakub Dawidek #include <fcntl.h>
35fbda685dSPawel Jakub Dawidek #include <errno.h>
36fbda685dSPawel Jakub Dawidek #include <stdint.h>
37fbda685dSPawel Jakub Dawidek #include <unistd.h>
38fbda685dSPawel Jakub Dawidek #include <string.h>
39fbda685dSPawel Jakub Dawidek #include <stdlib.h>
40fbda685dSPawel Jakub Dawidek #include <paths.h>
41fbda685dSPawel Jakub Dawidek 
42fbda685dSPawel Jakub Dawidek #include <libgeom.h>
43fbda685dSPawel Jakub Dawidek 
44f805f204SUlf Lilleengen static char	*g_device_path_open(const char *, int *, int);
45f805f204SUlf Lilleengen 
46fbda685dSPawel Jakub Dawidek /*
47fbda685dSPawel Jakub Dawidek  * Open the given provider and at least check if this is a block device.
48fbda685dSPawel Jakub Dawidek  */
49fbda685dSPawel Jakub Dawidek int
g_open(const char * name,int dowrite)509ac6b8aeSPawel Jakub Dawidek g_open(const char *name, int dowrite)
51fbda685dSPawel Jakub Dawidek {
52f805f204SUlf Lilleengen 	char *path;
53fbda685dSPawel Jakub Dawidek 	int fd;
54fbda685dSPawel Jakub Dawidek 
55f805f204SUlf Lilleengen 	path = g_device_path_open(name, &fd, dowrite);
56f805f204SUlf Lilleengen 	if (path != NULL)
57f805f204SUlf Lilleengen 		free(path);
58fbda685dSPawel Jakub Dawidek 	return (fd);
59fbda685dSPawel Jakub Dawidek }
60fbda685dSPawel Jakub Dawidek 
61fbda685dSPawel Jakub Dawidek int
g_close(int fd)62fbda685dSPawel Jakub Dawidek g_close(int fd)
63fbda685dSPawel Jakub Dawidek {
64fbda685dSPawel Jakub Dawidek 
65fbda685dSPawel Jakub Dawidek 	return (close(fd));
66fbda685dSPawel Jakub Dawidek }
67fbda685dSPawel Jakub Dawidek 
68fbda685dSPawel Jakub Dawidek static int
g_ioctl_arg(int fd,unsigned long cmd,void * arg)69fbda685dSPawel Jakub Dawidek g_ioctl_arg(int fd, unsigned long cmd, void *arg)
70fbda685dSPawel Jakub Dawidek {
71fbda685dSPawel Jakub Dawidek 	int ret;
72fbda685dSPawel Jakub Dawidek 
73fbda685dSPawel Jakub Dawidek 	if (arg != NULL)
74fbda685dSPawel Jakub Dawidek 		ret = ioctl(fd, cmd, arg);
75fbda685dSPawel Jakub Dawidek 	else
76fbda685dSPawel Jakub Dawidek 		ret = ioctl(fd, cmd);
77fbda685dSPawel Jakub Dawidek 	return (ret >= 0 ? 0 : -1);
78fbda685dSPawel Jakub Dawidek }
79fbda685dSPawel Jakub Dawidek 
80fbda685dSPawel Jakub Dawidek static int
g_ioctl(int fd,unsigned long cmd)81fbda685dSPawel Jakub Dawidek g_ioctl(int fd, unsigned long cmd)
82fbda685dSPawel Jakub Dawidek {
83fbda685dSPawel Jakub Dawidek 
84fbda685dSPawel Jakub Dawidek 	return (g_ioctl_arg(fd, cmd, NULL));
85fbda685dSPawel Jakub Dawidek }
86fbda685dSPawel Jakub Dawidek 
87fbda685dSPawel Jakub Dawidek /*
88fbda685dSPawel Jakub Dawidek  * Return media size of the given provider.
89fbda685dSPawel Jakub Dawidek  */
90fbda685dSPawel Jakub Dawidek off_t
g_mediasize(int fd)91fbda685dSPawel Jakub Dawidek g_mediasize(int fd)
92fbda685dSPawel Jakub Dawidek {
93fbda685dSPawel Jakub Dawidek 	off_t mediasize;
94fbda685dSPawel Jakub Dawidek 
95fbda685dSPawel Jakub Dawidek 	if (g_ioctl_arg(fd, DIOCGMEDIASIZE, &mediasize) == -1)
96fbda685dSPawel Jakub Dawidek 		mediasize = -1;
97fbda685dSPawel Jakub Dawidek 	return (mediasize);
98fbda685dSPawel Jakub Dawidek }
99fbda685dSPawel Jakub Dawidek 
100fbda685dSPawel Jakub Dawidek /*
101fbda685dSPawel Jakub Dawidek  * Return sector size of the given provider.
102fbda685dSPawel Jakub Dawidek  */
103fbda685dSPawel Jakub Dawidek ssize_t
g_sectorsize(int fd)104fbda685dSPawel Jakub Dawidek g_sectorsize(int fd)
105fbda685dSPawel Jakub Dawidek {
106fbda685dSPawel Jakub Dawidek 	u_int sectorsize;
107fbda685dSPawel Jakub Dawidek 
108fbda685dSPawel Jakub Dawidek 	if (g_ioctl_arg(fd, DIOCGSECTORSIZE, &sectorsize) == -1)
109fbda685dSPawel Jakub Dawidek 		return (-1);
110fbda685dSPawel Jakub Dawidek 	return ((ssize_t)sectorsize);
111fbda685dSPawel Jakub Dawidek }
112fbda685dSPawel Jakub Dawidek 
113fbda685dSPawel Jakub Dawidek /*
11435daa28fSXin LI  * Return stripe size of the given provider.
11535daa28fSXin LI  */
11635daa28fSXin LI off_t
g_stripesize(int fd)11735daa28fSXin LI g_stripesize(int fd)
11835daa28fSXin LI {
11935daa28fSXin LI 	off_t stripesize;
12035daa28fSXin LI 
12135daa28fSXin LI 	if (g_ioctl_arg(fd, DIOCGSTRIPESIZE, &stripesize) == -1)
12235daa28fSXin LI 		return (-1);
12335daa28fSXin LI 	return (stripesize);
12435daa28fSXin LI }
12535daa28fSXin LI 
12635daa28fSXin LI /*
12735daa28fSXin LI  * Return stripe size of the given provider.
12835daa28fSXin LI  */
12935daa28fSXin LI off_t
g_stripeoffset(int fd)13035daa28fSXin LI g_stripeoffset(int fd)
13135daa28fSXin LI {
13235daa28fSXin LI 	off_t stripeoffset;
13335daa28fSXin LI 
13435daa28fSXin LI 	if (g_ioctl_arg(fd, DIOCGSTRIPEOFFSET, &stripeoffset) == -1)
13535daa28fSXin LI 		return (-1);
13635daa28fSXin LI 	return (stripeoffset);
13735daa28fSXin LI }
13835daa28fSXin LI 
13935daa28fSXin LI /*
140f805f204SUlf Lilleengen  * Return the correct provider name.
141f805f204SUlf Lilleengen  */
142f805f204SUlf Lilleengen char *
g_providername(int fd)143f805f204SUlf Lilleengen g_providername(int fd)
144f805f204SUlf Lilleengen {
145f805f204SUlf Lilleengen 	char name[MAXPATHLEN];
146f805f204SUlf Lilleengen 
147f805f204SUlf Lilleengen 	if (g_ioctl_arg(fd, DIOCGPROVIDERNAME, name) == -1)
148f805f204SUlf Lilleengen 		return (NULL);
149f805f204SUlf Lilleengen 	return (strdup(name));
150f805f204SUlf Lilleengen }
151f805f204SUlf Lilleengen 
152f805f204SUlf Lilleengen /*
153fbda685dSPawel Jakub Dawidek  * Call BIO_FLUSH for the given provider.
154fbda685dSPawel Jakub Dawidek  */
155fbda685dSPawel Jakub Dawidek int
g_flush(int fd)156fbda685dSPawel Jakub Dawidek g_flush(int fd)
157fbda685dSPawel Jakub Dawidek {
158fbda685dSPawel Jakub Dawidek 
159fbda685dSPawel Jakub Dawidek 	return (g_ioctl(fd, DIOCGFLUSH));
160fbda685dSPawel Jakub Dawidek }
161fbda685dSPawel Jakub Dawidek 
162fbda685dSPawel Jakub Dawidek /*
163fbda685dSPawel Jakub Dawidek  * Call BIO_DELETE for the given range.
164fbda685dSPawel Jakub Dawidek  */
165fbda685dSPawel Jakub Dawidek int
g_delete(int fd,off_t offset,off_t length)166fbda685dSPawel Jakub Dawidek g_delete(int fd, off_t offset, off_t length)
167fbda685dSPawel Jakub Dawidek {
168fbda685dSPawel Jakub Dawidek 	off_t arg[2];
169fbda685dSPawel Jakub Dawidek 
170fbda685dSPawel Jakub Dawidek 	arg[0] = offset;
171fbda685dSPawel Jakub Dawidek 	arg[1] = length;
172fbda685dSPawel Jakub Dawidek 	return (g_ioctl_arg(fd, DIOCGDELETE, arg));
173fbda685dSPawel Jakub Dawidek }
174fbda685dSPawel Jakub Dawidek 
175fbda685dSPawel Jakub Dawidek /*
176fbda685dSPawel Jakub Dawidek  * Return ID of the given provider.
177fbda685dSPawel Jakub Dawidek  */
178fbda685dSPawel Jakub Dawidek int
g_get_ident(int fd,char * ident,size_t size)179fbda685dSPawel Jakub Dawidek g_get_ident(int fd, char *ident, size_t size)
180fbda685dSPawel Jakub Dawidek {
181fbda685dSPawel Jakub Dawidek 	char lident[DISK_IDENT_SIZE];
182fbda685dSPawel Jakub Dawidek 
183fbda685dSPawel Jakub Dawidek 	if (g_ioctl_arg(fd, DIOCGIDENT, lident) == -1)
184fbda685dSPawel Jakub Dawidek 		return (-1);
185fbda685dSPawel Jakub Dawidek 	if (lident[0] == '\0') {
186fbda685dSPawel Jakub Dawidek 		errno = ENOENT;
187fbda685dSPawel Jakub Dawidek 		return (-1);
188fbda685dSPawel Jakub Dawidek 	}
189fbda685dSPawel Jakub Dawidek 	if (strlcpy(ident, lident, size) >= size) {
190fbda685dSPawel Jakub Dawidek 		errno = ENAMETOOLONG;
191fbda685dSPawel Jakub Dawidek 		return (-1);
192fbda685dSPawel Jakub Dawidek 	}
193fbda685dSPawel Jakub Dawidek 	return (0);
194fbda685dSPawel Jakub Dawidek }
195fbda685dSPawel Jakub Dawidek 
196fbda685dSPawel Jakub Dawidek /*
197fbda685dSPawel Jakub Dawidek  * Return name of the provider, which has the given ID.
198fbda685dSPawel Jakub Dawidek  */
199fbda685dSPawel Jakub Dawidek int
g_get_name(const char * ident,char * name,size_t size)200fbda685dSPawel Jakub Dawidek g_get_name(const char *ident, char *name, size_t size)
201fbda685dSPawel Jakub Dawidek {
202fbda685dSPawel Jakub Dawidek 	int fd;
203fbda685dSPawel Jakub Dawidek 
204fbda685dSPawel Jakub Dawidek 	fd = g_open_by_ident(ident, 0, name, size);
205fbda685dSPawel Jakub Dawidek 	if (fd == -1)
206fbda685dSPawel Jakub Dawidek 		return (-1);
207fbda685dSPawel Jakub Dawidek 	g_close(fd);
208fbda685dSPawel Jakub Dawidek 	return (0);
209fbda685dSPawel Jakub Dawidek }
210fbda685dSPawel Jakub Dawidek 
211fbda685dSPawel Jakub Dawidek /*
212fbda685dSPawel Jakub Dawidek  * Find provider name by the given ID.
213fbda685dSPawel Jakub Dawidek  */
214fbda685dSPawel Jakub Dawidek int
g_open_by_ident(const char * ident,int dowrite,char * name,size_t size)2159ac6b8aeSPawel Jakub Dawidek g_open_by_ident(const char *ident, int dowrite, char *name, size_t size)
216fbda685dSPawel Jakub Dawidek {
217fbda685dSPawel Jakub Dawidek 	char lident[DISK_IDENT_SIZE];
218fbda685dSPawel Jakub Dawidek 	struct gmesh mesh;
219fbda685dSPawel Jakub Dawidek 	struct gclass *mp;
220fbda685dSPawel Jakub Dawidek 	struct ggeom *gp;
221fbda685dSPawel Jakub Dawidek 	struct gprovider *pp;
222fbda685dSPawel Jakub Dawidek 	int error, fd;
223fbda685dSPawel Jakub Dawidek 
224fbda685dSPawel Jakub Dawidek 	error = geom_gettree(&mesh);
225fbda685dSPawel Jakub Dawidek 	if (error != 0) {
226fbda685dSPawel Jakub Dawidek 		errno = error;
227fbda685dSPawel Jakub Dawidek 		return (-1);
228fbda685dSPawel Jakub Dawidek 	}
229fbda685dSPawel Jakub Dawidek 
230fbda685dSPawel Jakub Dawidek 	error = ENOENT;
231fbda685dSPawel Jakub Dawidek 	fd = -1;
232fbda685dSPawel Jakub Dawidek 
233fbda685dSPawel Jakub Dawidek 	LIST_FOREACH(mp, &mesh.lg_class, lg_class) {
234fbda685dSPawel Jakub Dawidek 		LIST_FOREACH(gp, &mp->lg_geom, lg_geom) {
235fbda685dSPawel Jakub Dawidek 			LIST_FOREACH(pp, &gp->lg_provider, lg_provider) {
2369ac6b8aeSPawel Jakub Dawidek 				fd = g_open(pp->lg_name, dowrite);
237fbda685dSPawel Jakub Dawidek 				if (fd == -1)
238fbda685dSPawel Jakub Dawidek 					continue;
239fbda685dSPawel Jakub Dawidek 				if (g_get_ident(fd, lident,
240fbda685dSPawel Jakub Dawidek 				    sizeof(lident)) == -1) {
241fbda685dSPawel Jakub Dawidek 					g_close(fd);
242fbda685dSPawel Jakub Dawidek 					continue;
243fbda685dSPawel Jakub Dawidek 				}
244fbda685dSPawel Jakub Dawidek 				if (strcmp(ident, lident) != 0) {
245fbda685dSPawel Jakub Dawidek 					g_close(fd);
246fbda685dSPawel Jakub Dawidek 					continue;
247fbda685dSPawel Jakub Dawidek 				}
248fbda685dSPawel Jakub Dawidek 				error = 0;
249fbda685dSPawel Jakub Dawidek 				if (name != NULL && strlcpy(name, pp->lg_name,
250fbda685dSPawel Jakub Dawidek 				    size) >= size) {
251fbda685dSPawel Jakub Dawidek 					error = ENAMETOOLONG;
252fbda685dSPawel Jakub Dawidek 					g_close(fd);
253fbda685dSPawel Jakub Dawidek 				}
254fbda685dSPawel Jakub Dawidek 				goto end;
255fbda685dSPawel Jakub Dawidek 			}
256fbda685dSPawel Jakub Dawidek 		}
257fbda685dSPawel Jakub Dawidek 	}
258fbda685dSPawel Jakub Dawidek end:
259fbda685dSPawel Jakub Dawidek 	geom_deletetree(&mesh);
260fbda685dSPawel Jakub Dawidek 	if (error != 0) {
261fbda685dSPawel Jakub Dawidek 		errno = error;
262fbda685dSPawel Jakub Dawidek 		return (-1);
263fbda685dSPawel Jakub Dawidek 	}
264fbda685dSPawel Jakub Dawidek 	return (fd);
265fbda685dSPawel Jakub Dawidek }
266f805f204SUlf Lilleengen 
267f805f204SUlf Lilleengen /*
268f805f204SUlf Lilleengen  * Return the device path device given a partial or full path to its node.
269f805f204SUlf Lilleengen  * A pointer can be provided, which will be set to an opened file descriptor of
270f805f204SUlf Lilleengen  * not NULL.
271f805f204SUlf Lilleengen  */
272f805f204SUlf Lilleengen static char *
g_device_path_open(const char * devpath,int * fdp,int dowrite)273f805f204SUlf Lilleengen g_device_path_open(const char *devpath, int *fdp, int dowrite)
274f805f204SUlf Lilleengen {
275f805f204SUlf Lilleengen 	char *path;
276f805f204SUlf Lilleengen 	int fd;
277f805f204SUlf Lilleengen 
278f805f204SUlf Lilleengen 	/* Make sure that we can fail. */
279f805f204SUlf Lilleengen 	if (fdp != NULL)
280f805f204SUlf Lilleengen 		*fdp = -1;
281b906c1a0SConrad Meyer 
282f805f204SUlf Lilleengen 	/* Use the device node if we're able to open it. */
283f805f204SUlf Lilleengen 	fd = open(devpath, dowrite ? O_RDWR : O_RDONLY);
284b906c1a0SConrad Meyer 	if (fd != -1) {
285f805f204SUlf Lilleengen 		if ((path = strdup(devpath)) == NULL) {
286f805f204SUlf Lilleengen 			close(fd);
287f805f204SUlf Lilleengen 			return (NULL);
288f805f204SUlf Lilleengen 		}
289b906c1a0SConrad Meyer 		goto fd_ok;
290b906c1a0SConrad Meyer 	}
291f805f204SUlf Lilleengen 
292f805f204SUlf Lilleengen 	/* If we're not given an absolute path, assume /dev/ prefix. */
293b906c1a0SConrad Meyer 	if (*devpath == '/')
294b906c1a0SConrad Meyer 		return (NULL);
295b906c1a0SConrad Meyer 
296f805f204SUlf Lilleengen 	asprintf(&path, "%s%s", _PATH_DEV, devpath);
297f805f204SUlf Lilleengen 	if (path == NULL)
298f805f204SUlf Lilleengen 		return (NULL);
299f805f204SUlf Lilleengen 	fd = open(path, dowrite ? O_RDWR : O_RDONLY);
300f805f204SUlf Lilleengen 	if (fd == -1) {
301f805f204SUlf Lilleengen 		free(path);
302f805f204SUlf Lilleengen 		return (NULL);
303f805f204SUlf Lilleengen 	}
304b906c1a0SConrad Meyer 
305b906c1a0SConrad Meyer fd_ok:
306f805f204SUlf Lilleengen 	/*
307b906c1a0SConrad Meyer 	 * Let try to get sectorsize, which will prove it is a GEOM provider.
308f805f204SUlf Lilleengen 	 */
309f805f204SUlf Lilleengen 	if (g_sectorsize(fd) == -1) {
310f805f204SUlf Lilleengen 		free(path);
311f805f204SUlf Lilleengen 		close(fd);
312f805f204SUlf Lilleengen 		errno = EFTYPE;
313f805f204SUlf Lilleengen 		return (NULL);
314f805f204SUlf Lilleengen 	}
315f805f204SUlf Lilleengen 	if (fdp != NULL)
316f805f204SUlf Lilleengen 		*fdp = fd;
317f805f204SUlf Lilleengen 	else
318f805f204SUlf Lilleengen 		close(fd);
319f805f204SUlf Lilleengen 	return (path);
320f805f204SUlf Lilleengen }
321f805f204SUlf Lilleengen 
322f805f204SUlf Lilleengen char *
g_device_path(const char * devpath)323f805f204SUlf Lilleengen g_device_path(const char *devpath)
324f805f204SUlf Lilleengen {
325f805f204SUlf Lilleengen 	return (g_device_path_open(devpath, NULL, 0));
326f805f204SUlf Lilleengen }
327