家電維修班,手機維修班,電腦維修班,電工班,焊工班,液晶電視維修班,電動工具維修班、電動車摩托車維修班、網絡營銷培訓、網站設計培訓、淘寶培訓---全國招生 家電維修班,手機維修班,電腦維修班,電工班,焊工班,液晶電視維修班,電動工具維修班、電動車摩托車維修班、網絡營銷培訓、網站設計培訓、淘寶培訓---全國招生

中日欧洲精品视频在线-中日韩在线观看-中日韩一区二区三区-中日韩精品视频在线观看-狂野欧美老黑粗又硬-狂野猛交xxxx吃奶

您的位置:網站首頁 > 電器維修資料網 > 正文 >

linux內核中一些常用的數據結構和操作

★★★★★【文章導讀】:linux內核中一些常用的數據結構和操作具體內容是:};鏈表頭的初始化,注意,結構中的指針為NULL并不是初始化,而是指向自身才是初始化,如果只是按普通情況下的置為NULL,而不是指向自身,系統會崩潰,這是一個容易犯的錯誤:#defineLIST_HEAD_INIT(name){&(nAME…

來源: 日期:2013-12-11 14:27:12 人氣:標簽:

linux內核中一些常用的數據結構和操作

    };鏈表頭的初始化,注意,結構中的指針為NULL并不是初始化,而是指向自身才是初始化,如果只是按普通情況下的置為NULL,而不是指向自身,系統會崩潰,這是一個容易犯的錯誤:#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);刪除鏈表節點:

    void list_del(struct list_head *entry);將節點移動到另一鏈表:

    void list_move(struct list_head *list, struct list_head *head);將節點移動到鏈表尾:

    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);取得節點指針:

    #define list_entry(ptr, type, member)

    ((type *)((char *)(ptr)-(unsigned lONg)(&((type *)0)->member)))遍歷鏈表中每個節點:

    #define list_for_each(pos, head)

    for (pos = (head)->next, prefetch(pos->next); pos != (head);

    pos = pos->next, prefetch(pos->next))逆向循環鏈表中每個節點:

    #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);

    }

    }

    在內存中形成如下結構的一個雙向鏈表:+---------------------------------------------------------------+

    | |

    | mylist 99 98 0 |

    | +----+ +---------+ +---------+ +---------+ |

    +->|next|--->|list.next|--->|list.next|--->...--->|list.next|---+

    |----| |---------| |---------| |---------|

    +--|prev|<---|list.prev|<---|list.prev|<---...<---|list.prev|<--+

    | +----+ |---------| |---------| |---------| |

    | | data | | data | | data | |

    | +---------+ +---------+ +---------+ |

    | |

    +---------------------------------------------------------------+知道了鏈表頭就能遍歷整個鏈表,如果是用list_add()插入新節點的話,從鏈表頭的next方向看是一個堆棧型。從鏈表中刪除節點很容易:staTIc void del_item(struct my_list *p)

    {

    list_del(&p->list, &mylist);

    kfree(p);

    } 重要的宏是list_entry,這個宏的思路是根據鏈表元素結構中鏈表頭結構list_head的地址推算出鏈表元素結構的實際地址:#define list_entry(ptr, type, member)

    ((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))ptr是鏈表元素結構(如struct my_list)中鏈表頭結構list_head的地址

    member是鏈表元素結構(如struct my_list)中鏈表頭結構list_head參數的名稱

    type是鏈表元素結構類型(如struct my_list)計算原理是根據鏈表頭結構list_head的地址減去其在鏈表元素結構中的偏移位置而得到鏈表元素結構的地址。例如: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=%dn", p->data);

    }

    }優點:這樣就可以用相同的數據處理方式來描述所有雙向鏈表,不用再單獨為各個鏈表編寫各種編輯函數。缺點:

    1) 鏈表頭中元素置為NULL不是初始化,與普通習慣不同;

    2) 仍然需要單獨編寫各自的刪除整個鏈表的函數,不能統一處理,因為不能保證所有鏈表元素結構中鏈表頭結構list_head的偏移地址都是相同的,當然如果把鏈表頭結構list_head都作為鏈表元素結構的第一個參數,就可以用統一的刪除整個鏈表的函數。

    3. HASH表HASH表適用于不需要對整個空間元素進行排序,而是只需要能快速找到某個元素的場合,是一種以空間換時間的方法,本質也是線性表,但由一個大 的線性表拆分為了多個小線性表,由于只需要查找小表,因此搜索速度就會線性查整個大表提高很多,理想情況下,有多少個小線性表,搜索速度就提高了多少倍, 通常把小線性表的表頭綜合為一個數組,大小就是HASH表的數量。HASH表速度的關鍵是HASH函數的設計,HASH函數根據每個元素中固定的參數進行計算,算出一個不大于HASH表數量的索引值,表示該元 素需要放在該索引號對應的那個表中,對于固定的參數,計算結果始終是固定的,但對于不同的參數值,希望計算出來的結果能盡可能地平均到每個索引值, HASH函數計算得越平均,表示每個小表中元素的數量都會差不多,這樣搜索性能將越好。HASH函數也要盡可能的簡單,以減少計算時間,常用的算法是將參 數累加求模,在include/linux/jhash.h中已經定義了一些HASH計算函數,可直接使用。HASH表在路由cache表,狀態連接表等處用得很多。舉例,連接跟蹤中根據tuple值計算HASH:// net/ipv4/netfiLTEr/ip_conntrack_core.cu_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內核定時器由以下結構描述:/* 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:到期函數,時間到期時調用的函數

    data:傳給到期函數的數據,實際應用中通常是一個指針轉化而來,該指針指向一個結構

    timer的操作:增加timer,將timer掛接到系統的timer鏈表:

    extern void add_timer(struct timer_list * timer);刪除timer,將timer從系統timer鏈表中拆除:

    extern int del_timer(struct timer_list * timer);

    (del_timer()函數可能會失敗,這是因為該timer本來已經不在系統timer鏈表中了,也就是已經刪除過了)對于SMP系統,刪除timer 好使用下面的函數來防止沖突:

    extern int del_timer_sync(struct timer_list * timer);修改timer,修改timer的到期時間:

    int mod_timer(struct timer_list *timer, unsigned long expires);通常用法:

    struct timer_list通常作為數據結構中的一個參數,在初始化結構的時候初始化timer,表示到期時要進行的操作,實現定時動作,通常更多的是作為超時 處理的,timer函數作為超時時的資源釋放函數。注意:如果超時了運行超時函數,此時系統是處在時鐘中斷的bottom half里的,不能進行很復雜的操作,如果要完成一些復雜操作,如到期后的數據發送,不能直接在到期函數中處理,而是應該在到期函數中發個信號給特定內核 線程轉到top half進行處理。為判斷時間的先后,內核中定義了以下宏來判斷:#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中的時間是無符號數,這里先將其轉換為有符號數后再判斷,就能解決時間回繞問題,當然只是一次回繞,回繞兩次當然是判斷不出來的,具體可自己實驗體會。5. 內核線程(kernel_thread)內核中新線程的建立可以用kernel_thread函數實現,該函數在kernel/fork.c中定義:long kernel_thread(int (*fn)(void *), void * arg, unsigned long flags)fn:內核線程主函數;

    arg:線程主函數的參數;

    flags:建立線程的標志;內核線程函數通常都調用daemonize()進行后臺化作為一個獨立的線程運行,然后設置線程的一些參數,如名稱,信號處理等,這也不是必須 的,然后就進入一個死循環,這是線程的主體部分,這個循環不能一直在運行,否則系統就死在這了,或者是某種事件驅動的,在事件到來前是睡眠的,事件到來后 喚醒進行操作,操作完后繼續睡眠;或者是定時睡眠,醒后操作完再睡眠;或者加入等待隊列通過schedule()調度獲得執行時間。總之是不能一直占著 CPU。以下是內核線程的一個實例,取自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. 結構地址在C中,結構地址和結構中第一個元素的地址是相同的,因此在linux內核中經常出現使用結構第一個元素的地址來表示結構地址的情況,在讀代碼時要注意這一點,這和list_entry宏的意思一樣。如:

    struct my_struct{

    int a;

    int b;

    }c;if(&c == &c.a){ // always true

    ...

    }


