进程睡眠原理(基于linux2.6.12.1)

进程是一个动态的实体,满足条件的情况下,他一直在执行,但是有时候,进程需要条件得不到满足的时候,他就会被挂起。但这是被动的,不是进程控制的,也就是说,进程访问一个资源的时候,如果不能被满足,进程会被系统挂起,等到条件满足的时候,系统会唤起进程。
    今天介绍的是一种进程主动睡眠的能力。即进程自己让自己挂起,等到一定时间后,被系统唤醒(时间到或者收到信号)。这个能力由sleep函数提供。

 unsigned int sleep(unsigned int seconds);

这个函数可以让进程自己挂起seconds秒。我们看看这个函数的一些说明。

On Linux, sleep() is implemented via nanosleep(2). See the
nanosleep(2) man page for a discussion of the clock used.

即sleep函数是由操作系统的nanosleep函数实现的。我们看一下核心代码。

asmlinkage long sys_nanosleep(struct timespec __user *rqtp, struct timespec __user *rmtp)
{
	struct timespec t;
	unsigned long expire;
	long ret;

	expire = timespec_to_jiffies(&t) + (t.tv_sec || t.tv_nsec);
	current->state = TASK_INTERRUPTIBLE;
	expire = schedule_timeout(expire);

}

算出超时时间,然后挂起进程(可中断挂起),然后调用schedule_timeout。

fastcall signed long __sched schedule_timeout(signed long timeout)
{
	struct timer_list timer;
	unsigned long expire;
	// 算出超时时间
	expire = timeout + jiffies;

	init_timer(&timer);
	// 超时时间
	timer.expires = expire;
	timer.data = (unsigned long) current;
	// 超时回调
	timer.function = process_timeout;
	// 添加定时器
	add_timer(&timer);
	// 进程调度
	schedule();
	// 删除定时器
	del_singleshot_timer_sync(&timer);
    // 超时或者被信号唤醒,被信号唤醒的话,可能还没有超时
	timeout = expire - jiffies;

 out:
	return timeout < 0 ? 0 : timeout;
}

接着往系统新增一个定时器,然后发送进程调度,该进程随即进入挂起状态。等到一定的时间后,进程会唤醒。另外我们注意到挂起的进程状态是TASK_INTERRUPTIBLE,即可中断的。意思是这种状态的进程可以被信号唤醒。而TASK_UNINTERRUPTIBLE是不能被信号唤醒的。
    等到超时的时候,执行process_timeout函数。

static void process_timeout(unsigned long __data)
{
	wake_up_process((task_t *)__data);
}

代码很简单,就是唤醒被挂起的进程。__data是在

timer.data = (unsigned long) current;

中设置的。这就是进程主动睡眠(sleep)的大致原理。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值