2 * Copyright (c) 1998 Michael Smith <msmith@freebsd.org>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 #include <sys/cdefs.h>
31 #include <sys/param.h>
32 #include "bootstrap.h"
35 #include "../zfs/libzfs.h"
37 static int i386_parsedev(struct i386_devdesc
**, const char *, const char **);
40 * Point dev at an allocated device specifier for the device matching the
41 * path in devspec. If it contains an explicit device specification,
42 * use that. If not, use the default device.
45 i386_getdev(void **vdev
, const char *devspec
, const char **path
)
47 struct i386_devdesc
**dev
= (struct i386_devdesc
**)vdev
;
51 * If it looks like this is just a path and no
52 * device, go with the current device.
54 if ((devspec
== NULL
) ||
55 (devspec
[0] == '/') || (strchr(devspec
, ':') == NULL
)) {
57 rv
= i386_parsedev(dev
, getenv("currdev"), NULL
);
58 if (rv
== 0 && path
!= NULL
)
64 * Try to parse the device name off the beginning of the devspec
66 return (i386_parsedev(dev
, devspec
, path
));
70 * Point (dev) at an allocated device specifier matching the string version
71 * at the beginning of (devspec). Return a pointer to the remaining
74 * In all cases, the beginning of (devspec) is compared to the names
75 * of known devices in the device switch, and then any following text
76 * is parsed according to the rules applied to the device type.
78 * For disk-type devices, the syntax is:
80 * disk<unit>[s<slice>][<partition>]:
84 i386_parsedev(struct i386_devdesc
**dev
, const char *devspec
, const char **path
)
86 struct i386_devdesc
*idev
;
92 /* minimum length check */
93 if (strlen(devspec
) < 2)
96 /* look for a device that matches */
97 for (i
= 0, dv
= NULL
; devsw
[i
] != NULL
; i
++) {
99 if (strncmp(devspec
, dv
->dv_name
, strlen(dv
->dv_name
)) == 0)
102 if (devsw
[i
] == NULL
)
105 np
= devspec
+ strlen(dv
->dv_name
);
109 switch (dv
->dv_type
) {
114 idev
= malloc(sizeof (struct i386_devdesc
));
118 err
= disk_parsedev((struct disk_devdesc
*)idev
, np
, path
);
124 idev
= malloc(sizeof (struct zfs_devdesc
));
128 err
= zfs_parsedev((struct zfs_devdesc
*)idev
, np
, path
);
134 idev
= malloc(sizeof (struct devdesc
));
141 if (*np
&& (*np
!= ':')) {
142 /* get unit number if present */
143 unit
= strtol(np
, &cp
, 0);
149 if (*cp
&& (*cp
!= ':')) {
154 idev
->dd
.d_unit
= unit
;
156 *path
= (*cp
== '\0') ? cp
: cp
+ 1;
175 i386_fmtdev(void *vdev
)
177 struct i386_devdesc
*dev
= (struct i386_devdesc
*)vdev
;
178 static char buf
[SPECNAMELEN
+ 1];
180 switch (dev
->dd
.d_dev
->dv_type
) {
182 strlcpy(buf
, "(no device)", sizeof (buf
));
186 return (disk_fmtdev(vdev
));
189 return (zfs_fmtdev(vdev
));
192 snprintf(buf
, sizeof (buf
), "%s%d:", dev
->dd
.d_dev
->dv_name
,
201 * Set currdev to suit the value being supplied in (value)
204 i386_setcurrdev(struct env_var
*ev
, int flags
, const void *value
)
206 struct i386_devdesc
*ncurr
;
209 if ((rv
= i386_parsedev(&ncurr
, value
, NULL
)) != 0)
213 env_setenv(ev
->ev_name
, flags
| EV_NOHOOK
, value
, NULL
, NULL
);