binder 添加多线程支持

binder进程间通讯 C程序代码示例添加多线程支持 下载链接 binder_c_thread

如果多个client同时向server发起请求,就有可能导致binder驱动,告知server需要创建线程。该逻辑在binder驱动中的binder_thread_read函数中。binder驱动源码路径:kernel\drivers\android\binder.c

done:

	*consumed = ptr - buffer;
	binder_inner_proc_lock(proc);
	if (proc->requested_threads == 0 && 
	    list_empty(&thread->proc->waiting_threads) &&
	    proc->requested_threads_started < proc->max_threads &&
	    (thread->looper & (BINDER_LOOPER_STATE_REGISTERED |
	     BINDER_LOOPER_STATE_ENTERED)) /* the user-space code fails to */
	     /*spawn a new thread if we leave this out */) {
		proc->requested_threads++;//1
		binder_inner_proc_unlock(proc);
		binder_debug(BINDER_DEBUG_THREADS,
			     "%d:%d BR_SPAWN_LOOPER\n",
			     proc->pid, thread->pid);
		if (put_user(BR_SPAWN_LOOPER, (uint32_t __user *)buffer))
			return -EFAULT;
		binder_stat_br(proc, thread, BR_SPAWN_LOOPER);
	} else
		binder_inner_proc_unlock(proc);

可以看出需要同时满足以下条件,才会告知server进程创建线程

  1. server进程中的 requested_threads 为0。这个代表创建线程的请求,在注释1处会自加,当binder驱动接收到BC_REGISTER_LOOPER命令时,会自减。所以server进程创建好线程后,需要发出BC_REGISTER_LOOPER,否则requested_threads 不可能为0
  2. server进程等待的线程为空
  3. 请求创建的线程数要小于max_threads最大线程数。max_threads默认为0,binder驱动通过接收到BINDER_SET_MAX_THREADS命令设置

满足条件后,就会向server进程发出BR_SPAWN_LOOPER,让server进程去创建线程

所以,如果server需要添加多线程支持,就要做以下工作:

1,设置最大线程数

void set_max_thread(struct binder_state *bs,int value){
	ioctl(bs->fd, BINDER_SET_MAX_THREADS, &value);
}

2,在处理函数中处理BR_SPAWN_LOOPER命令,创建线程

void *my_binder_thread_func(struct my_thread_data *data){
   /*进入循环等待数据*/
	my_binder_loop(data->my_bs, data->func);
   	return NULL;
}

case BR_SPAWN_LOOPER:{
			/*创建新线程*/
			pthread_t my_pthread_t;
			struct my_thread_data my_data ;
			my_data.my_bs=bs;
			my_data.func=func;
			pthread_create(&my_pthread_t,NULL,my_binder_thread_func,&my_data);
			break;
}

注意在 my_binder_loop 中需要发出BC_REGISTER_LOOPER

void my_binder_loop(struct binder_state *bs, binder_handler func)
{
   int res;
   struct binder_write_read bwr;
   uint32_t readbuf[32];

   bwr.write_size = 0;
   bwr.write_consumed = 0;
   bwr.write_buffer = 0;

   readbuf[0] = BC_REGISTER_LOOPER;
   binder_write(bs, readbuf, sizeof(uint32_t));

   for (;;) {
	   bwr.read_size = sizeof(readbuf);
	   bwr.read_consumed = 0;
	   bwr.read_buffer = (uintptr_t) readbuf;

	   res = ioctl(bs->fd, BINDER_WRITE_READ, &bwr);

	   if (res < 0) {
		   ALOGE("binder_loop: ioctl failed (%s)\n", strerror(errno));
		   break;
	   }

	   res = binder_parse(bs, 0, (uintptr_t) readbuf, bwr.read_consumed, func);
	   if (res == 0) {
		   ALOGE("binder_loop: unexpected reply?!\n");
		   break;
	   }
	   if (res < 0) {
		   ALOGE("binder_loop: io error %d %s\n", res, strerror(errno));
		   break;
	   }
   }
}

添加以上内容后,经测试,server进程是可以支持多线程的

# ps -t | grep test_server
root      24257 17186 4720   1588  binder_thr b3ddf8e8 S test_server
root      24329 24257 4720   1588  binder_thr b3ddf8e8 S test_server

可以看出已经多了一个线程。如果不加上面的内容,就只有一个主线程。

总结
binder多线程由binder驱动发起,server进程处理。处理之后需要发送BC_REGISTER_LOOPER。

具体的代码实现可参考上面的下载链接中的文件。

  • 6
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
binder_open是Binder机制中的一个函数,用于打开Binder驱动并返回一个Binder状态结构体。在该函数中,会调用copy_from_user函数从用户空间获取一个struct binder_write_read结构体,并将其作为参数传递给binder_thread_write和binder_thread_read函数。\[1\] 在Binder机制的Native层实现中,binder_open函数被用于开启Binder,并将自身注册为Binder进程的上下文,然后通过调用binder_loop函数来启动Binder循环。\[2\] 在binder_loop函数中,如果收到了Binder的读写消息信息,会调用binder_parse函数进行处理。binder_parse函数会将读取到的消息进行解析,并调用相应的处理函数进行处理。\[3\] #### 引用[.reference_title] - *1* [05.Binder系统:第6课第2节_Binder系统_驱动情景分析_打印数据交互过](https://blog.csdn.net/weixin_43013761/article/details/88171380)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [【Binder 机制】Native 层 Binder 机制分析 ( binder_loop | svcmgr_handler | binder.c | binder_parse )](https://blog.csdn.net/han1202012/article/details/120345228)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值