uloop_process
- uloop_process主要数据结构
- struct list_head processes
- struct uloop_process
- uloop_process_handler
- uloop_process主要函数
- uloop_handle_processes
- uloop_process原理
- uloop_process使用示例
uloop_process 是 Libubox 提供的一个进程管理工具,它并不会帮你创建进程,它主要用来等待子进程工作的结束,然后调用自定义的回调函数,所以一般需要配合
fork
一起使用。
uloop_process主要数据结构
struct list_head processes
static struct list_head processes = LIST_HEAD_INIT(processes);
processes
是一个全局链表,用于管理多个struct uloop_process
.
struct uloop_process
struct uloop_process
{
struct list_head list;
bool pending;
uloop_process_handler cb;
pid_t pid;
};
struct uloop_process
用来描述一个子进程,其中:
- list 用于将
struct uloop_process
结构加入全局processes
链表 - pending标志此进程是否需要挂起(初始化时此参数务必置0)
- cb是此进程运行结束后的回调函数
- pid是进程ID(通常=fork()的返回值)
uloop_process_handler
typedef void (*uloop_process_handler)(struct uloop_process *c, int ret);
uloop_process_handler是子进程退出后会触发调用此回调函数(不是子进程自己调用,是运行uloop_run的进程会去调用),其中子进程的一些退出状态码可以通过参数ret
获取。
如果子进程使用exit(150);
函数退出,那这里的ret
就是150.
更多有关进程状态码的信息参考这位大佬的文章:wait/waitpid 的 status 参数
uloop_process主要函数
int uloop_process_add(struct uloop_process *p);
int uloop_process_delete(struct uloop_process *p);
-
uloop_process_add()
函数是向全局链表中新增一个uloop_process
-
uloop_process_delete()
函数是从全局链表中删除指定的uloop_process
uloop_handle_processes
static void uloop_handle_processes(void)
uloop_handle_processes()
函数只会在uloop_run
中被调用,用户不会主动调用此函数,它的作用是等待uloop_process ->pid
对应的子进程运行结束,然后调用uloop_process ->cb
函数。
uloop_process原理
在uloop_init()
中会重定向SIGCHLD
信号处理函数为uloop_sigchld()
,在这个函数中会设置do_sigchld=1
。
当子进程主动退出或者被Kill后,就会向父进程发送SIGCHLD
信号
在uloop_run()
中会根据do_sigchld
决定是否要处理子进程,如果需要处理,首先会调用uloop_process_delete
从全局链表中删除子进程,然后调用struct uloop_process->cb
函数。
uloop_process使用示例
示例中,父进程fork
出一个子进程,然后让子进程执行test
程序,随后将子进程添加到uloop进程管理队列,在子进程结束后,父进程会调用uloop_process->cb
函数。
父进程源码:
#include <unistd.h>
#include <signal.h>
#include "libubus.h"
void u_process_cb(struct uloop_process *c, int ret)
{
printf("[%s]currnt pid=%d\n",__FUNCTION__,getpid());
printf("[%s]child process exec end,ret=%d\n",__FUNCTION__,ret);
}
int main(int argc, char **argv)
{
static char *arg[]={"./test",NULL};
struct uloop_process u;
memset((void *)&u,0,sizeof(u));
uloop_init();
u.cb = u_process_cb;
u.pid = fork();
if(u.pid == 0)//child process
{
execvp(arg[0],arg);
}
if(uloop_process_add(&u) == 0)
{
printf("[father] father pid=%d child pid=%d\n",getpid(),u.pid);
printf("[father] add pid(%d) to uloop_process success!\n",u.pid);
}
else
{
printf("[father] add pid(%d) to uloop_process failed!\n",u.pid);
}
uloop_run();
uloop_done();
return 0;
}
子进程源码如下:
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main(int argc, char **argv)
{
printf("[child]Enter\n");
sleep(2);
printf("[child]Exit\n");
return 0;
}
执行结果如下:
# ./uloop_process
[father] father pid=25305 child pid=25306
[father] add pid(25306) to uloop_process success!
[child]Enter # 子进程开始运行
[child]Exit # 子进程退出
[u_process_cb]currnt pid=25305 # 父进程调用cb
[u_process_cb]child process exec end,ret=0