青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 297,  comments - 15,  trackbacks - 0
本文檔的Copyleft歸yfydz所有,使用GPL發(fā)布,可以自由拷貝,轉(zhuǎn)載,轉(zhuǎn)載時請保持文檔的完整性,嚴(yán)禁用于任何商業(yè)用途。
msn: yfydz_no1@hotmail.com
來源:http://yfydz.cublog.cn

1. 前言
 
本文介紹linux內(nèi)核中一些常用的數(shù)據(jù)結(jié)構(gòu)和操作。
 
2. 雙向鏈表(list)
 
linux內(nèi)核中的雙向鏈表通過結(jié)構(gòu) struct list_head來將各個節(jié)點連接起來,此結(jié)構(gòu)會作為鏈表元素結(jié)構(gòu)中的一個參數(shù):
struct list_head {
 struct list_head *next, *prev;
};
 
鏈表頭的初始化,注意,結(jié)構(gòu)中的指針為NULL并不是初始化,而是指向自身才是初始化,如果只是按普通情況下的置為NULL,而不是指向自身,系統(tǒng)會崩潰,這是一個容易犯的錯誤:
 
#define LIST_HEAD_INIT(name) { &(name), &(name) }
#define LIST_HEAD(name) \
 struct list_head name = LIST_HEAD_INIT(name)
#define INIT_LIST_HEAD(ptr) do { \
 (ptr)->next = (ptr); (ptr)->prev = (ptr); \
} while (0)
 
最常用的鏈表操作:
插入到鏈表頭:
void list_add(struct list_head *new, struct list_head *head);
 
插入到鏈表尾:
void list_add_tail(struct list_head *new, struct list_head *head);
 
刪除鏈表節(jié)點:
void list_del(struct list_head *entry);
 
將節(jié)點移動到另一鏈表:
void list_move(struct list_head *list, struct list_head *head);
 
將節(jié)點移動到鏈表尾:
void list_move_tail(struct list_head *list,struct list_head *head);
 
判斷鏈表是否為空,返回1為空,0非空
int list_empty(struct list_head *head);
 
把兩個鏈表拼接起來:
void list_splice(struct list_head *list, struct list_head *head);
 
取得節(jié)點指針:
#define list_entry(ptr, type, member) \
 ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))
 
遍歷鏈表中每個節(jié)點:
#define list_for_each(pos, head) \
 for (pos = (head)->next, prefetch(pos->next); pos != (head); \
         pos = pos->next, prefetch(pos->next))
 
逆向循環(huán)鏈表中每個節(jié)點:
#define list_for_each_prev(pos, head) \
 for (pos = (head)->prev, prefetch(pos->prev); pos != (head); \
         pos = pos->prev, prefetch(pos->prev))
 
舉例:
 
LISH_HEAD(mylist);
 
struct my_list{
 struct list_head list;
 int data;
};
 
static int ini_list(void)
{
 struct my_list *p;
 int i;
 for(i=0; i<100; i++){
  p=kmalloc(sizeof(struct my_list), GFP_KERNEL);
  list_add(&p->list, &mylist);
 }
}

在內(nèi)存中形成如下結(jié)構(gòu)的一個雙向鏈表:
 
  +---------------------------------------------------------------+
  |                                                               |
  |  mylist         99            98                     0        |
  |  +----+    +---------+    +---------+           +---------+   |
  +->|next|--->|list.next|--->|list.next|--->...--->|list.next|---+
     |----|    |---------|    |---------|           |---------|
  +--|prev|<---|list.prev|<---|list.prev|<---...<---|list.prev|<--+
  |  +----+    |---------|    |---------|           |---------|   |
  |            |  data   |    |  data   |           |  data   |   |
  |            +---------+    +---------+           +---------+   |
  |                                                               |
  +---------------------------------------------------------------+
 
知道了鏈表頭就能遍歷整個鏈表,如果是用list_add()插入新節(jié)點的話,從鏈表頭的next方向看是一個堆棧型。
 
從鏈表中刪除節(jié)點很容易:
static void del_item(struct my_list *p)
{
 list_del(&p->list, &mylist);
 kfree(p);
}
 
最重要的宏是list_entry,這個宏的思路是根據(jù)鏈表元素結(jié)構(gòu)中鏈表頭結(jié)構(gòu)list_head的地址推算出鏈表元素結(jié)構(gòu)的實際地址:
 
#define list_entry(ptr, type, member) \
 ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))
 
