Android BlueDroid分析: 配置文件(bt_stack.conf bt_vendor.conf )的加载与分析

2024-03-04 12:18

本文主要是介绍Android BlueDroid分析: 配置文件(bt_stack.conf bt_vendor.conf )的加载与分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

说明

在Android BlueDroid启动,即stack启动的时候,回去加载好几个配置文件, 然后BlueDroid Stack根据这几个配置文件会进行调整, 例如Device ID(did), Log相关的Trace Level, COD(即Class of Device), BT snoop log相关配置等等.下面结合代码和配置文件一起来说明分析.

配置文件说明

配置文件分为运行时动态加载和编译的时候直接解析使用的.主要有下面三个, 冒号后面是简要的作用说明.

编译完成后,这些配置文件都是位于: /system/etc/bluetooth/

如果是编译之前,那么是在device相关的vendor board中, 或者是默认的stack里面自带的文件.


 bt_stack.conf: 对stack进行配置, 包括不同的stack里面的log level, bt snoop的控制

 bt_did.conf : 记录和配置BLE Controller的信息, 例如VendorID可能是QualComm也可以是Broadcom等等.

 auto_pair_devlist.conf: 将某些slave/peripheral加入到BlackList中不让其配对

config格式

绝大部分都是用等号表示的string=value对, 也有使用{}与逗号区分的一串数据, 例如COD.

代码分析

配置文件的解析位于bte_config.c中, 其含有下面这些函数:

▼ functionsdevice_name_cfg(char *p_conf_name, char *p_conf_value)device_class_cfg(char *p_conf_name, char *p_conf_value)logging_cfg_onoff(char *p_conf_name, char *p_conf_value)logging_set_filepath(char *p_conf_name, char *p_conf_value)trace_cfg_onoff(char *p_conf_name, char *p_conf_value)bte_load_conf(const char *p_path)-bte_parse_did_conf(const char *p_path, UINT32 num, tKEY_VALUE_PAIRS *conf_pairs, UINT32 conf_pairs_num)bte_load_did_conf(const char *p_path)

其中有trace log level解析相关的函数, 对于不同的文件解析使用的函数不同, 例如did使用did相关函数, 而bt_stack.conf使用bte_load_conf来解析.

而代码的实现也比较简单, 使用最多的函数就是strtok, 即对token的解析(可以参考C++程序设计原理与实践). 

解析完成后进行赋值完事.

那么对于did都有哪些assignment item:

/*******************************************************************************
**
** Function        bte_load_did_conf
**
** Description     Set local Device ID records, reading from configuration files
**
** Returns         None
**
*******************************************************************************/void bte_load_did_conf (const char *p_path)
{tBTA_DI_RECORD rec;UINT32 rec_num, i, j;for (i=1; i<=BTA_DI_NUM_MAX; i++) {for (j=0; j<CONF_DID_MAX; j++) {*did_conf_pairs[j].value = 0;}if (bte_parse_did_conf(p_path, i, did_conf_pairs, CONF_DID_MAX)) {memset(&rec, 0, sizeof(rec));if (*did_conf_pairs[CONF_DID_RECORD_NUM].value) {rec_num = (UINT32)(strtoul(did_conf_pairs[CONF_DID_RECORD_NUM].value, NULL, 0)-1);} else {debug("[%d] Unknown %s", (unsigned int)i, did_conf_pairs[CONF_DID_RECORD_NUM].key);continue;}if (*did_conf_pairs[CONF_DID_VENDOR_ID].value) {rec.vendor = (UINT16)strtoul(did_conf_pairs[CONF_DID_VENDOR_ID].value, NULL, 0);} else {rec.vendor = LMP_COMPID_BROADCOM;}if (*did_conf_pairs[CONF_DID_VENDOR_ID_SOURCE].value) {rec.vendor_id_source = (UINT16)strtoul(did_conf_pairs[CONF_DID_VENDOR_ID_SOURCE].value, NULL, 0);} else {rec.vendor_id_source = DI_VENDOR_ID_SOURCE_BTSIG;}if ((*did_conf_pairs[CONF_DID].value == 0) ||(rec_num >= BTA_DI_NUM_MAX) ||(!((rec.vendor_id_source >= DI_VENDOR_ID_SOURCE_BTSIG) &&(rec.vendor_id_source <= DI_VENDOR_ID_SOURCE_USBIF))) ||(rec.vendor == DI_VENDOR_ID_DEFAULT)) {error("DID record #%u not set", (unsigned int)i);for (j=0; j<CONF_DID_MAX; j++) {error("%s:%s", did_conf_pairs[j].key, did_conf_pairs[j].value);}continue;}rec.product = (UINT16)strtoul(did_conf_pairs[CONF_DID_PRODUCT_ID].value, NULL, 0);rec.version = (UINT16)strtoul(did_conf_pairs[CONF_DID_VERSION].value, NULL, 0);strncpy(rec.client_executable_url,did_conf_pairs[CONF_DID_CLIENT_EXECUTABLE_URL].value,SDP_MAX_ATTR_LEN);strncpy(rec.service_description,did_conf_pairs[CONF_DID_SERVICE_DESCRIPTION].value,SDP_MAX_ATTR_LEN);strncpy(rec.documentation_url,did_conf_pairs[CONF_DID_DOCUMENTATION_URL].value,SDP_MAX_ATTR_LEN);for (j=0; j<strlen(did_conf_pairs[CONF_DID_PRIMARY_RECORD].value); j++) {did_conf_pairs[CONF_DID_PRIMARY_RECORD].value[j] =tolower(did_conf_pairs[CONF_DID_PRIMARY_RECORD].value[j]);}if ((!strcmp(did_conf_pairs[CONF_DID_PRIMARY_RECORD].value, "true")) ||(!strcmp(did_conf_pairs[CONF_DID_PRIMARY_RECORD].value, "1"))) {rec.primary_record = TRUE;} else {rec.primary_record = FALSE;}info("[%u] primary_record=%d vendor_id=0x%04X vendor_id_source=0x%04X product_id=0x%04X version=0x%04X",(unsigned int)rec_num+1, rec.primary_record, rec.vendor,rec.vendor_id_source, rec.product, rec.version);if (*rec.client_executable_url) {info(" client_executable_url=%s", rec.client_executable_url);}if (*rec.service_description) {info(" service_description=%s", rec.service_description);}if (*rec.documentation_url) {info(" documentation_url=%s", rec.documentation_url);}if (BTA_DmSetLocalDiRecord(&rec, &rec_num) != BTA_SUCCESS) {error("SetLocalDiInfo failed for #%u!", (unsigned int)i);}}}
}

