蓝牙设备也有VendorId ProductId ?

本文探讨了蓝牙HID设备的驱动实现过程,特别是在内核3.4环境下如何解决VIDPID未知的问题。通过分析内核代码及蓝牙协议,明确了VIDPID的获取方式,并介绍了3.5以上内核版本中对HID设备驱动的改进。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在做一个蓝牙HID设备的时候,从机使用的SensorTile,主机使用的Android 5.0平板(内核3.4),在内核中的HID驱动和HID设备是通过VID PID匹配的,而这个SensorTile的VID PID一直是0x00, 0x00。为了快速验证,我把generic-usb或者hid-multitouchhid_device_id表中添加了如下代码以兜底这个VID PID均为0x00的蓝牙HID设备,经测试是可以顺利驱动的。

// for generic-usb
{ HID_BLUETOOTH_DEVICE(HID_ANY_ID, HID_ANY_ID) },
// for hid-multitouch
{ .driver_data = MT_CLS_DEFAULT, HID_BLUETOOTH_DEVICE(HID_ANY_ID,HID_ANY_ID) },

必经上面这种改法只是为了验证,最终这样改不合适,那么蓝牙设备的VID PID是在哪里设置的呢?我作了进一步追踪。
内核的上一级是uhid,通过查看在external/bluetooth/bluedroid/btif/co/bta_hh_co.cbta_hh_co_send_hid_info这里进入到uhid的。

/*******************************************************************************
**
** Function         bta_hh_co_send_hid_info
**
** Description      This function is called in btif_hh.c to process DSCP received.
**
** Parameters       dev_handle  - device handle
**                  dscp_len    - report descriptor length
**                  *p_dscp     - report descriptor
**
** Returns          void
*******************************************************************************/
void bta_hh_co_send_hid_info(btif_hh_device_t *p_dev, char *dev_name, UINT16 vendor_id,
                             UINT16 product_id, UINT16 version, UINT8 ctry_code,
                             int dscp_len, UINT8 *p_dscp)
{
    ...
    APPL_TRACE_WARNING("%s: vendor_id = 0x%04x, product_id = 0x%04x, version= 0x%04x,"
                                                                    "ctry_code=0x%02x",__FUNCTION__,
                                                                    vendor_id, product_id,
                                                                    version, ctry_code);
    ...
}

向上再追踪到了:

/*******************************************************************************
**
** Function         SDP_GetDiRecord
**
** Description      This function retrieves a remote device's DI record from
**                  the specified database.
**
** Returns          SDP_SUCCESS if record retrieved, else error
**
*******************************************************************************/
UINT16 SDP_GetDiRecord( UINT8 get_record_index, tSDP_DI_GET_RECORD *p_device_info,
                        tSDP_DISCOVERY_DB *p_db )
{
        ...
        p_curr_attr = SDP_FindAttributeInRec( p_curr_record, ATTR_ID_VENDOR_ID );
        if ( p_curr_attr )
            p_device_info->rec.vendor = p_curr_attr->attr_value.v.u16;
        else
            result = SDP_ERR_ATTR_NOT_PRESENT;
        ...
}

通过查看得知该函数是通过SDP协议获取VENDOR_ID的,在蓝牙协议官网上一查,还真就查到了:
SDP协议中的VendorId
来源:https://www.bluetooth.com/specifications/assigned-numbers/service-discovery

那么现在就清楚了,蓝牙中也是有VID 和 PID,方便区别不同的厂商,这样主机系统驱动中就可以加以区分。需要具体到从机的SDP协议中查看一下。下回分解。

更新:
经查看3.5以上的内核不需要在内核中指定VID PID了,根据HID描述符来确定是否为多点触控设备,如果描述符中包含HID_DG_CONTACTID则就是多点触摸设备,自动使用hid-multitouch进行驱动。这也就是为什么在Ubuntu 16.04上什么都不用改就可以驱动起来了。

《完》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

袁保康

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值