Bluetooth Hands-Free Profile (HFP) for Raspberry Pi
This is another part of adding a Raspberry Pi 5 to my 3D-printed Mandalorian helmet. This post will cover how I set up the Pi to function as a Bluetooth Hands-Free device for my phone so I can take calls while wearing the helmet since I will also have internal speakers and a microphone.
Installing the Necessary Libraries
I found these two articles on setting up a Raspberry Pi as a Hands-Free device: Enabling Hands-Free Profile on Raspberry Pi (Raspbian Stretch) by using PulseAudio and HFP on Raspberry Pi. Both of those articles were written in 2017, so some of the information is out of date. For example, the OS for the Raspberry Pi isn't even called "Raspbian" anymore. At the time of writing this, the latest version is the Raspberry Pi OS "Bookworm". So here are the steps that I took to get HFP working on my Pi 5.
$ sudo apt-get install pulseaudio pulseaudio-module-bluetooth
The first step is to install PulseAudio, along with the Bluetooth module for PulseAudio. The most recent of the two articles I mentioned, which was written in September of 2017, mentions that the version of PulseAudio which shipped with Raspbian Stretch was version 5.0, while the project needs version 6.0 or higher. That isn't an issue with Bookworm because PulseAudio no longer ships with Raspberry Pi OS. So whatever version that gets installed by the above command is going to be 6.0 or higher. In the article, version 7.1 was already released. For me, version 16.1 was installed.
While we are installing packages, do this as well:
sudo apt-get install dbus python3-dbus libdbus-1-dev libglib2.0-dev libudev-dev
I didn't figure out that I needed D-Bus until later, but it will be necessary. I also found out later that I needed to have GLib installed in order to build Ofono. Specifically the development version, because GLib-2.0 comes installed with Raspberry Pi OS by default. The development version does not.
Installing Ofono
$ wget https://www.kernel.org/pub/linux/network/ofono/ofono-2.9.tar.gz
$ tar -xzvf ofono-2.9.tar.gz
$ cd ofono-2.9 ## If necessary
The next step is to install the Ofono library, which will provide telephony services that are apparently necessary for the Pi to be recognized by the phone as a Bluetooth Hands-Free device, and that should, in theory, allow the Pi to display caller ID and call duration information, as well as allow the Pi to issue the "accept call" or "end call" commands to the phone. In my case, I'll be pairing an iPhone to the Pi.
The simplest way to compile this package is:
1. `cd' to the directory containing the package's source code and type
`./configure' to configure the package for your system. If you're
using `csh' on an old version of System V, you might need to type
`sh ./configure' instead to prevent `csh' from trying to execute
`configure' itself.
Running `configure' takes awhile. While running, it prints some
messages telling which features it is checking for.
2. Type `make' to compile the package.
3. Optionally, type `make check' to run any self-tests that come with
the package.
4. Type `make install' to install the programs and any data files and
documentation.
5. You can remove the program binaries and object files from the
source code directory by typing `make clean'. To also remove the
files that `configure' created (so you can compile the package for
a different kind of computer), type `make distclean'. There is
also a `make maintainer-clean' target, but that is intended mainly
for the package's developers. If you use it, you may have to get
all sorts of other programs in order to regenerate files that came
with the distribution.
The above text is from the INSTALL file in the ofonos-2.9 directory that gets decompressed in the previous step. The README said to run the configure script with these arguments:
./configure --prefix=/usr --mandir=/usr/share/man --sysconfdir=/etc --localstatedir=/var
I ran that command in the same directory as the README and INSTALL files since that is where the configure script is located. After some trial and error, I finally got all the proper libraries installed and the configure script ran without issue. Once the script was successful, I ran `sudo make && sudo make install`, which generated build files. If either make or make install are not run as super user, they won't be able to do what they need to do.
$sudo systemctl daemon-reload
$sudo systemctl start ofonos
$sudo systemctl status ofonos
After `sudo make install` successfully finishes running, it will probably be necessary to reload the system daemons because the Ofonos one will have changed due to the installation process. After that, the System Control commands will start Ofonos as a service, and the last command will show the status, which should say "active (running)".
Setting Up PulseAudio
$ sudo nano /etc/pulse/default.pa
### Automatically load driver modules for Bluetooth hardware
.ifexists module-bluetooth-policy.so
load-module module-bluetooth-policy
.endif
.ifexists module-bluetooth-discover.so
load-module module-bluetooth-discover autodetect_mtu=yes headset=ofono
.endif
Once PulseAudio and Ofono are installed, run the nano command to open the config file for the service and add the part about `headset=ofono` to the `load-module module-bluetooth-discover autodetect_mtu=yes` line. Also, check for this code block in that config file, and add it if missing:
### Echo cancel and noise reduction
.ifexists module-echo-cancel.so
load-module module-echo-cancel aec_method=webrtc source_name=ec_out sink_name=ec_ref
set-default-source ec_out
set-default-sink ec_ref
.endif
Since the mic and speaker are both inside a helmet, this seems like an important addition to have, and was mentioned in the older of the two articles.
pulseaudio --start
Finally, start the PulseAudio service.
Pairing My Phone to the Pi 5
bluethoothctl
power on
discoverable on
pair on
agent on
default-agent
This will enable the Bluetooth on the Pi 5 to be discoverable on the phone's Bluetooth settings page. Select the name of the Pi once it appears (I called mine "helmet"). On my iPhone, a pop-up screen asked me to verify the code matched the one displayed on the Pi, which meant checking the command line and typing "yes" if they matched while clicking "OK" on my phone screen. That was all that I had to do in order to pair my phone to the Pi.
Testing it Out
~/ofono-2.9/test $ python dial-number 1234567890
After all these steps had been completed, I navigated to the `test` directory and ran the script `dial-number`, giving it a full ten-digit telephone number. A few seconds later, my phone started ringing as it dialed the number! I haven't had a chance to test the audio input or output using the Pi itself, but my speakers and microphone should be delivered soon.