kernel源码学习:进程

linux使用上百个双向链表来存储内核的各种数据结构, 其中包括进程描述符的task_struct结构.
cpp 代码
 
  1. struct list_head {  
  2.     struct list_head *next, *prev;  
  3. };  
在list.h中,形如__xxx的函数用来操作整个链表,而不是单个链表元素.


进程从pid得到进程描述符pd要一种高效的方法, 提到高效方法,那就用哈希表啦.  内核在这儿的哈希方法为一道宏, 表中有 PIDHASH_SZ个元素(一般情况下 PIDHASH_SZ= 1024):
cpp 代码
 
  1. #define pid_hashfn(x)   ((((x) >> 8) ^ (x)) & (PIDHASH_SZ - 1))  

哈希表解决冲突的方法以是采取双向链表. 这是数据结构中一种比较常见的手段. 因为数字比较大的pid很少用到, 这样的pid一般放在链表尾端。 而数字较小的pid经常用到, 而它们如果冲突也排在链表很前的位置, 所以采取这样的数据结构速度很快.
   内核提供了hash_pid()来加入一个元素到表中, unhash_pid()从表中删除一元素. 
   这个哈希表的主要目的是通过pid找到进程描述符, find_task_by_pid ()提供了这项功能.
cpp 代码
 
  1. static inline void hash_pid(struct task_struct *p)  
  2. {  
  3.     struct task_struct **htable = &pidhash[pid_hashfn(p->pid)];  
  4.   
  5.     if((p->pidhash_next = *htable) != NULL)  
  6.         (*htable)->pidhash_pprev = &p->pidhash_next;  
  7.     *htable = p;  
  8.     p->pidhash_pprev = htable;  
  9. }  
  10.   
  11. static inline void unhash_pid(struct task_struct *p)  
  12. {  
  13.     if(p->pidhash_next)  
  14.         p->pidhash_next->pidhash_pprev = p->pidhash_pprev;  
  15.     *p->pidhash_pprev = p->pidhash_next;  
  16. }  
  17.   
  18. static inline struct task_struct *find_task_by_pid(int pid)  
  19. {  
  20.     struct task_struct *p, **htable = &pidhash[pid_hashfn(pid)];  
  21.   
  22.     for(p = *htable; p && p->pid != pid; p = p->pidhash_next)  
  23.         ;  
  24.   
  25.     return p;  
  26. }  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值