<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > 如何在Linux系統(tǒng)中添加新的系統(tǒng)調用

          如何在Linux系統(tǒng)中添加新的系統(tǒng)調用

          作者: 時間:2016-09-12 來源:網絡 收藏

          是應用程序和操作系統(tǒng)內核之間的功能接口。其主要目的是使得用戶可以使用操作系統(tǒng)提供的有關設備管理、輸入/輸入系統(tǒng)、文件系統(tǒng)和進程控制、通信以及存儲管理等方面的功能,而不必了解系統(tǒng)程序的內部結構和有關硬件細節(jié),從而起到減輕用戶負擔和保護系統(tǒng)以及提高資源利用率的作用。

          本文引用地址:http://www.ex-cimer.com/article/201609/303761.htm

          Linux操作系統(tǒng)作為自由軟件的代表,它優(yōu)良的性能使得它的應用日益廣泛,不僅得到專業(yè)人士的肯定,而且商業(yè)化的應用也是如火如荼。在Linux中,大部分的包含在Linux的libc庫中,通過標準的C函數(shù)調用方法可以調用這些。那么,對Linux的發(fā)燒友來說,如何在Linux中增加新的系統(tǒng)調用呢?

          1 調用機制

          中,系統(tǒng)調用是作為一種異常類型實現(xiàn)的。它將執(zhí)行相應的機器代碼指令來產生異常信號。產生中斷或異常的重要效果是系統(tǒng)自動將用戶態(tài)切換為核心態(tài)來對它進行處理。這就是說,執(zhí)行系統(tǒng)調用異常指令時,自動地將系統(tǒng)切換為核心態(tài),并安排異常處理程序的執(zhí)行。Linux用來實現(xiàn)系統(tǒng)調用異常的實際指令是:

          QUOTE:

          Int $0x80

          這一指令使用中斷/異常向量號128(即16進制的80)將控制權轉移給內核。為達到在使用系統(tǒng)調用時不必用機器指令編程,在標準的C語言庫中為每一系統(tǒng)調用提供了一段短的子程序,完成機器代碼的編程工作。事實上,機器代碼段非常簡短。它所要做的工作只是將送給系統(tǒng)調用的參數(shù)加載到CPU寄存器中,接著執(zhí)行int $0x80指令。然后運行系統(tǒng)調用,系統(tǒng)調用的返回值將送入CPU的一個寄存器中,標準的庫子程序取得這一返回值,并將它送回用戶程序。

          為使系統(tǒng)調用的執(zhí)行成為一項簡單的任務,Linux提供了一組預處理宏指令。它們可以用在程序中。這些宏指令取一定的參數(shù),然后擴展為調用指定的系統(tǒng)調用的函數(shù)。

          這些宏指令具有類似下面的名稱格式:

          QUOTE:

          _syscallN(parameters)

          其中N是系統(tǒng)調用所需的參數(shù)數(shù)目,而parameters則用一組參數(shù)代替。這些參數(shù)使宏指令完成適合于特定的系統(tǒng)調用的擴展。例如,為了建立調用setuid()系統(tǒng)調用的函數(shù),應該使用:

          QUOTE:

          _syscall1( int, setuid, uid_t, uid )

          syscallN( )宏指令的第1個參數(shù)int說明產生的函數(shù)的返回值的類型是整型,第2個參數(shù)setuid說明產生的函數(shù)的名稱。后面是系統(tǒng)調用所需要的每個參數(shù)。這一宏指令后面還有兩個參數(shù)uid_t和uid分別用來指定參數(shù)的類型和名稱。

          另外,用作系統(tǒng)調用的參數(shù)的數(shù)據類型有一個限制,它們的容量不能超過四個字節(jié)。這是因為執(zhí)行int $0x80指令進行系統(tǒng)調用時,所有的參數(shù)值都存在32位的CPU寄存器中。使用CPU寄存器傳遞參數(shù)帶來的另一個限制是可以傳送給系統(tǒng)調用的參數(shù)的數(shù)目。這個限制是最多可以傳遞5個參數(shù)。所以Linux一共定義了6個不同的_syscallN()宏指令,從_syscall0()、_syscall1()直到_syscall5()。

          一旦_syscallN()宏指令用特定系統(tǒng)調用的相應參數(shù)進行了擴展,得到的結果是一個與系統(tǒng)調用同名的函數(shù),它可以在用戶程序中執(zhí)行這一系統(tǒng)調用。

          2 添加新的系統(tǒng)調用

          如果用戶在Linux中添加新的系統(tǒng)調用,應該遵循幾個步驟才能添加成功,下面幾個步驟詳細說明了添加系統(tǒng)調用的相關內容。

          (1) 添加源代碼

          第一個任務是編寫加到內核中的源程序,即將要加到一個內核文件中去的一個函數(shù),該函數(shù)的名稱應該是新的系統(tǒng)調用名稱前面加上sys_標志。假設新加的系統(tǒng)調用為mycall(int number),在/usr/src/linux/kernel/sys.c文件中添加源代碼,如下所示:

          QUOTE:

          asmlinkage int sys_mycall(int number)

          {

          return number;

          }

          作為一個最簡單的例子,我們新加的系統(tǒng)調用僅僅返回一個整型值。

          (2) 連接新的系統(tǒng)調用

          添加新的系統(tǒng)調用后,下一個任務是使Linux內核的其余部分知道該程序的存在。為了從已有的內核程序中增加到新的函數(shù)的連接,需要編輯兩個文件。

          在我們所用的Linux內核版本(RedHat 6.0,內核為2.2.5-15)中,第一個要修改的文件是:

          QUOTE:

          /usr/src/linux/include/asm-i386/unistd.h

          該文件中包含了系統(tǒng)調用清單,用來給每個系統(tǒng)調用分配一個唯一的號碼。文件中每一行的格式如下:

          QUOTE:

          #define __NR_name NNN

          其中,name用系統(tǒng)調用名稱代替,而NNN則是該系統(tǒng)調用對應的號碼。應該將新的系統(tǒng)調用名稱加到清單的最后,并給它分配號碼序列中下一個可用的系統(tǒng)調用號。我們的系統(tǒng)調用如下:

          QUOTE:

          #define __NR_mycall 191

          系統(tǒng)調用號為191,之所以系統(tǒng)調用號是191,是因為Linux-2.2內核自身的系統(tǒng)調用號碼已經用到190。

          第二個要修改的文件是:

          QUOTE:

          /usr/src/linux/arch/i386/kernel/entry.S

          該文件中有類似如下的清單:

          QUOTE:

          .long SYMBOL_NAME()

          該清單用來對sys_call_table[]數(shù)組進行初始化。該數(shù)組包含指向內核中每個系統(tǒng)調用的指針。這樣就在數(shù)組中增加了新的內核函數(shù)的指針。我們在清單最后添加一行:

          QUOTE:

          .long SYMBOL_NAME(sys_mycall)

          (3) 重建新的Linux內核

          為使新的系統(tǒng)調用生效,需要重建Linux的內核。這需要以超級用戶身份登錄。

          QUOTE:

          #pwd

          /usr/src/linux

          #

          超級用戶在當前工作目錄(/usr/src/linux)下,才可以重建內核。

          QUOTE:

          #make config

          #make dep

          #make clearn

          #make bzImage

          編譯完畢后,系統(tǒng)生成一可用于安裝的、壓縮的內核映象文件:

          QUOTE:

          /usr/src/linux/arch/i386/boot/bzImage

          (4) 用新的內核啟動系統(tǒng)

          要使用新的系統(tǒng)調用,需要用重建的新內核重新引導系統(tǒng)。為此,需要修改/etc/lilo.conf文件,在我們的系統(tǒng)中,該文件內容如下:

          QUOTE:

          boot=/dev/hda

          map=/boot/map

          install=/boot/boot.b

          prompt

          timeout=50

          image=/boot/vmlinuz-2.2.5-15


          上一頁 1 2 下一頁

          評論


          相關推薦

          技術專區(qū)

          關閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();