如何在 Android 中实例化 L2Cap 套接字?
How can I instantiate a L2Cap socket in Android?
我看到 Bluetooth socket 可以是 TYPE_L2CAP 类型,但是 BluetoothSocket 的构造函数似乎是私有的,我只能找到一个方法来实例化 RFCOMM 类型的套接字。
如何获得和使用 L2CAP 套接字? Android真的支持吗?
2019 年 10 月更新
支持!享受新的 API:
- BluetoothDevice.createL2capChannel(int)
- BluetoothDevice.createInsecureL2capChannel(int)
- BluetoothAdapter.listenUsingL2capChannel()
- BluetoothAdapter.listenUsingInsecureL2capChannel()
更新 03/2019
随着第一个 Android Q 测试版的推出,支持似乎终于到来了。
https://developer.android.com/sdk/api_diff/q-beta1/changes.html
2018 年 10 月更新
获悉 L2CAP CoC 将从 Android Q 开始支持。
很遗憾,我无法引用官方来源。
旧答案
我会 post 我自己的结果,希望能帮助别人,如果我遗漏了什么,可能会得到提示。
TL;DR: 看来我不能
无法通过 public API
获得 L2CAP CoC 套接字
Android 文档中的 This 页面声称“LE 面向连接的通道”(l2cap 通道)自 Android 8.0 起可用。
在 Android API 中有一个 BluetoothSocket
class 可以有类型 L2_CAP
。但是由于 class 的元素的私有可见性,无法获得具有这种类型的 BluetoothSocket 实例。 public Android java API: stuck.
所以?
在 public java API 的表面下挖掘。 android 版本 8+ 的源代码提供了一种获取随时间发展的 l2cap 套接字的方法:
createL2capSocket
方法 android 8 (source)
createL2capCoCSocket
方法 android 9 (source)
- A
createL2capChannel
在 android 项目的 master 分支中(我猜是未来版本?) (source)
但是在每个发布的版本中这段代码都标有@hide
注释,这意味着none这些方法实际上是可用的。
在 Android 之前,至少可以访问 9 个隐藏方法,在某些设备上,java 反射。这就是我们如何能够在 android 8 上进行测试,结果很糟糕。
从 OS 的第 9 版开始,这不再可能了:平台 actively blocks attempts 无法正确访问 APIs.
而且..
除了实例化套接字的方法不可见之外,还有证据表明 API 还缺乏调整 l2cap 连接参数的方法。
本地化?
我没有使用 NDK 的经验,但我认为可以编写 C 代码来利用隐藏方法所使用的相同低级函数。除非 android 也没有阻止这种用法,这很可能。
此外,这种用法会与 phone 上蓝牙堆栈的常规用法冲突:正常用法通常都由 Android 服务过滤。绕过此服务会导致不可预测的后果,并且它不适合生产应用程序。
总结一下
在 Android 8.1 上的反射测试导致了非常不令人满意的结果。自 Android 8.0 以来“面向 LE 连接的通道”可用的说法尽管付出了努力,但仍无法得到证实。
在较新的版本中,创建 l2cap 通道的隐藏方法在不断发展,最近隐藏注释是 removed。在我们手中拥有支持此功能的设备之前,很难说这在时间上意味着什么 API。很可能需要数年时间,l2cap 频道才会变得可靠并可供很大一部分 android 用户使用。
我看到 Bluetooth socket 可以是 TYPE_L2CAP 类型,但是 BluetoothSocket 的构造函数似乎是私有的,我只能找到一个方法来实例化 RFCOMM 类型的套接字。 如何获得和使用 L2CAP 套接字? Android真的支持吗?
2019 年 10 月更新
支持!享受新的 API:
- BluetoothDevice.createL2capChannel(int)
- BluetoothDevice.createInsecureL2capChannel(int)
- BluetoothAdapter.listenUsingL2capChannel()
- BluetoothAdapter.listenUsingInsecureL2capChannel()
更新 03/2019 随着第一个 Android Q 测试版的推出,支持似乎终于到来了。 https://developer.android.com/sdk/api_diff/q-beta1/changes.html
2018 年 10 月更新
获悉 L2CAP CoC 将从 Android Q 开始支持。
很遗憾,我无法引用官方来源。
旧答案
我会 post 我自己的结果,希望能帮助别人,如果我遗漏了什么,可能会得到提示。
TL;DR: 看来我不能
无法通过 public API
获得 L2CAP CoC 套接字 Android 文档中的This 页面声称“LE 面向连接的通道”(l2cap 通道)自 Android 8.0 起可用。
在 Android API 中有一个 BluetoothSocket
class 可以有类型 L2_CAP
。但是由于 class 的元素的私有可见性,无法获得具有这种类型的 BluetoothSocket 实例。 public Android java API: stuck.
所以?
在 public java API 的表面下挖掘。 android 版本 8+ 的源代码提供了一种获取随时间发展的 l2cap 套接字的方法:
createL2capSocket
方法 android 8 (source)createL2capCoCSocket
方法 android 9 (source)- A
createL2capChannel
在 android 项目的 master 分支中(我猜是未来版本?) (source)
但是在每个发布的版本中这段代码都标有@hide
注释,这意味着none这些方法实际上是可用的。
在 Android 之前,至少可以访问 9 个隐藏方法,在某些设备上,java 反射。这就是我们如何能够在 android 8 上进行测试,结果很糟糕。
从 OS 的第 9 版开始,这不再可能了:平台 actively blocks attempts 无法正确访问 APIs.
而且..
除了实例化套接字的方法不可见之外,还有证据表明 API 还缺乏调整 l2cap 连接参数的方法。
本地化?
我没有使用 NDK 的经验,但我认为可以编写 C 代码来利用隐藏方法所使用的相同低级函数。除非 android 也没有阻止这种用法,这很可能。 此外,这种用法会与 phone 上蓝牙堆栈的常规用法冲突:正常用法通常都由 Android 服务过滤。绕过此服务会导致不可预测的后果,并且它不适合生产应用程序。
总结一下
在 Android 8.1 上的反射测试导致了非常不令人满意的结果。自 Android 8.0 以来“面向 LE 连接的通道”可用的说法尽管付出了努力,但仍无法得到证实。
在较新的版本中,创建 l2cap 通道的隐藏方法在不断发展,最近隐藏注释是 removed。在我们手中拥有支持此功能的设备之前,很难说这在时间上意味着什么 API。很可能需要数年时间,l2cap 频道才会变得可靠并可供很大一部分 android 用户使用。