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
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
18 commits
Select commit Hold shift + click to select a range
702f477
gh-129288: Add optional bdaddr_type in BTPROTO_L2CAP socket address t…
themadinventor Jan 25, 2025
d45523c
gh-129288: Expose socket.BDADDR_BREDR, socket.BDADDR_LE_PUBLIC, socke…
themadinventor Jan 25, 2025
e9b2a54
gh-129288: Add optional CID field in BTPROTO_L2CAP socket address tuple
themadinventor Jan 26, 2025
117ccd4
📜🤖 Added by blurb_it.
blurb-it[bot] Jan 26, 2025
4c123f6
gh-129288: Fix NEWS blurb formatting
themadinventor Jan 26, 2025
4259dca
Merge branch 'main' into gh-129288
themadinventor Jan 27, 2025
d771afb
gh-129288: Update docs for socket to include new BTPROTO_L2CAP addres…
themadinventor Feb 23, 2025
508ef7d
Merge branch 'main' into gh-129288
themadinventor Feb 23, 2025
60abd6e
gh-129288: Improve docs for socket on BTPROTO_L2CAP, BDADDR_BREDR et al
themadinventor Feb 25, 2025
2c537ba
gh-129288: Move BDADDR_BREDR et al macros to use BTPROTO_L2CAP guard
themadinventor Feb 25, 2025
fe0b993
Expand the docs & provide link targets for the constants
encukou Feb 25, 2025
b25005e
gh-129288: Update comment re discarding cid for BR/EDR L2CAP socket
themadinventor Feb 25, 2025
05de5ee
Merge pull request #1 from encukou/gh-129288
themadinventor Feb 25, 2025
8532c5b
gh-129288: Add tests for BTPROTO_L2CAP socket
themadinventor Feb 25, 2025
1af3e88
gh-129288: Add HAVE_SOCKET_BLUETOOTH_L2CAP to socket tests
themadinventor Feb 26, 2025
5a1eaa3
gh-129288 Revert skip criteria for testCreateL2capSocket
themadinventor Feb 26, 2025
38e32a7
Update Doc/library/socket.rst
encukou Feb 27, 2025
b731ae7
Never commit from GitHub UI, right?
encukou Feb 27, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
32 changes: 30 additions & 2 deletions Doc/library/socket.rst
Original file line number Diff line number Diff line change
Expand Up @@ -137,8 +137,19 @@ created. Socket addresses are represented as follows:
- :const:`AF_BLUETOOTH` supports the following protocols and address
formats:

- :const:`BTPROTO_L2CAP` accepts ``(bdaddr, psm)`` where ``bdaddr`` is
the Bluetooth address as a string and ``psm`` is an integer.
- :const:`BTPROTO_L2CAP` accepts a tuple
``(bdaddr, psm[, cid[, bdaddr_type]])`` where:

- ``bdaddr`` is a string specifying the Bluetooth address.
- ``psm`` is an integer specifying the Protocol/Service Multiplexer.
- ``cid`` is an optional integer specifying the Channel Identifier.
If not given, defaults to zero.
- ``bdaddr_type`` is an optional integer specifying the address type;
one of :const:`BDADDR_BREDR` (default), :const:`BDADDR_LE_PUBLIC`,
:const:`BDADDR_LE_RANDOM`.

.. versionchanged:: next
Added ``cid`` and ``bdaddr_type`` fields.

- :const:`BTPROTO_RFCOMM` accepts ``(bdaddr, channel)`` where ``bdaddr``
is the Bluetooth address as a string and ``channel`` is an integer.
Expand Down Expand Up @@ -626,6 +637,14 @@ Constants
This constant contains a boolean value which indicates if IPv6 is supported on
this platform.

.. data:: AF_BLUETOOTH
BTPROTO_L2CAP
BTPROTO_RFCOMM
BTPROTO_HCI
BTPROTO_SCO

Integer constants for use with Bluetooth addresses.

.. data:: BDADDR_ANY
BDADDR_LOCAL

Expand All @@ -634,6 +653,15 @@ Constants
any address when specifying the binding socket with
:const:`BTPROTO_RFCOMM`.

.. data:: BDADDR_BREDR
BDADDR_LE_PUBLIC
BDADDR_LE_RANDOM

These constants describe the Bluetooth address type when binding or
connecting a :const:`BTPROTO_L2CAP` socket.

.. versionadded:: next

.. data:: HCI_FILTER
HCI_TIME_STAMP
HCI_DATA_DIR
Expand Down
40 changes: 40 additions & 0 deletions Lib/test/test_socket.py
Original file line number Diff line number Diff line change
Expand Up @@ -177,6 +177,17 @@ def _have_socket_bluetooth():
return True


def _have_socket_bluetooth_l2cap():
"""Check whether BTPROTO_L2CAP sockets are supported on this host."""
try:
s = socket.socket(socket.AF_BLUETOOTH, socket.SOCK_SEQPACKET, socket.BTPROTO_L2CAP)
except (AttributeError, OSError):
return False
else:
s.close()
return True


def _have_socket_hyperv():
"""Check whether AF_HYPERV sockets are supported on this host."""
try:
Expand Down Expand Up @@ -219,6 +230,8 @@ def socket_setdefaulttimeout(timeout):

HAVE_SOCKET_BLUETOOTH = _have_socket_bluetooth()

