-
-
Notifications
You must be signed in to change notification settings - Fork 50
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Help wanted: Using L2CAP connection to read from and write settings to AirPods #215
Comments
See here for details from the MagicPods project:
and here is a python script that can connect to AirPods from a desktop computer: import bluetooth
# Change to MAC address of your AirPods
address = "MAC_ADDRESS"
aap_service = "74EC2172-0BAD-4D01-8F77-997B2BE0722A"
aap_port = 0x1001
# Commands taken from https://github.com/steam3d/MagicPodsCore/blob/master/src/aap/Aap.h
cmd_handshake = b"\x00\x00\x04\x00\x01\x00\x02\x00\x00\x00\x00\x00\x00\x00\x00\x00"
cmd_off = b"\x04\x00\x04\x00\x09\x00\x0d\x01\x00\x00\x00"
cmd_on = b"\x04\x00\x04\x00\x09\x00\x0d\x02\x00\x00\x00"
cmd_transparency = b"\x04\x00\x04\x00\x09\x00\x0d\x03\x00\x00\x00"
services = bluetooth.find_service(address=address)
service = [s for s in services if s["service-classes"] == [aap_service]]
if not service:
print("Device does not have AAP service")
exit()
sock = bluetooth.BluetoothSocket(bluetooth.L2CAP)
sock.connect((address, aap_port))
print("Connected to AirPods")
print("Sending handshake...")
sock.send(cmd_handshake)
while True:
# Ignore the responses because I am unsure what they mean
# res = sock.recv(1024)
# print(f"Response: {res.hex()}")
print("Select command:")
print("1. Turn off")
print("2. Turn on")
print("3. Toggle transparency")
print("4. Exit")
cmd = input("Enter command: ")
if cmd == "1":
sock.send(cmd_off)
elif cmd == "2":
sock.send(cmd_on)
elif cmd == "3":
sock.send(cmd_transparency)
elif cmd == "4":
break
sock.close() |
This comment was marked as outdated.
This comment was marked as outdated.
Tried playing with this and I didn't get very far unfortunately. Was able to establish the connection on a desktop computer too, but didn't manage to get anything to happen on Android 14. The names for |
Thanks for trying @vulpes2 🍻. I had some hope that Android 15 would add more APIs here, but didn't see anything in the changelogs so far 😦 |
Seems like L2CAP on BR/EDR was never supported for 3rd party use on Android, which is kind of surprising. Not sure why did they make a new API to create L2CAP sockets for LE, but still wouldn't allow it over BR/EDR to this day. |
Any chance someone with an understanding of L2CAP (I'm certainly not an L2CAP understander) could do a PR into AOSP? |
The function exists, you are just not allowed to use it from a normal app. Neither Android nor iOS allow you to create raw L2CAP connections on BR/EDR, but obviously both use them heavily at the system level. |
@vulpes2 So it might be not possible at all unless you root your device? |
Effectively yes, LSPosed requires Magisk. |
I reverse engineered the beats app too. The difference between AirPods and Beats headphones (both on H1 and H2 chips) is that Beats headphones expose the RFCOMM interface, but it seems that Apple intentionally turned RFCOMM off. |
I have also been working on this myself (without knowing magicpods existed :| ), but have had no luck on android (unlike linux)... i am planning to publish my findings (and a small app for linux) that i found while reverse engineering using the packetlogger on mac (set ANC mode, etc.), but i think it's useless for now considering the psm value thing.. |
I've been poking at this for a little while now, and have managed to map out most of the packet formats, opcodes and configuration keys as of version |
@vulpes2 I have fully implemented the work of all functions of all versions of airpods in MagicPods, it works for any firmware version. The main problem is that the functions are hardcoded, and not obtained dynamically as apple does. Apple uses several configuration requests, but I still don't understand the logic behind it requests for them. It seems like it depends on the protocol version. |
The capability stuff and notification type bitfield is a total nightmare to deal with, I have managed to decipher a few of them, but not enough to make a difference just yet. By configuration requests, do you mean opcode 0x9, or the initial capability response with opcode 0x2? |
@d4rken could you provide a simple apk with xposed with 3 simple toggles for ANC modes? or maybe even publish the code? I'll try it on my phone and try experimenting.. |
mine doesn't work using the lsposed hiddenapibypass, the Here's my error when using
|
adolfintel/OpenPods#58 (comment) TL;DR Got it working by commenting a few lines in the bluetooth stack's code for managing l2c connections (the ones which were blocking when sending data to the airpods, and getting no response). I made a small app, using the hidden method to create br/edr l2c socket. i can change listening mode, get battery data, and in-ear detection works, for now. |
I've started working on AAP protocol documentation in python, decided to abandon c++ in favor of simplicity. |
I've created a issue on issuetracker for this! https://issuetracker.google.com/issues/371713238. please upvote :) |
Just want to say that this feature is a banger and could be a huge deal breaker for a lot of people to start using airpods in android, maybe even buying just to use on android |
@aikooo7 you can try a beta app if you have a rooted phone right now! |
I do have root, where can I get the app? Note: I will use it if it is open-source. |
Are you comfortable with working with a shell environment? If you are then, you will have to create a shell script in post-data-fs for overlaying the library file (I'll upload it on my repo later today) in /apex (where the bluetooth services live), and then another root module, or a script (ksu or magisk) to overlay the same library in /system/lib64 (because apex files have signature verification). Then you should be able to use my app (same repo).. (The app works best for pro 2, and is very much a work in progress!!) |
@aikooo7, instructions and files on my first release |
@kavishdevar |
Our sister project for Windows, MagicPods, has reverse engineered reading and writing data to AirPods.
The good news is that this actually works and I could reproduce it on a desktop computer running linux. So support for changing settings (ANC mode, microphone etc.) and getting more reliable information would be possible
The bad news is that I'm unable to get the connection established on an Android device.
If anyone can figure out how to establish the connection, then I could start on adding support for this.
AirPods require a Bluetooth connection that uses the
L2CAP
protocol. I was unable to establish this connection on a Pixel 6 running Android 15 and Pixel 8 running Android 14.This is what I tried so far
Use the public
createInsecureL2capChannel
methodbut that seems to have limits on the
PSM
value that you can providehttps://cs.android.com/android/platform/superproject/main/+/main:packages/modules/Bluetooth/system/stack/l2cap/l2c_api.cc;l=416?q=%22Invalid%20BLE%20PSM%20value%22
Then I tried the hidden method
createInsecureL2capSocket
https://cs.android.com/android/platform/superproject/main/+/main:packages/modules/Bluetooth/framework/java/android/bluetooth/BluetoothDevice.java;l=2924?q=createInsecureL2capSocket&sq=&ss=android%2Fplatform%2Fsuperproject%2Fmain
Which needed some convincing to execute (https://github.com/LSPosed/AndroidHiddenApiBypass) but
still no dice.
A friend has tried simulating AirPods on a Desktop computer and was able to connect to that mock via
createInsecureL2capSocket
from an Android phone, but wasn't able to do this with actual AirPods.The text was updated successfully, but these errors were encountered: