成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux內核完全剖析---頭文件

系統 Linux
程序在使用一個函數之前,應該首先聲明該函數。為了便于使用,通常的做法是把同一類函數或數據結構以及常數的聲明放在一個頭文件(header file)中。

程序在使用一個函數之前,應該首先聲明該函數。為了便于使用,通常的做法是把同一類函數或數據結構以及常數的聲明放在一個頭文件(header file)中。頭文件中也可以包括任何相關的類型定義和宏(macros)。在程序源代碼文件中則使用預處理指令“#include”來引用相關的頭文件。
程序中如下形式的一條控制行語句將會使得該行被文件filename的內容替換掉:

# include

當然,文件名filename中不能包含 > 和換行字符以及 "、'、\、或 /* 字符。編譯系統會在定義的一系列地方搜索這個文件。類似地,下面形式的控制行會讓編譯器首先在源程序所在目錄中搜索filename文件:

# include "filename"

如果沒有找到,編譯器再執行同上面一樣的搜索過程。在這種形式中,文件名filename中不能包含換行字符和 "、'、\、或 /* 字符,但允許使用 > 字符。

在一般應用程序源代碼中,頭文件與開發環境中的庫文件有著不可分割的緊密聯系,庫中的每個函數都需要在頭文件中加以聲明。應用程序開發環境中的頭文件(通常放置在系統/usr/include/目錄中)可以看作是其所提供函數庫(例如libc.a)中函數的一個組成部分,是庫函數的使用說明或接口聲明。在編譯器把源代碼程序轉換成目標模塊后,鏈接程序(linker)會把程序所有的目標模塊組合在一起,包括用到的任何庫文件中的模塊。從而構成一個可執行的程序。

對于標準C函數庫來講,其最基本的頭文件有15個。每個頭文件都表示出一類特定函數的功能說明或結構定義,例如I/O操作函數、字符處理函數等。有關標準函數庫的詳細說明及其實現可參照Plauger編著的《The Standard C Library》一書。

而對于本書所描述的內核源代碼,其中涉及的頭文件則可以看作是對內核及其函數庫所提供服務的一個概要說明,是內核及其相關程序專用的頭文件。在這些頭文件中主要描述了內核所用到的所有數據結構、初始化數據、常數和宏定義,也包括少量的程序代碼。除了幾個專用的頭文件以外(例如塊設備頭文件blk.h),Linux 0.12內核中所用到的頭文件都放在內核代碼樹的include/目錄中。因此編譯Linux 0.12內核無需使用開發環境提供的位于/usr/include/目錄下的任何頭文件。當然,tools/build.c程序除外。因為這個程序雖然被包含在內核源代碼樹中,但它只是一個用于組合創建內核映像文件的工具程序或應用程序,不會被鏈接到內核代碼中。

從0.95版開始,內核代碼樹中的頭文件需要復制到/usr/include/linux目錄下才能順利地編譯內核。即從該版內核開始頭文件已經與開發環境使用的頭文件合二為一。

14.1 include/目錄下的文件

內核所用到的頭文件都保存在include/目錄下。該目錄下的文件如表11-1所示。這里需要說明一點:為了方便使用和兼容性,Linus在編制內核程序頭文件時所使用的命名方式與標準C庫頭文件的命名方式相似,許多頭文件的名稱甚至其中的一些內容都與標準C庫的頭文件基本相同,但這些內核頭文件仍然是內核源代碼或與內核有緊密聯系的程序專用的。在一個Linux系統中,它們與標準庫的頭文件并存。通常的做法是將這些頭文件放置在標準庫頭文件目錄中的子目錄下,以讓需要用到內核數據結構或常數的程序使用。

另外,也由于版權問題,Linus試圖重新編制一些頭文件以取代具有版權限制的標準C庫的頭文件。因此這些內核源代碼中的頭文件與開發環境中的頭文件有一些重疊的地方。在Linux系統中,列表14-1中的asm/、linux/和sys/三個子目錄下的內核頭文件通常需要復制到標準C庫頭文件所在的目錄(/usr/include)中,而其他一些文件若與標準庫的頭文件沒有沖突則可以直接放到標準庫頭文件目錄下,或者改放到這里的三個子目錄中。

