make_fcountext、jump_fcontext

咱們先弄清如何進行協程的切換,程序能夠在某個地方掛起,跳轉到另外的流程中執行,而且能夠從新在掛起處繼續運行。那如何實現呢?app

咱們先來看一個例子,有下面2個函數,若是在一個單線程中讓輸出結果依次是 funcA1 funcB1 funcA2 funcB2 ... ,你會怎麼作呢?函數

void funcA(){
    int i = 0;
    while(true){
        //to do something
        
        printf("funcA%d ",i);
        i++;
    }
}

void funcB(){
    int i = 0;
    while(true){
        //to do something
        
        printf("funcB%d ",i);
        i++;
    }
}

若是從c代碼的角度來看,若是單線程運行到func1 的 while循環中,如何能調用到func2的while循環呢?必須使用跳轉。spa

首先想到是goto。goto是能夠實現跳轉,可是goto不能實現函數間的跳轉。沒法知足這個要求。即便能夠實現函數間跳轉,難道就可行嗎?

那這裏不得不說下C函數調用過程.net

具體相見這篇文章https://blog.csdn.net/jelly_9/article/details/53239718
子程序或者稱爲函數,在全部語言中都是層級調用,好比A調用B,B在執行過程當中又調用了C,
C執行完畢返回,B執行完畢返回,最後是A執行完畢。
因此子程序調用是經過棧實現的,子程序調用老是一個入口,一次返回,調用順序是明確的。

程序運行有2個部分,指令,數據。棧保存的數據,指令經過寄存器(rip)控制。線程

2個函數內部的跳轉必須保證棧是正確,因此跳轉以前須要保存好當前的棧信息,而後跳轉。
另外咱們能夠獲得另外一個信息,在一個棧上實現多個流程直接的跳轉是不能實現的。
因此須要多個棧來維護。

那咱們來看jump_fcontext是怎麼實現跳轉指針

c語言函數聲明
int jump_fcontext(fcontext_t *ofc, fcontext_t nfc, void* vp, bool preserve_fpu);


彙編代碼以下
.text
.globl jump_fcontext
.type jump_fcontext,@function
.align 16
jump_fcontext:
    pushq  %rbp  /* save RBP */
    pushq  %rbx  /* save RBX */
    pushq  %r15  /* save R15 */
    pushq  %r14  /* save R14 */
    pushq  %r13  /* save R13 */
    pushq  %r12  /* save R12 */

    /* prepare stack for FPU */
    leaq  -0x8(%rsp), %rsp

    /* test for flag preserve_fpu */
    cmp  $0, %rcx
    je  1f

    /* save MMX control- and status-word */
    stmxcsr  (%rsp)
    /* save x87 control-word */
    fnstcw   0x4(%rsp)

1:
    /* store RSP (pointing to context-data) in RDI */
    movq  %rsp, (%rdi)

    /* restore RSP (pointing to context-data) from RSI */
    movq  %rsi, %rsp

    /* test for flag preserve_fpu */
    cmp  $0, %rcx
    je  2f

    /* restore MMX control- and status-word */
    ldmxcsr  (%rsp)
    /* restore x87 control-word */
    fldcw  0x4(%rsp)

2:
    /* prepare stack for FPU */
    leaq  0x8(%rsp), %rsp

    popq  %r12  /* restrore R12 */
    popq  %r13  /* restrore R13 */
    popq  %r14  /* restrore R14 */
    popq  %r15  /* restrore R15 */
    popq  %rbx  /* restrore RBX */
    popq  %rbp  /* restrore RBP */

    /* restore return-address */
    popq  %r8

    /* use third arg as return-value after jump */
    movq  %rdx, %rax
    /* use third arg as first arg in context function */
    movq  %rdx, %rdi

    /* indirect jump to context */
    jmp  *%r8
.size jump_fcontext,.-jump_fcontext