ptr是鏈表元素結(jié)構(gòu)(如struct my_list)中鏈表頭結(jié)構(gòu)list_head的地址
member是鏈表元素結(jié)構(gòu)(如struct my_list)中鏈表頭結(jié)構(gòu)list_head參數(shù)的名稱
type是鏈表元素結(jié)構(gòu)類型(如struct my_list)
計算原理是根據(jù)鏈表頭結(jié)構(gòu)list_head的地址減去其在鏈表元素結(jié)構(gòu)中的偏移位置而得到鏈表元素結(jié)構(gòu)的地址。
 
例如:
static void print_list(void)
{
 struct list_head *cur;
 struct my_list *p;
 list_for_each(cur, &mylist){
  p=list_entry(cur, struct my_list, list);
  printk("data=%d\n", p->data);
 }
}
 
優(yōu)點:
這樣就可以用相同的數(shù)據(jù)處理方式來描述所有雙向鏈表,不用再單獨為各個鏈表編寫各種編輯函數(shù)。
 
缺點:
1) 鏈表頭中元素置為NULL不是初始化,與普通習(xí)慣不同;
2) 仍然需要單獨編寫各自的刪除整個鏈表的函數(shù),不能統(tǒng)一處理,因為不能保證所有鏈表元素結(jié)構(gòu)中鏈表頭結(jié)構(gòu)list_head的偏移地址都是相同的,當(dāng)然如果把鏈表頭結(jié)構(gòu)list_head都作為鏈表元素結(jié)構(gòu)的第一個參數(shù),就可以用統(tǒng)一的刪除整個鏈表的函數(shù)。

3. HASH表
 
HASH表適用于不需要對整個空間元素進(jìn)行排序,而是只需要能快速找到某個元素的場合,是一種以空間換時間的方法,本質(zhì)也是線性表,但由一個大的線性表拆分為了多個小線性表,由于只需要查找小表,因此搜索速度就會線性查整個大表提高很多,理想情況下,有多少個小線性表,搜索速度就提高了多少倍,通常把小線性表的表頭綜合為一個數(shù)組,大小就是HASH表的數(shù)量。
 
HASH表速度的關(guān)鍵是HASH函數(shù)的設(shè)計,HASH函數(shù)根據(jù)每個元素中固定的參數(shù)進(jìn)行計算,算出一個不大于HASH表數(shù)量的索引值,表示該元素需要放在該索引號對應(yīng)的那個表中,對于固定的參數(shù),計算結(jié)果始終是固定的,但對于不同的參數(shù)值,希望計算出來的結(jié)果能盡可能地平均到每個索引值,HASH函數(shù)計算得越平均,表示每個小表中元素的數(shù)量都會差不多,這樣搜索性能將越好。HASH函數(shù)也要盡可能的簡單,以減少計算時間,常用的算法是將參數(shù)累加求模,在include/linux/jhash.h中已經(jīng)定義了一些HASH計算函數(shù),可直接使用。
 
HASH表在路由cache表,狀態(tài)連接表等處用得很多。
 
舉例,連接跟蹤中根據(jù)tuple值計算HASH:
// net/ipv4/netfilter/ip_conntrack_core.c
u_int32_t
hash_conntrack(const struct ip_conntrack_tuple *tuple)
{
#if 0
 dump_tuple(tuple);
#endif
 return (jhash_3words(tuple->src.ip,
                      (tuple->dst.ip ^ tuple->dst.protonum),
                      (tuple->src.u.all | (tuple->dst.u.all << 16)),
                      ip_conntrack_hash_rnd) % ip_conntrack_htable_size);
}
 
// include/linux/jhash.h
static inline u32 jhash_3words(u32 a, u32 b, u32 c, u32 initval)
{
 a += JHASH_GOLDEN_RATIO;
 b += JHASH_GOLDEN_RATIO;
 c += initval;
 __jhash_mix(a, b, c);
 return c;
}
 
4. 定時器(timer)
 
linux內(nèi)核定時器由以下結(jié)構(gòu)描述:
 
/* include/linux/timer.h */
struct timer_list {
 struct list_head list;
 unsigned long expires;
 unsigned long data;
 void (*function)(unsigned long);
};
list:timer鏈表
expires:到期時間
function:到期函數(shù),時間到期時調(diào)用的函數(shù)
data:傳給到期函數(shù)的數(shù)據(jù),實際應(yīng)用中通常是一個指針轉(zhuǎn)化而來,該指針指向一個結(jié)構(gòu)