asm/目錄下主要用于存放與計算機體系結構密切相關的函數聲明或數據結構的頭文件。例如Intel CPU 端口IO匯編宏文件io.h、中斷描述符設置匯編宏頭文件system.h等。linux/目錄下是Linux內核程序使用的一些頭文件。其中包括調度程序使用的頭文件sched.h、內存管理頭文件mm.h和終端管理數據結構文件tty.h等。而sys/目錄下存放著幾個與內核資源相關頭文件。不過從0.98版開始,內核目錄樹下sys/目錄中的頭文件被全部移到了linux/目錄下。

Linux 0.12版內核中共有32個頭文件(*.h),其中asm/子目錄中含有4個,linux/子目錄中含有10個,sys/子目錄中含有5個。從下一節開始我們首先描述include/目錄下的13個頭文件,然后依次說明每個子目錄中的文件。說明順序按照文件名稱排序進行。

  

Linux內核完全剖析---頭文件


#p#14.2 a.out.h文件

14.2.1 功能描述

在Linux 內核中,a.out.h文件用于定義被加載的可執行文件結構。主要用于加載程序fs/exec.c中。該文件不屬于標準C庫,它是內核專用的頭文件。但由于與標準庫的頭文件名沒有沖突,因此在Linux系統中一般可以放/usr/include/目錄下,以供涉及相關內容的程序使用。該頭文件中定義了目標文件的一種a.out(Assembly out)格式。Linux 0.12系統中使用的.o文件和可執行文件就采用了這種目標文件格式。

a.out.h文件包括三個數據結構定義和一些相關的宏定義,因此文件可被相應地分成三個部分:

◆第1~108行給出并描述了目標文件執行頭結構和相關的宏定義。

◆第109~185行對符號表項結構的定義和說明。

◆第186~217行對重定位表項結構進行定義和說明。

由于該文件內容比較多,因此對其中三個數據結構以及相關宏定義的詳細說明放在程序列表后。

從0.96版內核開始,Linux系統直接采用了GNU的同名頭文件a.out.h。因此造成在Linux 0.9x下編譯的程序不能在Linux 0.1x系統上運行。下面對兩個a.out頭文件的不同之處進行分析,并說明如何讓0.9x下編譯的一些不是用動態鏈接庫的執行文件也能在0.1x下運行。

Linux 0.12使用的a.out.h文件與GNU同名文件的主要區別在于exec結構的第一個字段a_magic。GNU的該文件字段名稱是a_info,并且把該字段又分成3個子域:標志域(Flags)、機器類型域(Machine Type)和魔數域(Magic Number)。同時為機器類型域定義了相應的宏N_MACHTYPE和N_FLAGS,如圖14-1所示。

Linux內核完全剖析---頭文件

在Linux 0.9x系統中,對于采用靜態庫連接的執行文件,圖中各域注釋中括號內的值是該字段的默認值。這種二進制執行文件開始處的4個字節是:

0x0b, 0x01, 0x64, 0x00

而這里的頭文件僅定義了魔數域。因此,在Linux 0.1x系統中一個a.out格式的二進制執行文件開始的4個字節是:

0x0b, 0x01, 0x00, 0x00

可以看出,采用GNU的a.out格式的執行文件與Linux 0.1x系統上編譯出的執行文件的區別僅在機器類型域。因此我們可以把Linux 0.9x上的a.out格式執行文件的機器類型域(第3個字節)清零,讓其運行在0.1x系統中。只要被移植的執行文件所調用的系統調用都已經在0.1x系統中實現即可。在開始重新組建Linux 0.1x根文件系統中的很多命令時,作者就采用了這種方法。

在其他方面,GNU的a.out.h頭文件與這里的a.out.h沒有什么區別。

#p#14.2.2 代碼注釋(附件下載,pdf格式)程序14.pdf

14.2.3 a.out執行文件格式

Linux內核0.12版僅支持a.out(Assembly out)執行文件和目標文件的格式,雖然這種格式目前已經漸漸不用,而使用功能更為齊全的ELF(Executable and Link Format)格式,但是由于其簡單性,作為入門的學習材料比較適用。下面全面介紹一下a.out格式。

