AVRCP的按键定义:

\sdk\emulator\keymaps\AVRCP.kl

key 200   MEDIA_PLAY_PAUSE    WAKE

key 201   MEDIA_PLAY_PAUSE    WAKE

key 166   MEDIA_STOP          WAKE

key 163   MEDIA_NEXT          WAKE

key 165   MEDIA_PREVIOUS      WAKE

key 168   MEDIA_REWIND        WAKE

key 208   MEDIA_FAST_FORWARD  WAKE

BCM(broadcom)宏定义需要打开:

BOARD_HAVE_BLUETOOTH_BCM := true

BT音频控制的代码

external\bluetooth\bluez\audio\control.c

(1) 按键的MAP

static struct {

const char *name;

uint8_t avrcp;

uint16_t uinput;

} key_map[] = {

{ "PLAY",     PLAY_OP,      KEY_PLAYCD },

{ "STOP",     STOP_OP,      KEY_STOPCD },

{ "PAUSE",    PAUSE_OP,     KEY_PAUSECD },

{ "FORWARD",      FORWARD_OP,       KEY_NEXTSONG },

{ "BACKWARD",     BACKWARD_OP,      KEY_PREVIOUSSONG },

{ "REWIND",       REWIND_OP,    KEY_REWIND },

{ "FAST FORWARD", FAST_FORWARD_OP,  KEY_FASTFORWARD },

{ NULL }

};

(2) 按键处理

