耳机插拔流程

1.1      耳机

Android系统中,有线耳机分两种,一种带mic,一种不带mic,带mic的耳机被称为Headset,不带mic的耳机被称为HeadPhone。在audio.h中,有以下几个设备来表示耳机:

       AUDIO_DEVICE_OUT_WIRED_HEADSET             = 0x4,

AUDIO_DEVICE_OUT_WIRED_HEADPHONE           = 0x8,

AUDIO_DEVICE_IN_WIRED_HEADSET         = AUDIO_DEVICE_BIT_IN | 0x10

         对于Headset的插入检测,一般通过jack耳机插座来完成,大概原理是使用带检测机构的耳机插座,将检测脚连到GPIO中断上,耳机插入时,是的检测脚的电平变化,引起中断。通过GPIO的值判断耳机是插入还是拔出。

1.2      Uevent

目前使用的耳机插拔事件使用的是UEvent。但是也可以使用InputEvent,这种设置在frameworks/base/core/res/res/values/config.xml中,设置项为

<boolname="config_useDevInputEventForAudioJack">false</bool>

值为false,所以目前kernel是用Uevent来通知应用层耳机插拔的。

什么是Uevent

1.    内核通知应用的一种方式

2.    目前使用socket进行内核和应用的通信

3.    uevent就是一个特殊格式的字符串,如下,这应该是插入鼠标的Uevent

"add@/class/input/input9/mouse2\0    // message

ACTION=add\0                         // action type

DEVPATH=/class/input/input9/mouse2\0 // path in /sys

SUBSYSTEM=input\0                    // subsystem (class)

SEQNUM=1064\0                        // sequence number

PHYSDEVPATH=/devices/pci0000:00/0000:00:1d.1/usb2/2­2/2­2:1.0\0  // device path in /sys

PHYSDEVBUS=usb\0      // bus

PHYSDEVDRIVER=usbhid\0 // driver

MAJOR=13\0            // major number

MINOR=34\0",          // minor number

内核发送uevent使用kobject_uevent_env,它还有一个wrapperkobject_uevent,这个函数使用的比较多

1.3      UEvent是怎么工作的?

1.3.1       Switch驱动

Switch是在android中实现的一个module,可动态加载,是一个驱动。 Switch相关的代码在在drivers\switch\switch_class.cdrivers\switch\switch_gpio.c中,其中switch_gpio基于platform_device框架,在switch_class.c中,Export了以下两个函数供switch_gpio使用

    EXPORT_SYMBOL_GPL(switch_dev_register);

EXPORT_SYMBOL_GPL(switch_dev_unregister);

Switch_gpio.c是驱动模块栈的更靠近设备的一层,这个设备就是gpio,初始化函数是

gpio_switch_probe{

         switch_dev_register(&switch_data->sdev);

         gpio_request(switch_data->gpio,pdev->name);

         gpio_direction_input(switch_data->gpio);

         gpio_to_irq(switch_data->gpio);

         request_irq//申请GPIO中断

         gpio_switch_work

}

这货是个platform驱动,它会调用

static int __init gpio_switch_init(void)

{

         returnplatform_driver_register(&gpio_switch_driver);

}

把自己注册成一个platform驱动。Platform是一种虚拟总线,和spiemmc等都是类似的。

基于platform device/driver框架,在probe函数中完成初始化,包括获取gpio的使用权限,设置gpio方向为输入,注册switch_dev设备,为gpio分配中断,指定中断服务程序,初始化一个gpio_switch_work工作,最后读取gpio初始状态。  

switch_class.c中,调用switch_set_state->switch_gpio_print_state输出GPIO状态到 sysfs

Sysfs是一个虚拟的文件系统,由linux内核提供,通过使用虚拟文件,sysfs向用户空间公布了关于内核子系统的信息,硬件设备和相关的设备驱动。但是这个文件系统并不是被上层直接使用的,它仅仅是作为比较直观的方式提供给human being检查状态用的。

而真正发送uevent是在switch_class.c的这个函数里:

void switch_set_state(struct switch_dev *sdev,int state)

最终通过kobject_uevent_env(&sdev->dev->kobj, KOBJ_CHANGE, envp)发送。

kobject_uevent_env的流程如下:

1.    调用kobject_uevent_env之前,系统已经调用uevent_net_init(struct net *net)完成了初始化工作,也就是创建了socket,并创建了netlink,如下:

         netlink_kernel_create(net,NETLINK_KOBJECT_UEVENT,1, NULL, NULL, THIS_MODULE);

                   sock_create_lite(PF_NETLINK,SOCK_DGRAM, unit, &sock)

                   __netlink_create(&init_net,sock, cb_mutex, unit)

2.    netlink_broadcast_filterednetlink广播uevent信息。

         sk_for_each_bound

                   do_one_broadcast(sk,&info);

上面提到了Netlink socket,下面看看这是啥:

Netlink socket是一种Linux特有的socket,用于实现用户进程与内核进程之间通信的一种特殊的进程间通信方式(IPC) ,也是网络应用程序与内核通信的最常用的接口。
Netlink 是一种在内核和用户应用间进行双向数据传输的非常好的方式,用户态应用使用标准的 socket API 就能使用 Netlink 提供的强大功能,内核态需要使用专门的内核 API 来使用 Netlink

1.3.2       Netlink Socket通信

