2 * Copyright (c) 1998 Michael Smith <msmith@freebsd.org>
3 * Copyright (c) 2006 Marcel Moolenaar
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28 #include <sys/cdefs.h>
32 #include <sys/disklabel.h>
33 #include <sys/param.h>
34 #include <bootstrap.h>
41 #include "loader_efi.h"
43 static int efi_parsedev(struct devdesc
**, const char *, const char **);
46 * Point (dev) at an allocated device specifier for the device matching the
47 * path in (devspec). If it contains an explicit device specification,
48 * use that. If not, use the default device.
51 efi_getdev(void **vdev
, const char *devspec
, const char **path
)
53 struct devdesc
**dev
= (struct devdesc
**)vdev
;
57 * If it looks like this is just a path and no device, then
58 * use the current device instead.
60 if (devspec
== NULL
|| *devspec
== '/' || !strchr(devspec
, ':')) {
61 rv
= efi_parsedev(dev
, getenv("currdev"), NULL
);
62 if (rv
== 0 && path
!= NULL
)
67 /* Parse the device name off the beginning of the devspec. */
68 return (efi_parsedev(dev
, devspec
, path
));
72 * Point (dev) at an allocated device specifier matching the string version
73 * at the beginning of (devspec). Return a pointer to the remaining
76 * In all cases, the beginning of (devspec) is compared to the names
77 * of known devices in the device switch, and then any following text
78 * is parsed according to the rules applied to the device type.
80 * For disk-type devices, the syntax is:
85 efi_parsedev(struct devdesc
**dev
, const char *devspec
, const char **path
)
93 /* minimum length check */
94 if (strlen(devspec
) < 2)
97 /* look for a device that matches */
98 for (i
= 0; devsw
[i
] != NULL
; i
++) {
100 if (strncmp(devspec
, dv
->dv_name
, strlen(dv
->dv_name
)) == 0)
103 if (devsw
[i
] == NULL
)
106 np
= devspec
+ strlen(dv
->dv_name
);
110 switch (dv
->dv_type
) {
115 idev
= malloc(sizeof (struct disk_devdesc
));
119 err
= disk_parsedev((struct disk_devdesc
*)idev
, np
, path
);
125 idev
= malloc(sizeof (struct zfs_devdesc
));
129 err
= zfs_parsedev((struct zfs_devdesc
*)idev
, np
, path
);
135 idev
= malloc(sizeof (struct devdesc
));
142 if (*np
!= '\0' && *np
!= ':') {
143 /* get unit number if present */
145 unit
= strtol(np
, &cp
, 0);
146 if (errno
!= 0 || cp
== np
) {
151 if (*cp
!= '\0' && *cp
!= ':') {
158 *path
= (*cp
== '\0') ? cp
: cp
+ 1;
163 idev
->d_type
= dv
->dv_type
;
177 efi_fmtdev(void *vdev
)
179 struct devdesc
*dev
= (struct devdesc
*)vdev
;
180 static char buf
[SPECNAMELEN
+ 1];
182 switch (dev
->d_type
) {
184 strlcpy(buf
, "(no device)", sizeof (buf
));
188 return (disk_fmtdev(vdev
));
191 return (zfs_fmtdev(dev
));
194 snprintf(buf
, sizeof (buf
), "%s%d:", dev
->d_dev
->dv_name
,
203 * Set currdev to suit the value being supplied in (value)
206 efi_setcurrdev(struct env_var
*ev
, int flags
, const void *value
)
208 struct devdesc
*ncurr
;
211 rv
= efi_parsedev(&ncurr
, value
, NULL
);
216 env_setenv(ev
->ev_name
, flags
| EV_NOHOOK
, value
, NULL
, NULL
);