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, §orsize) == -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