[Android 源码解析]bluez中几个重要plugin的初始化--audio,input,network,health

本文主要是介绍[Android 源码解析]bluez中几个重要plugin的初始化--audio,input,network,health,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本来只想分析一下hciops的初始化就不再管了,后来发现别的plugin的初始化在后面还是会有涉及,心中想,既然已经分析了这么多,咋就索性都分析了算了,反正也不差这一点代码,其它的plugin还有audioinputnetworkhealth。我们只看两个函数init或者setup。所以看起来应该会比较简单一点。

 

2.3.7.1 audioinit分析

 audio 这个插件的分析

static int audio_init(void)
{
GKeyFile *config;
gboolean enable_sco;
//得到系统的dbus
connection = dbus_bus_get(DBUS_BUS_SYSTEM, NULL);
if (connection == NULL)
return -EIO;
//得到audio.config文件
config = load_config_file(CONFIGDIR "/audio.conf");
//根据config来进行init
if (audio_manager_init(connection, config, &enable_sco) < 0)
goto failed;
//没有使能sco,就直接return了,我们这里的确没有使能sco,所以,没什么好说的,直接return吧
if (!enable_sco)
return 0;
……
return -EIO;
}

1audio.conf的内容简介如下:

[General]
Enable=Sink,Control
Disable=Headset,Gateway,Source
Master=false
FastConnectable=false
[A2DP]
SBCSources=1
MPEG12Sources=0
[AVRCP]
InputDeviceName=AVRCP

2)根据config来进行init

int audio_manager_init(DBusConnection *conn, GKeyFile *conf,
gboolean *enable_sco)
{
char **list;
int i;
gboolean b;
GError *err = NULL;
//dbus的connection的ref+1
connection = dbus_connection_ref(conn);
//没有config文件,直接结束
if (!conf)
goto proceed;
//config是一个全局变量
config = conf;
/*
//从上面1)中可以得到
Enable=Sink,Control
Disable=Headset,Gateway,Source
*/
list = g_key_file_get_string_list(config, "General", "Enable",
NULL, NULL);
//enable的有哪些,加粗的就是enable的
for (i = 0; list && list[i] != NULL; i++) {
if (g_str_equal(list[i], "Headset"))
enabled.headset = TRUE;
else if (g_str_equal(list[i], "Gateway"))
enabled.gateway = TRUE;
else if (g_str_equal(list[i], "Sink"))
enabled.sink = TRUE;
else if (g_str_equal(list[i], "Source"))
enabled.source = TRUE;
else if (g_str_equal(list[i], "Control"))
enabled.control = TRUE;
else if (g_str_equal(list[i], "Socket"))
enabled.socket = TRUE;
else if (g_str_equal(list[i], "Media"))
enabled.media = TRUE;
}
g_strfreev(list);
//再看disable的,加粗的就是disable的
list = g_key_file_get_string_list(config, "General", "Disable",
NULL, NULL);
for (i = 0; list && list[i] != NULL; i++) {
if (g_str_equal(list[i], "Headset"))
enabled.headset = FALSE;
else if (g_str_equal(list[i], "Gateway"))
enabled.gateway = FALSE;
else if (g_str_equal(list[i], "Sink"))
enabled.sink = FALSE;
else if (g_str_equal(list[i], "Source"))
enabled.source = FALSE;
else if (g_str_equal(list[i], "Control"))
enabled.control = FALSE;
else if (g_str_equal(list[i], "Socket"))
enabled.socket = FALSE;
else if (g_str_equal(list[i], "Media"))
enabled.media = FALSE;
}
//除了上面这些设置,下面就还有几个默认的
/*
//这里是一些默认
//hsp是没有的,hfp有的
static struct enabled_interfaces enabled = {
.hfp		= TRUE,
.headset	= TRUE,
.gateway	= FALSE,
.sink		= TRUE,
.source		= FALSE,
.control	= TRUE,
.socket		= TRUE, //主要就是这个没有配置了
.media		= FALSE
};
*/
b = g_key_file_get_boolean(config, "General", "AutoConnect", &err);
if (err) { //没有,所以就是not found,使用默认值,是true
DBG("audio.conf: %s", err->message);
g_clear_error(&err);
} else
auto_connect = b; 
b = g_key_file_get_boolean(config, "Headset", "HFP",
&err);
if (err)
g_clear_error(&err);
else
enabled.hfp = b; //默认是true是支持的
err = NULL;
i = g_key_file_get_integer(config, "Headset", "MaxConnected",
&err);
if (err) {
DBG("audio.conf: %s", err->message);
g_clear_error(&err);
} else
max_connected_headsets = i; //没有设置,默认是1
proceed:
//socket是肯定要的
if (enabled.socket)
unix_init();
//media默认是false
if (enabled.media)
btd_register_adapter_driver(&media_server_driver);
//headset也是false
if (enabled.headset)
btd_register_adapter_driver(&headset_server_driver);
//gateway也是flase的
if (enabled.gateway)
btd_register_adapter_driver(&gateway_server_driver);
//这两个是enable的,把对应的driver加入到adapter_drivers列表中,并调用driver的probe函数,同时把这个driver加入到adapter->loaded_drivers列表中。然而这些有一个前提就是adapter需要up,若是没有也是不会做的,后面自然会有分析他们的地方,我们到时再继续分析。
if (enabled.source || enabled.sink)
btd_register_adapter_driver(&a2dp_server_driver);
//control也是enable的
if (enabled.control)
btd_register_adapter_driver(&avrcp_server_driver);
//audio也是enable的,不过这是一个device driver
btd_register_device_driver(&audio_driver);
//这里是false,sco是没有enable的
*enable_sco = (enabled.gateway || enabled.headset);
return 0;
}


 