在頭文件a.out.h中聲明了三個數據結構以及一些宏。這些數據結構描述了系統上目標文件的結構。在Linux 0.12系統中,編譯產生的目標模塊文件(簡稱模塊文件)和鏈接生成的二進制可執行文件均采用a.out格式。這里統稱為目標文件。一個目標文件由7部分(7節)組成。它們依次為:

(1)執行頭部分(exec header)。該部分中含有一些參數(exec結構),內核使用這些參數把執行文件加載到內存中并執行,而鏈接程序(ld)使用這些參數將一些模塊文件組合成一個可執行文件。這是目標文件唯一必要的組成部分。

(2)代碼段部分(text segment)。含有程序執行時被加載到內存中的指令代碼和相關數據。可以以只讀形式被加載。

(3)數據段部分(data segment)。這部分含有已經初始化過的數據,總是被加載到可讀寫的內存中。

(4)代碼重定位部分(text relocations)。這部分含有供鏈接程序使用的記錄數據。在組合目標模塊文件時用于定位代碼段中的指針或地址。

(5)數據重定位部分(data relocations)。類似于代碼重定位部分的作用,但是用于數據段中指針的重定位。

(6)符號表部分(symbol table)。這部分同樣含有供鏈接程序使用的記錄數據,用于在二進制目標模塊文件之間對命名的變量和函數(符號)進行交叉引用。

(7)字符串表部分(string table)。該部分含有與符號名對應的字符串。

每個目標文件均以一個執行數據結構(exec structure)開始。該數據結構的形式如下:

Linux內核完全剖析---頭文件

各個字段的功能如下:

1)a_magic——該字段含有三個子字段,分別是標志字段、機器類型標識字段和魔數字段,參見圖11-1。不過對于Linux 0.12系統其目標文件只使用了其中的魔數子字段,并使用宏N_MAGIC()來訪問,它唯一地確定了二進制執行文件與其他加載的文件之間的區別。該子字段中必須包含以下值之一:

◆OMAGIC。表示代碼和數據段緊隨在執行頭后面并且是連續存放的。內核將代碼和數據段都加載到可讀寫內存中。編譯器編譯出的目標文件的魔數是OMAGIC(八進制0407)。

◆NMAGIC。同OMAGIC一樣,代碼和數據段緊隨在執行頭后面并且是連續存放的。然而內核將代碼加載到了只讀內存中,并把數據段加載到代碼段后下一頁可讀寫內存邊界開始。

◆ZMAGIC。內核在必要時從二進制執行文件中加載獨立的頁面。執行頭部、代碼段和數據段都被鏈接程序處理成多個頁面大小的塊。內核加載的代碼頁面是只讀的,而數據段的頁面是可寫的。鏈接生成的可執行文件的魔數即是ZMAGIC(0413,即0x10b)。

2)a_text——該字段含有代碼段的長度值,字節數。

3)a_data——該字段含有數據段的長度值,字節數。

4)a_bss——含有bss段的長度,內核用其設置在數據段后初始的break(brk)。內核在加載程序時,這段可寫內存顯現出處于數據段后面,并且初始時為全零。

5)a_syms——含有符號表部分的字節長度值。

6)a_entry——含有內核將執行文件加載到內存中以后,程序執行起始點的內存地址。

7)a_trsize——該字段含有代碼重定位表的大小,是字節數。

8)a_drsize——該字段含有數據重定位表的大小,是字節數。

#p#在a.out.h頭文件中定義了幾個宏,這些宏使用exec結構來測試一致性或者定位執行文件中各個部分(節)的位置偏移值。這些宏有:

  ◆N_BADMAG(exec)。如果a_magic字段不能被識別,則返回非零值。

  ◆N_TXTOFF(exec)。代碼段的起始位置字節偏移值。

  ◆N_DATOFF(exec)。數據段的起始位置字節偏移值。

  ◆N_DRELOFF(exec)。數據重定位表的起始位置字節偏移值。

  ◆N_TRELOFF(exec)。代碼重定位表的起始位置字節偏移值。

  ◆N_SYMOFF(exec)。符號表的起始位置字節偏移值。

  ◆N_STROFF(exec)。字符串表的起始位置字節偏移值。

