淺析go語(yǔ)言如何實(shí)現(xiàn)協(xié)程的搶占式調(diào)度的
詳解協(xié)程搶占式調(diào)度
函數(shù)調(diào)用間進(jìn)行搶占式調(diào)度
假設(shè)我們現(xiàn)在有這樣一個(gè)協(xié)程,它會(huì)進(jìn)行函數(shù)嵌套調(diào)用,代碼如下所示:
func foo1() {
fmt.Println("foo1調(diào)用foo2")
foo2()
}
func foo2() {
fmt.Println("foo2調(diào)用foo3")
foo3()
}
func foo3() {
fmt.Println("foo3")
}
func main() {
//設(shè)置WaitGroup等待協(xié)程運(yùn)行結(jié)束
var wg sync.WaitGroup
wg.Add(1)
//通過(guò)協(xié)程調(diào)用foo1
go func() {
defer wg.Done()
foo1()
}()
//等待協(xié)程運(yùn)行結(jié)束
wg.Wait()
}
我們給出運(yùn)行結(jié)果:
foo1調(diào)用foo2
foo2調(diào)用foo3
foo3
基于這段代碼示例,我們通過(guò)這段指令獲取plan9匯編碼:
go build -gcflags -S main.go
可以看到在foo1插入runtime.morestack_noctxt方法,該方法是用于檢查當(dāng)前協(xié)程是否有足夠的堆??臻g以保證函數(shù)的正常調(diào)用,基于這一點(diǎn),go就會(huì)在進(jìn)行這部檢查時(shí)順帶檢查協(xié)程的執(zhí)行時(shí)長(zhǎng),一旦超過(guò)10ms該方法就會(huì)將協(xié)程設(shè)置為標(biāo)記可被搶占:
0x0061 00097 (F:\github\test\main.go:8) CALL runtime.morestack_noctxt(SB)
如下圖,我們的調(diào)用的函數(shù)都會(huì)被插入一個(gè)morestack通過(guò)這個(gè)標(biāo)記判斷當(dāng)前協(xié)程執(zhí)行耗時(shí),一旦發(fā)現(xiàn)超過(guò)10ms則會(huì)直接通過(guò)搶占式調(diào)度的方法g0協(xié)程直接調(diào)用schedule方法獲取另外的協(xié)程進(jìn)行調(diào)用:

這一點(diǎn)我們可以在asm_amd64.s看到morestack的newstack的代碼,而newstack就是實(shí)現(xiàn)搶占式調(diào)度的核心:
TEXT runtime·morestack(SB),NOSPLIT,$0-0
// Cannot grow scheduler stack (m->g0).
get_tls(CX)
MOVQ g(CX), BX
MOVQ g_m(BX), BX
MOVQ m_g0(BX), SI
CMPQ g(CX), SI
JNE 3(PC)
CALL runtime·badmorestackg0(SB)
CALL runtime·abort(SB)
//......
//函數(shù)調(diào)用前會(huì)調(diào)用newstack進(jìn)行搶占式的檢查
CALL runtime·newstack(SB)
CALL runtime·abort(SB) // crash if newstack returns
RET
上述的newstack方法在stack.go中,如果當(dāng)前協(xié)程可被搶占則會(huì)調(diào)用gopreempt_m回到g0調(diào)用schedule方法從協(xié)程隊(duì)列中拿到新的協(xié)程執(zhí)行任務(wù):
func newstack() {
preempt := stackguard0 == stackPreempt
//如果preempt 為true,則直接當(dāng)前協(xié)程被標(biāo)記為搶占直接調(diào)用gopreempt_m讓出線程執(zhí)行權(quán)
if preempt {
if gp == thisg.m.g0 {
throw("runtime: preempt g0")
}
//......
// Act like goroutine called runtime.Gosched.
gopreempt_m(gp) // never return
}
}
基于系統(tǒng)調(diào)用發(fā)起信號(hào)的搶占式調(diào)度
假設(shè)我們的協(xié)程沒(méi)有進(jìn)行額外的函數(shù)調(diào)用,是否就意味著當(dāng)前協(xié)程的線程不能被搶占呢?很明顯不是這樣:
網(wǎng)絡(luò)傳輸過(guò)程中需要發(fā)送某些緊急消息希望通過(guò)已有連接迅速將消息通知給對(duì)端時(shí),就會(huì)產(chǎn)生SIGURG信號(hào),go語(yǔ)言就會(huì)在收到此信號(hào)時(shí)觸發(fā)搶占式調(diào)度。

