三種方法實現(xiàn)Linux系統(tǒng)調(diào)用
系統(tǒng)調(diào)用(System Call)是操作系統(tǒng)為在用戶態(tài)運行的進程與硬件設備(如CPU、磁盤、打印機等)進行交互提供的一組接口。當用戶進程需要發(fā)生系統(tǒng)調(diào)用時,CPU 通過軟中斷切換到內(nèi)核態(tài)開始執(zhí)行內(nèi)核系統(tǒng)調(diào)用函數(shù)。下面介紹Linux 下三種發(fā)生系統(tǒng)調(diào)用的方法:
一、通過 glibc 提供的庫函數(shù)
glibc 是 Linux 下使用的開源的標準 C 庫,它是 GNU 發(fā)布的 libc 庫,即運行時庫。glibc 為程序員提供豐富的 API(Application Programming Interface),除了例如字符串處理、數(shù)學運算等用戶態(tài)服務之外,最重要的是封裝了操作系統(tǒng)提供的系統(tǒng)服務,即系統(tǒng)調(diào)用的封裝。那么glibc提供的系統(tǒng)調(diào)用API與內(nèi)核特定的系統(tǒng)調(diào)用之間的關(guān)系是什么呢?
- 通常情況,每個特定的系統(tǒng)調(diào)用對應了至少一個 glibc 封裝的庫函數(shù),如系統(tǒng)提供的打開文件系統(tǒng)調(diào)用 sys_open 對應的是 glibc 中的 open 函數(shù);
- 其次,glibc 一個單獨的 API 可能調(diào)用多個系統(tǒng)調(diào)用,如 glibc 提供的 printf 函數(shù)就會調(diào)用如 sys_open、sys_mmap、sys_write、sys_close 等等系統(tǒng)調(diào)用;
- 另外,多個 API 也可能只對應同一個系統(tǒng)調(diào)用,如glibc 下實現(xiàn)的 malloc、calloc、free 等函數(shù)用來分配和釋放內(nèi)存,都利用了內(nèi)核的 sys_brk 的系統(tǒng)調(diào)用。
舉例來說,我們通過 glibc 提供的chmod 函數(shù)來改變文件 etc/passwd 的屬性為 444:
#include <sys/types.h> #include <sys/stat.h> #include <errno.h> #include <stdio.h> int main() { int rc; rc = chmod("/etc/passwd", 0444); if (rc == -1) fprintf(stderr, "chmod failed, errno = %d\n", errno); else printf("chmod success!\n"); return 0; }
在普通用戶下編譯運用,輸出結(jié)果為:
chmod failed, errno = 1
上面系統(tǒng)調(diào)用返回的值為-1,說明系統(tǒng)調(diào)用失敗,錯誤碼為1,在 /usr/include/asm-generic/errno-base.h 文件中有如下錯誤代碼說明:
#define EPERM 1 /* Operation not permitted */
即無權(quán)限進行該操作,我們以普通用戶權(quán)限是無法修改 /etc/passwd 文件的屬性的,結(jié)果正確。
二、使用 syscall 直接調(diào)用
使用上面的方法有很多好處,首先你無須知道更多的細節(jié),如 chmod 系統(tǒng)調(diào)用號,你只需了解 glibc 提供的 API 的原型;其次,該方法具有更好的移植性,你可以很輕松將該程序移植到其他平臺,或者將 glibc 庫換成其它庫,程序只需做少量改動。
但有點不足是,如果 glibc 沒有封裝某個內(nèi)核提供的系統(tǒng)調(diào)用時,我就沒辦法通過上面的方法來調(diào)用該系統(tǒng)調(diào)用。如我自己通過編譯內(nèi)核增加了一個系統(tǒng)調(diào)用,這時 glibc 不可能有你新增系統(tǒng)調(diào)用的封裝 API,此時我們可以利用 glibc 提供的syscall 函數(shù)直接調(diào)用。該函數(shù)定義在 unistd.h 頭文件中,函數(shù)原型如下:
long int syscall (long int sysno, ...)
- sysno 是系統(tǒng)調(diào)用號,每個系統(tǒng)調(diào)用都有唯一的系統(tǒng)調(diào)用號來標識。在 sys/syscall.h 中有所有可能的系統(tǒng)調(diào)用號的宏定義。
- ... 為剩余可變長的參數(shù),為系統(tǒng)調(diào)用所帶的參數(shù),根據(jù)系統(tǒng)調(diào)用的不同,可帶0~5個不等的參數(shù),如果超過特定系統(tǒng)調(diào)用能帶的參數(shù),多余的參數(shù)被忽略。
- 返回值 該函數(shù)返回值為特定系統(tǒng)調(diào)用的返回值,在系統(tǒng)調(diào)用成功之后你可以將該返回值轉(zhuǎn)化為特定的類型,如果系統(tǒng)調(diào)用失敗則返回 -1,錯誤代碼存放在 errno 中。
還以上面修改 /etc/passwd 文件的屬性為例,這次使用 syscall 直接調(diào)用:
#include <stdio.h> #include <unistd.h> #include <sys/syscall.h> #include <errno.h> int main() { int rc; rc = syscall(SYS_chmod, "/etc/passwd", 0444); if (rc == -1) fprintf(stderr, "chmod failed, errno = %d\n", errno); else printf("chmod succeess!\n"); return 0; }
在普通用戶下編譯執(zhí)行,輸出的結(jié)果與上例相同。
三、通過 int 指令陷入
如果我們知道系統(tǒng)調(diào)用的整個過程的話,應該就能知道用戶態(tài)程序通過軟中斷指令int 0x80 來陷入內(nèi)核態(tài)(在Intel Pentium II 又引入了sysenter指令),參數(shù)的傳遞是通過寄存器,eax 傳遞的是系統(tǒng)調(diào)用號,ebx、ecx、edx、esi和edi 來依次傳遞最多五個參數(shù),當系統(tǒng)調(diào)用返回時,返回值存放在 eax 中。
仍然以上面的修改文件屬性為例,將調(diào)用系統(tǒng)調(diào)用那段寫成內(nèi)聯(lián)匯編代碼:
#include <stdio.h> #include <sys/types.h> #include <sys/syscall.h> #include <errno.h> int main() { long rc; char *file_name = "/etc/passwd"; unsigned short mode = 0444; asm( "int $0x80" : "=a" (rc) : "0" (SYS_chmod), "b" ((long)file_name), "c" ((long)mode) ); if ((unsigned long)rc >= (unsigned long)-132) { errno = -rc; rc = -1; } if (rc == -1) fprintf(stderr, "chmode failed, errno = %d\n", errno); else printf("success!\n"); return 0; }
如果 eax 寄存器存放的返回值(存放在變量 rc 中)在 -1~-132 之間,就必須要解釋為出錯碼(在/usr/include/asm-generic/errno.h 文件中定義的最大出錯碼為 132),這時,將錯誤碼寫入 errno 中,置系統(tǒng)調(diào)用返回值為 -1;否則返回的是 eax 中的值。
上面程序在 32位Linux下以普通用戶權(quán)限編譯運行結(jié)果與前面兩個相同!
以上就是本文的詳細內(nèi)容,希望大家喜歡。
相關(guān)文章
關(guān)于Apache shiro實現(xiàn)一個賬戶同一時刻只有一個人登錄(shiro 單點登錄)
今天和同事在一起探討shiro如何實現(xiàn)一個賬戶同一時刻只有一session存在的問題,下面小編把核心代碼分享到腳本之家平臺,需要的朋友參考下2017-09-09ubuntu環(huán)境下安裝memcache及啟動的方法
這篇文章主要介紹了ubuntu環(huán)境下安裝memcache及啟動的方法,結(jié)合實例形式簡單分析了ubuntu環(huán)境下memcache的安裝、啟動及簡單測試方法,需要的朋友可以參考下2018-03-03linux下查看已經(jīng)安裝的jdk 并卸載jdk的方法(推薦)
下面小編就為大家?guī)硪黄猯inux下查看已經(jīng)安裝的jdk 并卸載jdk的方法(推薦)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-11-11CentOs 7.*中配置安裝phpMyAdmin的完整步驟記錄
phpMyAdmin是一個以PHP為基礎(chǔ),以Web-Base方式架構(gòu)在網(wǎng)站主機上的MySQL的資料庫管理工具。下面這篇文章主要給大家介紹了關(guān)于CentOs 7.*中配置安裝phpMyAdmin的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧2018-07-07linux安裝mysql數(shù)據(jù)庫以及配置Java項目的圖文詳解
在本篇文章里小編給大家整理的是關(guān)于linux安裝mysql配置Java的相關(guān)知識點內(nèi)容,有需要的朋友們可以跟著學習下。2019-11-11Centos7如何備份和還原Redis數(shù)據(jù)的方法
這篇文章主要介紹了Centos7如何備份和還原Redis數(shù)據(jù)的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-06-06詳解Linux常用命令的用法(二)————文本編輯器命令vi/vim
這篇文章主要介紹了Linux常用命令的用法(二)————文本編輯器命令vi/vim,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-04-04