/* i386 support code for fibers and multithreading. Copyright (C) 2019-2022 Free Software Foundation, Inc. This file is part of GCC. GCC is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. GCC is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. Under Section 7 of GPL version 3, you are granted additional permissions described in the GCC Runtime Library Exception, version 3.1, as published by the Free Software Foundation. You should have received a copy of the GNU General Public License and a copy of the GCC Runtime Library Exception along with this program; see the files COPYING3 and COPYING.RUNTIME respectively. If not, see . */ #include "../common/threadasm.S" /* NB: Generate the CET marker for -fcf-protection. */ #ifdef __CET__ # include #endif #if !defined(__CET__) # if defined(__ELF__) # if defined(__i386__) .text .globl CSYM(fiber_switchContext) .type CSYM(fiber_switchContext), @function .align 16 CSYM(fiber_switchContext): .cfi_startproc // save current stack state push %ebp mov %esp, %ebp push %edi push %esi push %ebx push %eax // store oldp again with more accurate address mov 8(%ebp), %eax mov %esp, (%eax) // load newp to begin context switch mov 12(%ebp), %esp // load saved state from new stack pop %eax pop %ebx pop %esi pop %edi pop %ebp // 'return' to complete switch ret .cfi_endproc .size CSYM(fiber_switchContext),.-CSYM(fiber_switchContext) # endif /* defined(__ELF__) && defined(__i386__) */ # if defined(__x86_64__) && !defined(__ILP32__) .text .globl CSYM(fiber_switchContext) .type CSYM(fiber_switchContext), @function .align 16 CSYM(fiber_switchContext): .cfi_startproc // Save current stack state.save current stack state push %rbp mov %rsp, %rbp push %rbx push %r12 push %r13 push %r14 push %r15 // store oldp again with more accurate address mov %rsp, (%rdi) // load newp to begin context switch mov %rsi, %rsp // load saved state from new stack pop %r15 pop %r14 pop %r13 pop %r12 pop %rbx pop %rbp // 'return' to complete switch ret .cfi_endproc .size CSYM(fiber_switchContext),.-CSYM(fiber_switchContext) # endif /* defined(__ELF__) && defined(__x86_64__) && !defined(__ILP32__) */ # endif /* defined(__ELF__) */ # if defined(__MACH__) # if defined(__i386__) .text .globl CSYM(fiber_switchContext) .p2align 4 CSYM(fiber_switchContext): LFB0: // save current stack state push %ebp mov %esp, %ebp push %edi push %esi push %ebx push %eax // store oldp again with more accurate address mov 8(%ebp), %eax mov %esp, (%eax) // load newp to begin context switch mov 12(%ebp), %esp // load saved state from new stack pop %eax pop %ebx pop %esi pop %edi pop %ebp // 'return' to complete switch ret LFE0: /* CFI */ .section __TEXT,__eh_frame,coalesced,no_toc+strip_static_syms+live_support EH_frame1: .set L$set$0,LECIE1-LSCIE1 .long L$set$0 # Length of Common Information Entry LSCIE1: .long 0 # CIE Identifier Tag .byte 0x1 # CIE Version .ascii "zR\0" # CIE Augmentation .byte 0x1 # uleb128 0x1; CIE Code Alignment Factor .byte 0x7c # sleb128 -4; CIE Data Alignment Factor .byte 0x8 # CIE RA Column .byte 0x1 # uleb128 0x1; Augmentation size .byte 0x10 # FDE Encoding (pcrel) .byte 0xc # DW_CFA_def_cfa .byte 0x5 # uleb128 0x5 .byte 0x4 # uleb128 0x4 .byte 0x88 # DW_CFA_offset, column 0x8 .byte 0x1 # uleb128 0x1 .p2align 2,0 LECIE1: /* minimal FDE - does not record the stack frame changes. */ LSFDE1: .set L$set$1,LEFDE1-LASFDE1 .long L$set$1 # FDE Length LASFDE1: .long LASFDE1-EH_frame1 # FDE CIE offset .long LFB0-. # FDE initial location .set L$set$2,LFE0-LFB0 .long L$set$2 # FDE address range .byte 0 # uleb128 0; Augmentation size .p2align 2,0 LEFDE1: # endif /* defined(__MACH__) && defined(__i386__) */ # if defined(__x86_64__) && !defined(__ILP32__) .text .globl CSYM(fiber_switchContext) .p2align 4 CSYM(fiber_switchContext): LFB0: // Save current stack state.save current stack state push %rbp mov %rsp, %rbp push %r15 push %r14 push %r13 push %r12 push %rbx // store oldp again with more accurate address mov %rsp, (%rdi) // load newp to begin context switch mov %rsi, %rsp // load saved state from new stack pop %rbx pop %r12 pop %r13 pop %r14 pop %r15 pop %rbp // 'return' to complete switch ret LFE0: /* CFI */ .section __TEXT,__eh_frame,coalesced,no_toc+strip_static_syms+live_support EH_frame1: .set L$set$0,LECIE1-LSCIE1 .long L$set$0 # Length of Common Information Entry LSCIE1: .long 0 # CIE Identifier Tag .byte 0x1 # CIE Version .ascii "zR\0" # CIE Augmentation .byte 0x1 # uleb128 0x1; CIE Code Alignment Factor .byte 0x78 # sleb128 -8; CIE Data Alignment Factor .byte 0x10 # CIE RA Column .byte 0x1 # uleb128 0x1; Augmentation size .byte 0x10 # FDE Encoding (pcrel) .byte 0xc # DW_CFA_def_cfa .byte 0x7 # uleb128 0x7 .byte 0x8 # uleb128 0x8 .byte 0x90 # DW_CFA_offset, column 0x10 .byte 0x1 # uleb128 0x1 .p2align 3,0 LECIE1: /* minimal FDE - does not record the stack frame changes. */ LSFDE1: .set L$set$1,LEFDE1-LASFDE1 .long L$set$1 # FDE Length LASFDE1: .long LASFDE1-EH_frame1 # FDE CIE offset .quad LFB0-. # FDE initial location .set L$set$2,LFE0-LFB0 .quad L$set$2 # FDE address range .byte 0 # uleb128 0; Augmentation size .p2align 3,0 LEFDE1: # endif /* defined(__MACH__) && defined(__x86_64__) && !defined(__ILP32__) */ # endif /* defined (__MACH__) */ #endif /* !defined(__CET__) */