Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

gh-129288: Add optional l2_cid and l2_bdaddr_type in BTPROTO_L2CAP socket address tuple #129293

Merged
merged 18 commits into from
Feb 27, 2025

Conversation

themadinventor
Copy link
Contributor

@themadinventor themadinventor commented Jan 25, 2025

To be able to connect L2CAP sockets to Bluetooth LE devices, the l2_bdaddr_type field in sockaddr_l2 must be set to BDADDR_LE_PUBLIC or BDADDR_LE_RANDOM.
Likewise, when opening a raw LE L2CAP socket to the ATT service, l2_cid must be set instead of l2_psm.

This change adds support for providing l2_cid and l2_bdaddr_type as optional, traliing elements in the address tuple passed to connect()

…ress tuple

To be able to connect L2CAP sockets to Bluetooth LE devices, the l2_bdaddr_type must be set to BDADDR_LE_PUBLIC or BDADDR_LE_RANDOM.
This change adds support for providing the l2_bdaddr_type as an optional, traliing element in the address tuple passed to connect()
Copy link

cpython-cla-bot bot commented Jan 25, 2025

All commit authors signed the Contributor License Agreement.
CLA signed

@bedevere-app
Copy link

bedevere-app bot commented Jan 25, 2025

Most changes to Python require a NEWS entry. Add one using the blurb_it web app or the blurb command-line tool.

If this change has little impact on Python users, wait for a maintainer to apply the skip news label instead.

… socket.BDADDR_RANDOM only if defined for target

Add check that BDADDR_BREDR is defined as Windows build may set USE_BLUETOOTH but not BDADDR_*
@bedevere-app
Copy link

bedevere-app bot commented Jan 25, 2025

Most changes to Python require a NEWS entry. Add one using the blurb_it web app or the blurb command-line tool.

If this change has little impact on Python users, wait for a maintainer to apply the skip news label instead.

…ss tuple

This change adds support for the CID field in the socket address tuple. This allows e.g raw LE ATT connections.
@bedevere-app
Copy link

bedevere-app bot commented Jan 26, 2025

Most changes to Python require a NEWS entry. Add one using the blurb_it web app or the blurb command-line tool.

If this change has little impact on Python users, wait for a maintainer to apply the skip news label instead.

@themadinventor themadinventor changed the title gh-129288: Add optional l2_bdaddr_type in BTPROTO_L2CAP socket address tuple gh-129288: Add optional l2_cid and l2_bdaddr_type in BTPROTO_L2CAP socket address tuple Jan 26, 2025
themadinventor and others added 2 commits February 23, 2025 10:47
…address fields

Update documentation to mention the new cid and bdaddr_type fields for BTPROTO_L2CAP connections.
Copy link
Member

@encukou encukou left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you!
I guess we don't have active Bluetooth experts, so I'll look at this mostly as a generic C dev.
Are you aware of some official docs that would explain what, for example, what a cid is? man l2cap doesn't help much.

Are there any tests you could add to Lib/test/test_socket.py?

@themadinventor
Copy link
Contributor Author

Are you aware of some official docs that would explain what, for example, what a cid is? man l2cap doesn't help much.

I'm afraid the best reference is the Bluetooth Core specification itself.
CID is defined in Volume 3, Part A, Chapter 2, which is on page 1088 in version 6.0. It's actually not bad.

Basically, a CID (Channel Identifier) is a port number; each L2CAP connection has one CID for each end. Some fundamental services (like security and discovery things) are defined in the spec to use fixed/hardcoded CID on both ends. This is the use case for passing cid to bind and connect.

Normal applications will however open L2CAP connections using PSM (Protocol/Service Multiplexer), which is kind of an abstract port number, to specify a specific service.
When using connecting using PSM, a handshake will occur, where the host and device assign dynamic CIDs for each end of the connection. This handshake is done over one of the fixed CID channels, the L2CAP Signalling Channel.

(The L2CAP connection handshake is described in Vol 3 Part 4 Chapter 4 in the Core spec.)

So, while it's not possible to provide both cid and psm when binding or connecting, an open connection may have both PSM and CID, although the CID in that case is mostly a curiosity, as it's ephemeral and restricted to a specific point-to-point link.

Are there any tests you could add to Lib/test/test_socket.py?

I'll look at it! 👍

@encukou
Copy link
Member

encukou commented Feb 25, 2025

I'm afraid the best reference is the Bluetooth Core specification itself.

Thanks! That looks like what I was looking for, it explains the terms and even clarifies the boundary between BlueZ & Bluetooth itself.
Still, it's too long to read all of it so I'll continue asking you :)

