A sample ASHA implementation for Linux designed to work with pipewire and bluez.
This project mostly follows the Hearing Aid Audio Support Using Bluetooth LE document created by Google. Note that the specification varies in minor ways from the actual implementation used in the android source code.
This project will recognize ASHA-compatible hearing devices that have been connected via bluetooth, and it will create virtual pipewire sinks that the user can select to stream audio to the hearing devices.
Android's ASHA has been superceded by Bluetooth LE audio. Bluez already supports LE Audio, and I rather suspect that future hearing devices will no longer support the older ASHA protocol. However:
- There are a large number of existing devices that use ASHA. Additionally, some of the newest releases are still ASHA-only. Since these devices tend to be very expensive, they have an extremely long replacement cycle. I suspect that many of these devices will be around for at least another ten years.
- LE Audio-enabled hearing devices are dual mode, meaning they can use either ASHA or LE Audio. This dual functionality has proven useful since LE Audio implementation by manufacturers has been problematic so far.
If you have the ability to use LE Audio, then you should definitely prefer it over this project because:
- LE Audio offers lower latency.
- Your devices can operate in hands-free mode, functioning as both receivers and input microphones (a feature not available with ASHA).
- LE Audio utilizes a vastly improved audio codec called LC3, replacing the outdated G.722 codec. While this upgrade is promising in theory, hearing devices are limited by the capabilities of their receivers, restricting their output to 8-12kHz. Consequently, there are improvements in sound quality, albeit marginal.
However, if your hearing devices only support ASHA, then read on.
The ASHA spec relies on the central manipulating the stream properties to match what the hearing device expects, but bluez is designed to allow the peripheral to set those properties itself (spoiler alert, they don't). In order to get a good listening experience, you will have to manually configure the bluetooth service and the bluetooth kernel module to set those properties yourself.
There is at least one effort (see @ford-prefect's branch here, along with the discussions here) to implement ASHA which just recently got merged into BlueZ. I am fully looking forward to throwing away my code and using integrated support instead, but I don't expect it to land into mainstream stable Linux distributions for some time.
You will need at least a bluetooth 5.0 adapter. I recommend 5.2 if you can get it.
Bluetooth adapters vary a lot in quality and compatibilty. I have tested this on an ASUS BT-500 usb adaptor and on an Intel AX200 wifi/bluetooth adapter. The ASUS device claimed 2M PHY and DLE support, but was still only able to stream to one hearing aid reliably. btmon captures showed that it was only sending 27 bytes of data at a time, apparently not using the DLE support. The Intel device was able to work reliably for both devices, but only after manually enabling 2M PHY.
The ASHA spec requires LE credit based flow control, which is turned off by default in the Linux bluetooth kernel module. This can be turned on using the enable_ecred
module paramter. On my system, I have created this file:
/etc/modprobe.d/bluetooth_asha.conf
options bluetooth enable_ecred=1
After adding this file, you will want to reload the bluetooth module. The easiest way is probably just to reboot your system.
The ASHA spec requires the central to set the connection interval to match the data transfer rate. Supposedly this is flexible, but I have only ever gotten a 20ms transfer rate to work on my device. Your results may vary. The default interval set by bluez is 30ms, but this can be adjusted by editing the bluetooth configuration. These configuration items will already already be present, but they are commented out, and have the wrong values. Note that these values are set in units of 1.25ms, so 20 / 1.25 = 16.
/etc/bluetooth/main.conf
[LE]
# LE default connection parameters. These values are superceeded by any
# specific values provided via the Load Connection Parameters interface
MinConnectionInterval=16
MaxConnectionInterval=16
ConnectionLatency=10
ConnectionSupervisionTimeout=100
You will want to restart bluetooth after making these config changes
sudo systemctl restart bluetooth
If you have a kernel older than 6.8, 2M PHY will not be automatically enabled. I have had success with both 1M and 2M PHY, but people have reported only getting it to work with 2M PHY. I haven't been able to find an option to configure this using dbus, the kernel, or the bluez configuration, so I have had to resort to using btmgmt commands to set the default PHY configuration. You can check your existing enabled PHYs with btmgmt phy
. If the Configurable phys
contains LE2MTX
and LE2MRX
, but the Selected phys
does not, then you can copy your existing Selected phys
, and add the values LE2MTX LE2MRX
to it. On my system, that looks like this:
# Check the existing phys
$ sudo btmgmt phy
Supported phys: BR1M1SLOT BR1M3SLOT BR1M5SLOT EDR2M1SLOT EDR2M3SLOT EDR2M5SLOT EDR3M1SLOT EDR3M3SLOT EDR3M5SLOT LE1MTX LE1MRX LE2MTX LE2MRX LECODEDTX LECODEDRX
Configurable phys: BR1M3SLOT BR1M5SLOT EDR2M1SLOT EDR2M3SLOT EDR2M5SLOT EDR3M1SLOT EDR3M3SLOT EDR3M5SLOT LE2MTX LE2MRX LECODEDTX LECODEDRX
Selected phys: BR1M1SLOT BR1M3SLOT BR1M5SLOT EDR2M1SLOT EDR2M3SLOT EDR2M5SLOT EDR3M1SLOT EDR3M3SLOT EDR3M5SLOT LE1MTX LE1MRX
# copy the Selected phys, and add the new LE2MTX LE2MRX values to it
$ sudo btmgmt phy BR1M1SLOT BR1M3SLOT BR1M5SLOT EDR2M1SLOT EDR2M3SLOT EDR2M5SLOT EDR3M1SLOT EDR3M3SLOT EDR3M5SLOT LE1MTX LE1MRX LE2MTX LE2MRX
I haven't yet found a configuration setting that will make this setup persist through a power cycle of the bluetooth adapter. This step must be run before connecting your devices.
- glib development packages
- pipewire development packages
- bluetooth development packages
- CMake 3.9
- C++14 compiler
On my debian development system, I can install all of this with:
sudo apt-get install build-essential cmake libglib2.0-dev libpipewire-0.3-dev libbluetooth-dev
Fairly standard cmake-based compile:
mkdir build
cd build
cmake ..
make
This must be done before connecting your hearing aids. Please read Enable 2M Phy for more details, but on my box, it looks like this:
sudo btmgmt phy BR1M1SLOT BR1M3SLOT BR1M5SLOT EDR2M1SLOT EDR2M3SLOT EDR2M5SLOT EDR3M1SLOT EDR3M3SLOT EDR3M5SLOT LE1MTX LE1MRX LE2MTX LE2MRX
Connect your hearing aids using your standard bluetooth device dialog.
Run the asha_connection_test
utility. This will check for common configuration mistakes and provide some debugging output that will be of use for troubleshooting.
Run asha_pipewire_sink
. If you want to see lots of debug output, you can set the environment variable G_MESSAGES_DEBUG=all
Once asha_pipewire_sink
detects your devices, it should create a new virtual pipewire sink that you can select as an audio device. You may have to select Show Virtual Devices
in the KDE volume control or Show: All Output Devices
in pavucontrol to see it.
If the audio is choppy, delayed, or sounds like it is shifting from ear to ear, then your adapter may not be able to keep up with the bandwidth requirements. Try connecting a single device and see if the quality improves.
Please use the issue reporter to let me know of your experiences, positive or negative, trying to get this to work. I would love to generate a compatibility table, and perhaps to troubleshoot and add support for additional devices.
Please check the wiki for current testing status of various devices.