Android BlueDroid分析: OSI中的List与alloctor的实现分析

2024-03-04 12:18

本文主要是介绍Android BlueDroid分析: OSI中的List与alloctor的实现分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

说明

在bluedroid中有许多数据是使用List来保存的, 而不同的数据的size各不相同,因此bluedroid中的osi封装了一个list.
不同的OS的malloc与free各不相同,因此为了屏蔽底层细节bluedroid对malloc与free进行了一个封装,成为alloctor.

alloctor

定义

alloctor是一对函数函数指针的结构体,因此先要定义来个函数Pointer 类型。

typedef void *(*alloc_fn)(size_t size);
typedef void (*free_fn)(void *ptr);

可以看到,这个和libc中的malloc与free arguments与return type都是一样的,因此可以认为是OSI为了抽象分隔不同的“OS Interface”(Memory中的Stack 管理)。因此如果我们需要将BlueDroid移植到其他OS的话,需要注意适配。


alloctor_t结构体包含两个函数Pointer,一个是分配的fn,一个是释放的fn,函数Pointer类型定义如下:

typedef struct {alloc_fn alloc;free_fn  free;
} allocator_t;

在Android中,使用的libc是bonic,和glibc一样,都实现C POSIX Library(参考Wikipedia),所以函数都是名称和传入参数都一样,也是malloc与free,因此:

void *osi_malloc(size_t size) {size_t real_size = allocation_tracker_resize_for_canary(size);return allocation_tracker_notify_alloc(alloc_allocator_id,malloc(real_size),size);
}

calloc

calloc比malloc多了一个初始化功能.

void *osi_calloc(size_t size) {size_t real_size = allocation_tracker_resize_for_canary(size);return allocation_tracker_notify_alloc(alloc_allocator_id,calloc(1, real_size),size);
}void osi_free(void *ptr) {free(allocation_tracker_notify_free(alloc_allocator_id, ptr));
}const allocator_t allocator_malloc = {osi_malloc,osi_free
};const allocator_t allocator_calloc = {osi_calloc,osi_free
};

其中allocation_tracker_resize_for_canary函数的实现与hashmap相关,这个在以后的文章中再说明.


List链表

定义与结构

struct list_node_t;
typedef struct list_node_t list_node_t;struct list_t;
typedef struct list_t list_t;typedef void (*list_free_cb)(void *data);

下面的data为void *,这样可以用来存储任何type的指针,一般都是指向数据的指针.

struct list_node_t {struct list_node_t *next;void *data;
};typedef struct list_t {list_node_t *head;list_node_t *tail;size_t length;list_free_cb free_cb;const allocator_t *allocator;
} list_t;

free的call_back函数, alloctor分配器,均为函数指针。

list包含一个头尾节点head与tail, 表示有多少个node的长度length, 用来free list node中的data指针指向数据的callback函数指针, 还有一个用来如何malloc与free list node(不是data的free)的alloctor.


List的其他utils函数

主要是:增删查改.包括empty判断,next, create,begin,end.需要说明的是创建与clear.


List的创建与使用

内核创建函数:

list_new_internal

BlueDroid中的osi/src中的各类封装都有一个特点,就是有一个XXX_new同时还有一个XXX_new_internal, internal代表者这个接口(函数)为osi下的source code相互调用,一般是对XXX_new的封装, 而其他地方如果想new的话,一般使用XXX_new来调用.

例如下面:

list_t *list_new(list_free_cbcallback){returnlist_new_internal(callback,&allocator_calloc);
}

list_new_internal的parameter是callback, 其类型为list_free_cb, 用于list的free. 对于传入的第二个参数:

allocator_calloc

就是前面alloctor中的封装系统POSIX的malloc/free对.



List的Clear/Delete

List本身就是malloc出来的, 然后其结构中包含了一串的node,这些node是使用list_t中的alloctor中的malloc创建,所以要销毁一个list,想要销毁其里面的node,因此直接从head到tail,一个个销毁节点,然后给对应指针与length赋值即可:

void list_clear(list_t*list){  
assert(list!= NULL);  for (list_node_t *node =list->head;node; )  
node = list_free_node_(list,node);  list->head= NULL;  
list->tail= NULL; 
list->length= 0;
}

而node的free就是调用alloctor中的free,即OSI封装的free,一般在linux中就是free:

static list_node_t *list_free_node_(list_t*list,list_node_t *node)
{ assert(list!= NULL);assert(node!= NULL); list_node_t *next =node->next; 
if (list->free_cb)  
list->free_cb(node->data);  
list->allocator->free(node);  
--list->length; return next;}

