- 论坛徽章:
- 0
|
本帖最后由 kgn28 于 2010-06-27 09:05 编辑
回复 1# retuor
关于他们的区别,ULK3是这么说的:“The main difference is that deferrable functions run in interrupt context while functions in work queues run in process context. Running in process context is the only way to execute functions that can block (for instance, functions that need to access some block of data on disk) because, as already observed in the section "Nested Execution of Exception and Interrupt Handlers" earlier in this chapter, no process switch can take place in interrupt context. ”
工作队列执行时可以发生进程切换,而中断函数执行时不能发生进程切换(因为抢占计数器非零)。
另外关于ksoftirqd线程做的事情:- from:2.6.34(lxr.linux.no)
- 695static int run_ksoftirqd(void * __bind_cpu)
- 696{
- 697 set_current_state(TASK_INTERRUPTIBLE);
- 698
- 699 while (!kthread_should_stop()) {
- 700 preempt_disable();
- 701 if (!local_softirq_pending()) {
- 702 preempt_enable_no_resched();
- 703 schedule();
- 704 preempt_disable();
- 705 }
- 706
- 707 __set_current_state(TASK_RUNNING);
- 708
- 709 while (local_softirq_pending()) {
- 710 /* Preempt disable stops cpu going offline.
- 711 If already offline, we'll be on wrong CPU:
- 712 don't process */
- 713 if (cpu_is_offline((long)__bind_cpu))
- 714 goto wait_to_die;
- 715 do_softirq();
- 716 preempt_enable_no_resched();
- 717 cond_resched();
- 718 preempt_disable();
- 719 rcu_sched_qs((long)__bind_cpu);
- 720 }
- 721 preempt_enable();
- 722 set_current_state(TASK_INTERRUPTIBLE);
- 723 }
- 724 __set_current_state(TASK_RUNNING);
- 725 return 0;
- 726
- 727wait_to_die:
- 728 preempt_enable();
- 729 /* Wait for kthread_stop */
- 730 set_current_state(TASK_INTERRUPTIBLE);
- 731 while (!kthread_should_stop()) {
- 732 schedule();
- 733 set_current_state(TASK_INTERRUPTIBLE);
- 734 }
- 735 __set_current_state(TASK_RUNNING);
- 736 return 0;
- 737}
复制代码 明显可以看到在do_softirq后,是做了reschedule(717cond_resched();)的,而ksoftirqd的优先级比较低,所以很有可能被其他进程所替代,但是在执行do_softirq的时候是处于非抢占的中断上下文(premmt_count计数器非零)的。这样设计ULK3里面关于为什么是这样 说:
The first strategy consists of ignoring new softirqs that occur while do_softirq( ) is running. In other words, the do_softirq( ) function could determine what softirqs are pending when the function is started and then execute their functions. Next, it would terminate without rechecking the pending softirqs. This solution is not good enough. Suppose that a softirq function is reactivated during the execution of do_softirq( ). In the worst case, the softirq is not executed again until the next timer interrupt, even if the machine is idle. As a result, softirq latency time is unacceptable for networking developers. The second strategy consists of continuously rechecking for pending softirqs. The do_softirq( ) function could keep checking the pending softirqs and would terminate only when none of them is pending. While this solution might satisfy networking developers, it can certainly upset normal users of the system: if a high-frequency flow of packets is received by a network card or a softirq function keeps activating itself, the do_softirq( ) function never returns, and the User Mode programs are virtually stopped.
无论是软中断还是硬中断,因为允许他们嵌套执行(nested),就不可能在执行中断函数的时候切走(switch_to)他们,因为这样就无法保证对中断的及时响应了。而软中断的设计目的应该是为了提高硬中断的吞吐率吧? |
|