【看看這篇文章在百度的收錄情況】

相關文章

聯系方式

  • 0731-85579057 , 0731-85569651
  • 點擊這里給我發消息點擊這里給我發消息點擊這里給我發消息
網站欄目導航: 培訓課程 手機硬件 手機軟件 綜合維修 學校資訊 考證指南 就業導航 招生指南 教學管理 入學須知 學校圖片 教學大綱 師資力量 學生感言 學校概況 教學實景 手機維修培訓資訊 電腦維修培訓 維修間故事 手機維修培訓 液晶電視維修培訓 家電維修資料網 電器維修資料網 招生地區 刷機教程 家電維修 手機技巧 老版網站 招生平臺網絡工程
友情鏈接: 監控安裝培訓 電動工具維修 家電維修學校 電工培訓學校 液晶電視維修 焊工培訓學校 電工焊工學校 電腦維修學校 家電維修培訓 電腦維修培訓 家裝電工培訓網絡安裝維護 主板維修 液晶顯示器 筆記本電腦維修 電腦組裝維護 電腦硬件維修 電腦維修 電工考證 電工證 裝修電工 水電工 維修電工 電工 焊接技術 電焊工 焊工 電動設備維修 電動工具維修 制冷維修 空調維修 冰箱維修  更多>>
陽光-手機維修教育品牌學校
點擊這里給我發消息 點擊這里給我發消息 點擊這里給我發消息
電工培訓學校 電動車維修學校 摩托車維修學校 摩托車維修培訓 手機維修培訓 家電維修培訓 電腦維修培訓 電動工具維修培訓 液晶電視維修培訓 安防監控培訓 空調維修培訓 網絡營銷培訓 網站設計培訓 淘寶網店培訓 電器維修培訓 家電維修學校 電工培訓 焊工培訓 電工學校 電工培訓學校 電動車維修學校 摩托車維修學校 摩托車維修培訓 手機維修培訓 家電維修培訓 電腦維修培訓 電動工具維修培訓 液晶電視維修培訓 安防監控培訓 空調維修培訓 網絡營銷培訓 網站設計培訓 淘寶網店培訓 電器維修培訓 家電維修學校 電工培訓 焊工培訓 電工學校 電工培訓學校 電動車維修學校 摩托車維修學校 摩托車維修培訓 手機維修培訓 家電維修培訓 電腦維修培訓 電動工具維修培訓 液晶電視維修培訓 安防監控培訓 空調維修培訓 網絡營銷培訓 網站設計培訓 淘寶網店培訓 電器維修培訓 家電維修學校 電工培訓 焊工培訓 電工學校 電工培訓學校 電動車維修學校 摩托車維修學校 摩托車維修培訓 手機維修培訓 家電維修培訓 電腦維修培訓 電動工具維修培訓 液晶電視維修培訓 安防監控培訓 空調維修培訓 網絡營銷培訓 網站設計培訓 淘寶網店培訓 電器維修培訓 家電維修學校 電工培訓 焊工培訓 電工學校
中山市,固原市,銀川市,玉樹,海東,隴南市,酒泉市,張掖市,天水市,金昌市,蘭州市,榆林市,延安市,渭南市,銅川市,阿里,山南,拉薩市,怒江,文山州,楚雄州,普洱市,昭通市,玉溪市,昆明市,畢節,銅仁,遵義市,貴陽市,甘孜州,資陽市,達州市,宜賓市,南充市,遂寧市,綿陽市,瀘州市,自貢市,三亞市,崇左市,河池市,玉林市,欽州市,梧州市,柳州市,梅州市,肇慶市,湛江市,佛山市,珠海市,韶關市,湘西州,懷化市,郴州市,張家界市,邵陽市,株洲市,仙桃市,隨州市,荊州市,荊門市,襄樊市,黃石市,駐馬店市,信陽市,南陽市,漯河市,中衛市,石嘴山市,海西,海南藏州,黃南州,海北,甘南,慶陽市,平涼市,武威市,白銀市,嘉峪關市,安康市,漢中市,咸陽市,寶雞市,林芝,日喀則,昌都,迪慶,德宏,大理,西雙版納,紅河州,臨滄市,麗江市,保山市,曲靖市,黔東州,黔西州,安順市,六盤水市,涼山州,阿壩州,雅安市,廣安市,眉山市,內江市,廣元市,德陽市,攀枝花市,成都市,海口市,來賓市,百色市,貴港市,北海市,桂林市,南寧市,云浮市,揭陽市,潮州市,清遠市,陽江市,汕尾市,惠州市,茂名市,江門市,汕頭市,深圳市,廣州市,婁底市,永州市,益陽市,岳陽市,湘潭市,長沙市,恩施州,黃岡市,孝感市,鄂州市,十堰市,武漢市,周口市,商丘市,三門峽市,許昌市,焦作市,安陽市,鶴壁市,平頂山市,開封市,鄭州市,聊城市,濱州市,德州市,萊蕪市,日照市,泰安市,煙臺市,濰坊市,東營市,淄博市,上饒市,濟南市,撫州市,宜春市,贛州市,新余市,九江市,景德鎮市,寧德市,南平市,泉州市,莆田市,廈門市,宣城市,亳州市,六安市,宿州市,黃山市,滁州市,安慶市,淮北市,馬鞍山市,蚌埠市,蕪湖市,合肥市,麗水市,舟山市,衢州市,金華市,湖州市,嘉興市,寧波市,宿遷市,鎮江市,鹽城市,連云港市,蘇州市,徐州市,南京市,綏化市,牡丹江市,佳木斯市,大慶市,鶴崗市,哈爾濱市,白城市,白山市,遼源市,吉林市,葫蘆島市,鐵嶺市,盤錦市,阜新市,錦州市,本溪市,鞍山市,沈陽市,錫林郭勒盟,通遼市,烏海市,呂梁市,忻州市,晉中市,晉城市,陽泉市,太原市,廊坊市,承德市,保定市,邯鄲市,唐山市,寧夏,甘肅省,西藏,貴州省,重慶市,廣西,湖南省,河南省,江西省,安徽省,江蘇省,黑龍江省,遼寧省,山西省,天津市,四平市,內蒙古,吳忠市,果洛,西寧市,定西市,商洛市,西安市,那曲,黔南州,巴中市,樂山市,賀州市,防城港市,東莞市,河源市,常德市,衡陽市,咸寧市,宜昌市,濮陽市,新鄉市,洛陽市,菏澤市,臨沂市,威海市,濟寧市,棗莊市,青島市,吉安市,鷹潭市,萍鄉市,南昌市,龍巖市,漳州市,三明市,福州市,池州市,巢湖市,阜陽市,銅陵市,淮南市,臺州市,紹興市,溫州市,杭州市,泰州市,揚州市,淮安市,南通市,常州市,無錫市,大興安嶺,黑河市,七臺河市,伊春市,雙鴨山市,雞西市,齊齊哈爾市,延邊,松原市,通化市,長春市,朝陽市,遼陽市,營口市,丹東市,撫順市,大連市,阿拉善盟,興安盟,烏蘭察布市,巴彥淖爾市,呼倫貝爾市,鄂爾多斯市,赤峰市,包頭市,呼和浩特市,臨汾市,運城市,朔州市,長治市,大同市,衡水市,滄州市,張家口市,邢臺市,秦皇島市,石家莊市,青海省,陜西省,云南省,四川省,海南省,廣東省,湖北省,山東省,福建省,浙江省,上海市,吉林省,河北省,北京市 主站蜘蛛池模板: 浴室里强摁做开腿呻吟的漫画男男| 国产高清超清在线播放| 亚洲免费网站在线观看| 亚洲爆乳少妇精品无码专区| 亚洲精品无码不卡| 在线免费观看日本| 亚洲婷婷天堂综合国产剧情| 天天干夜夜曰| 亚洲免费在线视频| 37pao成人国产永久免费视频| 夜色伊甸园| 88福利视频| 4455永久在线毛片观看| AV久久久囯产果冻传媒| 在线观看国产精美视频| 99久久精品国产一区二区三区| 在线视频一区二区三区在线播放| 97视频视频人人碰视频| 丰满的美女射精动态图| 国偷自产视频一区二区久| 美女张开腿露尿口给男人亲| 精品爽爽久久久久久蜜臀| 免费在线a| 欧美白人极品性喷潮| 色姊姊真舒服| 手机看片国产免费| 无人区日本电影在线观看| 亚洲中文字幕欧美自拍一区| bbw美女与zooxx| 国产手机在线视频| 果冻传媒在线观看视频| 麻豆官网入口| 欧美末成年videos在线| 香蕉59tv视频| 亚洲色图在线观看视频| 99久久伊人一区二区yy5099 | 被cao的奶水直喷高H| 被室友C哭调教双性| 果冻传媒我的女老板 | 寂寞骚妇女被后入式抽插| 女人张腿让男人桶免费|