重定位記錄具有標準的格式,它使用重定位信息(relocation_info)結構來描述,如下所示:

  

Linux內核完全剖析---頭文件

該結構中各字段的含義如下:

1)r_address——該字段含有需要鏈接程序處理(編輯)的指針的字節偏移值。代碼重定位的偏移值是從代碼段開始處計數的,數據重定位的偏移值是從數據段開始處計算的。鏈接程序會將已經存儲在該偏移處的值與使用重定位記錄計算出的新值相加。

2)r_symbolnum——該字段含有符號表中一個符號結構的序號值(不是字節偏移值)。鏈接程序在算出符號的絕對地址以后,就將該地址加到正在進行重定位的指針上。(如果r_extern比特位是0,那么情況就不同,見下面。)

3)r_pcrel——如果設置了該位,鏈接程序就認為正在更新一個指針,該指針使用pc相關尋址方式,是屬于機器碼指令部分。當運行程序使用這個被重定位的指針時,該指針的地址被隱式地加到該指針上。

4)r_length——該字段含有指針長度的2的次方值:0表示1字節長,1表示2字節長,2表示4字節長。

5)r_extern——如果被置位,表示該重定位需要一個外部引用;此時鏈接程序必須使用一個符號地址來更新相應指針。當該位是0時,則重定位是“局部”的。鏈接程序更新指針以反映各個段加載地址中的變化,而不是反映一個符號值的變化。在這種情況下,r_symbolnum字段的內容是一個n_type值;這類字段告訴鏈接程序被重定位的指針指向那個段。

6)r_pad——Linux系統中沒有使用的4個比特位。在寫一個目標文件時最好全置0。

符號將名稱映射為地址(或者更通俗地講是字符串映射到值)。由于鏈接程序對地址的調整,一個符號的名稱必須用來表示其地址,直到已被賦予一個絕對地址值。符號是由符號表中固定長度的記錄以及字符串表中的可變長度名稱組成。符號表是nlist結構的一個數組,如下所示:

  

Linux內核完全剖析---頭文件


其中各字段的含義為:

1)n_un.n_strx——含有本符號的名稱在字符串表中的字節偏移值。當程序使用nlist()函數訪問一個符號表時,該字段被替換為n_un.n_name字段,這是內存中字符串的指針。

2)n_type——用于鏈接程序確定如何更新符號的值。使用第146~154行開始的位屏蔽(bitmasks)碼可以將8比特寬度的n_type字段分割成三個子字段,如圖14-2所示。對于N_EXT類型位置位的符號,鏈接程序將它們看作是“外部的”符號,并且允許其他二進制目標文件對它們的引用。N_TYPE屏蔽碼用于鏈接程序感興趣的比特位:

 

◆N_UNDF。一個未定義的符號。鏈接程序必須在其他二進制目標文件中定位一個具有相同名稱的外部符號,以確定該符號的絕對數據值。特殊情況下,如果n_type字段是非零值,并且沒有二進制文件定義了這個符號,則鏈接程序在BSS段中將該符號解析為一個地址,保留長度等于n_value的字節。如果符號在多于一個二進制目標文件中都沒有定義并且這些二進制目標文件對其長度值不一致,則鏈接程序將選擇所有二進制目標文件中最大的長度。

◆N_ABS。一個絕對符號。鏈接程序不會更新一個絕對符號。

◆N_TEXT。一個代碼符號。該符號的值是代碼地址,鏈接程序在合并二進制目標文件時會更新其值。

◆N_DATA。一個數據符號。與N_TEXT類似,但是用于數據地址。對應代碼和數據符號的值不是文件的偏移值而是地址;為了找出文件的偏移,就有必要確定相關部分開始加載的地址并減去它,然后加上該部分的偏移。

◆N_BSS。一個BSS符號。與代碼或數據符號類似,但在二進制目標文件中沒有對應的偏移。

