系統(tǒng)調(diào)用:計(jì)算機(jī)中的“服務(wù)員”
一、什么是系統(tǒng)調(diào)用
想象一下,你在一家餐廳就餐,你需要通過(guò)服務(wù)員來(lái)點(diǎn)菜、支付等。系統(tǒng)調(diào)用就像是這個(gè)服務(wù)員,它在軟件和操作系統(tǒng)之間起到了橋梁的作用。當(dāng)軟件需要操作系統(tǒng)提供的某項(xiàng)服務(wù)時(shí),它就像顧客一樣,通過(guò)點(diǎn)菜(調(diào)用API)來(lái)告訴服務(wù)員(系統(tǒng)調(diào)用)它的需求。本質(zhì)上,系統(tǒng)調(diào)用就是用戶(hù)程序與操作系統(tǒng)之間的接口程序。
二、為什么需要系統(tǒng)調(diào)用
保護(hù)資源
就像在餐廳里,你不能直接進(jìn)廚房做飯,而需要通過(guò)服務(wù)員來(lái)點(diǎn)菜。同樣,操作系統(tǒng)會(huì)將可能產(chǎn)生多個(gè)程序訪問(wèn)沖突的資源保護(hù)起來(lái),提供API,軟件只能通過(guò)系統(tǒng)調(diào)用這些API來(lái)操作對(duì)應(yīng)的資源。比如應(yīng)用程序要訪問(wèn)網(wǎng)絡(luò)、讀寫(xiě)文件等都需要通過(guò)系統(tǒng)調(diào)用來(lái)完成。
簡(jiǎn)化軟件開(kāi)發(fā)
就像你只需要告訴服務(wù)員你想吃什么,而不需要自己去廚房做飯。操作系統(tǒng)為簡(jiǎn)化上層軟件開(kāi)發(fā),對(duì)某些資源和基礎(chǔ)能力進(jìn)行封裝,提供API,軟件通過(guò)系統(tǒng)調(diào)用這些API可以輕松集成能力。
操作系統(tǒng)是基礎(chǔ)軟件
操作系統(tǒng)是基礎(chǔ)軟件,與其它軟件是不同的進(jìn)程。其它軟件使用操作系統(tǒng)提供的能力,可以用進(jìn)程間通信,但是各種進(jìn)程間通信機(jī)制也是基于系統(tǒng)調(diào)用,所以直接使用系統(tǒng)調(diào)用更為便捷。
三、系統(tǒng)調(diào)用過(guò)程
系統(tǒng)調(diào)用和普通庫(kù)函數(shù)調(diào)用非常相似,只是系統(tǒng)調(diào)用由操作系統(tǒng)內(nèi)核提供,運(yùn)行于內(nèi)核態(tài),而普通的庫(kù)函數(shù)調(diào)用由函數(shù)庫(kù)或用戶(hù)自己提供,運(yùn)行于用戶(hù)態(tài)。
圖片
軟中斷模式
在軟中斷模式下,用戶(hù)程序會(huì)調(diào)用標(biāo)準(zhǔn)庫(kù),這就像顧客看菜單點(diǎn)菜。然后,標(biāo)準(zhǔn)庫(kù)執(zhí)行軟中斷指令,CPU切換上下文,由用戶(hù)態(tài)進(jìn)入內(nèi)核態(tài),這就像服務(wù)員把菜單帶到廚房。內(nèi)核通過(guò)中斷向量表查找到中斷處理程序,并執(zhí)行它,這就像廚師根據(jù)菜單開(kāi)始做菜。系統(tǒng)調(diào)用執(zhí)行完畢,從中斷處理程序返回,這就像服務(wù)員把做好的菜端給顧客。
在軟中斷模式中,CPU只需要執(zhí)行一個(gè)INT指令就可以了,比較簡(jiǎn)單。
快速調(diào)用
快速調(diào)用是為了避免上下文切換的開(kāi)銷(xiāo)。這就像顧客直接告訴廚師他們想吃什么,而不需要通過(guò)服務(wù)員,使得通信的速度更快,更有效率。
快速調(diào)用需要CPU的支持,以x86 CPU為例,快速調(diào)用的實(shí)現(xiàn)主要使用了兩個(gè)指令:sysenter和sysexit。
sysenter指令
sysenter指令是一個(gè)特殊的CPU指令,它用于從用戶(hù)態(tài)切換到內(nèi)核態(tài)。當(dāng)一個(gè)程序需要進(jìn)行系統(tǒng)調(diào)用時(shí),它會(huì)執(zhí)行sysenter指令。這個(gè)指令會(huì)使CPU切換到內(nèi)核態(tài),并跳轉(zhuǎn)到操作系統(tǒng)內(nèi)核中預(yù)設(shè)的系統(tǒng)調(diào)用入口點(diǎn)。這個(gè)過(guò)程中,CPU不需要執(zhí)行中斷,也不需要切換上下文,因此,執(zhí)行sysenter指令的開(kāi)銷(xiāo)比執(zhí)行軟中斷的開(kāi)銷(xiāo)要小。
sysexit指令
與sysenter指令相對(duì)應(yīng),sysexit指令用于從內(nèi)核態(tài)切換回用戶(hù)態(tài)。當(dāng)系統(tǒng)調(diào)用完成后,操作系統(tǒng)內(nèi)核會(huì)執(zhí)行sysexit指令。這個(gè)指令會(huì)使CPU切換回用戶(hù)態(tài),并跳轉(zhuǎn)回到執(zhí)行sysenter指令之后的下一條指令。這個(gè)過(guò)程同樣不需要執(zhí)行中斷和切換堆棧,因此,執(zhí)行sysexit指令的開(kāi)銷(xiāo)也比執(zhí)行軟中斷的開(kāi)銷(xiāo)要小。
不同的CPU可能會(huì)有不同的快速調(diào)用方式,不過(guò)應(yīng)用程序不需要關(guān)心CPU具體是怎么做的,操作系統(tǒng)會(huì)做好封裝,標(biāo)準(zhǔn)庫(kù)會(huì)提供操作的API。
四、代碼示例
在Go語(yǔ)言中,我們可以使用syscall包來(lái)進(jìn)行系統(tǒng)調(diào)用。下面是一個(gè)使用syscall包進(jìn)行系統(tǒng)調(diào)用的示例,該示例使用系統(tǒng)調(diào)用獲取系統(tǒng)時(shí)間:
package main
import (
"fmt"
"syscall"
"time"
)
func main() {
// 創(chuàng)建一個(gè)syscall.Timeval結(jié)構(gòu)體用于存儲(chǔ)時(shí)間
var tv syscall.Timeval
// 使用syscall.Gettimeofday函數(shù)獲取當(dāng)前時(shí)間
err := syscall.Gettimeofday(&tv)
if err != nil {
fmt.Println("Error:", err)
return
}
// 將秒和微秒轉(zhuǎn)換為time.Time類(lèi)型
t := time.Unix(tv.Sec, tv.Usec*1000)
// 打印時(shí)間
fmt.Println("Current time:", t)
}
在這個(gè)示例中,我們首先創(chuàng)建了一個(gè)syscall.Timeval結(jié)構(gòu)體用于存儲(chǔ)時(shí)間。然后,我們使用syscall.Gettimeofday函數(shù)來(lái)獲取當(dāng)前時(shí)間,并將其存儲(chǔ)在我們之前創(chuàng)建的syscall.Timeval結(jié)構(gòu)體中。最后,我們將syscall.Timeval中的秒和微秒轉(zhuǎn)換為time.Time類(lèi)型,并打印出來(lái)。