1*22ce4affSfengbojiang /*
2*22ce4affSfengbojiang * CDDL HEADER START
3*22ce4affSfengbojiang *
4*22ce4affSfengbojiang * The contents of this file are subject to the terms of the
5*22ce4affSfengbojiang * Common Development and Distribution License (the "License").
6*22ce4affSfengbojiang * You may not use this file except in compliance with the License.
7*22ce4affSfengbojiang *
8*22ce4affSfengbojiang * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*22ce4affSfengbojiang * or http://www.opensolaris.org/os/licensing.
10*22ce4affSfengbojiang * See the License for the specific language governing permissions
11*22ce4affSfengbojiang * and limitations under the License.
12*22ce4affSfengbojiang *
13*22ce4affSfengbojiang * When distributing Covered Code, include this CDDL HEADER in each
14*22ce4affSfengbojiang * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*22ce4affSfengbojiang * If applicable, add the following below this CDDL HEADER, with the
16*22ce4affSfengbojiang * fields enclosed by brackets "[]" replaced with your own identifying
17*22ce4affSfengbojiang * information: Portions Copyright [yyyy] [name of copyright owner]
18*22ce4affSfengbojiang *
19*22ce4affSfengbojiang * CDDL HEADER END
20*22ce4affSfengbojiang */
21*22ce4affSfengbojiang
22*22ce4affSfengbojiang /*
23*22ce4affSfengbojiang * Copyright 2007 Sun Microsystems, Inc. All rights reserved.
24*22ce4affSfengbojiang * Use is subject to license terms.
25*22ce4affSfengbojiang *
26*22ce4affSfengbojiang * Copyright (c) 2016, Intel Corporation.
27*22ce4affSfengbojiang */
28*22ce4affSfengbojiang
29*22ce4affSfengbojiang #include <sys/types.h>
30*22ce4affSfengbojiang #include <sys/param.h>
31*22ce4affSfengbojiang #include <sys/stat.h>
32*22ce4affSfengbojiang #include <libudev.h>
33*22ce4affSfengbojiang #include <errno.h>
34*22ce4affSfengbojiang #include <stdio.h>
35*22ce4affSfengbojiang #include <stdlib.h>
36*22ce4affSfengbojiang #include <string.h>
37*22ce4affSfengbojiang #include <fcntl.h>
38*22ce4affSfengbojiang
39*22ce4affSfengbojiang /*
40*22ce4affSfengbojiang * Linux persistent device strings for vdev labels
41*22ce4affSfengbojiang *
42*22ce4affSfengbojiang * based on udev_device_get_devid() at zfs/lib/libzfs/libzfs_import.c
43*22ce4affSfengbojiang */
44*22ce4affSfengbojiang
45*22ce4affSfengbojiang #define DEV_BYID_PATH "/dev/disk/by-id/"
46*22ce4affSfengbojiang
47*22ce4affSfengbojiang static int
udev_device_get_devid(struct udev_device * dev,char * bufptr,size_t buflen)48*22ce4affSfengbojiang udev_device_get_devid(struct udev_device *dev, char *bufptr, size_t buflen)
49*22ce4affSfengbojiang {
50*22ce4affSfengbojiang struct udev_list_entry *entry;
51*22ce4affSfengbojiang const char *bus;
52*22ce4affSfengbojiang char devbyid[MAXPATHLEN];
53*22ce4affSfengbojiang
54*22ce4affSfengbojiang /* The bus based by-id path is preferred */
55*22ce4affSfengbojiang bus = udev_device_get_property_value(dev, "ID_BUS");
56*22ce4affSfengbojiang
57*22ce4affSfengbojiang if (bus == NULL) {
58*22ce4affSfengbojiang const char *dm_uuid;
59*22ce4affSfengbojiang
60*22ce4affSfengbojiang /*
61*22ce4affSfengbojiang * For multipath nodes use the persistent uuid based identifier
62*22ce4affSfengbojiang *
63*22ce4affSfengbojiang * Example: 'dm-uuid-mpath-35000c5006304de3f'
64*22ce4affSfengbojiang */
65*22ce4affSfengbojiang dm_uuid = udev_device_get_property_value(dev, "DM_UUID");
66*22ce4affSfengbojiang if (dm_uuid != NULL) {
67*22ce4affSfengbojiang (void) snprintf(bufptr, buflen, "dm-uuid-%s", dm_uuid);
68*22ce4affSfengbojiang return (0);
69*22ce4affSfengbojiang }
70*22ce4affSfengbojiang return (ENODATA);
71*22ce4affSfengbojiang }
72*22ce4affSfengbojiang
73*22ce4affSfengbojiang /*
74*22ce4affSfengbojiang * locate the bus specific by-id link
75*22ce4affSfengbojiang *
76*22ce4affSfengbojiang * Example: 'scsi-MG03SCA300_350000494a8cb3d67-part1'
77*22ce4affSfengbojiang */
78*22ce4affSfengbojiang (void) snprintf(devbyid, sizeof (devbyid), "%s%s-", DEV_BYID_PATH, bus);
79*22ce4affSfengbojiang entry = udev_device_get_devlinks_list_entry(dev);
80*22ce4affSfengbojiang while (entry != NULL) {
81*22ce4affSfengbojiang const char *name;
82*22ce4affSfengbojiang
83*22ce4affSfengbojiang name = udev_list_entry_get_name(entry);
84*22ce4affSfengbojiang if (strncmp(name, devbyid, strlen(devbyid)) == 0) {
85*22ce4affSfengbojiang name += strlen(DEV_BYID_PATH);
86*22ce4affSfengbojiang (void) stpncpy(bufptr, name, buflen - 1);
87*22ce4affSfengbojiang bufptr[buflen - 1] = '\0';
88*22ce4affSfengbojiang return (0);
89*22ce4affSfengbojiang }
90*22ce4affSfengbojiang entry = udev_list_entry_get_next(entry);
91*22ce4affSfengbojiang }
92*22ce4affSfengbojiang
93*22ce4affSfengbojiang return (ENODATA);
94*22ce4affSfengbojiang }
95*22ce4affSfengbojiang
96*22ce4affSfengbojiang /*
97*22ce4affSfengbojiang * Usage: devname2devid <devicepath>
98*22ce4affSfengbojiang *
99*22ce4affSfengbojiang * Examples:
100*22ce4affSfengbojiang * # ./devname2devid /dev/sda1
101*22ce4affSfengbojiang * devid scsi-350000394a8caede4-part1
102*22ce4affSfengbojiang *
103*22ce4affSfengbojiang * # ./devname2devid /dev/dm-1
104*22ce4affSfengbojiang * devid: 'dm-uuid-mpath-35000c5006304de3f'
105*22ce4affSfengbojiang *
106*22ce4affSfengbojiang * This program accepts a disk or disk slice path and prints a
107*22ce4affSfengbojiang * device id.
108*22ce4affSfengbojiang *
109*22ce4affSfengbojiang * Exit values:
110*22ce4affSfengbojiang * 0 - means success
111*22ce4affSfengbojiang * 1 - means failure
112*22ce4affSfengbojiang *
113*22ce4affSfengbojiang */
114*22ce4affSfengbojiang int
main(int argc,char * argv[])115*22ce4affSfengbojiang main(int argc, char *argv[])
116*22ce4affSfengbojiang {
117*22ce4affSfengbojiang struct udev *udev;
118*22ce4affSfengbojiang struct udev_device *dev = NULL;
119*22ce4affSfengbojiang char devid[128], nodepath[MAXPATHLEN];
120*22ce4affSfengbojiang char *device, *sysname;
121*22ce4affSfengbojiang int ret;
122*22ce4affSfengbojiang
123*22ce4affSfengbojiang if (argc == 1) {
124*22ce4affSfengbojiang (void) printf("%s <devicepath> [search path]\n", argv[0]);
125*22ce4affSfengbojiang exit(1);
126*22ce4affSfengbojiang }
127*22ce4affSfengbojiang device = argv[1];
128*22ce4affSfengbojiang
129*22ce4affSfengbojiang if ((udev = udev_new()) == NULL) {
130*22ce4affSfengbojiang perror("udev_new");
131*22ce4affSfengbojiang exit(1);
132*22ce4affSfengbojiang }
133*22ce4affSfengbojiang
134*22ce4affSfengbojiang /* resolve path to a runtime device node instance */
135*22ce4affSfengbojiang if (realpath(device, nodepath) == NULL) {
136*22ce4affSfengbojiang perror("realpath");
137*22ce4affSfengbojiang exit(1);
138*22ce4affSfengbojiang }
139*22ce4affSfengbojiang sysname = strrchr(nodepath, '/') + 1;
140*22ce4affSfengbojiang
141*22ce4affSfengbojiang if ((dev = udev_device_new_from_subsystem_sysname(udev, "block",
142*22ce4affSfengbojiang sysname)) == NULL) {
143*22ce4affSfengbojiang perror(sysname);
144*22ce4affSfengbojiang exit(1);
145*22ce4affSfengbojiang }
146*22ce4affSfengbojiang
147*22ce4affSfengbojiang if ((ret = udev_device_get_devid(dev, devid, sizeof (devid))) != 0) {
148*22ce4affSfengbojiang udev_device_unref(dev);
149*22ce4affSfengbojiang errno = ret;
150*22ce4affSfengbojiang perror(sysname);
151*22ce4affSfengbojiang exit(1);
152*22ce4affSfengbojiang }
153*22ce4affSfengbojiang
154*22ce4affSfengbojiang (void) printf("devid %s\n", devid);
155*22ce4affSfengbojiang
156*22ce4affSfengbojiang udev_device_unref(dev);
157*22ce4affSfengbojiang udev_unref(udev);
158*22ce4affSfengbojiang
159*22ce4affSfengbojiang return (0);
160*22ce4affSfengbojiang }
161