【openwrt】libubox组件——uloop_process

uloop_process是Libubox中的进程管理工具,用于等待并处理子进程的结束。它包括全局链表`processes`、结构体`uloop_process`和回调函数`uloop_process_handler`。`uloop_process`结构体包含子进程信息,如回调函数和PID。`uloop_process_handler`在子进程结束时调用。主要函数有`uloop_process_add()`和`uloop_process_delete()`,分别用于添加和删除子进程。`uloop_handle_processes()`内部处理子进程结束,并调用回调函数。使用示例展示了如何创建子进程,添加到uloop管理,并在子进程结束时调用回调函数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


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
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

知否,知否

来一杯冰美式把

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值