/* Mark that we don't need executable stack.  */
.section .note.GNU-stack,"",%progbits

寄存器的用途能夠先了解下https://www.jianshu.com/p/571...rest

一、保存寄存器code

pushq  %rbp  /* save RBP */
pushq  %rbx  /* save RBX */
pushq  %r15  /* save R15 */
pushq  %r14  /* save R14 */
pushq  %r13  /* save R13 */
pushq  %r12  /* save R12 */

「被調函數有義務保證 rbp rbx r12~r15 這幾個寄存器的值在進出函數先後一致」
rbx 是基址寄存器 做用存放存儲區的起始地址 被調用者保存
rbp (base pointer)基址指針寄存器,用於提供堆棧內某個單元的偏移地址,與rss段寄存器聯用, 
能夠訪問堆棧中的任一個存儲單元,被調用者保存

二、預留fpu 8個字節空間視頻

/* prepare stack for FPU */
leaq  -0x8(%rsp), %rsp
表示 %rsp中的內容減8。因爲棧是從高到底,此處的意思表示預留8字節的棧空間。
FPU:(Float Point Unit,浮點運算單元)

三、判斷是否保存fpu協程

cmp  $0, %rcx
je  1f

rcx是第四個參數,判斷是否等於0。若是爲0,跳轉到1標示的位置。也就是preserve_fpu 。

當preserve_fpu = true 的時候,須要執行2個指令是將浮點型運算的2個32位寄存器數據保存到第2步中預留的8字節空間。
/* save MMX control- and status-word */
stmxcsr  (%rsp)
/* save x87 control-word */
fnstcw   0x4(%rsp)

四、修改rsp 此時已經改變到其餘棧
將rsp 保存到第一參數(第一個參數保存在rdi)指向的內存。fcontext_t *ofc 第一參數ofc指向的內存中保存是 rsp 的指針。第二條指令,實現了將第二個參數複製到 rsp.

1:
/* store RSP (pointing to context-data) in RDI */
movq  %rsp, (%rdi)

/* restore RSP (pointing to context-data) from RSI */
movq  %rsi, %rsp

五、判斷是否保存了fpu,若是保存了就恢復保存在nfx 棧上的 fpu相關數據到響應的寄存器。

/* test for flag preserve_fpu */
cmp  $0, %rcx
je  2f

/* restore MMX control- and status-word */
ldmxcsr  (%rsp)
/* restore x87 control-word */
fldcw  0x4(%rsp)

六、將rsp 存儲的地址+8(8字節fpu),按順序將棧中數據恢復到寄存器中。

2:
/* prepare stack for FPU */
leaq  0x8(%rsp), %rsp

popq  %r12  /* restrore R12 */
popq  %r13  /* restrore R13 */
popq  %r14  /* restrore R14 */
popq  %r15  /* restrore R15 */
popq  %rbx  /* restrore RBX */
popq  %rbp  /* restrore RBP */

七、設置返回值,實現指令跳轉。
接下來繼續pop 數據,那棧上存的是什麼呢,在c函數調用文章中能夠知道,call的時候會保存rip(指令寄存器)到棧。因此此時POP的數據是rip 也就是下一條指令。這是下一條指令是nfx 棧保存的,因此這是另外一個協程的下一條指令。保存到 r8。最後跳轉下一條指令就恢復到另外一個協程運行 jmp *%r8。

movq %rdx, %rax 是將上一個協程A jump_fcontext第三個參數做爲當前協程B jump_fcontext 的返回值,能夠實現2個協程直接的數據傳遞。

movq %rdx, %rdi 若是跳轉過去的新的協程,將第三個參數做爲協程B 啓動入口void func(int param)的第一參數。

/* restore return-address */
popq  %r8

/* use third arg as return-value after jump */
movq  %rdx, %rax
/* use third arg as first arg in context function */
movq  %rdx, %rdi

/* indirect jump to context */
jmp  *%r8

