1、有多个client发送请求时server会忙不过来,导致创建多个线程。谁来判断忙不过来呢?
client请求时会将数据放至todo链表,并且会唤醒等待wait队列的线程,如果有线程在wait队列中等待表示server忙得过来,如果没有表示忙不过来。此时驱动会向应用程序反馈,你应该多创建一些线程来处理。
驱动向APP发出“创建新线程请求”的条件
proc->requested_threads=0, 未处理的新线程请求。
proc->ready_threads为0,空闲的线程数
proc->requested_threads_started < proc->max_threads。 已启动的线程数<max_threads
*consumed = ptr - buffer;
if (proc->requested_threads + proc->ready_threads == 0 &&
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++;
binder_debug(BINDER_DEBUG_THREADS,
"binder: %d:%d BR_SPAWN_LOOPER\n",
proc->pid, thread->pid);
SAMPLE_INFO("%s (%d, %d) [%s]", proc->tsk->comm, proc->pid, thread->pid, binder_cmd_name(BR_SPAWN_LOOPER));
if (put_user(BR_SPAWN_LOOPER, (uint32_t __user *)buffer))
return -EFAULT;
}
return 0;
当创建好新线程后,新线程会执行ioctl表明已经进入了loop
binder_thread_write
case BC_REGISTER_LOOPER:
proc->requested_threads_started++;在此设置了thread_started增加线程的数量
如何创建新线程
1、设置max_threads
proc->max_threads = 0。
binder_ioctl
case BINDER_SET_MAX_THREADS
copy_from_user(&proc->max_threads, ubuf, sizeof(proc->max_threads)
2、收到BR_SPAWN_LOOPER。
创建线程时为什么使用pthread_create()而不使用fork。因为fork会复制mmap的地址,mmap地址设置为不可复制
3、再ioctl BC_REGISTER_LOOPER表示已经正常运行了。
在驱动中会更新如下两个变量。
switch BC_REGISTER_LOOPER
proc->requested_threads--;
proc->requested_threads_started++;
4、向主线一进进入循环体,读驱动、处理。