chiark / gitweb /
a6dfbf1a089dd644ea8e50923cf519b9f6463967
[elogind.git] / extras / path_id / path_id
1 #!/bin/sh
2
3 # provide the shortest possible unique hardware path to a device
4 # for the Linux Persistent Device Naming scheme
5 #
6 # Copyright (C) 2005-2006 SUSE Linux Products GmbH
7 # Author:
8 #       Hannes Reinecke <hare@suse.de>
9 #
10 #       This program is free software; you can redistribute it and/or modify it
11 #       under the terms of the GNU General Public License as published by the
12 #       Free Software Foundation version 2 of the License.
13 #
14 # to be called from a udev rule to return the name for a symlink
15 #       DEVPATH=<devpath>; path_id
16 #       path_id <devpath>
17
18 # examples for all block devices on a system:
19 #       for i in `find /sys/class/block`; do DEVPATH="`echo $i | sed -e 's@^/sys\|/dev@@g'`"; path_id; done
20
21 SYSFS=/sys
22 RESULT=1
23 TYPE=
24 OPWD="`pwd`"
25 full_sysfs_path=
26 full_sysfs_device_path=
27
28 if [ -z "$DEVPATH" -a -z "$1" ] ; then
29         exit 1
30 fi
31
32 if [ -z "$DEVPATH" ] ; then
33         case "$1" in
34                 $SYSFS/*)
35                         DEVPATH="${1#$SYSFS}"
36                         ;;
37                 *)
38                         DEVPATH=$1
39                         ;;
40         esac
41 fi
42
43 if [ ! -e $SYSFS$DEVPATH/dev ] ; then
44         exit 1
45 fi
46
47 case "$DEVPATH" in
48         /devices/*)
49                 cd "$SYSFS$DEVPATH/subsystem";
50                 TYPE="`pwd -P`"
51                 cd "$OPWD"
52                 TYPE="${TYPE##*/}"
53                 ;;
54         /class/*)
55                 TYPE="${DEVPATH#/class/}"
56                 TYPE="${TYPE%%/*}"
57                 ;;
58         /block/*)
59                 TYPE=block
60                 ;;
61         *)
62                 exit 1
63                 ;;
64 esac
65
66 get_port_offset () {
67         local type offset port
68         type=$1
69         offset=$2
70         for i in $type[0-9]* ; do
71                 : i $i
72                 port="${i#$type}"
73                 if [ "$port" -lt "$offset" ] ; then offset=$port ; fi
74         done
75         echo $offset
76 }
77
78 handle_pci () {
79         local DEV=$1
80         cd -P $1
81         DEV=${PWD}
82         pci_id=${DEV##*/}
83         host_dev_path=$DEV
84         while [ ! -z "$host_dev_path" ] ; do
85                 case "$host_dev_path" in
86                         */pci[0-9]*)
87                                 host_dev_path=${host_dev_path%/*}
88                                 ;;
89                         *)
90                                 break
91                                 ;;
92                 esac
93         done
94         d="pci-$pci_id-$d"
95         D="$host_dev_path"
96         RESULT=0
97 }
98
99 handle_platform () {
100         local DEV=$1
101         cd -P $1
102         DEV=${PWD}
103         platform_id=${DEV##*/}
104         host_dev_path=$DEV
105         while [ ! -z "$host_dev_path" ] ; do
106                 case "$host_dev_path" in
107                         */platform*)
108                                 host_dev_path=${host_dev_path%/*}
109                                 ;;
110                         *)
111                                 break
112                                 ;;
113                 esac
114         done
115         if [ "$d" ]; then
116                 d="platform-$platform_id-$d"
117         else
118                 d="platform-$platform_id"
119         fi
120         D="$host_dev_path"
121         RESULT=0
122 }
123
124 handle_serio () {
125         local DEV=$1
126         cd -P $1
127         DEV=${PWD}
128         serio_id=${DEV##*/serio}
129         host_dev_path=$DEV
130         while [ ! -z "$host_dev_path" ] ; do
131                 case "$host_dev_path" in
132                         */serio*)
133                                 host_dev_path=${host_dev_path%/*}
134                                 ;;
135                         *)
136                                 break
137                                 ;;
138                 esac
139         done
140         if [ "$d" ]; then
141                 d="serio-$serio_id-$d"
142         else
143                 d="serio-$serio_id"
144         fi
145         D="$host_dev_path"
146         RESULT=0
147 }
148
149 handle_ide () {
150         : handle_ide $*
151         local DEV=$1
152         local port idedev idecontroller
153         # IDE
154         : DEV $DEV
155         port=${DEV##*/}
156         idedev=${DEV%/*}
157         idecontroller=${idedev%/*}
158         # port info if the controller has more than one interface
159         port="${port#ide}"
160         : port $port d $d
161         : idedev $idedev kernel_port $port
162         case "${port#*.}" in
163                 0)
164                         channel=0
165                         ;;
166                 1)
167                         channel=1
168                         ;;
169                 *)
170                         echo "Error: $idedev is neither master or slave" >&2
171                         ;;
172         esac
173         cd $idecontroller
174         offset="`get_port_offset ide ${port%.*}`"
175         cd "$OPWD"
176         :  port offset $offset
177         port=$((${port%.*} - $offset))
178         if [ "$d" ] ; then
179                 d="ide-${port}:$channel-$d"
180         else
181                 d="ide-${port}:$channel"
182         fi
183         D=$idecontroller
184         RESULT=0
185 }
186
187 handle_scsi () {
188         : handle_scsi $*
189         local DEV=$1
190         local cil controller_port controller_dev
191         # SCSI device
192         cil="${DEV##*/}"
193         cil="${cil#*:}"
194         target_dev=${DEV%/*}
195         target_id=${target_dev##*/target}
196         cd "$target_dev"
197         target_num=0
198         for tid in ${target_id}* ; do
199                 target_num=$(( $target_num + 1 ))
200         done
201         controller_port=${target_dev%/*}
202         controller_dev="${controller_port%/*}"
203         : controller_dev $controller_dev
204         : controller_port $controller_port
205         # a host controller may have more than one interface/port
206         controller_port="${controller_port##*/host}"
207         #
208         cd "$controller_dev"
209         controller_offset=$(get_port_offset host $controller_port)
210         cd "$OPWD"
211         controller_port=$(( $controller_port - $controller_offset))
212         scsi_id="scsi-${controller_port}:${cil}"
213         if [ "$d" ] ; then
214                 d="${scsi_id}-$d"
215         else
216                 d="$scsi_id"
217         fi
218         D="$controller_dev"
219         RESULT=0
220 }
221
222 handle_firewire () {
223         :handle_firewire $*
224         local DEV=$1
225         if [ -f "$D/ieee1394_id" ] ; then
226                 read ieee1394_id < $D/ieee1394_id
227         fi
228         if [ -z "$ieee1394_id" ] ; then
229                 : no IEEE1394 ID
230                 RESULT=1
231                 return
232         fi
233         fw_host_dev=${DEV%/fw-host*}
234         # IEEE1394 devices are always endpoints
235         d="ieee1394-0x$ieee1394_id"
236         D="$fw_host_dev"
237         RESULT=0
238 }
239
240 handle_fc () {
241         : handle_fc $*
242         local DEV=$1
243         local cil controller_port controller_dev
244         # SCSI-FC device
245         fc_tgt_hcil="${DEV##*/}"
246         fc_tgt_lun="${fc_tgt_hcil##*:}"
247         fc_tgt_path="${DEV%/*}"
248         fc_tgt_num="${fc_tgt_path##*/}"
249         fc_tgt_dev="${fc_tgt_path}/fc_transport:${fc_tgt_num}"
250         if [ -e "$fc_tgt_dev/port_name" ]; then
251                 read wwpn < $fc_tgt_dev/port_name
252         fi
253         if [ -z "$wwpn" ] ; then
254                 : no WWPN
255                 RESULT=1
256                 return
257         fi
258         # Linux currently knows about 32bit luns
259         tmp_lun3=$(printf "%04x" $(($fc_tgt_lun & 0xFFFF)))
260         tmp_lun2=$(printf "%04x" $(( ($fc_tgt_lun >> 16) & 0xFFFF)))
261         tmp_lun1="0000"
262         tmp_lun0="0000"
263         if (($fc_tgt_lun == 0)) ; then
264                 lun="0x0000000000000000"
265         else
266                 lun="0x${tmp_lun3}${tmp_lun2}${tmp_lun1}${tmp_lun0}"
267         fi
268         controller_dev="${fc_tgt_path%/host[0-9]*}"
269         # FC devices are always endpoints
270         d="fc-${wwpn}:${lun}"
271         D="$controller_dev"
272         RESULT=0
273 }
274
275 handle_sas () {
276         : handle_sas $*
277         local DEV=$1
278         local cil adapter controller_dev
279         # SAS device
280         sas_host_path="${DEV%%/phy*}"
281         sas_phy_path="${DEV#*/host*/}"
282         sas_phy_path="${sas_phy_path%%/target*}"
283         sas_phy_id="${sas_phy_path%%/*}"
284         sas_rphy_id="${sas_phy_path##*/}"
285         sas_phy_dev="/sys/class/sas_phy/${sas_phy_id}"
286         if [ -e "$sas_phy_dev/sas_address" ]; then
287                 read phy_address < $sas_phy_dev/sas_address
288                 read phy_port < $sas_phy_dev/port_identifier
289                 read phy_id < $sas_phy_dev/phy_identifier
290         fi
291         if [ -z "$phy_address" ] ; then
292                 : no initiator address
293                 RESULT=1
294                 return
295         fi
296         sas_phy_address="$phy_address:$phy_port:$phy_id"
297         sas_rphy_dev="/sys/class/sas_device/${sas_rphy_id}"
298         if [ -e "$sas_rphy_dev/sas_address" ]; then
299                 read rphy_address < $sas_rphy_dev/sas_address
300                 read rphy_id < $sas_rphy_dev/phy_identifier
301         fi
302         if [ -z "$rphy_address" ] ; then
303                 : no initiator address
304                 RESULT=1
305                 return
306         fi
307         sas_rphy_address="$rphy_address:$rphy_id"
308         controller_dev="${sas_host_path%/host[0-9]*}"
309         # SAS devices are always endpoints
310         d="sas-${sas_phy_address}-${sas_rphy_address}"
311         D="$controller_dev"
312         RESULT=0
313 }
314
315 handle_iscsi() {
316         local DEV=$1
317         local iscsi_session_dir
318         local iscsi_session iscsi_session_path
319         local iscsi_connection iscsi_connection_path
320         # iSCSI device
321         iscsi_session_dir="${DEV%%/target*}"
322         iscsi_session="${iscsi_session_dir##*/}"
323         iscsi_session_path=/sys/class/iscsi_session/${iscsi_session}
324         if [ ! -d "$iscsi_session_path" ] ; then
325             : no iSCSI session path
326             RESULT=1
327             return
328         fi
329         # Currently we're not doing MC/S
330         for conn in ${iscsi_session_dir}/connection* ; do
331             iscsi_conn_num=${conn##*:}
332             if [ "$iscsi_conn_num" = '0' ] ; then
333                 iscsi_connection=$(basename $conn)
334             fi
335         done
336         if [ -z "$iscsi_connection" ] ; then
337             : no iSCSI connection found
338             RESULT=1
339             return
340         fi
341         iscsi_connection_path=/sys/class/iscsi_connection/${iscsi_connection}
342         if [ ! -d "$iscsi_connection_path" ] ; then
343             : no iSCSI connection path
344             RESULT=1
345             return
346         fi
347         if [ -e "${iscsi_session_path}/targetname" ]; then
348             read iscsi_tgtname < ${iscsi_session_path}/targetname
349         fi
350         if [ -z "$iscsi_tgtname" ] ; then
351             : No iSCSI Targetname
352             RESULT=1
353             return
354         fi
355         if [ -e "${iscsi_connection_path}/persistent_address" ] ; then
356             read iscsi_address < ${iscsi_connection_path}/persistent_address
357         fi
358         if [ -z "$iscsi_address" ] ; then
359             : No iSCSI Target address
360             RESULT=1
361             return
362         fi
363         if [ -e "${iscsi_connection_path}/persistent_port" ] ; then
364             read iscsi_port < ${iscsi_connection_path}/persistent_port
365         fi
366         d="ip-${iscsi_address}:${iscsi_port}-iscsi-${iscsi_tgtname}"
367         RESULT=0
368 }
369
370 handle_usb () {
371 : handle_usb $*
372         local DEV=$1
373         cd -P $1
374         DEV=${PWD}
375         port_id=${DEV##*/}
376         port_num=${port_id#*-}
377         host_dev_path=$DEV
378         while [ ! -z "$host_dev_path" ] ; do
379                 case "$host_dev_path" in
380                         */usb*)
381                                 usb_host_path=$host_dev_path
382                                 host_dev_path="${host_dev_path%/*}"
383                                 ;;
384                         *)
385                                 break
386                                 ;;
387                 esac
388         done
389         : host_dev_path $host_dev_path
390         usb_host_num=${usb_host_path##*/usb}
391
392         cd "$host_dev_path"
393         usb_host_offset=$(get_port_offset usb $usb_host_num)
394         usb_host_port=$(($usb_host_num - $usb_host_offset))
395         cd "$OPWD"
396         if [ "$d" ] ; then
397                 d="usb-$usb_host_port:$port_num-${d}"
398         else
399                 d="usb-$usb_host_port:$port_num"
400         fi
401         D="$host_dev_path"
402         RESULT=0
403 }
404
405 handle_device () {
406         full_sysfs_path="$SYSFS$DEVPATH"
407         if [ -L $full_sysfs_path/subsystem ]; then
408                 # new sysfs block layout
409                 full_sysfs_path="${full_sysfs_path%/*}"
410                 cd "$full_sysfs_path/subsystem";
411                 subsys="`pwd -P`"
412                 cd "$OPWD"
413                 subsys="${subsys##*/}"
414                 if [ "$subsys" = "block" ]; then
415                         # parent is "block", it's a partition, move one up
416                         full_sysfs_path="${full_sysfs_path%/*}"
417                 fi
418                 cd $full_sysfs_path
419         else
420                 # old sysfs block layout
421                 if [ ! -L $full_sysfs_path/device ] ; then
422                         if [ -f $full_sysfs_path/range ] ; then return ; fi
423                         full_sysfs_path="${full_sysfs_path%/*}"
424                         : full_sysfs_path "$full_sysfs_path"
425                         if [ ! -L $full_sysfs_path/device -o ! -f $full_sysfs_path/dev ] ; then
426                                 return
427                         fi
428                 fi
429                 cd $full_sysfs_path/device
430         fi
431         full_sysfs_device_path="`pwd -P`"
432         cd "$OPWD"
433         D=$full_sysfs_device_path
434         while [ ! -z "$D" ] ; do
435                 case "$D" in
436                         */ide[0-9]/[0-9].[0-9]*|*/ide[0-9][0-9]/[0-9][0-9].[0-9]*)
437                                 handle_ide "$D"
438                                 ;;
439                         */css0/*)
440                                 if [ -r $full_sysfs_device_path/wwpn ]; then
441                                         read wwpn < $full_sysfs_device_path/wwpn
442                                 fi
443                                 if [ -r $full_sysfs_device_path/fcp_lun ]; then
444                                         read lun < $full_sysfs_device_path/fcp_lun
445                                 fi
446                                 if [ -r $full_sysfs_device_path/hba_id ]; then
447                                         read bus_id < $full_sysfs_device_path/hba_id
448                                 fi
449                                 if [ "$bus_id" -a "$wwpn" -a "$lun" ]; then
450                                         # S/390 zfcp adapter
451                                         d="ccw-$bus_id-zfcp-$wwpn:$lun"
452                                         RESULT=0
453                                 else
454                                         # DASD devices
455                                         bus="ccw"
456                                         adapter=${D##*/}
457                                         d="$bus-$adapter"
458                                         RESULT=0
459                                 fi
460                                 D=
461                                 ;;
462                         */rport-[0-9]*:[0-9]*-[0-9]*/*)
463                                 handle_fc "$D"
464                                 ;;
465                         */phy-[0-9]*:[0-9]*/*)
466                                 handle_sas "$D"
467                                 ;;
468                         */fw-host[0-9]*/*)
469                                 handle_firewire "$D"
470                                 ;;
471                         */session[0-9]*/*)
472                                 handle_iscsi "$D"
473                                 D=
474                                 ;;
475                         */host[0-9]*/[0-9]*:[0-9]*:[0-9]*:[0-9]*)
476                                 handle_scsi "$D"
477                                 ;;
478                         */usb[0-9]*/[0-9]*/*)
479                                 handle_usb "$D"
480                                 ;;
481                         */pci[0-9]*:[0-9]*)
482                                 handle_pci "$D"
483                                 ;;
484                         */serio[0-9]*)
485                                 handle_serio "$D"
486                                 ;;
487                         */platform/*)
488                                 handle_platform "$D"
489                                 ;;
490                         */devices)
491                                 D=
492                                 ;;
493                         *)
494                                 : not handled
495                                 RESULT=1
496                                 return
497                                 ;;
498                 esac
499         done
500         if [ "$TYPE" = "scsi_tape" ] ; then
501                 devname=${full_sysfs_path##*/}
502                 rewind="${devname%%st*}"
503                 mode="${devname##*st}"
504                 case "$mode" in
505                         *l)
506                                 mode="l"
507                                 ;;
508                         *m)
509                                 mode="m"
510                                 ;;
511                         *a)
512                                 mode="a"
513                                 ;;
514                         *)
515                                 mode=""
516                            ;;
517                 esac
518                 if [ "$d" ]; then
519                         d="$d-${rewind}st${mode}"
520                 fi
521         fi
522 }
523
524 case "$TYPE" in
525         block)
526                 handle_device
527                 echo "ID_PATH=$d"
528                 ;;
529         scsi_tape)
530                 handle_device
531                 echo "ID_PATH=$d"
532                 ;;
533         input)
534                 handle_device
535                 echo "ID_PATH=$d"
536                 ;;
537         *)
538                 RESULT=1
539                 ;;
540 esac
541
542 exit $RESULT