進(jìn)行GC工作時(shí)像目標(biāo)線程發(fā)送信號(hào)由此實(shí)現(xiàn)搶占式調(diào)度。
對(duì)于第一點(diǎn)我們可以在signal_unix.go的sighandler方法得以印證,可以看到它會(huì)判斷sig 是否為_SIGURG若是則調(diào)用doSigPreempt進(jìn)行搶占式調(diào)度
func sighandler(sig uint32, info *siginfo, ctxt unsafe.Pointer, gp *g) {
//如果傳入的信號(hào)為_(kāi)SIGURG則調(diào)用doSigPreempt回到schedule實(shí)現(xiàn)搶占式調(diào)度
if sig == sigPreempt && debug.asyncpreemptoff == 0 && !delayedSignal {
// Might be a preemption signal.
doSigPreempt(gp, c)
}
//......
}
doSigPreempt會(huì)通過(guò)調(diào)用asyncPreempt最終執(zhí)行到preempt.go的asyncPreempt2調(diào)用到和上文函數(shù)調(diào)用搶占式調(diào)度方法gopreempt_m回到schedule方法從而完成搶占式調(diào)度:
func doSigPreempt(gp *g, ctxt *sigctxt) {
//......
if wantAsyncPreempt(gp) {
if ok, newpc := isAsyncSafePoint(gp, ctxt.sigpc(), ctxt.sigsp(), ctxt.siglr()); ok {
// 調(diào)用asyncPreempt內(nèi)部會(huì)得到一個(gè)和上文函數(shù)調(diào)用時(shí)搶占式調(diào)度的方法gopreempt_m的調(diào)用從而回到schedule方法
ctxt.pushCall(abi.FuncPCABI0(asyncPreempt), newpc)
}
}
//......
}
到此這篇關(guān)于淺析go語(yǔ)言如何實(shí)現(xiàn)協(xié)程的搶占式調(diào)度的的文章就介紹到這了,更多相關(guān)go協(xié)程搶占式調(diào)度內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語(yǔ)言排序算法之插入排序與生成隨機(jī)數(shù)詳解
從這篇文章開(kāi)始將帶領(lǐng)大家學(xué)習(xí)Go語(yǔ)言的經(jīng)典排序算法,比如插入排序、選擇排序、冒泡排序、希爾排序、歸并排序、堆排序和快排,二分搜索,外部排序和MapReduce等,本文將先詳細(xì)介紹插入排序,并給大家分享了go語(yǔ)言生成隨機(jī)數(shù)的方法,下面來(lái)一起看看吧。2017-11-11
go語(yǔ)言中Timer和Ticker兩種計(jì)時(shí)器的使用
go語(yǔ)言中有Timer和Ticker這樣的兩種計(jì)時(shí)器,兩種計(jì)時(shí)器分別實(shí)現(xiàn)了不同的計(jì)時(shí)功能,本文主要介紹了go語(yǔ)言中Timer和Ticker兩種計(jì)時(shí)器的使用,感興趣的可以了解一下2024-08-08
Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn)
go語(yǔ)言是一門功能強(qiáng)大的編程語(yǔ)言,它提供了眾多的網(wǎng)絡(luò)編程庫(kù),其中包括tcp/ip,本文主要介紹了Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以來(lái)了解一下2024-06-06
Go語(yǔ)言學(xué)習(xí)之結(jié)構(gòu)體和方法使用詳解
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言中結(jié)構(gòu)體和方法的使用,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定的幫助,需要的可以參考一下2022-04-04
Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究
這篇文章主要為大家介紹了Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01

