对当前进程记账

http://book.51cto.com/art/200810/93784.htm

 Linux 2.6内核标准教程》第6章时间度量,这一章围绕墙上时间xtims和相对时间jiffies两个主体进行阐述,从硬件支持到软件架构;从时间度量模块的初始化到如何使用时间度量的工作机制;从软件定时器的使用到软件定时器的工作原理。本节讲述的是如何对当前被中断的进程(用户进程或内核线程)进行记帐,以及记帐时完成了那些工作。

AD:


6.4.3  对当前进程记账

下面分析时钟中断处理过程如何对当前被中断的进程(用户进程或内核线程)进行记帐,以及记帐时完成了那些工作。下面是对负责进程记帐的函数update_process_times()的分析。

代码清单6.15--函数update_process_times

功能简介:该函数负责对当前进程进行记账,减小进程剩余可用的时间片。然后激活其他内核模块的处理函数,如同步机制RCU的处理函数、内核定时器的处理函数。

文件:src/kernel/timer.c

784  void update_process_times(int user_tick)
785  {
786       struct task_struct *p = current;
787       int cpu = smp_processor_id();
788  
789       /* Note: this timer irq context must be accounted for as well. */
790       if (user_tick)
791            account_user_time(p, jiffies_to_cputime(1));
792       else
793            account_system_time(p, HARDIRQ_OFFSET, jiffies_to_cputime(1));
794       run_local_timers();
795       if (rcu_pending(cpu))
796            rcu_check_callbacks(cpu, user_tick);
797       scheduler_tick();
798       run_posix_cpu_timers(p);
799  } 
第784行,我们首先分析一下do_timer_interrupt_hook()调用该函数时传递进来的实际参数user_mode(regs),该参数使用函数user_mode()来判断被中断时系统处于用户态还是内核态。该函数根据被打断的上下文所使用的指令段选择子寄存器cs的CPL的字段是否为3,如果为3则返回1,表示是用户态;否则返回0,表示为内核态。该函数在文件src/include/asm- i386/ptrace.h中的第71行开始定义,代码如下:
static inline int user_mode(struct pt_regs *regs) 
{
return (regs->xcs & 3) != 0;
}

第786~787行代码通过宏定义current、smp_processor_id()获得当前进程描述符指针和当前处理器编号,并将其分别保存到指针变量p和整型cpu中。

第790~793行代码根据被中断进程运行于用户态还是系统态分别调用account_ user_time()、account_system_time()对被中断进程进行记账。它们分别将进程描述符中用户态时间字段utime、内核态时间字段stime的值加上jiffies_to_cputime(1),表示进程在用户态或者内核又运行了一个系统时钟滴答。然后更新处理器历史统计信息。其中jiffies_to_cputime()是一个宏定义,该宏定义用于将一个时钟中断转换为处理器时间。

第794行调用函数run_local_timers()设置时钟中断处理的下半部处理标记、激活时钟中断处理的下半部。该下半部负责维护、更新内核定时器链表,对于超时的内核定时器执行相应的超时处理函数,并将超时的定时器移出内核定时器链表。详情请参见6.5节。其中函数run_local_timers()在文件src/kernel/timer.c中的第867行定义如下。关于函数raise_softirq的功能请参见5.5.1小节。

void run_local_timers(void)
{
raise_softirq(TIMER_SOFTIRQ); 
}

第795~796行代码用于处理内核同步机制RCU所使用的数据,将老版本的数据删除。详情请参见8.4.2小节。

第797行调用函数scheduler_tick()判断被中断进程的时间片是否用完。如果用完设置该进程描述中的调度标记,在中断处理完毕、中断返回时系统会检查该标记;如果设置了该标记,系统会重新调度选择合适的进程运行。

第798行调用函数run_posix_cpu_timers()处理向用户进程提供的POSIX标准的时钟,这里不作更深入的介绍。

你可能感兴趣的:(kernel与Android)