[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常见环境管理工具超全解析》在Python开发中,管理多个项目及其依赖项通常是一个挑战,下面:本文主要介绍python常见环境管理工具的相关资料,文中通过代码介绍的非常详细,需要的朋友... 目录1. conda2. pip3. uvuv 工具自动创建和管理环境的特点4. setup.py5.

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

全面解析HTML5中Checkbox标签

《全面解析HTML5中Checkbox标签》Checkbox是HTML5中非常重要的表单元素之一,通过合理使用其属性和样式自定义方法,可以为用户提供丰富多样的交互体验,这篇文章给大家介绍HTML5中C... 在html5中,Checkbox(复选框)是一种常用的表单元素,允许用户在一组选项中选择多个项目。本

Python包管理工具核心指令uvx举例详细解析

《Python包管理工具核心指令uvx举例详细解析》:本文主要介绍Python包管理工具核心指令uvx的相关资料,uvx是uv工具链中用于临时运行Python命令行工具的高效执行器,依托Rust实... 目录一、uvx 的定位与核心功能二、uvx 的典型应用场景三、uvx 与传统工具对比四、uvx 的技术实

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

Redis过期删除机制与内存淘汰策略的解析指南

《Redis过期删除机制与内存淘汰策略的解析指南》在使用Redis构建缓存系统时,很多开发者只设置了EXPIRE但却忽略了背后Redis的过期删除机制与内存淘汰策略,下面小编就来和大家详细介绍一下... 目录1、简述2、Redis http://www.chinasem.cn的过期删除策略(Key Expir

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析

《Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析》InstantiationAwareBeanPostProcessor是Spring... 目录一、什么是InstantiationAwareBeanPostProcessor?二、核心方法解

深入解析 Java Future 类及代码示例

《深入解析JavaFuture类及代码示例》JavaFuture是java.util.concurrent包中用于表示异步计算结果的核心接口,下面给大家介绍JavaFuture类及实例代码,感兴... 目录一、Future 类概述二、核心工作机制代码示例执行流程2. 状态机模型3. 核心方法解析行为总结:三

springboot项目中使用JOSN解析库的方法

《springboot项目中使用JOSN解析库的方法》JSON,全程是JavaScriptObjectNotation,是一种轻量级的数据交换格式,本文给大家介绍springboot项目中使用JOSN... 目录一、jsON解析简介二、Spring Boot项目中使用JSON解析1、pom.XML文件引入依