上面的代码看起来有一片,但是根据数组的下表宏,可以很清晰的知道只有这么几个:

▼ __anon3* : enum[enumerators]-CONF_DID-CONF_DID_RECORD_NUM-CONF_DID_PRIMARY_RECORD-CONF_DID_VENDOR_ID-CONF_DID_VENDOR_ID_SOURCE-CONF_DID_PRODUCT_ID-CONF_DID_VERSION-CONF_DID_CLIENT_EXECUTABLE_URL-CONF_DID_SERVICE_DESCRIPTION-CONF_DID_DOCUMENTATION_URL

对应的意义如下:

static tKEY_VALUE_PAIRS did_conf_pairs[CONF_DID_MAX] = {{ "[DID]",               "" },{ "recordNumber",        "" },{ "primaryRecord",       "" },{ "vendorId",            "" },{ "vendorIdSource",      "" },{ "productId",           "" },{ "version",             "" },{ "clientExecutableURL", "" },{ "serviceDescription",  "" },{ "documentationURL",    "" },
};

对于bt_stack.conf的解析分为两类, 一类是Trace Log Level, 还有就是其他的,具体见下面代码中的注释, 下面就是所有的可用被用来配置的string:

static const conf_entry_t conf_table[] = {/*{"Name", device_name_cfg},{"Class", device_class_cfg},*/{"BtSnoopLogOutput", logging_cfg_onoff},{"BtSnoopFileName", logging_set_filepath},{"TraceConf", trace_cfg_onoff},{(const char *) NULL, NULL}
};
解析的代码如下:

void bte_load_conf(const char *p_path)
{FILE    *p_file;char    *p_name;char    *p_value;conf_entry_t    *p_entry;char    line[CONF_MAX_LINE_LEN+1]; /* add 1 for \0 char */BOOLEAN name_matched;ALOGI("Attempt to load stack conf from %s", p_path);if ((p_file = fopen(p_path, "r")) != NULL){/* read line by line */while (fgets(line, CONF_MAX_LINE_LEN+1, p_file) != NULL){if (line[0] == CONF_COMMENT)continue;p_name = strtok(line, CONF_DELIMITERS);if (NULL == p_name){continue;}p_value = strtok(NULL, CONF_VALUES_DELIMITERS);if (NULL == p_value){ALOGW("bte_load_conf: missing value for name: %s", p_name);continue;}name_matched = FALSE;p_entry = (conf_entry_t *)conf_table;while (p_entry->conf_entry != NULL){if (strcmp(p_entry->conf_entry, (const char *)p_name) == 0)//判断是否是前面list中的{name_matched = TRUE;if (p_entry->p_action != NULL)p_entry->p_action(p_name, p_value);break;}p_entry++;}if ((name_matched == FALSE) && (trace_conf_enabled == TRUE)) //判断是否是TraceLevel{/* Check if this is a TRC config item */bte_trace_conf(p_name, p_value);}}fclose(p_file);}else{ALOGI( "bte_load_conf file >%s< not found", p_path);}
}

总结

实际上,除了上面三个conf文件, 我们还会看到一个额外的bt_vendor.conf, 在某些设备上面, 这个配置文件是给libbt-vendor.so中用的,用来配置串口与firmware. 

这篇关于Android BlueDroid分析: 配置文件(bt_stack.conf bt_vendor.conf )的加载与分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

Spring Boot 配置文件之类型、加载顺序与最佳实践记录

《SpringBoot配置文件之类型、加载顺序与最佳实践记录》SpringBoot的配置文件是灵活且强大的工具,通过合理的配置管理,可以让应用开发和部署更加高效,无论是简单的属性配置,还是复杂... 目录Spring Boot 配置文件详解一、Spring Boot 配置文件类型1.1 applicatio

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

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

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

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

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

SpringBoot项目启动报错"找不到或无法加载主类"的解决方法

《SpringBoot项目启动报错找不到或无法加载主类的解决方法》在使用IntelliJIDEA开发基于SpringBoot框架的Java程序时,可能会出现找不到或无法加载主类com.example.... 目录一、问题描述二、排查过程三、解决方案一、问题描述在使用 IntelliJ IDEA 开发基于

SpringBoot3使用Jasypt实现加密配置文件

《SpringBoot3使用Jasypt实现加密配置文件》这篇文章主要为大家详细介绍了SpringBoot3如何使用Jasypt实现加密配置文件功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编... 目录一. 使用步骤1. 添加依赖2.配置加密密码3. 加密敏感信息4. 将加密信息存储到配置文件中5