淺析go語言如何實(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) //通過協(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
基于這段代碼示例,我們通過這段指令獲取plan9
匯編碼:
go build -gcflags -S main.go
可以看到在foo1
插入runtime.morestack_noctxt
方法,該方法是用于檢查當(dāng)前協(xié)程是否有足夠的堆棧空間以保證函數(shù)的正常調(diào)用,基于這一點(diǎn),go
就會(huì)在進(jìn)行這部檢查時(shí)順帶檢查協(xié)程的執(zhí)行時(shí)長(zhǎng),一旦超過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
通過這個(gè)標(biāo)記判斷當(dāng)前協(xié)程執(zhí)行耗時(shí),一旦發(fā)現(xiàn)超過10ms則會(huì)直接通過搶占式調(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é)程沒有進(jìn)行額外的函數(shù)調(diào)用,是否就意味著當(dāng)前協(xié)程的線程不能被搶占呢?很明顯不是這樣:
網(wǎng)絡(luò)傳輸過程中需要發(fā)送某些緊急消息希望通過已有連接迅速將消息通知給對(duì)端時(shí),就會(huì)產(chǎn)生SIGURG
信號(hào),go
語言就會(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)為_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ì)通過調(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語言如何實(shí)現(xiàn)協(xié)程的搶占式調(diào)度的的文章就介紹到這了,更多相關(guān)go協(xié)程搶占式調(diào)度內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言排序算法之插入排序與生成隨機(jī)數(shù)詳解
從這篇文章開始將帶領(lǐng)大家學(xué)習(xí)Go語言的經(jīng)典排序算法,比如插入排序、選擇排序、冒泡排序、希爾排序、歸并排序、堆排序和快排,二分搜索,外部排序和MapReduce等,本文將先詳細(xì)介紹插入排序,并給大家分享了go語言生成隨機(jī)數(shù)的方法,下面來一起看看吧。2017-11-11go語言中Timer和Ticker兩種計(jì)時(shí)器的使用
go語言中有Timer和Ticker這樣的兩種計(jì)時(shí)器,兩種計(jì)時(shí)器分別實(shí)現(xiàn)了不同的計(jì)時(shí)功能,本文主要介紹了go語言中Timer和Ticker兩種計(jì)時(shí)器的使用,感興趣的可以了解一下2024-08-08Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn)
go語言是一門功能強(qiáng)大的編程語言,它提供了眾多的網(wǎng)絡(luò)編程庫(kù),其中包括tcp/ip,本文主要介紹了Golang TCP網(wǎng)絡(luò)編程的具體實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以來了解一下2024-06-06Go語言學(xué)習(xí)之結(jié)構(gòu)體和方法使用詳解
這篇文章主要為大家詳細(xì)介紹了Go語言中結(jié)構(gòu)體和方法的使用,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語言有一定的幫助,需要的可以參考一下2022-04-04Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究
這篇文章主要為大家介紹了Golang實(shí)現(xiàn)EasyCache緩存庫(kù)實(shí)例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01