◆N_FN。一個文件名符號。在合并二進制目標文件時,鏈接程序會將該符號插入在二進制文件中的符號之前。符號的名稱就是給予鏈接程序的文件名,而其值是二進制文件中首個代碼段地址。鏈接和加載時不需要文件名符號,但對于調式程序非常有用。

◆N_STAB。屏蔽碼用于選擇符號調式程序(例如gdb)感興趣的位。其值在stab()中說明。

3)n_other——該字段按照n_type確定的段,提供有關符號重定位操作的符號獨立性信息。目前,n_other字段的最低4位含有兩個值之一:AUX_FUNC和AUX_OBJECT(有關定義參見)。AUX_FUNC將符號與可調用的函數相關,AUX_OBJECT將符號與數據相關,無論它們是位于代碼段還是數據段。該字段主要用于鏈接程序ld,用于動態可執行程序的創建。

4)n_desc——保留給調式程序使用;鏈接程序不對其進行處理。不同的調試程序將該字段用作不同的用途。

5)n_value——含有符號的值。對于代碼、數據和BSS符號,這是一個地址;對于其他符號(例如調式程序符號),值可以是任意的。

字符串表由長度為unsigned long后跟一null結尾的符號字符串組成。長度代表整個表的字節大小,所以在32位的機器上其最小值(即第1個字符串的偏移)總是4。

14.3 const.h文件

14.3.1 功能描述

該文件定義了i節點中文件屬性和類型i_mode字段所用到的一些標志位常量符號。

14.3.2 代碼注釋

Linux內核完全剖析---頭文件




【編輯推薦】

  1. Linux下查看用戶登陸后的操作記錄
  2. 專家指導:Linux操作系統密碼恢復方法
  3. Linux系統下手動分析病毒樣本技巧
責任編輯:趙寧寧 來源: IT專家網
相關推薦

2009-06-17 13:03:42

Linux內核

2009-06-17 11:58:19

Linux

2010-01-07 13:44:54

Linux內核代碼

2020-11-20 07:55:55

Linux內核映射

2011-06-15 11:03:54

Linux內核Xen

2025-04-01 02:00:22

2025-04-02 00:33:00

2025-04-18 04:05:00

2011-09-09 10:31:40

Xen虛擬化linux內核

2022-08-03 11:00:20

Linux內核

2020-04-01 10:28:12

Apache HBas數據結構算法

2010-01-07 10:22:49

Linux內核

2009-10-22 16:31:48

2009-10-28 13:44:40

linux庫文件路徑

2010-03-05 15:02:09

Linux文件編碼

2010-03-05 16:54:47

2021-02-20 06:08:07

LinuxWindows內核

2009-12-31 14:34:02

ISDN終端

2017-08-01 17:34:47

Linux內核驅動文件讀寫

2017-03-30 10:13:11

Linux內核文件系統
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久精品在线 | 国产精品视频观看 | 久久骚| 成人免费观看男女羞羞视频 | 欧美精品综合在线 | 一级高清 | 亚洲电影一区二区三区 | 午夜私人影院在线观看 | 欧美黑人一级爽快片淫片高清 | 欧美一区二区三区在线观看 | 亚洲人成人一区二区在线观看 | 日韩av最新网址 | 欧美精品一区三区 | 国产免费一级一级 | 国产精品一区久久久久 | 亚洲精品一区中文字幕 | 国产高清在线 | 国产精品美女久久久久久免费 | av网站免费看 | 自拍中文字幕 | 亚洲国产精品成人久久久 | 99久热在线精品视频观看 | 日韩成人在线网站 | .国产精品成人自产拍在线观看6 | 中文字幕亚洲视频 | 欧洲精品一区 | 欧美激情欧美激情在线五月 | 一区二区三区小视频 | 欧美寡妇偷汉性猛交 | 色综合一区二区 | 伊人久久免费 | 国产精品自产拍在线观看蜜 | 精品久久香蕉国产线看观看亚洲 | 日本欧美在线观看视频 | 久久久久国产一区二区三区 | 99这里只有精品视频 | 日韩精品在线观看一区二区三区 | 亚州精品天堂中文字幕 | 亚洲精品九九 | 手机看片1 | 狠狠躁天天躁夜夜躁婷婷老牛影视 |