瞭解了程序是如何跳轉後,我門在看下如何建立一個協程棧呢。make_fcontext
c語言函數聲明 fcontext_t make_fcontext(void sp, size_t size, void (fn)(int));

.text
.globl make_fcontext
.type make_fcontext,@function
.align 16
make_fcontext:
    /* first arg of make_fcontext() == top of context-stack */
    movq  %rdi, %rax

    /* shift address in RAX to lower 16 byte boundary */
    andq  $-16, %rax

    /* reserve space for context-data on context-stack */
    /* size for fc_mxcsr .. RIP + return-address for context-function */
    /* on context-function entry: (RSP -0x8) % 16 == 0 */
    leaq  -0x48(%rax), %rax

    /* third arg of make_fcontext() == address of context-function */
    movq  %rdx, 0x38(%rax)

    /* save MMX control- and status-word */
    stmxcsr  (%rax)
    /* save x87 control-word */
    fnstcw   0x4(%rax)

    /* compute abs address of label finish */
    leaq  finish(%rip), %rcx
    /* save address of finish as return-address for context-function */
    /* will be entered after context-function returns */
    movq  %rcx, 0x40(%rax)

    ret /* return pointer to context-data */

finish:
    /* exit code is zero */
    xorq  %rdi, %rdi
    /* exit application */
    call  _exit@PLT
    hlt
.size make_fcontext,.-make_fcontext

/* Mark that we don't need executable stack. */
.section .note.GNU-stack,"",%progbits

一、第一個參數是程序申請的內存地址高位(棧是從高到低),將第一個參數放到rax,將地址取16的整數倍。
andq $-16, %rax 表示低4位取0。 -16 的補碼錶示爲0xfffffffff0.

/* first arg of make_fcontext() == top of context-stack */
movq  %rdi, %rax

/* shift address in RAX to lower 16 byte boundary */
andq  $-16, %rax

二、預留72字節棧空間,將第3個參數(void (*fn)(int)函數指針)保存在當前偏移0x38位置(大小8字節)。

/* reserve space for context-data on context-stack */
/* size for fc_mxcsr .. RIP + return-address for context-function */
/* on context-function entry: (RSP -0x8) % 16 == 0 */
leaq  -0x48(%rax), %rax

/* third arg of make_fcontext() == address of context-function */
movq  %rdx, 0x38(%rax)

三、保存fpu 和jump_fcontext 相似總大小8字節。

/* save MMX control- and status-word */
stmxcsr  (%rax)
/* save x87 control-word */
fnstcw   0x4(%rax)

四、計算finish的絕對地址,保存到棧的0x40位置。
leaq finish(%rip), %rcx 表示finish是相對位置+rip 就是finish的函數的地址。

/* compute abs address of label finish */
leaq  finish(%rip), %rcx
/* save address of finish as return-address for context-function */
/* will be entered after context-function returns */
movq  %rcx, 0x40(%rax)

五、返回,rax 做爲返回值,目前的指向能夠當作新棧的棧頂,至關於rsp

ret /* return pointer to context-data */

咱們回頭在看看爲何會預留72字節大小。首先知道jump_fcontext 在新棧須要 pop 的大小爲,fpu(8字節)+ rbp rbx r12 ~ r15 (8*6 = 48 字節) = 56 字節。還會繼續POP rip 8 字節,因此能夠看到第二步中 movq %rdx, 0x38(%rax),就是將rip 保存到這個位置。
目前已經64字節了,棧還有存儲什麼呢,協程(fn 函數)運行完成後會退出調用ret,其實就是POP到 rip.因此保存是finish 函數指針 大小8字節。總共 72 字節。

make_fcontext 建立協程的棧。jump_fcontext實現跳轉。

網校內部培訓視頻by李樂 https://biglive.xueersi.com/L...

本站公眾號
   歡迎關注本站公眾號,獲取更多信息