systemd/udev/bluez:连接时删除其他蓝牙音频设备

systemd/udev/bluez:连接时删除其他蓝牙音频设备

我的笔记本电脑的蓝牙适配器似乎不是很强大,当我同时连接多个音频设备时,音频由于某种原因变得断断续续。

因此,我想编写一个服务,当有新设备连接时,它会断开所有其他蓝牙音频设备的连接。我遇到的主要问题是,我不知道如何捕捉设备添加事件,以便随后识别哪些设备已添加,这样我就可以断开其余设备。

我一直在尝试的是 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 识别该设备,但与刚刚添加的设备不匹配,则会断开连接。

相关内容