• 【openwrt】libubox组件——uloop_process



    uloop_process 是 Libubox 提供的一个进程管理工具,它并不会帮你创建进程,它主要用来等待子进程工作的结束,然后调用自定义的回调函数,所以一般需要配合 fork一起使用。

    uloop_process主要数据结构

    struct list_head processes

    static struct list_head processes = LIST_HEAD_INIT(processes);
    
    • 1

    processes是一个全局链表,用于管理多个struct uloop_process.

    struct uloop_process

    struct uloop_process
    {
    	struct list_head list;
    	bool pending;
    
    	uloop_process_handler cb;
    	pid_t pid;
    };
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    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);
    
    • 1

    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);
    
    • 1
    • 2
    • uloop_process_add()函数是向全局链表中新增一个uloop_process
    • uloop_process_delete()函数是从全局链表中删除指定的uloop_process

    uloop_handle_processes

    static void uloop_handle_processes(void)
    
    • 1

    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 
    #include 
    #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;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41

    子进程源码如下:

    #include 
    #include 
    #include 
    
    int main(int argc, char **argv)
    {
    	printf("[child]Enter\n");
    	sleep(2);
    	printf("[child]Exit\n");
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    执行结果如下:

    # ./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
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
  • 相关阅读:
    详解字符串比较函数:strcmp函数及其模拟实现
    2023年汉字小达人市级比赛在线模拟题来了,四种练习助力好成绩
    ch12、字符串(string)与字符编码
    养殖自动化管理系统:开启智慧养殖新篇章
    Base64编码知识详解
    某60区块链安全之整数溢出漏洞实战学习记录
    Jupyter Notebook:使用Jupyter NoteBook的时候遇到的一些坑。
    (1-线性回归问题)线性回归(Linear regression)Lasso回归和Ridge回归的区别
    6条优势,anzo capital昂首资本相信MT5替代MT4的原因
    Learn Prompt-ChatGPT 精选案例:写作&博客
  • 原文地址:https://blog.csdn.net/qq_24835087/article/details/127854652