static void handle_panel_passthrough(struct control *control,

{

for (i = 0; key_map[i].name != NULL; i++) {

uint8_t key_quirks;

if ((operands[0] & 0x7F) != key_map[i].avrcp)

continue;

DBG("AVRCP: %s %s", key_map[i].name, status);

key_quirks = control->key_quirks[key_map[i].avrcp];

if (key_quirks & QUIRK_NO_RELEASE) {

if (!pressed) {

DBG("AVRCP: Ignoring release");

break;

}

DBG("AVRCP: treating key press as press + release");

send_key(control->uinput, key_map[i].uinput, 1);

send_key(control->uinput, key_map[i].uinput, 0);

break;

}

send_key(control->uinput, key_map[i].uinput, pressed);

break;

}

}

HCIDUMP数据分析

以Sony耳机DRC-BT15为例

#adb shell

#hcidump –X

左键:

> ACL data: handle 12 flags 0x02 dlen 12

L2CAP(d): cid 0x0042 len 8 [psm 0]

0000: 30 11 0e 00 48 7c4c 00                           0...H|L.

< HCI Command: Exit Sniff Mode (0x02|0x0004) plen 2

0000: 0c 00                                             ..

< ACL data: handle 12 flags 0x00 dlen 12

0000: 08 00 4b 00 32 11 0e 09  48 7c4c 00              ..K.2...H|L.

> HCI Event: Max Slots Change (0x1b) plen 3

0000: 0c 00 05                                          ...

> HCI Event: Mode Change (0x14) plen 6

0000: 00 0c 00 00 00 00                                 ......

> HCI Event: Command Status (0x0f) plen 4

0000: 0c 01 04 08                                       ....

> ACL data: handle 12 flags 0x02 dlen 12

L2CAP(d): cid 0x0042 len 8 [psm 0]

0000: 40 11 0e 00 48 7c cc 00                           @...H|?

< ACL data: handle 12 flags 0x00 dlen 12

0000: 08 00 4b 00 42 11 0e 09  48 7c cc 00              ..K.B...H|?

> HCI Event: Number of Completed Packets (0x13) plen 5

0000: 01 0c 00 02 00                                    .....

播放/暂停:

> ACL data: handle 12 flags 0x02 dlen 12

L2CAP(d): cid 0x0042 len 8 [psm 0]

0000: 50 11 0e 00 48 7c 4b 00                           P...H|K.

< ACL data: handle 12 flags 0x00 dlen 12

0000: 08 00 4b 00 52 11 0e 09  48 7c 4b 00              ..K.R...H|K.

> ACL data: handle 12 flags 0x02 dlen 12

L2CAP(d): cid 0x0042 len 8 [psm 0]

0000: 60 11 0e 00 48 7c cb 00                           `...H|?

< ACL data: handle 12 flags 0x00 dlen 12

0000: 08 00 4b 00 62 11 0e 09  48 7c cb 00              ..K.b...H|?

> HCI Event: Number of Completed Packets (0x13) plen 5

0000: 01 0c 00 02 00                                    .....

右键:

> ACL data: handle 12 flags 0x02 dlen 12

L2CAP(d): cid 0x0042 len 8 [psm 0]

0000: 70 11 0e 00 48 7c 46 00                           p...H|F.

< ACL data: handle 12 flags 0x00 dlen 12

0000: 08 00 4b 00 72 11 0e 09  48 7c 46 00              ..K.r...H|F.

> ACL data: handle 12 flags 0x02 dlen 12

L2CAP(d): cid 0x0042 len 8 [psm 0]

0000: 80 11 0e 00 48 7c c6 00                           ....H|?

< ACL data: handle 12 flags 0x00 dlen 12

0000: 08 00 4b 00 82 11 0e 09  48 7c c6 00              ..K.....H|?

> HCI Event: Number of Completed Packets (0x13) plen 5

0000: 01 0c 00 02 00                                    .....

> HCI Event: Max Slots Change (0x1b) plen 3

0000: 0c 00 01                                          ...

> HCI Event: Mode Change (0x14) plen 6

0000: 00 0c 00 02 c8 00                                 ....?

然后将control.c的日志打印出来:

按一次”“播放/暂停键”:

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = 46   对应PAUSE_OP

D/ACRVP   (  237): key_quirks = 0, pressed = 1按键按下

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 201对应MEDIA_PLAY_PAUSE

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = c6  (= 0x46 | 0x80 表示按键释放了)

D/ACRVP   (  237): key_quirks = 0, pressed = 0按键释放

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 201 对应MEDIA_PLAY_PAUSE

再按一次”“播放/暂停键”:

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = 44   对应PLAY_OP

D/ACRVP   (  237): key_quirks = 0, pressed = 1 按键按下

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 200  对应MEDIA_PLAY_PAUSE

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = c4

D/ACRVP   (  237): key_quirks = 0, pressed = 0  按键释放

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 200 对应MEDIA_PLAY_PAUSE

next key:

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = 4b对应FORWARD_OP

D/ACRVP   (  237): key_quirks = 0, pressed = 1

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 163 对应MEDIA_NEXT

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = cb

D/ACRVP   (  237): key_quirks = 0, pressed = 0

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 163

prev key:

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = 4c 对应BACKWARD_OP

D/ACRVP   (  237): key_quirks = 0, pressed = 1

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 165 对应

D/ACRVP   (  237): --- handle_panel_passthrough ----

D/ACRVP   (  237): operands[0] = cc

D/ACRVP   (  237): key_quirks = 0, pressed = 0

D/ACRVP   (  237): control->uinput = fffffffe,send_key = 165 MEDIA_PREVIOUS

从上面可以看到bluetooth的协议栈blueZ是没有问题的

将 frameworks\base\libs\ui\EventHub.cpp的LOG打开,只能看到了control.c的日志,EventHub的getEvent完全不响应

观察所有log日志发现,openDevice里也没有装载AVRCP.kl

初步判断event有问题

 

event分析:

$ adb shell

# cd /proc/bus/input

# cat devices

cat devices

# cat devices

cat devices

I: Bus=0019 Vendor=0001 Product=0001 Version=0001   参考s3c-keypad.c

N: Name="s3c-keypad"        input_dev->name = DEVICE_NAME;

P: Phys=s3c-keypad/input0       input_dev->phys = "s3c-keypad/input0";

S: Sysfs=/devices/virtual/input/input0   virtual的?

U: Uniq=

H: Handlers=event0

B: EV=3

B: KEY=4000400 0

I: Bus=0019 Vendor=0001 Product=0001 Version=0100参考vpad_buttons.c

N: Name="s3c-eintkey"          input->name = pdev->name,  gpio_keys_device_driver.name    = "s3c-eintkey",

P: Phys=gpio-keys/input0         input->phys = "gpio-keys/input0"

S: Sysfs=/devices/platform/s3c-eintkey/input/input1   为什么这里是platform目录?

U: Uniq=

H: Handlers=event1

B: EV=3

B: KEY=100000 0 0 0

I: Bus=0018 Vendor=0000 Product=0000 Version=0000 参考amri_ts.c

N: Name="amri_ts"        amri_ts_driver.name = "amri_ts"

P: Phys=            没有定义

S: Sysfs=/devices/platform/s3c2440-i2c.0/i2c-0/0-0033/input/input2   为什么这里是platform目录?

U: Uniq=

H: Handlers=event2

B: EV=b

B: KEY=400 0 0 0 0 0 40000800 40 0 0 10000

B: ABS=2650000 0

I: Bus=0000 Vendor=0000 Product=0000 Version=0000

N: Name="ecompass_data"

P: Phys=

S: Sysfs=/devices/virtual/input/input3

U: Uniq=

H: Handlers=event3

B: EV=9

B: ABS=307bf

从上面可以看到,完全没有AVRCP的event。

解决办法:

 

Kernel

$ make menuconfig

CONFIG_INPUT_UINPUT

解决后的状况:

$ adb shell

# cd /proc/bus/input

# cat devices

显示增加了一个event

I: Bus=0005 Vendor=0000 Product=0000 Version=0000

N: Name="AVRCP"

P: Phys=

S: Sysfs=/devices/virtual/input/input4

U: Uniq=

H: Handlers=event4

B: EV=100007

B: KEY=10300 168 0 0 0 0 0

B: REL=0

看openDevice的信息:

D/EventHub(   84): EventHub::readNotify nfd: 87

D/EventHub(   84): Opening device: /dev/input/event4

D/EventHub(   84): Getting keys...

D/EventHub(   84): Getting absolute controllers...

D/EventHub(   84): keylayoutFilename = /system/usr/keylayout/AVRCP.kl

I/EventHub(   84): New keyboard: device->id=0x10003 devname='AVRCP' propName='hw.keyboards.65539.devname' keylayout='/system/usr/keylayout/AVRCP.kl'

I/EventHub(   84): New device: path=/dev/input/event4 name=AVRCP id=0x10003 (of 0x4) index=4 fd=196 classes=0x1

D/EventHub(   84): Adding device /dev/input/event4 0x361800 at 4, id = 3, classes = 0x1

D/EventHub(   84): Reporting device opened: id=0x10003, name=/dev/input/event4

 

按键时也可以看到EventHub的信息了:

Log如下:

D/ACRVP   (  236): --- handle_panel_passthrough ----

D/ACRVP   (  236): operands[0] = 46

D/ACRVP   (  236): key_quirks = 0, pressed = 1

D/ACRVP   (  236): control->uinput = 14,send_key = 201     control.c发出201号键,按键按下了

D/EventHub(   84): /dev/input/event4 got: t0=937, t1=582930, type=1, code=201, v=1   表示从event4得到201号键 MEDIA_PLAY_PAUSE

D/EventHub(   84): iev.code=201 keyCode=85 flags=0x00000001 err=0

D/EventHub(   84): /dev/input/event4 got: t0=937, t1=600241, type=0, code=0, v=0

D/AudioHardware(   61): AudioStreamOutALSA::setParameters() routing=0

D/ACRVP   (  236): --- handle_panel_passthrough ----

D/ACRVP   (  236): operands[0] = c6

D/ACRVP   (  236): key_quirks = 0, pressed = 0

D/ACRVP   (  236): control->uinput = 14,send_key = c9

D/EventHub(   84): /dev/input/event4 got: t0=937, t1=664391, type=1, code=201, v=0 control.c发出201号键,按键释放了

D/EventHub(   84): iev.code=201 keyCode=85 flags=0x00000001 err=0

D/EventHub(   84): /dev/input/event4 got: t0=937, t1=664406, type=0, code=0, v=0

D/A2DP    (   61): a2dp_stop

D/A2DP    (   61): bluetooth_stop

E/BluetoothEventLoop.cpp(   84): event_filter: Received signal org.bluez.AudioSink:Stopped from /org/bluez/236/hci0/dev_00_1D_BA_A5_D8_1C

E/BluetoothEventLoop.cpp(   84): event_filter: Received signal org.bluez.AudioSink:PropertyChanged from /org/bluez/236/hci0/dev_00_1D_BA_A5_D8_1C

E/BluetoothEventLoop.cpp(   84): event_filter: Received signal org.bluez.AudioSink:PropertyChanged from /org/bluez/236/hci0/dev_00_1D_BA_A5_D8_1C

V/BluetoothEventRedirector(  244): Received android.bluetooth.a2dp.action.SINK_STATE_CHANGED

D/CachedBluetoothDevice(  244): onProfileStateChanged: profile A2DP newProfileState 2

D/BluetoothA2dpService(   84): A2DP state : device: 00:1D:BA:A5:D8:1C State:4->2

D/A2DP    (   61): Received BT_RESPONSE - BT_STOP_STREAM

D/dalvikvm(  285): GC_EXPLICIT freed 87K, 56% free 2640K/5895K, external 1625K/2137K, paused 75ms

转载于:https://www.cnblogs.com/chenyready/p/7986712.html

Android-蓝牙AVRCP功能实现【转】相关推荐

  1. Android蓝牙通信功能开发

    1. 概述 Bluetooth 是几乎现在每部手机标准配备的功能,多用于耳机 mic 等设备与手机的连接,除此之外,还可以多部手机之间建立 bluetooth 通信,本文就通过 SDK 中带的一个聊天 ...

  2. Android蓝牙打印机功能开发完整Demo

    蓝牙便携式打印机的种类繁多,支持的打印格式也不尽相同.按照指令集可划分为:ESC指令集.CPCL指令集,实现原理基本相同,我这里以佳博便携式打印机为例,进行蓝牙搜索配对并发送打印数据. 完整代码地址在 ...

  3. android 蓝牙hf编程,基于Android蓝牙Inband ring功能实现.doc

    基于Android蓝牙Inband ring功能实现 基于Android蓝牙Inband ring功能实现 [摘要] 蓝牙作为一种短距无线数据与语音传输的开放性全球规范,目前在整个世界范围内都得到了很 ...

  4. android 蓝牙打印机(ESC/POS 热敏打印机),打印菜单小票和图片,对蓝牙配对和连接打印功能进行了封装,让你超快实现蓝牙打印功能

    BluetoothPrint 项目地址:liuGuiRong18/BluetoothPrint  简介:android 蓝牙打印机(ESC/POS 热敏打印机),打印菜单小票和图片,对蓝牙配对和连接打 ...

  5. Android Studio开发——蓝牙聊天功能

    Android Studio开发--蓝牙聊天功能 蓝牙工作流程 功能要求 实现要点 声明蓝牙权限 添加程序运行的状态描述文本及配色代码 布局文件 蓝牙会话的服务组件ChatService Activi ...

  6. Android Studio在类微信程序完成“蓝牙聊天功能”实现蓝牙通信

    Android Studio在类微信程序完成"蓝牙聊天功能"实现蓝牙通信 项目运行截图 通信原理 蓝牙权限 strings.xml tab01.xml 菜单文件option_men ...

  7. android 蓝牙锁应用开发实例(三)蓝牙相关功能实现【第一部分】

    本人水平有限,文章中如果出现什么不正确或者模糊的地方,还请各位小伙伴留下评论,多多指教 : ) 正式开始前的话 蓝牙开发梳理 整体思路 核心API BlueToothAdapter 简介 getDef ...

  8. android 蓝牙设置界面高级选项功能解析

    1.文件传输服务:是指蓝牙 File Transfer Profile (FTP),允许支持蓝牙FTP客户端的蓝牙设备进行连接,实现对服务端文件的访问和操作.   2.远程SIM卡模式:是指蓝牙SIM ...

  9. Android 蓝牙对战五子棋项目实现(含人机对战功能)

    上周花了一周时间做的课程设计的项目,实现的功能如下: 基本功能: (1) 该APP能够通过蓝牙自动搜索周围其他使用了该APP的手机,用户可选择其中某一个APP发起对战的要求,被发起方可以同意或者拒绝: ...

  10. Android蓝牙服务

    From 蓝牙服务 通过蓝牙,设备可以传输数据并供各种交互式服务(例如音频.短信和电话)使用.您可以在以下位置找到用于不同服务的蓝牙配置文件: include/hardware/bluetooth.h ...

最新文章

  1. Java并发编程:线程封闭和ThreadLocal详解
  2. 前沿 | 历时十二年!曼彻斯特百万级神经元的类脑超算终开启
  3. MySLQ排序后标记排行
  4. 可手工拖拽alv行记录的实例
  5. 通达oa 不允许从该ip登陆_通达OA-命令执行漏洞复现
  6. 40张图全面解析TCP 三次握手和四次挥手
  7. c#在当前窗体的按钮事件调用另一封…
  8. 数据结构与算法之-----总览
  9. 得力考勤机excel密码_考勤机
  10. lede 内核 单 编_openwrt和lede有何区别?
  11. RANSAC算法详解
  12. 工业机器人图册 索罗门采夫_机械手控制系统设计(完整图纸)
  13. 把自己曾经写的一个客户间聊天的程序与大家分享(2)
  14. canvas绘制虚线图表
  15. 华为项目管理的精髓干货!可收藏
  16. Flutter Sliver大家族之SliverPersistentHeader()和SliverToBoxAdapter()组件(实现固定头布局)③
  17. 【word2019】公式中如何让连等式等号对齐
  18. Keras学习笔记11——keras.applications
  19. 手把手教你搭建实时大数据引擎FLINK
  20. pscad 与 matlab 接口,PSCAD与MATLAB的接口问题

热门文章

  1. 吉林大学超星学习通05
  2. 你会用JSON.stringify()?
  3. Unity Webm格式视频报错
  4. 万能启动利器FbinstTool引导工具教程
  5. 火狐浏览器的hoxx附件还能用吗_Haspit
  6. JavaScript快速入门-基础
  7. 修改hosts文件,解决端口占用方法
  8. 数据集获取方式和数据加强方式
  9. ElasticSearch 安装IK分词器
  10. 《GO语言实战》笔记