exit()函数_complete函数

c74dc8d61a76f3b550d064e4428f16f1.png

上次讲解了一下add_wait_queue和add_wait_queue_exclusive两个函数,今天来讲另一个函数complete函数。

文件包含:

#include<linux/completion.h>

这个函数也是与内核进程调度相关的一个函数,因此它也位于kernel/sched/目录下,具体的实现在completion.c中,我们可以看看它的具体实现,这样对于我们更好的理解它的工作原理会很有帮助。

void complete(struct completion *x)
{
	unsigned long flags;

	spin_lock_irqsave(&x->wait.lock, flags);
	x->done++;
	__wake_up_locked(&x->wait, TASK_NORMAL, 1);
	spin_unlock_irqrestore(&x->wait.lock, flags);
}

我们可以看到,事实上complete函数调用了__wake_up_locked函数,只是在调用时采用了自旋锁并且关闭了中断。

函数功能:

complete函数用于唤醒等待队列中的睡眠进程,并记录等待队列中的唤醒次数,保存在字段done中。函数的参数是completion的一个结构体。

struct completion {
	unsigned int done;
	wait_queue_head_t wait;
};

completion结构体除了done字段之外,就是一个wait_queue_head_t的等待队列头。需要注意的是能唤醒的进程状态只能是TASK_INTERRUPTIBLE或者TASK_UNINTERRUPTIBLE,并且唤醒进程不是同步的,就是说只能唤醒一个然后再去唤醒另一个。

实例解析:

#include<linux/init.h>
#include<linux/module.h>
#include<linux/sched.h>
#include<linux/wait.h>
#include<linux/completion.h>
#include<linux/kthread.h>

MODULE_LICENSE("GPL");
MODULE_AUTHOR("viz.xu, xujiweigo@163.com");

static struct completion comple;
static struct task_struct* old_task;

int my_test_function(void* args)
{
	printk("enter kernel function!n");
	printk("the current pid is %dn",current->pid);
	printk("the value of done of the comple:%dn",comple.done);
	printk("the state of the init thread is %ldn",old_task->state);
	
	complete(&comple);

	printk("the value of done of the comple:%dn",comple.done);
	printk("the state of the init thread is %ldn",old_task->state);

return 0;
}

int __init complete_init(void)
{
	printk("complete init! current->pid = %dn",current->pid);

	struct task_struct* new_task = NULL;

	old_task = current;
	
	new_task = kthread_create_on_node(my_test_function,NULL,-1,"comple");

	init_completion(&comple);

	wait_queue_entry_t data;
	init_waitqueue_entry(&data,current);	

	wake_up_process(new_task);

	add_wait_queue(&(comple.wait),&data);
	
	long left_time = schedule_timeout_uninterruptible(1000);

	printk("the pid of new_task is %dn",new_task->pid);

	printk("the return result of the schedule_timeout_uninterruptible is %dn",left_time);

return 0;
}

void __exit complete_exit(void)
{
	printk("complete exit!n");
return;
}

module_init(complete_init);
module_exit(complete_exit);

结果分析:

[ 2814.212263] complete init! current->pid = 40020
[ 2814.212397] enter kernel function!
[ 2814.212399] the current pid is 40021
[ 2814.212400] the value of done of the comple:0
[ 2814.212401] the state of the init thread is 2
[ 2814.212403] the value of done of the comple:1
[ 2814.212404] the state of the init thread is 0
[ 2814.212406] the pid of new_task is 40021
[ 2814.212407] the return result of the schedule_timeout_uninterruptible is 1000

从结果可以看出,子进程和父进程都执行了,并且子进程在父进程之前执行完毕。在my_test_ function中调用complete函数,父进程的状态发生了改变2->0,这是因为我们在init函数中调用了schedule_timeout_uninterruptible,使得父进程处于TASK_UNINTERRUPTIBLE状态。同是,调用complete函数之后,comple的done字段变为1,说明参数在等待队列被唤醒了一次。

### Python 中 `Traceback` 错误的原因及解决方案 当遇到 `Traceback` 报告中的错误时,通常意味着程序执行过程中遇到了异常情况。对于提到的具体案例: #### 关于 `SystemExit` 在给定的例子中,调用了 `sys.exit()` 函数来终止程序运行[^1]。此函数会引发一个名为 `SystemExit` 的特殊异常,该异常可以被正常捕获并处理。 如果希望避免因 `sys.exit()` 导致的立即退出行为,则应考虑其他方式结束应用程序逻辑流程或者确保当前上下文中允许安全地触发进程关闭动作。 #### 处理未定义符号问题 另一个提及到的是关于 `_io.so` 文件存在未定义符号的问题[^2]。这通常是由于库版本不匹配或编译环境配置不当引起的。建议验证所使用的 Python 解释器及其扩展模块是否来自同一发行版,并保持一致性和兼容性;必要时重新安装官方发布的二进制包或将源码按照说明文档正确构建。 #### 使用 `-m` 参数启动脚本 通过命令行参数 `-m` 来指定要作为主模块加载的文件路径是一种常见做法,在这里指定了 `test.__main__` 作为入口点[^3]。需要注意的是,这种方式下实际被执行的是位于相应包内的 `__main__.py` 文件而非直接传入的名字字符串本身。因此应当确认目标位置确实存在这样一个初始化文件并且其内部实现了预期的功能接口。 #### 捕捉特定类型的异常 针对除零操作产生的 `ZeroDivisionError` 可以利用 try-except 结构来进行优雅降级处理[^4]: ```python try: result = numerator / denominator except ZeroDivisionError as e: print(f"Caught an error: {e}") # Handle the exceptional case here... ``` 以上措施有助于提高代码健壮性的同时也便于调试期间快速定位潜在缺陷所在之处。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值