背景:我想在我的 usb-uart 设备连接时启动一个日志脚本。
我的/etc/udev/rules.d/10-local.rules
:
SUBSYSTEM=="tty", ATTRS{idVendor}=="1366", ATTRS{idProduct}=="1015", ATTRS{serial}=="000621000000", SYMLINK+="ttymkw", TAG+="systemd", ENV{SYSTEMD_USER_WANTS}+="offnet-uart-log@$env{ID_SERIAL_SHORT}.service"
我的~/.config/systemd/user/[email protected]
:
[Unit]
Description=Start log of UART
[Service]
ExecStart=sh -c "echo %I >> /tmp/systemd.test"
这并没有给我的期望值ID_SERIAL_SHORT
。相反,我得到了一个很长的系统路径:
sys/devices/pci0000:00/0000:00:1c.0/0000:01:00.0/0000:02:02.0/0000:39:00.0/usb3/3-1/3-1.7/3-1.7.2/3-1.7.2:1.0/tty/ttyACM0
当我尝试获取此路径的属性时(我需要删除前导sys
),我可以看到ID_SERIAL_SHORT
:
$ udevadm info --query=property --path /devices/pci0000:00/0000:00:1c.0/0000:01:00.0/0000:02:02.0/0000:39:00.0/usb3/3-1/3-1.7/3-1.7.2/3-1.7.2:1.0/tty/ttyACM0DEVPATH=//devices/pci0000:00/0000:00:1c.0/0000:01:00.0/0000:02:02.0/0000:39:00.0/usb3/3-1/3-1.7/3-1.7.2/3-1.7.2:1.0/tty/ttyACM0
DEVNAME=/dev/ttyACM0
MAJOR=166
MINOR=0
SUBSYSTEM=tty
USEC_INITIALIZED=1633350773062
[email protected]
ID_BUS=usb
ID_VENDOR_ID=1366
ID_MODEL_ID=1015
ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
ID_PCI_INTERFACE_FROM_DATABASE=XHCI
ID_VENDOR_FROM_DATABASE=SEGGER
ID_MODEL_FROM_DATABASE=DSL6340 USB 3.1 Controller [Alpine Ridge]
ID_VENDOR=SEGGER
ID_VENDOR_ENC=SEGGER
ID_MODEL=J-Link
ID_MODEL_ENC=J-Link
ID_REVISION=0100
ID_SERIAL=SEGGER_J-Link_000621000000
ID_SERIAL_SHORT=000621000000 ### <--- here
ID_TYPE=generic
ID_USB_INTERFACES=:020201:0a0000:ffffff:080650:
ID_USB_INTERFACE_NUM=00
ID_USB_DRIVER=cdc_acm
ID_USB_CLASS_FROM_DATABASE=Miscellaneous Device
ID_USB_PROTOCOL_FROM_DATABASE=Interface Association
ID_PATH=pci-0000:39:00.0-usb-0:1.7.2:1.0
ID_PATH_TAG=pci-0000_39_00_0-usb-0_1_7_2_1_0
ID_MM_CANDIDATE=1
DEVLINKS=/dev/ttymkw /dev/serial/by-id/usb-SEGGER_J-Link_000621000000-if00 /dev/serial/by-path/pci-0000:39:00.0-usb-0:1.7.2:1.0
TAGS=:systemd:
$env{ID_SERIAL_SHORT}
在我的 udev 规则中似乎没有被属性的值所取代。
udev 的手册页说该$env{key}
替换仅适用于 NAME、SYMLINK、PROGRAM、OWNER、GROUP、MODE、SECLABEL 和 RUN 字段。
但是,我已经看到人们在ENV{SYSTEMD_USER_WANTS}
. 显然,它对我不起作用。
如何将设备的序列号传递给 systemd 模板?
编辑:我也试过这个,但不是:
SUBSYSTEM=="tty", ATTRS{idVendor}=="1366", ATTRS{idProduct}=="1015", ATTRS{serial}=="000621000000", SYMLINK+="ttymkw", PROGRAM="/bin/systemd-escape -p [email protected] $env{ID_SERIAL_SHORT}", ENV{SYSTEMD_WANTS}+="%c"
我正在寻找的参数是
$attr{serial}
. 我不确定为什么其他解决方案在对其他人有用时不起作用,我猜它们已经过时了(编辑:可能是因为我的规则文件以 10- 开头,是在创建环境变量 ID_SERIAL 的规则文件之前读取的,从 60-) 开始。这是有效的:udev规则:
系统模板:
显然我不需要
TAG
,我需要使用$attr
而不是$env
.系统版本是 241。
对我有帮助的问答。
编辑:不工作的原因
$env
可能ID_SERIAL_SHORT
是在 60 级的规则上创建的,而我的文件是10-local.rules
. 谢谢@yuwata。(未验证。)作为另一种解决方法,如果systemd服务在其ExecStart语句中调用脚本,
udevadm
则问题中显示的相同调用可以包含在该脚本中以提取ID_SERIAL_SHORT
viagrep
。作为一种解决方法,我可以在 udev 规则中创建自己的环境变量:
(为了便于阅读,反斜杠应该在一行上。)
或者真的,直接在
@
.但是,这些不应该是必需的。