timer的操作:
 
增加timer,將timer掛接到系統(tǒng)的timer鏈表:
extern void add_timer(struct timer_list * timer);
 
刪除timer,將timer從系統(tǒng)timer鏈表中拆除:
extern int del_timer(struct timer_list * timer);
(del_timer()函數(shù)可能會失敗,這是因為該timer本來已經(jīng)不在系統(tǒng)timer鏈表中了,也就是已經(jīng)刪除過了)
 
對于SMP系統(tǒng),刪除timer最好使用下面的函數(shù)來防止沖突:
extern int del_timer_sync(struct timer_list * timer);
 
修改timer,修改timer的到期時間:
int mod_timer(struct timer_list *timer, unsigned long expires);
 
通常用法:
    struct timer_list通常作為數(shù)據(jù)結(jié)構(gòu)中的一個參數(shù),在初始化結(jié)構(gòu)的時候初始化timer,表示到期時要進(jìn)行的操作,實現(xiàn)定時動作,通常更多的是作為超時處理的,timer函數(shù)作為超時時的資源釋放函數(shù)。注意:如果超時了運行超時函數(shù),此時系統(tǒng)是處在時鐘中斷的bottom half里的,不能進(jìn)行很復(fù)雜的操作,如果要完成一些復(fù)雜操作,如到期后的數(shù)據(jù)發(fā)送,不能直接在到期函數(shù)中處理,而是應(yīng)該在到期函數(shù)中發(fā)個信號給特定內(nèi)核線程轉(zhuǎn)到top half進(jìn)行處理。
 
為判斷時間的先后,內(nèi)核中定義了以下宏來判斷:
#define time_after(a,b)  ((long)(b) - (long)(a) < 0)
#define time_before(a,b) time_after(b,a)
#define time_after_eq(a,b) ((long)(a) - (long)(b) >= 0)
#define time_before_eq(a,b) time_after_eq(b,a)
這里用到了一個技巧,由于linux中的時間是無符號數(shù),這里先將其轉(zhuǎn)換為有符號數(shù)后再判斷,就能解決時間回繞問題,當(dāng)然只是一次回繞,回繞兩次當(dāng)然是判斷不出來的,具體可自己實驗體會。
 
5. 內(nèi)核線程(kernel_thread)
 
內(nèi)核中新線程的建立可以用kernel_thread函數(shù)實現(xiàn),該函數(shù)在kernel/fork.c中定義:
long kernel_thread(int (*fn)(void *), void * arg, unsigned long flags)
fn:內(nèi)核線程主函數(shù);
arg:線程主函數(shù)的參數(shù);
flags:建立線程的標(biāo)志;
 
內(nèi)核線程函數(shù)通常都調(diào)用daemonize()進(jìn)行后臺化作為一個獨立的線程運行,然后設(shè)置線程的一些參數(shù),如名稱,信號處理等,這也不是必須的,然后就進(jìn)入一個死循環(huán),這是線程的主體部分,這個循環(huán)不能一直在運行,否則系統(tǒng)就死在這了,或者是某種事件驅(qū)動的,在事件到來前是睡眠的,事件到來后喚醒進(jìn)行操作,操作完后繼續(xù)睡眠;或者是定時睡眠,醒后操作完再睡眠;或者加入等待隊列通過schedule()調(diào)度獲得執(zhí)行時間。總之是不能一直占著 CPU。
 
以下是內(nèi)核線程的一個實例,取自kernel/context.c:
 