So, while it's not possible to provide both cid and psm when binding or connecting, an open connection may have both PSM and CID, although the CID in that case is mostly a curiosity, as it's ephemeral and restricted to a specific point-to-point link.

Thanks!
But, the case I'm worried about is an address with bdaddr_type=BDADDR_BREDR but a non-zero cid. With the current PR, makesockaddr will return the old 2-tuple format, so the cid is discarded.
Is that OK?

Why I'm asking: for backwards compatibility where a tuple-based API needs to be extended, we can use StructSequence. Here we could define a 2-tuple-like object with additional named (but not numbered) fields.
If the cid is generally not important, it would be best to leave that out of scope of this PR. If cid is sometimes useful with bdaddr_type=BDADDR_BREDR, we can expose it in a follow-up PR.

@themadinventor
Copy link
Contributor Author

Right!

I don't think the case with cid and bdaddr_type=BDADDR_BREDR is very common. I haven't worked much with BR/EDR, so I'll have to do some tests to see what the kernel even allows. For LE connections, only the ATT CID (for BLE GATT interaction) is accepted from userspace.

Overall, I don't think cid is very interesting to read back from the socket. StructSequence as you suggested sounds like a neat solution for a separate PR.

I can add a comment that we're discarding the cid for now.

@encukou
Copy link
Member

encukou commented Feb 26, 2025

For the tests, could you set HAVE_SOCKET_BLUETOOTH_L2CAP like the existing HAVE_SOCKET_BLUETOOTH?
The tests should run on all systems where this is exposed. If they fail in CI, we got some assumptions wrong. If they fail for someone on an platform we don't support, they'll know what they need to port.

@encukou encukou added the 🔨 test-with-buildbots Test PR w/ buildbots; report in status section label Feb 26, 2025
@encukou
Copy link
Member

encukou commented Feb 26, 2025

!buildbot .*

@bedevere-bot
Copy link

🤖 New build scheduled with the buildbot fleet by @encukou for commit 5a1eaa3 🤖

Results will be shown at:

https://buildbot.python.org/all/#/grid?branch=refs%2Fpull%2F129293%2Fmerge

The command will test the builders whose names match following regular expression: .*