用户空间程序只需要创建一个NETLINK socket描述符,就可以侦听Uevent了,如下:

sockfd=socket(AF_NETLINK,SOCK_RAW,NETLINK_KOBJECT_UEVENT);

AF_NETLINK 是一种“domain”,定义了通信使用的协议,比如我们通常使用的IPV4协议。

<sys/socket.h>定义的所有的协议如下: 

AF_UNIX,

AF_LOCAL

AF_INET

IPv4

AF_INET6

IPv6

AF_IPX

IPX

AF_NETLINK

Kernel

AF_X25

ITU-T

AF_AX25

Amateur

AF_ATMPVC

Access

AF_APPLETALK

AppleTal

AF_PACKET

Low

AF_ALG

Interface

事实上,Framework就是写了一个很简单的socket网络程序:

uevent.c (hardware\libhardware_legacy\uevent)中,看函数uevent_init

s =socket(PF_NETLINK, SOCK_DGRAM, NETLINK_KOBJECT_UEVENT);

setsockopt(s,SOL_SOCKET, SO_RCVBUFFORCE, &sz, sizeof(sz));

bind(s,(struct sockaddr *) &addr, sizeof(addr)

再看uevent_next_event(char* buffer, intbuffer_length)

int count =recv(fd, buffer, buffer_length, 0);

其实就是读取socket的内容到buffer中。

1.3.3       观察者模式

拿到socket的内容后,需要通知对它感兴趣的模块。这里就使用了观察者模式。Androidframeworks/base/services/java/com/android/server/WiredAccessoryManager. java中实现对UEvent的处理。在这个文件中,从UEventObserver中继承了类WiredAccessoryObserver,在makeObservedUEventList中将要观察的事件加入到UEvent系统中:

 

UeventObserver持有一个静态的UEventThread对象,所有子类Observer都共享一个Thread

private staticUEventThread sThread;

UeventThread是一个静态类,有一个Array

private finalArrayList<Object> mKeysAndObservers = new ArrayList<Object>();

各种子类Observer都添加keyobservermKeysAndObservers。比如,WiredAccessoryObserver添加(“DEVPATH= /devices/virtual/switch/h2w”,this),(“DEVPATH= /devices/virtual/switch/usb_audio”,this),(“DEVPATH= /devices/virtual/switch/hdmi”,this)到mKeysAndObservers。这个Thread不断的去读取AF_NETLINK中的数据,并对注册进去的key字符串进行匹配,匹配上的调用改key对应的observeronUevent函数。

如何匹配的?

private voidsendEvent(String message){

for;;{

if(message.contains(key))

mTempObserversToSignal.add(observer);

}

for (;;) {

final UEventObserverobserver = mTempObserversToSignal.get(i);

observer.onUEvent(event);

}

}

这个onUEvent经过一系列无聊的调用,最终调到

AudioManager:: setWiredDeviceConnectionState(intdevice, int state, String name),自此进入Audio子系统

1.4      Audio子系统

最终会到Audio Policy中的setDeviceConnectionState来,它有两个参数,一个表示设备id,另一个表示该设备是插入还是拔出。比如设备id 4代表的是HEADSET,而当stateAUDIO_POLICY_DEVICE_STATE_AVAILABLE时表示插入,当AUDIO_POLICY_DEVICE_STATE_UNAVAILABLE时表示拔出。当设备有hw module与之相对应时(getModuleForDevice),会增加或减少mAvailableOutputDevicesmAvailableInputDevices

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
我可以为您提供一个TWS蓝牙耳机程序的基本流程图,但由于具体的实现方式和细节可能会因开发者和设备而异,因此以下流程图仅供参考: ```mermaid graph LR A(初始化) --> B(开启蓝牙) B --> C(扫描设备) C --> D(连接设备) D --> E(设备配对) E --> F(配对成功) F --> G(TWS连接) G --> H(开启音频传输) H --> I(等待音频数据) I --> J(音频解码) J --> K(音频输出) K --> L(结束) ``` 具体来说,TWS蓝牙耳机程序的基本流程如下: 1. 初始化:启动程序并初始化相关变量和参数,包括蓝牙、音频解码器、音频输出设备等。 2. 开启蓝牙:启动蓝牙模块并打开蓝牙设备,使其可以被其他蓝牙设备扫描到。 3. 扫描设备:在可用设备列表中扫描并显示蓝牙耳机设备。 4. 连接设备:根据用户选择,连接到指定的蓝牙耳机设备。 5. 设备配对:如果该蓝牙耳机设备未曾与此前的设备配对过,则需要进行配对操作。 6. 配对成功:如果配对成功,则进行TWS连接操作。 7. TWS连接:通过蓝牙协议建立TWS连接,以便在两个耳机之间进行音频传输。 8. 开启音频传输:打开音频传输通道,允许音频数据流从主耳机传输到副耳机。 9. 等待音频数据:等待主耳机发送音频数据。 10. 音频解码:将接收到的音频数据进行解码,以便输出到耳机扬声器。 11. 音频输出:将解码后的音频数据输出到耳机扬声器,以便用户听到音频。 12. 结束:关闭程序并释放相关资源。 请注意,这只是一个基本的TWS蓝牙耳机程序流程图,实际的程序可能会更加复杂,例如支持多种音频格式、噪音抑制等功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值