淺析go語言如何實現(xiàn)協(xié)程的搶占式調(diào)度的
詳解協(xié)程搶占式調(diào)度
函數(shù)調(diào)用間進行搶占式調(diào)度
假設(shè)我們現(xiàn)在有這樣一個協(xié)程,它會進行函數(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é)程運行結(jié)束 var wg sync.WaitGroup wg.Add(1) //通過協(xié)程調(diào)用foo1 go func() { defer wg.Done() foo1() }() //等待協(xié)程運行結(jié)束 wg.Wait() }
我們給出運行結(jié)果:
foo1調(diào)用foo2
foo2調(diào)用foo3
foo3
基于這段代碼示例,我們通過這段指令獲取plan9
匯編碼:
go build -gcflags -S main.go
可以看到在foo1
插入runtime.morestack_noctxt
方法,該方法是用于檢查當(dāng)前協(xié)程是否有足夠的堆棧空間以保證函數(shù)的正常調(diào)用,基于這一點,go
就會在進行這部檢查時順帶檢查協(xié)程的執(zhí)行時長,一旦超過10ms
該方法就會將協(xié)程設(shè)置為標(biāo)記可被搶占:
0x0061 00097 (F:\github\test\main.go:8) CALL runtime.morestack_noctxt(SB)
如下圖,我們的調(diào)用的函數(shù)都會被插入一個morestack
通過這個標(biāo)記判斷當(dāng)前協(xié)程執(zhí)行耗時,一旦發(fā)現(xiàn)超過10ms則會直接通過搶占式調(diào)度的方法g0
協(xié)程直接調(diào)用schedule
方法獲取另外的協(xié)程進行調(diào)用:
這一點我們可以在asm_amd64.s
看到morestack
的newstack
的代碼,而newstack就是實現(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)用前會調(diào)用newstack進行搶占式的檢查 CALL runtime·newstack(SB) CALL runtime·abort(SB) // crash if newstack returns RET
上述的newstack
方法在stack.go
中,如果當(dāng)前協(xié)程可被搶占則會調(diào)用gopreempt_m
回到g0
調(diào)用schedule
方法從協(xié)程隊列中拿到新的協(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ā)起信號的搶占式調(diào)度
假設(shè)我們的協(xié)程沒有進行額外的函數(shù)調(diào)用,是否就意味著當(dāng)前協(xié)程的線程不能被搶占呢?很明顯不是這樣:
網(wǎng)絡(luò)傳輸過程中需要發(fā)送某些緊急消息希望通過已有連接迅速將消息通知給對端時,就會產(chǎn)生SIGURG
信號,go
語言就會在收到此信號時觸發(fā)搶占式調(diào)度。
進行GC
工作時像目標(biāo)線程發(fā)送信號由此實現(xiàn)搶占式調(diào)度。
對于第一點我們可以在signal_unix.go
的sighandler
方法得以印證,可以看到它會判斷sig 是否為_SIGURG
若是則調(diào)用doSigPreempt
進行搶占式調(diào)度
func sighandler(sig uint32, info *siginfo, ctxt unsafe.Pointer, gp *g) { //如果傳入的信號為_SIGURG則調(diào)用doSigPreempt回到schedule實現(xiàn)搶占式調(diào)度 if sig == sigPreempt && debug.asyncpreemptoff == 0 && !delayedSignal { // Might be a preemption signal. doSigPreempt(gp, c) } //...... }
doSigPreempt
會通過調(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)部會得到一個和上文函數(shù)調(diào)用時搶占式調(diào)度的方法gopreempt_m的調(diào)用從而回到schedule方法 ctxt.pushCall(abi.FuncPCABI0(asyncPreempt), newpc) } } //...... }
到此這篇關(guān)于淺析go語言如何實現(xiàn)協(xié)程的搶占式調(diào)度的的文章就介紹到這了,更多相關(guān)go協(xié)程搶占式調(diào)度內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Golang TCP網(wǎng)絡(luò)編程的具體實現(xiàn)
go語言是一門功能強大的編程語言,它提供了眾多的網(wǎng)絡(luò)編程庫,其中包括tcp/ip,本文主要介紹了Golang TCP網(wǎng)絡(luò)編程的具體實現(xiàn),具有一定的參考價值,感興趣的可以來了解一下2024-06-06Go語言學(xué)習(xí)之結(jié)構(gòu)體和方法使用詳解
這篇文章主要為大家詳細(xì)介紹了Go語言中結(jié)構(gòu)體和方法的使用,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)Go語言有一定的幫助,需要的可以參考一下2022-04-04Golang實現(xiàn)EasyCache緩存庫實例探究
這篇文章主要為大家介紹了Golang實現(xiàn)EasyCache緩存庫實例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01