值得注意的是,这里面调用了free_cb, 即节点callback函数被调用了, 可以设想一下,如果这个callback fn为空,那么意味着其节点node中的data(void *data)数据就是一个数据, 如果不为空的话, 那么说明void *data指向的很可能是一片数据, 而这个数据直接调用free并传入data这个指针无法回收. 例如:

btif/src/btif_sock_sco.c
92:  sco_sockets = list_new((list_free_cb)sco_socket_free_locked);

其中其回收函数中逐层对sco_socket进行free, 最后才是对sco_socket指针的回收:

static void sco_socket_free_locked(sco_socket_t *sco_socket) {if (!sco_socket)return;if (sco_socket->sco_handle != BTM_INVALID_SCO_INDEX)BTM_RemoveSco(sco_socket->sco_handle);socket_free(sco_socket->socket);osi_free(sco_socket);
}

GKI_getbuf 与 GKI_freebuf

但是在有一些地方使用的是GKI相关的free与alloc,这个是malloc下的二次封装, 为stack启动时候分配了一个pool,然后从这个pool中使用GKI_getbuf与GKI_freebuf来使用. 这个时候就不需要传入free callback fn了, 例如:

    BT_HDR *fragment = GKI_getbuf(client->remote_mtu + L2CAP_MIN_OFFSET);fragment->offset = L2CAP_MIN_OFFSET;fragment->len = client->remote_mtu;memcpy(fragment->data + fragment->offset, bt_packet->data + bt_packet->offset, client->remote_mtu);list_append(client->outbound_fragments, fragment);

而其free的则通过获取节点数据(即void *data指针)来完成

GKI_freebuf(list_node(node));
void *list_node(constlist_node_t *node){ assert(node!= NULL);return node->data;}

List中新添加Node:list_append

在system/bt中有下面这些地方用到了:

$ ag list_append
stack/l2cap/l2c_link.c
1122:        list_append(p_lcb->link_xmit_data_q, p_buf);
stack/l2cap/l2cap_client.c
116:  list_append(l2cap_clients, ret);
383:        list_append(client->outbound_fragments, bt_packet);
394:    list_append(client->outbound_fragments, fragment);
stack/l2cap/l2c_main.c
166:                list_append(l2cb.rcv_pending_q, p_msg);
bta/av/bta_av_main.c
1231:                    list_append(p_scbi->a2d_list, p_new);
bta/av/bta_av_aact.c
2287:                list_append(p_scb->a2d_list, p_buf);
append为将数据data放入一个新创建的node中.



这篇关于Android BlueDroid分析: OSI中的List与alloctor的实现分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

[职场] 公务员的利弊分析 #知识分享#经验分享#其他

公务员的利弊分析     公务员作为一种稳定的职业选择,一直备受人们的关注。然而,就像任何其他职业一样,公务员职位也有其利与弊。本文将对公务员的利弊进行分析,帮助读者更好地了解这一职业的特点。 利: 1. 稳定的职业:公务员职位通常具有较高的稳定性,一旦进入公务员队伍,往往可以享受到稳定的工作环境和薪资待遇。这对于那些追求稳定的人来说,是一个很大的优势。 2. 薪资福利优厚:公务员的薪资和

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

android 免费短信验证功能

没有太复杂的使用的话,功能实现比较简单粗暴。 在www.mob.com网站中可以申请使用免费短信验证功能。 步骤: 1.注册登录。 2.选择“短信验证码SDK” 3.下载对应的sdk包,我这是选studio的。 4.从头像那进入后台并创建短信验证应用,获取到key跟secret 5.根据技术文档操作(initSDK方法写在setContentView上面) 6.关键:在有用到的Mo

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

Android我的二维码扫描功能发展史(完整)

最近在研究下二维码扫描功能,跟据从网上查阅的资料到自己勉强已实现扫描功能来一一介绍我的二维码扫描功能实现的发展历程: 首页通过网络搜索发现做android二维码扫描功能看去都是基于google的ZXing项目开发。 2、搜索怎么使用ZXing实现自己的二维码扫描:从网上下载ZXing-2.2.zip以及core-2.2-source.jar文件,分别解压两个文件。然后把.jar解压出来的整个c

android 带与不带logo的二维码生成

该代码基于ZXing项目,这个网上能下载得到。 定义的控件以及属性: public static final int SCAN_CODE = 1;private ImageView iv;private EditText et;private Button qr_btn,add_logo;private Bitmap logo,bitmap,bmp; //logo图标private st