Convert a Raspberry Pi into a HID relay that translates Bluetooth keyboard and mouse input to USB. Minimal configuration. Zero hassle.
The issue with Bluetooth devices is that you usually can't use them to:
- wake up sleeping devices,
- access the BIOS or OS select menu (e.g., GRUB),
- access devices without Bluetooth interface (e.g., devices in a restricted environment or most KVM switches).
Sounds familiar? Congratulations! You just found the solution!
Linux's gadget mode allows a Raspberry Pi to act as USB HID (Human Interface Device). Therefore, from the host's perspective, it appears like a regular USB keyboard or mouse. You may think of your Pi as a multi-device Bluetooth dongle.
- 1. Features
- 2. Requirements
- 3. Installation
- 4. Usage
- 5. Updating
- 6. Uninstallation
- 7. Troubleshooting
- 8. Bonus points
- 9. Contributing
- 10. License
- 11. Acknowledgments
- Simple installation and highly automated setup
- Supports multiple input devices (currently keyboard and mouse - more than one of each kind simultaneously)
- Supports 146 multimedia keys (e.g., mute, volume up/down, launch browser, etc.)
- Auto-discovery feature for input devices
- Auto-reconnect feature for input devices (power off, energy saving mode, out of range, etc.)
- Pause/resume relaying input devices via configurable shortcut
- Robust error handling and logging
- Installation as a systemd service
- Reliable concurrency using state-of-the-art TaskGroups
- Clean and actively maintained code base
- A Raspberry Pi with Bluetooth and USB OTG support required for USB gadgets in so-called device mode. Supported models include:
- Raspberry Pi Zero W(H): Includes Bluetooth 4.1 and supports USB OTG with the lowest price tag.
- Raspberry Pi Zero 2 W: Similar to the Raspberry Pi Zero W, it has Bluetooth 4.1 and USB OTG support while providing additional processing power.
- Raspberry Pi 4B/5: Offers Bluetooth 5.0 and USB-C OTG support for device mode, providing the best performance.
- Raspberry Pi OS (Bookworm-based)
- Python 3.11+ for using TaskGroups.
Note
Raspberry Pi 3 Models feature Bluetooth 4.2 but no native USB gadget mode support. Earlier models like Raspberry Pi 1 and 2 do not support Bluetooth natively and have no USB gadget mode support.
Note
The latest version of Raspberry Pi OS, based on Debian Bookworm, supports Python 3.11 through the official package repositories. For older versions, you may build it from source. Note that building may take anything between a few minutes (Pi 4B) and more than an hour (Pi 0W).
Follow these steps to install and configure the project:
-
Install Raspberry Pi OS on your Raspberry Pi (e.g., using Pi Imager)
-
Connect to a network via Ethernet cable or Wi-Fi. Make sure this network has Internet access.
-
(optional, recommended) Enable SSH, if you intend to access the Pi remotely.
Note
These settings above may be configured during imaging (recommended), on first boot or afterwards.
-
Connect to the Pi and make sure
git
is installed:sudo apt update && sudo apt install -y git
-
Pair and trust any Bluetooth devices you wish to relay, either via GUI or via CLI:
bluetoothctl scan on
... wait for your devices to show up and note their MAC addresses (you may also type the first characters and hit
TAB
for auto-completion in the following commands) ...trust A1:B2:C3:D4:E5:F6 pair A1:B2:C3:D4:E5:F6 connect A1:B2:C3:D4:E5:F6 exit
Note
Replace A1:B2:C3:D4:E5:F6
by your input device's Bluetooth MAC address
-
On the Pi, clone the repository to your home directory:
cd ~ && git clone https://github.com/quaxalber/bluetooth_2_usb.git
-
Run the installation script as root:
sudo ~/bluetooth_2_usb/scripts/install.sh
-
Reboot:
sudo reboot
-
Verify that the service is running:
service bluetooth_2_usb status
It should look something like this and say
Active: active (running)
:● bluetooth_2_usb.service - Bluetooth to USB HID relay Loaded: loaded (/etc/systemd/system/bluetooth_2_usb.service; enabled; preset: enabled) Active: active (running) since Mon 2025-01-20 23:10:33 CET; 12min ago Main PID: 15598 (bash) Tasks: 3 (limit: 374) CPU: 13.454s CGroup: /system.slice/bluetooth_2_usb.service ├─15598 bash /usr/bin/bluetooth_2_usb --auto_discover --grab_devices └─15601 python3 /home/user/bluetooth_2_usb/bluetooth_2_usb.py --auto_discover --grab_devices Jan 20 23:10:33 pi0w systemd[1]: Started bluetooth_2_usb.service - Bluetooth to USB HID relay. Jan 20 23:10:35 pi0w bluetooth_2_usb[15601]: 25-01-20 23:10:35 [INFO] Launching Bluetooth 2 USB v0.9.0 Jan 20 23:10:39 pi0w bluetooth_2_usb[15601]: 25-01-20 23:10:39 [INFO] Activated relay for device /dev/input/event2, name "AceRK Keyboard", phys "b8:27:eb:be:dc:81" Jan 20 23:10:39 pi0w bluetooth_2_usb[15601]: 25-01-20 23:10:39 [INFO] Activated relay for device /dev/input/event3, name "AceRK Mouse", phys "b8:27:eb:be:dc:81"
Note
Something seems off? Try yourself in Troubleshooting!
Connect the USB-C power port of your Pi 4B/5 via cable with a USB port on your target device. You should hear the USB connection sound (depending on the target device) and be able to access your target device wirelessly using your Bluetooth keyboard or mouse. In case the Pi solely draws power from the host, it will take some time for the Pi to boot.
Important
It's essential to use the small power port instead of the bigger USB-A ports, since only the power port has the OTG feature required for USB gadgets.
For the Pi Zero, the situation is quite the opposite: Do not use the power port to connect to the target device, use the other port instead (typically labeled "DATA" or "USB"). You may connect the power port to a stable power supply.
Currently you can provide the following CLI arguments:
user@pi0w:~ $ bluetooth_2_usb -h
usage: bluetooth_2_usb.py [--device_ids DEVICE_IDS] [--auto_discover] [--grab_devices] [--interrupt_shortcut INTERRUPT_SHORTCUT] [--list_devices] [--log_to_file] [--log_path LOG_PATH] [--debug] [--version] [--help]
Bluetooth to USB HID relay. Handles Bluetooth keyboard and mouse events from multiple input devices and translates them to USB using Linux's gadget mode.
options:
--device_ids DEVICE_IDS, -i DEVICE_IDS
Comma-separated list of identifiers for input devices to be relayed.
An identifier is either the input device path, the MAC address or any case-insensitive substring of the device name.
Example: --device_ids '/dev/input/event2,a1:b2:c3:d4:e5:f6,0A-1B-2C-3D-4E-5F,logi'
Default: None
--auto_discover, -a Enable auto-discovery mode. All readable input devices will be relayed automatically.
Default: disabled
--grab_devices, -g Grab the input devices, i.e., suppress any events on your relay device.
Devices are not grabbed by default.
--interrupt_shortcut INTERRUPT_SHORTCUT, -s INTERRUPT_SHORTCUT
A plus-separated list of key names to press simultaneously in order to toggle relaying (pause/resume). Example: CTRL+SHIFT+Q
Default: None (feature disabled)
--list_devices, -l List all available input devices and exit.
--log_to_file, -f Add a handler that logs to file, additionally to stdout.
--log_path LOG_PATH, -p LOG_PATH
The path of the log file
Default: /var/log/bluetooth_2_usb/bluetooth_2_usb.log
--debug, -d Enable debug mode (Increases log verbosity)
Default: disabled
--version, -v Display the version number of this software and exit.
--help, -h Show this help message and exit.
The API is designed such that it may be consumed both via CLI and from within external Python code. More details on this coming soon!
You may update to the latest stable release by running:
sudo ~/bluetooth_2_usb/scripts/update.sh
Note
The update script performs a clean reinstallation, that is run uninstall.sh
, delete the repo folder, clone again and run the install script. The current branch will be maintained.
You may uninstall Bluetooth 2 USB by running:
sudo ~/bluetooth_2_usb/scripts/uninstall.sh
This is likely due to the limited power the Pi can draw from the host's USB port. Try these steps:
- If available, connect your Pi to a USB 3 port on the host / target device (usually blue) or preferably USB-C.
Important
Do not use the blue (or black) USB-A ports of your Pi to connect. This won't work.
Do use the small USB-C power port (in case of Pi 4B/5). For Pi Zero, use the data port to connect to the host and attach the power port to a dedicated power supply.
-
Try to connect to the Pi via SSH instead of attaching a display directly and remove any unnecessary peripherals.
-
Install a lite version of your OS on the Pi (without GUI)
-
For Pi 4B/5: Get a USB-C Data/Power Splitter and draw power from a dedicated power supply. This should ultimately resolve any power-related issues, and your Pi 4B will no longer be dependent on the host's power supply.
Note
The Pi Zero is recommended to have a 1.2 A power supply for stable operation, the Pi Zero 2 requires 2.0 A, the Pi 4B 3.0 A and the Pi 5 even 5.0 A, while hosts may typically only supply up to 0.5/0.9 A through USB-A 2.0/3.0 ports. However, this may be sufficient depending on your specific soft- and hardware configuration. For more information see the Raspberry Pi documentation.
This could be due to a number of reasons. Try these steps:
-
Verify that the service is running:
service bluetooth_2_usb status
-
Verify that you specified the correct input devices in
bluetooth_2_usb.service
-
Verify that your Bluetooth devices are paired, trusted, connected and not blocked:
bluetoothctl info A1:B2:C3:D4:E5:F6
It should look like this:
user@pi0w:~ $ bluetoothctl Agent registered [CHG] Controller 0A:1B:2C:3D:4E:5F Pairable: yes [AceRK]# info A1:B2:C3:D4:E5:F6 Device A1:B2:C3:D4:E5:F6 (random) Name: AceRK Alias: AceRK Paired: yes <--- Trusted: yes <--- Blocked: no <--- Connected: yes <--- WakeAllowed: no LegacyPairing: no UUID: Generic Access Profile (00001800-0000-1000-8000-00805f9b34fb) UUID: Generic Attribute Profile (00001801-0000-1000-8000-00805f9b34fb) UUID: Device Information (0000180a-0000-1000-8000-00805f9b34fb) UUID: Human Interface Device (00001812-0000-1000-8000-00805f9b34fb) UUID: Nordic UART Service (6e400001-b5a3-f393-e0a9-e50e24dcca9e)
Note
Replace A1:B2:C3:D4:E5:F6
by your input device's Bluetooth MAC address
-
Reload and restart service:
sudo systemctl daemon-reload && sudo service bluetooth_2_usb restart
-
Reboot Pi
sudo reboot
-
Re-connect the Pi to the host and check that the cable is capable of transmitting data, not power only
-
Try a different USB port on the host
-
Try connecting to a different host
This is a common issue, especially when the device gets paired with multiple hosts. One simple fix/workaround is to re-pair the device:
bluetoothctl
power off
power on
block A1:B2:C3:D4:E5:F6
remove A1:B2:C3:D4:E5:F6
scan on
trust A1:B2:C3:D4:E5:F6
pair A1:B2:C3:D4:E5:F6
connect A1:B2:C3:D4:E5:F6
exit
If the issue persists, it's worth trying to delete the cache:
sudo -i
cd '/var/lib/bluetooth/0A:1B:2C:3D:4E:5F/cache'
rm -rf 'A1:B2:C3:D4:E5:F6'
exit
Note
Replace 0A:1B:2C:3D:4E:5F
by your Pi's Bluetooth controller's MAC and A1:B2:C3:D4:E5:F6
by your input device's MAC
Here's a few things you could try:
- Check the log files (default at
/var/log/bluetooth_2_usb/
) for errors
Note
Logging to file requires the -f
flag
-
You may also query the journal to inspect the service logs in real-time:
journalctl -u bluetooth_2_usb.service -n 50 -f
-
For easier degguging, you may temporarily stop the service and run the script manually, modifying arguments as required, e.g., increase log verbosity by appending
-d
:sudo service bluetooth_2_usb stop && sudo bluetooth_2_usb -ad ; sudo service bluetooth_2_usb start
-
When you interact with your Bluetooth devices with
-d
set, you should see debug output in the logs such as:user@pi0w:~ $ sudo service bluetooth_2_usb stop && sudo bluetooth_2_usb -gads "CTRL+SHIFT+F12" ; sudo service bluetooth_2_usb start 25-01-26 13:21:39 [DEBUG] CLI args: device_ids=None, auto_discover=True, grab_devices=True, interrupt_shortcut=['CTRL', 'SHIFT', 'F12'], list_devices=False, log_to_file=False, log_path=/var/log/bluetooth_2_usb/bluetooth_2_usb.log, debug=True, version=False 25-01-26 13:21:39 [DEBUG] Logging to stdout 25-01-26 13:21:39 [INFO] Launching Bluetooth 2 USB v0.9.0 25-01-26 13:21:39 [DEBUG] Configuring global interrupt shortcut: {'KEY_F12', 'KEY_LEFTCTRL', 'KEY_LEFTSHIFT'} 25-01-26 13:21:42 [DEBUG] USB HID gadgets re-initialized: [boot mouse gadget (/dev/hidg0), keyboard gadget (/dev/hidg1), consumer control gadget (/dev/hidg2)] 25-01-26 13:21:42 [DEBUG] UdevEventMonitor initialized (observer not started yet). 25-01-26 13:21:42 [DEBUG] UdevEventMonitor started observer. 25-01-26 13:21:42 [DEBUG] RelayController: TaskGroup started. 25-01-26 13:21:42 [DEBUG] Created task for device /dev/input/event2, name "AceRK Keyboard", phys "b8:27:eb:be:dc:81". 25-01-26 13:21:42 [DEBUG] Created task for device /dev/input/event3, name "AceRK Mouse", phys "b8:27:eb:be:dc:81". 25-01-26 13:21:42 [INFO] Activated relay for device /dev/input/event2, name "AceRK Keyboard", phys "b8:27:eb:be:dc:81" 25-01-26 13:21:42 [INFO] Activated relay for device /dev/input/event3, name "AceRK Mouse", phys "b8:27:eb:be:dc:81" 25-01-26 13:21:53 [DEBUG] Received event at 1737894113.194308, code 04, type 04, val 458756 from AceRK Keyboard 25-01-26 13:21:53 [DEBUG] Received key event at 1737894113.194308, 30 (KEY_A), down from AceRK Keyboard 25-01-26 13:21:53 [DEBUG] Converted evdev scancode 0x1E (KEY_A) to HID UsageID 0x04 (A) 25-01-26 13:21:53 [DEBUG] Pressing A (0x04) 25-01-26 13:21:53 [DEBUG] Received synchronization event at 1737894113.194308, SYN_REPORT from AceRK Keyboard 25-01-26 13:21:53 [DEBUG] Received event at 1737894113.243246, code 04, type 04, val 458756 from AceRK Keyboard 25-01-26 13:21:53 [DEBUG] Received key event at 1737894113.243246, 30 (KEY_A), up from AceRK Keyboard 25-01-26 13:21:53 [DEBUG] Converted evdev scancode 0x1E (KEY_A) to HID UsageID 0x04 (A) 25-01-26 13:21:53 [DEBUG] Releasing A (0x04) 25-01-26 13:21:53 [DEBUG] Received synchronization event at 1737894113.243246, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:02 [DEBUG] Received relative axis event at 1737894122.359603, REL_X from AceRK Mouse 25-01-26 13:22:02 [DEBUG] Moving mouse (x=125, y=0, mwheel=0) 25-01-26 13:22:02 [DEBUG] Received synchronization event at 1737894122.359603, SYN_REPORT from AceRK Mouse 25-01-26 13:22:21 [DEBUG] Received event at 1737894141.518490, code 04, type 04, val 458977 from AceRK Keyboard 25-01-26 13:22:21 [DEBUG] Received key event at 1737894141.518490, 42 (KEY_LEFTSHIFT), down from AceRK Keyboard 25-01-26 13:22:21 [DEBUG] Converted evdev scancode 0x2A (KEY_LEFTSHIFT) to HID UsageID 0xE1 (LEFT_SHIFT) 25-01-26 13:22:21 [DEBUG] Pressing LEFT_SHIFT (0xE1) 25-01-26 13:22:21 [DEBUG] Received synchronization event at 1737894141.518490, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:22 [DEBUG] Received event at 1737894142.444821, code 04, type 04, val 458976 from AceRK Keyboard 25-01-26 13:22:22 [DEBUG] Received key event at 1737894142.444821, 29 (KEY_LEFTCTRL), down from AceRK Keyboard 25-01-26 13:22:22 [DEBUG] Converted evdev scancode 0x1D (KEY_LEFTCTRL) to HID UsageID 0xE0 (CONTROL) 25-01-26 13:22:22 [DEBUG] Pressing CONTROL (0xE0) 25-01-26 13:22:22 [DEBUG] Received synchronization event at 1737894142.444821, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:24 [INFO] ShortcutToggler: Relaying is now OFF. 25-01-26 13:22:24 [DEBUG] Ungrabbed device /dev/input/event2, name "AceRK Keyboard", phys "b8:27:eb:be:dc:81" 25-01-26 13:22:34 [DEBUG] Ungrabbed device /dev/input/event3, name "AceRK Mouse", phys "b8:27:eb:be:dc:81" 25-01-26 13:22:46 [INFO] ShortcutToggler: Relaying is now ON. 25-01-26 13:22:46 [DEBUG] Grabbed device /dev/input/event2, name "AceRK Keyboard", phys "b8:27:eb:be:dc:81" 25-01-26 13:22:46 [DEBUG] Received key event at 1737894166.576799, 88 (KEY_F12), down from AceRK Keyboard 25-01-26 13:22:46 [DEBUG] Converted evdev scancode 0x58 (KEY_F12) to HID UsageID 0x45 (F12) 25-01-26 13:22:46 [DEBUG] Pressing F12 (0x45) 25-01-26 13:22:46 [DEBUG] Received synchronization event at 1737894166.576799, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:46 [DEBUG] Received event at 1737894166.625320, code 04, type 04, val 458821 from AceRK Keyboard 25-01-26 13:22:46 [DEBUG] Received key event at 1737894166.625320, 88 (KEY_F12), up from AceRK Keyboard 25-01-26 13:22:46 [DEBUG] Converted evdev scancode 0x58 (KEY_F12) to HID UsageID 0x45 (F12) 25-01-26 13:22:46 [DEBUG] Releasing F12 (0x45) 25-01-26 13:22:46 [DEBUG] Received synchronization event at 1737894166.625320, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:49 [DEBUG] Received event at 1737894169.452812, code 04, type 04, val 458976 from AceRK Keyboard 25-01-26 13:22:49 [DEBUG] Received key event at 1737894169.452812, 29 (KEY_LEFTCTRL), up from AceRK Keyboard 25-01-26 13:22:49 [DEBUG] Converted evdev scancode 0x1D (KEY_LEFTCTRL) to HID UsageID 0xE0 (CONTROL) 25-01-26 13:22:49 [DEBUG] Releasing CONTROL (0xE0) 25-01-26 13:22:49 [DEBUG] Received synchronization event at 1737894169.452812, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:50 [DEBUG] Received event at 1737894170.378918, code 04, type 04, val 458977 from AceRK Keyboard 25-01-26 13:22:50 [DEBUG] Received key event at 1737894170.378918, 42 (KEY_LEFTSHIFT), up from AceRK Keyboard 25-01-26 13:22:50 [DEBUG] Converted evdev scancode 0x2A (KEY_LEFTSHIFT) to HID UsageID 0xE1 (LEFT_SHIFT) 25-01-26 13:22:50 [DEBUG] Releasing LEFT_SHIFT (0xE1) 25-01-26 13:22:50 [DEBUG] Received synchronization event at 1737894170.378918, SYN_REPORT from AceRK Keyboard 25-01-26 13:22:52 [DEBUG] Grabbed device /dev/input/event3, name "AceRK Mouse", phys "b8:27:eb:be:dc:81" 25-01-26 13:22:52 [DEBUG] Received relative axis event at 1737894172.475360, REL_X from AceRK Mouse 25-01-26 13:22:52 [DEBUG] Moving mouse (x=50, y=0, mwheel=0) 25-01-26 13:22:52 [DEBUG] Received synchronization event at 1737894172.475360, SYN_REPORT from AceRK Mouse 25-01-26 13:23:07 [DEBUG] Received signal: SIGINT. Requesting graceful shutdown. 25-01-26 13:23:07 [DEBUG] Shutdown event triggered. Cancelling relay task... 25-01-26 13:23:07 [DEBUG] Relay cancelled for device device /dev/input/event3, name "AceRK Mouse", phys "b8:27:eb:be:dc:81". 25-01-26 13:23:07 [DEBUG] Cancelling relay for /dev/input/event3. 25-01-26 13:23:07 [DEBUG] Relay cancelled for device device /dev/input/event2, name "AceRK Keyboard", phys "b8:27:eb:be:dc:81". 25-01-26 13:23:07 [DEBUG] Cancelling relay for /dev/input/event2. 25-01-26 13:23:07 [DEBUG] RelayController: TaskGroup exited. 25-01-26 13:23:07 [DEBUG] UdevEventMonitor stopped observer.
-
Still not resolved? Double-check the installation instructions
-
For more help, open an issue in the GitHub repository
Absolutely! Here's how.
After successfully setting up your Pi as a HID proxy for your Bluetooth devices, you may consider making Raspberry Pi OS read-only. That helps preventing the SD card from wearing out and the file system from getting corrupted when powering off the Raspberry forcefully.
Contributions are welcome! Please read the CONTRIBUTING.md file for guidelines.
This project is licensed under the MIT License - see the LICENSE file for details.
Bluetooth to USB Overview image by Laura T. is licensed under a Creative Commons Attribution-NonCommercial 4.0 International License.
- Mike Redrobe for the idea and the basic code logic and HeuristicPerson's bluetooth_2_hid based off this.
- Georgi Valkov for python-evdev making reading input devices a walk in the park.
- The folks at Adafruit for CircuitPython HID and Blinka providing super smooth access to USB gadgets.
- Special thanks to the open-source community for various other libraries and tools.