int start_context_thread(void)
{
 static struct completion startup __initdata = COMPLETION_INITIALIZER(startup);
 kernel_thread(context_thread, &startup, CLONE_FS | CLONE_FILES);
 wait_for_completion(&startup);
 return 0;
}
static int context_thread(void *startup)
{
 struct task_struct *curtask = current;
 DECLARE_WAITQUEUE(wait, curtask);
 struct k_sigaction sa;
 daemonize();
 strcpy(curtask->comm, "keventd");
 keventd_running = 1;
 keventd_task = curtask;
 spin_lock_irq(&curtask->sigmask_lock);
 siginitsetinv(&curtask->blocked, sigmask(SIGCHLD));
 recalc_sigpending(curtask);
 spin_unlock_irq(&curtask->sigmask_lock);
 complete((struct completion *)startup);
 /* Install a handler so SIGCLD is delivered */
 sa.sa.sa_handler = SIG_IGN;
 sa.sa.sa_flags = 0;
 siginitset(&sa.sa.sa_mask, sigmask(SIGCHLD));
 do_sigaction(SIGCHLD, &sa, (struct k_sigaction *)0);
 /*
  * If one of the functions on a task queue re-adds itself
  * to the task queue we call schedule() in state TASK_RUNNING
  */
 for (;;) {
  set_task_state(curtask, TASK_INTERRUPTIBLE);
  add_wait_queue(&context_task_wq, &wait);
  if (TQ_ACTIVE(tq_context))
   set_task_state(curtask, TASK_RUNNING);
  schedule();
  remove_wait_queue(&context_task_wq, &wait);
  run_task_queue(&tq_context);
  wake_up(&context_task_done);
  if (signal_pending(curtask)) {
   while (waitpid(-1, (unsigned int *)0, __WALL|WNOHANG) > 0)
    ;
   spin_lock_irq(&curtask->sigmask_lock);
   flush_signals(curtask);
   recalc_sigpending(curtask);
   spin_unlock_irq(&curtask->sigmask_lock);
  }
 }
}
 
6. 結(jié)構(gòu)地址
 
在C中,結(jié)構(gòu)地址和結(jié)構(gòu)中第一個元素的地址是相同的,因此在linux內(nèi)核中經(jīng)常出現(xiàn)使用結(jié)構(gòu)第一個元素的地址來表示結(jié)構(gòu)地址的情況,在讀代碼時要注意這一點,這和list_entry宏的意思一樣。
 
如:
struct my_struct{
 int a;
 int b;
}c;
if(&c == &c.a){  // always true
...
}



from:

http://blog.chinaunix.net/u/12313/showart_109612.html
posted on 2010-04-01 19:59 chatler 閱讀(357) 評論(0)  編輯 收藏 引用 所屬分類: linux kernel
<2010年4月>
28293031123
45678910
11121314151617
18192021222324
2526272829301
2345678

常用鏈接

留言簿(10)

隨筆分類(307)

隨筆檔案(297)

algorithm

Books_Free_Online

C++

database

Linux

Linux shell

linux socket

misce

  • cloudward
  • 感覺這個博客還是不錯,雖然做的東西和我不大相關(guān),覺得看看還是有好處的

network

OSS

  • Google Android
  • Android is a software stack for mobile devices that includes an operating system, middleware and key applications. This early look at the Android SDK provides the tools and APIs necessary to begin developing applications on the Android platform using the Java programming language.
  • os161 file list

