| // Copyright 2016 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| #include "go_asm.h" |
| #include "textflag.h" |
| #include "asm_ppc64x.h" |
| #include "cgo/abi_ppc64x.h" |
| |
| TEXT _rt0_ppc64le_linux(SB),NOSPLIT,$0 |
| XOR R0, R0 // Make sure R0 is zero before _main |
| BR _main<>(SB) |
| |
| TEXT _rt0_ppc64le_linux_lib(SB),NOSPLIT|NOFRAME,$0 |
| // This is called with ELFv2 calling conventions. Convert to Go. |
| // Allocate argument storage for call to newosproc0. |
| STACK_AND_SAVE_HOST_TO_GO_ABI(16) |
| |
| MOVD R3, _rt0_ppc64le_linux_lib_argc<>(SB) |
| MOVD R4, _rt0_ppc64le_linux_lib_argv<>(SB) |
| |
| // Synchronous initialization. |
| MOVD $runtime·libpreinit(SB), R12 |
| MOVD R12, CTR |
| BL (CTR) |
| |
| // Create a new thread to do the runtime initialization and return. |
| MOVD _cgo_sys_thread_create(SB), R12 |
| CMP $0, R12 |
| BEQ nocgo |
| MOVD $_rt0_ppc64le_linux_lib_go(SB), R3 |
| MOVD $0, R4 |
| MOVD R12, CTR |
| BL (CTR) |
| BR done |
| |
| nocgo: |
| MOVD $0x800000, R12 // stacksize = 8192KB |
| MOVD R12, 8+FIXED_FRAME(R1) |
| MOVD $_rt0_ppc64le_linux_lib_go(SB), R12 |
| MOVD R12, 16+FIXED_FRAME(R1) |
| MOVD $runtime·newosproc0(SB),R12 |
| MOVD R12, CTR |
| BL (CTR) |
| |
| done: |
| // Restore and return to ELFv2 caller. |
| UNSTACK_AND_RESTORE_GO_TO_HOST_ABI(16) |
| RET |
| |
| TEXT _rt0_ppc64le_linux_lib_go(SB),NOSPLIT,$0 |
| MOVD _rt0_ppc64le_linux_lib_argc<>(SB), R3 |
| MOVD _rt0_ppc64le_linux_lib_argv<>(SB), R4 |
| MOVD $runtime·rt0_go(SB), R12 |
| MOVD R12, CTR |
| BR (CTR) |
| |
| DATA _rt0_ppc64le_linux_lib_argc<>(SB)/8, $0 |
| GLOBL _rt0_ppc64le_linux_lib_argc<>(SB),NOPTR, $8 |
| DATA _rt0_ppc64le_linux_lib_argv<>(SB)/8, $0 |
| GLOBL _rt0_ppc64le_linux_lib_argv<>(SB),NOPTR, $8 |
| |
| TEXT _main<>(SB),NOSPLIT,$-8 |
| // In a statically linked binary, the stack contains argc, |
| // argv as argc string pointers followed by a NULL, envv as a |
| // sequence of string pointers followed by a NULL, and auxv. |
| // The TLS pointer should be initialized to 0. |
| // |
| // In an ELFv2 compliant dynamically linked binary, R3 contains argc, |
| // R4 contains argv, R5 contains envp, R6 contains auxv, and R13 |
| // contains the TLS pointer. |
| // |
| // When loading via glibc, the first doubleword on the stack points |
| // to NULL a value. (that is *(uintptr)(R1) == 0). This is used to |
| // differentiate static vs dynamically linked binaries. |
| // |
| // If loading with the musl loader, it doesn't follow the ELFv2 ABI. It |
| // passes argc/argv similar to the linux kernel, R13 (TLS) is |
| // initialized, and R3/R4 are undefined. |
| MOVD (R1), R12 |
| CMP R0, R12 |
| BEQ tls_and_argcv_in_reg |
| |
| // Arguments are passed via the stack (musl loader or a static binary) |
| MOVD 0(R1), R3 // argc |
| ADD $8, R1, R4 // argv |
| |
| // Did the TLS pointer get set? If so, don't change it (e.g musl). |
| CMP R0, R13 |
| BNE tls_and_argcv_in_reg |
| |
| MOVD $runtime·m0+m_tls(SB), R13 // TLS |
| ADD $0x7000, R13 |
| |
| tls_and_argcv_in_reg: |
| BR main(SB) |
| |
| TEXT main(SB),NOSPLIT,$-8 |
| MOVD $runtime·rt0_go(SB), R12 |
| MOVD R12, CTR |
| BR (CTR) |