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