所以,这里一共是3driver注册了。

3unix_init分析

//这里其实是新建一个本地的socket,作为server端,后期会有client可以通过这里进行交互和通信。
int unix_init(void)
{
GIOChannel *io;
struct sockaddr_un addr = {
AF_UNIX, BT_IPC_SOCKET_NAME
};
int sk, err;
//新建一个stram的socket
sk = socket(PF_LOCAL, SOCK_STREAM, 0);
if (sk < 0) {
err = errno;
error("Can't create unix socket: %s (%d)", strerror(err), err);
return -err;
}
//和这个地址绑定在一起
if (bind(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
error("Can't bind unix socket: %s (%d)", strerror(errno),
errno);
close(sk);
return -1;
}
//设置为non block
set_nonblocking(sk);
//最大连接数为1
if (listen(sk, 1) < 0) {
error("Can't listen on unix socket: %s (%d)",
strerror(errno), errno);
close(sk);
return -1;
}
unix_sock = sk;
//这边加一个io的watch,有了数据后就会去调用accept
io = g_io_channel_unix_new(sk);
//server_cb中会accept
g_io_add_watch(io, G_IO_IN | G_IO_HUP | G_IO_ERR | G_IO_NVAL,
server_cb, NULL);
g_io_channel_unref(io);
DBG("Unix socket created: %d", sk);
return 0;
}


 

2.3.7.2 inputinit函数分析

input 这个插件的初始化

static int input_init(void)
{
GKeyFile *config;
//老规矩,不多说
connection = dbus_bus_get(DBUS_BUS_SYSTEM, NULL);
if (connection == NULL)
return -EIO;
//加载input.conf,内容见1)
config = load_config_file(CONFIGDIR "/input.conf");
//根据config进行初始化
if (input_manager_init(connection, config) < 0) {
dbus_connection_unref(connection);
return -EIO;
}
if (config)
g_key_file_free(config);
return 0;
}


 

1input.conf的内容

内容简单的分析就如下:

[General]

别的什么都没有了,他其实能有的也就一个参数#IdleTimeout=30,这里用的是默认值。

2input_manager_init分析

int input_manager_init(DBusConnection *conn, GKeyFile *config)
{
GError *err = NULL;
//得到idle的timeout参数,没有设置,就是默认值0
if (config) {
idle_timeout = g_key_file_get_integer(config, "General",
"IdleTimeout", &err);
if (err) {
DBG("input.conf: %s", err->message);
g_error_free(err);
}
}
//dbus的ref+1
connection = dbus_connection_ref(conn);
//初始化了input server,同样因为adapter没有up,所以没有做对应的probe
btd_register_adapter_driver(&input_server_driver);
//注册了两个device driver
btd_register_device_driver(&input_hid_driver);
btd_register_device_driver(&input_headset_driver);
return 0;
}


 

2.3.7.3 networkinit函数分析

static int network_init(void)
{
connection = dbus_bus_get(DBUS_BUS_SYSTEM, NULL);
if (connection == NULL)
return -EIO;
//根据network.conf来进行各种配置
if (network_manager_init(connection) < 0) {
dbus_connection_unref(connection);
return -EIO;
}
return 0;
}
int network_manager_init(DBusConnection *conn)
{
//读取network.conf文件,network.conf里面也没有内容,设置conf_security,默认为true
read_config(CONFIGDIR "/network.conf");
//新建一个bnep的socket
if (bnep_init()) {
error("Can't init bnep module");
return -1;
}
/*
* There is one socket to handle the incomming connections. NAP,
* GN and PANU servers share the same PSM. The initial BNEP message
* (setup connection request) contains the destination service
* field that defines which service the source is connecting to.
*/
//初始化security
if (server_init(conn, conf_security) < 0)
return -1;
//加入到adapter_drivers列表中
/* Register network server if it doesn't exist */
btd_register_adapter_driver(&network_server_driver);
//dbus connection
if (connection_init(conn) < 0)
return -1;
//初始化了3个device driver
btd_register_device_driver(&network_panu_driver);
btd_register_device_driver(&network_gn_driver);
btd_register_device_driver(&network_nap_driver);
connection = dbus_connection_ref(conn);
return 0;
}


 

2.3.7.4 healthinit

health总得来说用得还是蛮少的,不过android4.0也是支持了,我们来看一下吧:

static int hdp_init(void)
{
connection = dbus_bus_get(DBUS_BUS_SYSTEM, NULL);
if (connection == NULL)
return -EIO;
//就是这个函数了
if (hdp_manager_init(connection) < 0) {
dbus_connection_unref(connection);
return -EIO;
}
return 0;
}
int hdp_manager_init(DBusConnection *conn)
{
//就是注册了一个health_manager_methods的接口,里面有creatapplication和destroyapplication两个函数
if (hdp_manager_start(conn))
return -1;
connection = dbus_connection_ref(conn);
//注册一个adapter drvier和一个device driver。
btd_register_adapter_driver(&hdp_adapter_driver);
btd_register_device_driver(&hdp_device_driver);
return 0;
}


 

至此,所有的plugin都已经完成了。总共注册了一下几个adapter_drivera2dp_server_driveravrcp_server_driverinput_server_drivernetwork_server_driverhdp_adapter_driver。另外还有几个devcie driveraudio_driverinput_hid_driverinput_headset_drivernetwork_panu_drivernetwork_gn_drivernetwork_nap_driverhdp_device_driver。在后面adapter up之后我们会加载这些driver,到时我们再来详细分析他们各自都做了些什么。

 

若您觉得该文章对您有帮助,请在下面用鼠标轻轻按一下“顶”,哈哈~~·

这篇关于[Android 源码解析]bluez中几个重要plugin的初始化--audio,input,network,health的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1079306

相关文章

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

Python 中的异步与同步深度解析(实践记录)

《Python中的异步与同步深度解析(实践记录)》在Python编程世界里,异步和同步的概念是理解程序执行流程和性能优化的关键,这篇文章将带你深入了解它们的差异,以及阻塞和非阻塞的特性,同时通过实际... 目录python中的异步与同步:深度解析与实践异步与同步的定义异步同步阻塞与非阻塞的概念阻塞非阻塞同步

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

Android自定义Scrollbar的两种实现方式

《Android自定义Scrollbar的两种实现方式》本文介绍两种实现自定义滚动条的方法,分别通过ItemDecoration方案和独立View方案实现滚动条定制化,文章通过代码示例讲解的非常详细,... 目录方案一:ItemDecoration实现(推荐用于RecyclerView)实现原理完整代码实现

Redis中高并发读写性能的深度解析与优化

《Redis中高并发读写性能的深度解析与优化》Redis作为一款高性能的内存数据库,广泛应用于缓存、消息队列、实时统计等场景,本文将深入探讨Redis的读写并发能力,感兴趣的小伙伴可以了解下... 目录引言一、Redis 并发能力概述1.1 Redis 的读写性能1.2 影响 Redis 并发能力的因素二、

SpringBoot整合jasypt实现重要数据加密

《SpringBoot整合jasypt实现重要数据加密》Jasypt是一个专注于简化Java加密操作的开源工具,:本文主要介绍详细介绍了如何使用jasypt实现重要数据加密,感兴趣的小伙伴可... 目录jasypt简介 jasypt的优点SpringBoot使用jasypt创建mapper接口配置文件加密

Spring MVC使用视图解析的问题解读

《SpringMVC使用视图解析的问题解读》:本文主要介绍SpringMVC使用视图解析的问题解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring MVC使用视图解析1. 会使用视图解析的情况2. 不会使用视图解析的情况总结Spring MVC使用视图

Android App安装列表获取方法(实践方案)

《AndroidApp安装列表获取方法(实践方案)》文章介绍了Android11及以上版本获取应用列表的方案调整,包括权限配置、白名单配置和action配置三种方式,并提供了相应的Java和Kotl... 目录前言实现方案         方案概述一、 androidManifest 三种配置方式

利用Python和C++解析gltf文件的示例详解

《利用Python和C++解析gltf文件的示例详解》gltf,全称是GLTransmissionFormat,是一种开放的3D文件格式,Python和C++是两个非常强大的工具,下面我们就来看看如何... 目录什么是gltf文件选择语言的原因安装必要的库解析gltf文件的步骤1. 读取gltf文件2. 提