<em id="0a85b"><option id="0a85b"></option></em>

<abbr id="0a85b"></abbr>

      <nobr id="0a85b"></nobr>
        <tr id="0a85b"></tr>
        9久久伊人精品综合,亚洲一区精品视频在线,成 人免费va视频,国产一区二区三区黄网,99国产精品永久免费视频,亚洲毛片多多影院,精品久久久无码人妻中文字幕,无码国产欧美一区二区三区不卡
        學習啦 > 學習電腦 > 操作系統(tǒng) > Linux教程 > linux系統(tǒng)調(diào)用方法

        linux系統(tǒng)調(diào)用方法

        時間: 春健736 分享

        linux系統(tǒng)調(diào)用方法

          系統(tǒng)調(diào)用:是通過軟件中斷向內(nèi)核發(fā)送一個明確的請求,系統(tǒng)調(diào)用實現(xiàn)是在內(nèi)核完成的,那么linux系統(tǒng)怎么調(diào)用呢?下面大家與學習啦小編一起來學習一下吧。

          linux系統(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

        學習啦 > 學習電腦 > 操作系統(tǒng) > Linux教程 > linux系統(tǒng)調(diào)用方法

        linux系統(tǒng)調(diào)用方法

        時間: 春健736 分享

        linux系統(tǒng)調(diào)用方法

          系統(tǒng)調(diào)用:是通過軟件中斷向內(nèi)核發(fā)送一個明確的請求,系統(tǒng)調(diào)用實現(xiàn)是在內(nèi)核完成的,那么linux系統(tǒng)怎么調(diào)用呢?下面大家與學習啦小編一起來學習一下吧。

          linux系統(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

        x80"

          : "=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é)果與前面兩個相同!

        看過“linux 系統(tǒng)調(diào)用方法 ”的人還看了:

        1.Linux內(nèi)核詳細介紹

        2.Linux系統(tǒng)常用命令合集

        3.Linux系統(tǒng)中ioctl函數(shù)如何實現(xiàn)

        4.初學者需要學會Linux的命令

        5.Linux系統(tǒng)中重啟命令如何使用

        655203 主站蜘蛛池模板: 久久一区二区三区黄色片| gogogo高清在线播放免费| 欧美大胆老熟妇乱子伦视频| 中文字幕乱码一区二区免费| 一本久道久久综合久久鬼色| 国产在线乱子伦一区二区| 国产成人精品一区二区三区免费| 无码天堂亚洲国产AV| 99久久精品免费看国产电影| 国产97视频人人做人人爱| 精品国产乱码久久久软件下载| 99精品国产中文字幕| 美丽的姑娘在线观看免费| 18禁动漫一区二区三区| 挺进朋友人妻雪白的身体韩国电影| 国产成人av三级在线观看| 天天夜碰日日摸日日澡性色av| 国产欧美日韩免费看AⅤ视频| 日本公与熄乱理在线播放| 尹人香蕉久久99天天拍欧美p7| 午夜通通国产精品福利| √天堂中文官网8在线| 一本一道av无码中文字幕麻豆| 国精品午夜福利视频不卡| 成人精品老熟妇一区二区| 久久精品免视看国产成人| 色综合久久精品中文字幕| 亚洲av影院一区二区三区| 口爆少妇在线视频免费观看| 东京热大乱系列无码| 国产AV影片麻豆精品传媒| 国产白袜脚足j棉袜在线观看| 少妇高潮喷水惨叫久久久久电影| 无码人妻人妻经典| 色伊人国产高清在线| 欧美巨大极度另类| 婷婷四房播播| 深夜视频国产在线观看| 最新国产精品剧情在线ss| 色一情一乱一伦麻豆| 亚洲女同精品一区二区久久 |