HAVE_SOCKET_BLUETOOTH_L2CAP = _have_socket_bluetooth_l2cap()

HAVE_SOCKET_HYPERV = _have_socket_hyperv()

# Size in bytes of the int type
Expand Down Expand Up @@ -2608,6 +2621,33 @@ def testCreateScoSocket(self):
with socket.socket(socket.AF_BLUETOOTH, socket.SOCK_SEQPACKET, socket.BTPROTO_SCO) as s:
pass

@unittest.skipUnless(HAVE_SOCKET_BLUETOOTH_L2CAP, 'Bluetooth L2CAP sockets required for this test')
def testBindLeAttL2capSocket(self):
with socket.socket(socket.AF_BLUETOOTH, socket.SOCK_SEQPACKET, socket.BTPROTO_L2CAP) as f:
# ATT is the only CID allowed in userspace by the Linux kernel
CID_ATT = 4
f.bind((socket.BDADDR_ANY, 0, CID_ATT, socket.BDADDR_LE_PUBLIC))
addr = f.getsockname()
self.assertEqual(addr, (socket.BDADDR_ANY, 0, CID_ATT, socket.BDADDR_LE_PUBLIC))

@unittest.skipUnless(HAVE_SOCKET_BLUETOOTH_L2CAP, 'Bluetooth L2CAP sockets required for this test')
def testBindLePsmL2capSocket(self):
with socket.socket(socket.AF_BLUETOOTH, socket.SOCK_SEQPACKET, socket.BTPROTO_L2CAP) as f:
# First user PSM in LE L2CAP
psm = 0x80
f.bind((socket.BDADDR_ANY, psm, 0, socket.BDADDR_LE_RANDOM))
addr = f.getsockname()
self.assertEqual(addr, (socket.BDADDR_ANY, psm, 0, socket.BDADDR_LE_RANDOM))

@unittest.skipUnless(HAVE_SOCKET_BLUETOOTH_L2CAP, 'Bluetooth L2CAP sockets required for this test')
def testBindBrEdrL2capSocket(self):
with socket.socket(socket.AF_BLUETOOTH, socket.SOCK_SEQPACKET, socket.BTPROTO_L2CAP) as f:
# First user PSM in BR/EDR L2CAP
psm = 0x1001
f.bind((socket.BDADDR_ANY, psm))
addr = f.getsockname()
self.assertEqual(addr, (socket.BDADDR_ANY, psm))


@unittest.skipUnless(HAVE_SOCKET_HYPERV,
'Hyper-V sockets required for this test.')
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Add optional ``l2_cid`` and ``l2_bdaddr_type`` fields to :mod:`socket` ``BTPROTO_L2CAP`` sockaddr tuple.
27 changes: 22 additions & 5 deletions Modules/socketmodule.c
Original file line number Diff line number Diff line change
Expand Up @@ -1495,9 +1495,20 @@ makesockaddr(SOCKET_T sockfd, struct sockaddr *addr, size_t addrlen, int proto)
PyObject *addrobj = makebdaddr(&_BT_L2_MEMB(a, bdaddr));
PyObject *ret = NULL;
if (addrobj) {
ret = Py_BuildValue("Oi",
addrobj,
_BT_L2_MEMB(a, psm));
/* Retain old format for non-LE address.
(cid may be set for BR/EDR, but we're discarding it for now)
*/
if (_BT_L2_MEMB(a, bdaddr_type) == BDADDR_BREDR) {
ret = Py_BuildValue("Oi",
addrobj,
_BT_L2_MEMB(a, psm));
} else {
ret = Py_BuildValue("OiiB",
addrobj,
_BT_L2_MEMB(a, psm),
_BT_L2_MEMB(a, cid),
_BT_L2_MEMB(a, bdaddr_type));
}
Py_DECREF(addrobj);
}
return ret;
Expand Down Expand Up @@ -2047,8 +2058,11 @@ getsockaddrarg(PySocketSockObject *s, PyObject *args,
struct sockaddr_l2 *addr = &addrbuf->bt_l2;
memset(addr, 0, sizeof(struct sockaddr_l2));
_BT_L2_MEMB(addr, family) = AF_BLUETOOTH;
if (!PyArg_ParseTuple(args, "si", &straddr,
&_BT_L2_MEMB(addr, psm))) {
_BT_L2_MEMB(addr, bdaddr_type) = BDADDR_BREDR;
if (!PyArg_ParseTuple(args, "si|iB", &straddr,
&_BT_L2_MEMB(addr, psm),
&_BT_L2_MEMB(addr, cid),
&_BT_L2_MEMB(addr, bdaddr_type))) {
PyErr_Format(PyExc_OSError,
"%s(): wrong format", caller);
return 0;
Expand Down Expand Up @@ -7782,6 +7796,9 @@ socket_exec(PyObject *m)
ADD_INT_MACRO(m, AF_BLUETOOTH);
#ifdef BTPROTO_L2CAP
ADD_INT_MACRO(m, BTPROTO_L2CAP);
ADD_INT_MACRO(m, BDADDR_BREDR);
ADD_INT_MACRO(m, BDADDR_LE_PUBLIC);
ADD_INT_MACRO(m, BDADDR_LE_RANDOM);
#endif /* BTPROTO_L2CAP */
#ifdef BTPROTO_HCI
ADD_INT_MACRO(m, BTPROTO_HCI);
Expand Down
Loading