The builders matched are:

  • PPC64LE CentOS9 LTO PR
  • AMD64 Windows10 PR
  • AMD64 FreeBSD Refleaks PR
  • aarch64 Fedora Rawhide Clang Installed PR
  • PPC64LE Fedora Stable Refleaks PR
  • AMD64 Arch Linux VintageParser PR
  • PPC64LE CentOS9 PR
  • PPC64LE Fedora Rawhide Clang PR
  • PPC64LE CentOS9 Refleaks PR
  • AMD64 Windows11 Non-Debug PR
  • AMD64 RHEL8 PR
  • s390x RHEL8 LTO PR
  • AMD64 Arch Linux Valgrind PR
  • AMD64 Windows11 Bigmem PR
  • PPC64LE RHEL8 LTO + PGO PR
  • aarch64 Ubuntu 22.04 BigMem PR
  • PPC64LE Fedora Stable Clang Installed PR
  • ARM64 MacOS M1 Refleaks NoGIL PR
  • s390x RHEL9 LTO + PGO PR
  • AMD64 FreeBSD PR
  • AMD64 FreeBSD14 PR
  • aarch64 RHEL8 PR
  • aarch64 Fedora Rawhide LTO + PGO PR
  • AMD64 Fedora Rawhide Clang PR
  • iOS ARM64 Simulator PR
  • AMD64 Fedora Rawhide Refleaks PR
  • PPC64LE Fedora Rawhide LTO PR
  • AMD64 CentOS9 FIPS No Builtin Hashes PR
  • AMD64 CentOS9 PR
  • SPARCv9 Oracle Solaris 11.4 PR
  • s390x RHEL8 LTO + PGO PR
  • AMD64 RHEL8 Refleaks PR
  • aarch64 Fedora Rawhide LTO PR
  • s390x RHEL9 LTO PR
  • PPC64LE CentOS9 LTO + PGO PR
  • AMD64 Fedora Stable LTO PR
  • ARM64 macOS PR
  • PPC64LE Fedora Rawhide NoGIL PR
  • PPC64 AIX PR
  • aarch64 Fedora Stable LTO PR
  • AMD64 CentOS9 NoGIL Refleaks PR
  • ARM Raspbian PR
  • AMD64 Fedora Rawhide NoGIL refleaks PR
  • s390x RHEL9 PR
  • ARM64 Windows Non-Debug PR
  • AMD64 Fedora Stable Clang PR
  • PPC64LE Fedora Stable LTO PR
  • AMD64 Debian root PR
  • PPC64LE Fedora Stable Clang PR
  • aarch64 Fedora Stable Refleaks PR
  • AMD64 Arch Linux Asan PR
  • AMD64 RHEL8 FIPS No Builtin Hashes PR
  • AMD64 FreeBSD15 PR
  • AMD64 Windows PGO PR
  • PPC64LE Fedora Stable LTO + PGO PR
  • PPC64LE RHEL8 LTO PR
  • PPC64LE RHEL8 PR
  • PPC64LE Fedora Rawhide NoGIL refleaks PR
  • x86-64 macOS PR
  • aarch64 Fedora Stable PR
  • ARM64 Windows PR
  • AMD64 Fedora Stable PR
  • AMD64 RHEL8 FIPS Only Blake2 Builtin Hash PR
  • aarch64 Fedora Stable Clang Installed PR
  • AMD64 Arch Linux Asan Debug PR
  • aarch64 Fedora Rawhide NoGIL refleaks PR
  • PPC64LE Fedora Stable PR
  • aarch64 RHEL8 Refleaks PR
  • s390x RHEL8 Refleaks PR
  • aarch64 Fedora Rawhide PR
  • AMD64 Arch Linux Usan PR
  • AMD64 Ubuntu PR
  • AMD64 Arch Linux TraceRefs PR
  • AMD64 Fedora Stable Refleaks PR
  • PPC64LE Fedora Rawhide Clang Installed PR
  • AMD64 Fedora Rawhide NoGIL PR
  • AMD64 Fedora Stable Clang Installed PR
  • AMD64 CentOS9 LTO + PGO PR
  • PPC64LE Fedora Rawhide Refleaks PR
  • AMD64 RHEL8 LTO PR
  • AMD64 RHEL8 LTO + PGO PR
  • AMD64 Alpine Linux PR
  • aarch64 Android PR
  • wasm32-wasi PR
  • AMD64 Windows11 Refleaks PR
  • AMD64 CentOS9 LTO PR
  • aarch64 Fedora Stable LTO + PGO PR
  • s390x RHEL8 PR
  • AMD64 Fedora Stable LTO + PGO PR
  • AMD64 Arch Linux Usan Function PR
  • wasm32-wasi Non-Debug PR
  • wasm32 WASI 8Core PR
  • AMD64 Windows Server 2022 NoGIL PR
  • aarch64 CentOS9 LTO + PGO PR
  • AMD64 CentOS9 NoGIL PR
  • AMD64 Fedora Rawhide LTO PR
  • riscv64 Ubuntu23 PR
  • AMD64 CentOS9 FIPS Only Blake2 Builtin Hash PR
  • AMD64 Arch Linux Perf PR
  • aarch64 CentOS9 LTO PR
  • aarch64 Fedora Rawhide NoGIL PR
  • aarch64 Fedora Rawhide Clang PR
  • x86 Debian Non-Debug with X PR
  • AMD64 Fedora Rawhide PR
  • AMD64 Fedora Rawhide LTO + PGO PR
  • aarch64 Fedora Stable Clang PR
  • PPC64LE Fedora Rawhide LTO + PGO PR
  • ARM64 MacOS M1 NoGIL PR
  • PPC64 AIX XLC PR
  • AMD64 CentOS9 Refleaks PR
  • x86-64 MacOS Intel NoGIL PR
  • PPC64LE RHEL8 Refleaks PR
  • AMD64 Android PR
  • x86-64 MacOS Intel ASAN NoGIL PR
  • PPC64LE Fedora Rawhide PR
  • s390x RHEL9 Refleaks PR
  • AMD64 Ubuntu Shared PR
  • AMD64 Windows PGO NoGIL PR
  • aarch64 Fedora Rawhide Refleaks PR
  • AMD64 Fedora Rawhide Clang Installed PR
  • aarch64 CentOS9 Refleaks PR
  • aarch64 RHEL8 LTO + PGO PR
  • aarch64 RHEL8 LTO PR
  • x86 Debian Installed with X PR

@encukou
Copy link
Member

encukou commented Feb 26, 2025

I've started the buildbots, which usually run after a change is merged, and check all the platforms & configurations.

Don't be alarmed if some fail and mark the PR red; not all of them are stable.

@encukou encukou removed the 🔨 test-with-buildbots Test PR w/ buildbots; report in status section label Feb 26, 2025
Copy link
Member

@encukou encukou left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good!
One more wording touchup for the field docs, which follow the where:

@encukou encukou enabled auto-merge (squash) February 27, 2025 12:30
@encukou encukou merged commit 45a24f5 into python:main Feb 27, 2025
42 checks passed
@themadinventor
Copy link
Contributor Author

Great! Thank you for the review and all the help! 🌟

# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants