我的笔记本电脑的蓝牙适配器似乎不是很强大,当我同时连接多个音频设备时,音频由于某种原因变得断断续续。
因此,我想编写一个服务,当有新设备连接时,它会断开所有其他蓝牙音频设备的连接。我遇到的主要问题是,我不知道如何捕捉设备添加事件,以便随后识别哪些设备已添加,这样我就可以断开其余设备。
我一直在尝试的是 udev。我已经有一个 udev 规则来捕获所有蓝牙添加事件,如下所示:
ACTION=="add", SUBSYSTEM=="bluetooth", TAG+="systemd", ENV{SYSTEMD_USER_WANTS}="bluetooth-add-event.service"
因此经过一番挖掘,我发现我可以将设备信息写入临时文件,然后在服务启动的脚本中引用它,所以我添加了这个进行测试:
RUN+="/bin/sh -c 'echo %k_%b_$driver_%E{HID_ID} > /run/btadd-id'"
但输出结果却毫无帮助:
hci0:256___
我不知道有什么方法可以实际使用它来识别已知的设备bluetoothctl
,它可以告诉我有关该设备的信息:
~$ bluetoothctl info 25:4E:4A:C7:43:C4
Device 25:4E:4A:C7:43:C4 (public)
Name: JR-CB1
Alias: JR-CB1
Class: 0x00240404
Icon: audio-headset
Paired: yes
Bonded: yes
Trusted: yes
Blocked: no
Connected: yes
LegacyPairing: no
UUID: Serial Port (00001101-0000-1000-8000-00805f9b34fb)
UUID: Audio Sink (0000110b-0000-1000-8000-00805f9b34fb)
UUID: A/V Remote Control Target (0000110c-0000-1000-8000-00805f9b34fb)
UUID: Advanced Audio Distribu.. (0000110d-0000-1000-8000-00805f9b34fb)
UUID: A/V Remote Control (0000110e-0000-1000-8000-00805f9b34fb)
UUID: Handsfree (0000111e-0000-1000-8000-00805f9b34fb)
UUID: PnP Information (00001200-0000-1000-8000-00805f9b34fb)
UUID: Vendor specific (fe010000-1234-5678-abcd-00805f9b34fb)
Modalias: bluetooth:v05D6p000Ad0240
另一方面,udev 给了我这个:
~$ udevadm info --path /devices/pci0000:00/0000:00:14.0/usb2/2-7/2-7:1.0/bluetooth/hci0/hci0:256
P: /devices/pci0000:00/0000:00:14.0/usb2/2-7/2-7:1.0/bluetooth/hci0/hci0:256
M: hci0:256
R: 256
U: bluetooth
T: link
E: DEVPATH=/devices/pci0000:00/0000:00:14.0/usb2/2-7/2-7:1.0/bluetooth/hci0/hci0:256
E: SUBSYSTEM=bluetooth
E: DEVTYPE=link
E: USEC_INITIALIZED=34817721040
E: SYSTEMD_USER_WANTS=bluetooth-add-event.service bluetooth.target
E: SYSTEMD_ALIAS=/sys/subsystem/bluetooth/devices/hci0:256
E: SYSTEMD_WANTS=bluetooth.target
E: TAGS=:systemd:
E: CURRENT_TAGS=:systemd:
这些属性都不对应。
说实话,我不知道还能尝试什么。有没有可行的方法可以跳过 udev 并直接通过 bluez 处理添加事件?
答案1
解决方案是使用 udev 的“输入”子系统。规则:
SUBSYSTEM=="input",KERNEL=="input*",ENV{ID_BUS}=="bluetooth",ACTION=="add",TAG+="systemd",RUN+="/bin/sh -c 'echo $attr{name} > /run/input-added'",ENV{SYSTEMD_USER_WANTS}+="pipewire-bt-add.service"
每次在蓝牙总线上添加输入时都会触发此命令。KERNEL
需要指定,以便它仅针对input#
事件触发,而不是input#/event#
。该RUN
命令将name
添加设备的属性(例如“JR-CB1 _AVRCP”)写入我的脚本可以访问的文件中。不幸的是,$attr{phys}
无法使用,因为它只是返回蓝牙适配器的 ID,而不是远程设备的 ID。
服务文件非常标准:
[Unit]
Description=Pipewire BT Device Add Event
Requires=bluetooth.target
After=bluetooth.target
[Service]
ExecStart=/home/lutra/.local/bin/pw-bt-add.sh
Type=exec
Restart=no
[Install]
WantedBy=default.target
脚本非常简单:
#!/bin/bash
bluetoothctl devices Connected | while read devType devId devName; do
if pw-cli list-objects | grep "$devName" && ! cat /run/input-added | grep "$devName"; then
bluetoothctl disconnect "$devId"
fi
done
它循环遍历已连接的蓝牙设备,然后从之前创建的文件中查找它们的设备名称。如果 pipewire 识别该设备,但与刚刚添加的设备不匹配,则会断开连接。