overall

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲午夜av在线| 欧美黑人在线观看| 亚洲日韩欧美一区二区在线| 亚洲一区二区三区四区五区午夜 | 欧美精品在线一区二区| 欧美一区二区三区啪啪| 亚洲美女黄网| 欧美国产免费| 久久资源在线| 欧美中文字幕| 亚洲综合精品自拍| 99精品热视频| 亚洲区第一页| 亚洲电影自拍| 精久久久久久久久久久| 亚欧成人精品| 亚洲一区精品视频| 亚洲视频视频在线| 亚洲最新在线视频| 亚洲精品久久久久中文字幕欢迎你 | 欧美特黄一级| 欧美精品在线免费观看| 欧美高清不卡| 欧美不卡在线视频| 久久综合五月| 老司机成人在线视频| 久久国产精品久久久| 亚洲综合社区| 亚洲在线成人| 亚洲图片欧美日产| 中文日韩在线视频| 一区二区动漫| 中文日韩电影网站| 亚洲一区999| 一二三区精品福利视频| 9色porny自拍视频一区二区| 日韩午夜黄色| 中文高清一区| 亚洲午夜91| 午夜精品婷婷| 久久精品最新地址| 久久午夜电影网| 亚洲国产精品一区| 亚洲日本无吗高清不卡| 日韩视频在线播放| 制服丝袜激情欧洲亚洲| 亚洲一二三四区| 欧美一区网站| 六月丁香综合| 亚洲国产成人在线播放| 日韩午夜在线播放| 亚洲自拍都市欧美小说| 欧美一区二区三区四区在线观看地址| 午夜国产精品影院在线观看 | 一本色道久久综合狠狠躁篇的优点| 亚洲日韩第九十九页| 亚洲欧美日韩精品久久亚洲区 | 欧美激情综合色| 亚洲精品美女在线观看| 亚洲一二三区在线| 久久成人国产精品| 欧美大片在线影院| 国产精品www色诱视频| 国产午夜精品久久久| 亚洲第一狼人社区| 亚洲性视频网站| 久久精品国产免费观看| 欧美激情片在线观看| 在线视频你懂得一区| 久久精品视频免费观看| 欧美日韩国产a| 久久久精品999| 欧美区国产区| 国产一区999| 日韩午夜激情| 久久精品首页| 亚洲精品人人| 久久av一区二区| 欧美日韩亚洲国产精品| 好吊视频一区二区三区四区| 日韩视频精品| 久久亚洲春色中文字幕久久久| 亚洲国产日韩欧美一区二区三区| 亚洲一区二区三区国产| 免费观看日韩| 国产婷婷一区二区| 一本到12不卡视频在线dvd| 久久精品中文| 夜夜夜久久久| 欧美成人有码| 午夜精品久久久久久久99热浪潮 | 欧美黄色日本| 午夜亚洲精品| 欧美日韩视频专区在线播放| 国产主播一区| 午夜日韩电影| 亚洲精品一区二区网址| 久久男人资源视频| 国产欧美日韩在线播放| 一本久久综合| 女女同性女同一区二区三区91| 亚洲性感激情| 欧美日韩精品免费观看视频| 黄色成人在线| 久久精品中文字幕免费mv| 一区二区三区国产在线| 欧美激情在线| 亚洲欧洲视频| 欧美成人免费网| 久久久精品一品道一区| 国产午夜精品久久久| 午夜在线播放视频欧美| 99v久久综合狠狠综合久久| 欧美高清在线精品一区| 在线国产亚洲欧美| 久久综合成人精品亚洲另类欧美| 亚洲男女自偷自拍| 国产伦精品一区二区| 亚洲欧美日本在线| 一区二区三区欧美成人| 欧美日韩成人激情| 一区二区日韩免费看| 亚洲日本va午夜在线影院| 欧美α欧美αv大片| 亚洲国产精品第一区二区三区| 久久夜色精品一区| 久久福利资源站| 激情文学一区| 欧美 日韩 国产在线 | 亚洲一区二区三区精品动漫| 亚洲国产一区二区在线| 欧美成人免费在线视频| 亚洲欧洲日本专区| 亚洲国产日韩综合一区| 欧美激情一区二区三级高清视频| 亚洲久久一区| 日韩视频免费观看高清在线视频| 亚洲作爱视频| 国产精品一区一区| 久久国内精品视频| 欧美在线视频一区二区| 伊人婷婷久久| 亚洲电影免费观看高清完整版在线观看 | 免费在线看一区| 亚洲美洲欧洲综合国产一区| 亚洲国产99| 一级日韩一区在线观看| 国产精品国产a级| 久久国产一区二区| 久久综合电影| 日韩一级精品| 亚洲欧美韩国| 黑人操亚洲美女惩罚| 欧美国产日韩一区二区| 欧美精品久久久久久久| 午夜精品久久久99热福利| 欧美一二三区在线观看| 悠悠资源网久久精品| 最新中文字幕亚洲| 国产精品久久婷婷六月丁香| 久久精品成人一区二区三区蜜臀| 久久久噜噜噜久久| 日韩一区二区精品| 午夜精品婷婷| 亚洲精品美女免费| 亚洲欧美久久久| 欧美成人国产va精品日本一级| 一区二区三区蜜桃网| 香蕉成人啪国产精品视频综合网| 国内成人自拍视频| 亚洲精品欧美专区| 国产亚洲精久久久久久| 亚洲高清一二三区| 国产精品亚洲激情 | 午夜亚洲性色福利视频| 1204国产成人精品视频| 一本一道久久综合狠狠老精东影业| 国产亚洲欧美中文| 亚洲三级影片| 激情欧美一区二区三区在线观看| 日韩一级黄色av| 在线免费不卡视频| 亚洲自拍电影| av不卡在线看| 久久五月婷婷丁香社区| 亚洲自拍偷拍麻豆| 免费人成网站在线观看欧美高清| 午夜精品福利一区二区蜜股av| 美腿丝袜亚洲色图| 久久高清一区| 欧美视频精品一区| 亚洲高清视频在线观看| 国产综合激情| 亚洲午夜精品福利| av成人黄色| 你懂的一区二区| 久久综合伊人77777麻豆| 国产精品欧美久久| 亚洲理伦电影| 亚洲精品久久久久久一区二区|