// Copyright 2019 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. // // System calls and other sys.stuff for arm64, NetBSD // #include "go_asm.h" #include "go_tls.h" #include "textflag.h" #define CLOCK_REALTIME 0 #define CLOCK_MONOTONIC 3 #define FD_CLOEXEC 1 #define F_SETFD 2 #define F_GETFL 3 #define F_SETFL 4 #define O_NONBLOCK 4 #define SYS_exit 1 #define SYS_read 3 #define SYS_write 4 #define SYS_open 5 #define SYS_close 6 #define SYS_getpid 20 #define SYS_kill 37 #define SYS_munmap 73 #define SYS_madvise 75 #define SYS_fcntl 92 #define SYS_mmap 197 #define SYS___sysctl 202 #define SYS___sigaltstack14 281 #define SYS___sigprocmask14 293 #define SYS_getcontext 307 #define SYS_setcontext 308 #define SYS__lwp_create 309 #define SYS__lwp_exit 310 #define SYS__lwp_self 311 #define SYS__lwp_kill 318 #define SYS__lwp_unpark 321 #define SYS___sigaction_sigtramp 340 #define SYS_kqueue 344 #define SYS_sched_yield 350 #define SYS___setitimer50 425 #define SYS___clock_gettime50 427 #define SYS___nanosleep50 430 #define SYS___kevent50 435 #define SYS_pipe2 453 #define SYS_openat 468 #define SYS____lwp_park60 478 // int32 lwp_create(void *context, uintptr flags, void *lwpid) TEXT runtime·lwp_create(SB),NOSPLIT,$0 MOVD ctxt+0(FP), R0 MOVD flags+8(FP), R1 MOVD lwpid+16(FP), R2 SVC $SYS__lwp_create BCC ok NEG R0, R0 ok: MOVW R0, ret+24(FP) RET TEXT runtime·lwp_tramp(SB),NOSPLIT,$0 CMP $0, R1 BEQ nog CMP $0, R2 BEQ nog MOVD R0, g_m(R1) MOVD R1, g nog: CALL (R2) MOVD $0, R0 // crash (not reached) MOVD R0, (R8) TEXT ·netbsdMstart(SB),NOSPLIT|TOPFRAME,$0 CALL ·netbsdMstart0(SB) RET // not reached TEXT runtime·osyield(SB),NOSPLIT,$0 SVC $SYS_sched_yield RET TEXT runtime·lwp_park(SB),NOSPLIT,$0 MOVW clockid+0(FP), R0 // arg 1 - clockid MOVW flags+4(FP), R1 // arg 2 - flags MOVD ts+8(FP), R2 // arg 3 - ts MOVW unpark+16(FP), R3 // arg 4 - unpark MOVD hint+24(FP), R4 // arg 5 - hint MOVD unparkhint+32(FP), R5 // arg 6 - unparkhint SVC $SYS____lwp_park60 MOVW R0, ret+40(FP) RET TEXT runtime·lwp_unpark(SB),NOSPLIT,$0 MOVW lwp+0(FP), R0 // arg 1 - lwp MOVD hint+8(FP), R1 // arg 2 - hint SVC $SYS__lwp_unpark MOVW R0, ret+16(FP) RET TEXT runtime·lwp_self(SB),NOSPLIT,$0 SVC $SYS__lwp_self MOVW R0, ret+0(FP) RET // Exit the entire program (like C exit) TEXT runtime·exit(SB),NOSPLIT,$-8 MOVW code+0(FP), R0 // arg 1 - exit status SVC $SYS_exit MOVD $0, R0 // If we're still running, MOVD R0, (R0) // crash // func exitThread(wait *uint32) TEXT runtime·exitThread(SB),NOSPLIT,$0-8 MOVD wait+0(FP), R0 // We're done using the stack. MOVW $0, R1 STLRW R1, (R0) SVC $SYS__lwp_exit JMP 0(PC) TEXT runtime·open(SB),NOSPLIT|NOFRAME,$-8 MOVD name+0(FP), R0 // arg 1 - pathname MOVW mode+8(FP), R1 // arg 2 - flags MOVW perm+12(FP), R2 // arg 3 - mode SVC $SYS_open BCC ok MOVW $-1, R0 ok: MOVW R0, ret+16(FP) RET TEXT runtime·closefd(SB),NOSPLIT,$-8 MOVW fd+0(FP), R0 // arg 1 - fd SVC $SYS_close BCC ok MOVW $-1, R0 ok: MOVW R0, ret+8(FP) RET TEXT runtime·read(SB),NOSPLIT|NOFRAME,$0 MOVW fd+0(FP), R0 // arg 1 - fd MOVD p+8(FP), R1 // arg 2 - buf MOVW n+16(FP), R2 // arg 3 - count SVC $SYS_read BCC ok NEG R0, R0 ok: MOVW R0, ret+24(FP) RET // func pipe() (r, w int32, errno int32) TEXT runtime·pipe(SB),NOSPLIT|NOFRAME,$0-12 ADD $8, RSP, R0 MOVW $0, R1 SVC $SYS_pipe2 BCC pipeok NEG R0, R0 pipeok: MOVW R0, errno+8(FP) RET // func pipe2(flags int32) (r, w int32, errno int32) TEXT runtime·pipe2(SB),NOSPLIT|NOFRAME,$0-20 ADD $16, RSP, R0 MOVW flags+0(FP), R1 SVC $SYS_pipe2 BCC pipe2ok NEG R0, R0 pipe2ok: MOVW R0, errno+16(FP) RET TEXT runtime·write1(SB),NOSPLIT,$-8 MOVD fd+0(FP), R0 // arg 1 - fd MOVD p+8(FP), R1 // arg 2 - buf MOVW n+16(FP), R2 // arg 3 - nbyte SVC $SYS_write BCC ok NEG R0, R0 ok: MOVW R0, ret+24(FP) RET TEXT runtime·usleep(SB),NOSPLIT,$24-4 MOVWU usec+0(FP), R3 MOVD R3, R5 MOVW $1000000, R4 UDIV R4, R3 MOVD R3, 8(RSP) // sec MUL R3, R4 SUB R4, R5 MOVW $1000, R4 MUL R4, R5 MOVD R5, 16(RSP) // nsec MOVD $8(RSP), R0 // arg 1 - rqtp MOVD $0, R1 // arg 2 - rmtp SVC $SYS___nanosleep50 RET TEXT runtime·lwp_kill(SB),NOSPLIT,$0-16 MOVW tid+0(FP), R0 // arg 1 - target MOVD sig+8(FP), R1 // arg 2 - signo SVC $SYS__lwp_kill RET TEXT runtime·raiseproc(SB),NOSPLIT,$16 SVC $SYS_getpid // arg 1 - pid (from getpid) MOVD sig+0(FP), R1 // arg 2 - signo SVC $SYS_kill RET TEXT runtime·setitimer(SB),NOSPLIT,$-8 MOVW mode+0(FP), R0 // arg 1 - which MOVD new+8(FP), R1 // arg 2 - itv MOVD old+16(FP), R2 // arg 3 - oitv SVC $SYS___setitimer50 RET // func walltime() (sec int64, nsec int32) TEXT runtime·walltime(SB), NOSPLIT, $32 MOVW $CLOCK_REALTIME, R0 // arg 1 - clock_id MOVD $8(RSP), R1 // arg 2 - tp SVC $SYS___clock_gettime50 MOVD 8(RSP), R0 // sec MOVD 16(RSP), R1 // nsec // sec is in R0, nsec in R1 MOVD R0, sec+0(FP) MOVW R1, nsec+8(FP) RET // int64 nanotime1(void) so really // void nanotime1(int64 *nsec) TEXT runtime·nanotime1(SB), NOSPLIT, $32 MOVD $CLOCK_MONOTONIC, R0 // arg 1 - clock_id MOVD $8(RSP), R1 // arg 2 - tp SVC $SYS___clock_gettime50 MOVD 8(RSP), R0 // sec MOVD 16(RSP), R2 // nsec // sec is in R0, nsec in R2 // return nsec in R2 MOVD $1000000000, R3 MUL R3, R0 ADD R2, R0 MOVD R0, ret+0(FP) RET TEXT runtime·getcontext(SB),NOSPLIT,$-8 MOVD ctxt+0(FP), R0 // arg 1 - context SVC $SYS_getcontext BCS fail RET fail: MOVD $0, R0 MOVD R0, (R0) // crash TEXT runtime·sigprocmask(SB),NOSPLIT,$0 MOVW how+0(FP), R0 // arg 1 - how MOVD new+8(FP), R1 // arg 2 - set MOVD old+16(FP), R2 // arg 3 - oset SVC $SYS___sigprocmask14 BCS fail RET fail: MOVD $0, R0 MOVD R0, (R0) // crash TEXT sigreturn_tramp<>(SB),NOSPLIT,$-8 MOVD g, R0 SVC $SYS_setcontext MOVD $0x4242, R0 // Something failed, return magic number SVC $SYS_exit TEXT runtime·sigaction(SB),NOSPLIT,$-8 MOVW sig+0(FP), R0 // arg 1 - signum MOVD new+8(FP), R1 // arg 2 - nsa MOVD old+16(FP), R2 // arg 3 - osa // arg 4 - tramp MOVD $sigreturn_tramp<>(SB), R3 MOVW $2, R4 // arg 5 - vers SVC $SYS___sigaction_sigtramp BCS fail RET fail: MOVD $0, R0 MOVD R0, (R0) // crash // XXX ??? TEXT runtime·sigfwd(SB),NOSPLIT,$0-32 MOVW sig+8(FP), R0 MOVD info+16(FP), R1 MOVD ctx+24(FP), R2 MOVD fn+0(FP), R11 BL (R11) RET TEXT runtime·sigtramp(SB),NOSPLIT,$192 // Save callee-save registers in the case of signal forwarding. // Please refer to https://golang.org/issue/31827 . MOVD R19, 8*4(RSP) MOVD R20, 8*5(RSP) MOVD R21, 8*6(RSP) MOVD R22, 8*7(RSP) MOVD R23, 8*8(RSP) MOVD R24, 8*9(RSP) MOVD R25, 8*10(RSP) MOVD R26, 8*11(RSP) MOVD R27, 8*12(RSP) MOVD g, 8*13(RSP) // Unclobber g for now (kernel uses it as ucontext ptr) // See https://github.com/golang/go/issues/30824#issuecomment-492772426 // This is only correct in the non-cgo case. // XXX should use lwp_getprivate as suggested. // 8*36 is ucontext.uc_mcontext.__gregs[_REG_X28] MOVD 8*36(g), g MOVD R29, 8*14(RSP) FMOVD F8, 8*15(RSP) FMOVD F9, 8*16(RSP) FMOVD F10, 8*17(RSP) FMOVD F11, 8*18(RSP) FMOVD F12, 8*19(RSP) FMOVD F13, 8*20(RSP) FMOVD F14, 8*21(RSP) FMOVD F15, 8*22(RSP) // this might be called in external code context, // where g is not set. // first save R0, because runtime·load_g will clobber it MOVD R0, 8(RSP) // signum MOVB runtime·iscgo(SB), R0 CMP $0, R0 // XXX branch destination BEQ 2(PC) BL runtime·load_g(SB) MOVD R1, 16(RSP) MOVD R2, 24(RSP) BL runtime·sigtrampgo(SB) // Restore callee-save registers. MOVD 8*4(RSP), R19 MOVD 8*5(RSP), R20 MOVD 8*6(RSP), R21 MOVD 8*7(RSP), R22 MOVD 8*8(RSP), R23 MOVD 8*9(RSP), R24 MOVD 8*10(RSP), R25 MOVD 8*11(RSP), R26 MOVD 8*12(RSP), R27 MOVD 8*13(RSP), g MOVD 8*14(RSP), R29 FMOVD 8*15(RSP), F8 FMOVD 8*16(RSP), F9 FMOVD 8*17(RSP), F10 FMOVD 8*18(RSP), F11 FMOVD 8*19(RSP), F12 FMOVD 8*20(RSP), F13 FMOVD 8*21(RSP), F14 FMOVD 8*22(RSP), F15 RET TEXT runtime·mmap(SB),NOSPLIT,$0 MOVD addr+0(FP), R0 // arg 1 - addr MOVD n+8(FP), R1 // arg 2 - len MOVW prot+16(FP), R2 // arg 3 - prot MOVW flags+20(FP), R3 // arg 4 - flags MOVW fd+24(FP), R4 // arg 5 - fd MOVW $0, R5 // arg 6 - pad MOVD off+28(FP), R6 // arg 7 - offset SVC $SYS_mmap BCS fail MOVD R0, p+32(FP) MOVD $0, err+40(FP) RET fail: MOVD $0, p+32(FP) MOVD R0, err+40(FP) RET TEXT runtime·munmap(SB),NOSPLIT,$0 MOVD addr+0(FP), R0 // arg 1 - addr MOVD n+8(FP), R1 // arg 2 - len SVC $SYS_munmap BCS fail RET fail: MOVD $0, R0 MOVD R0, (R0) // crash TEXT runtime·madvise(SB),NOSPLIT,$0 MOVD addr+0(FP), R0 // arg 1 - addr MOVD n+8(FP), R1 // arg 2 - len MOVW flags+16(FP), R2 // arg 3 - behav SVC $SYS_madvise BCC ok MOVD $-1, R0 ok: MOVD R0, ret+24(FP) RET TEXT runtime·sigaltstack(SB),NOSPLIT,$0 MOVD new+0(FP), R0 // arg 1 - nss MOVD old+8(FP), R1 // arg 2 - oss SVC $SYS___sigaltstack14 BCS fail RET fail: MOVD $0, R0 MOVD R0, (R0) // crash TEXT runtime·sysctl(SB),NOSPLIT,$0 MOVD mib+0(FP), R0 // arg 1 - name MOVW miblen+8(FP), R1 // arg 2 - namelen MOVD out+16(FP), R2 // arg 3 - oldp MOVD size+24(FP), R3 // arg 4 - oldlenp MOVD dst+32(FP), R4 // arg 5 - newp MOVD ndst+40(FP), R5 // arg 6 - newlen SVC $SYS___sysctl BCC ok NEG R0, R0 ok: MOVW R0, ret+48(FP) RET // int32 runtime·kqueue(void) TEXT runtime·kqueue(SB),NOSPLIT,$0 MOVD $0, R0 SVC $SYS_kqueue BCC ok NEG R0, R0 ok: MOVW R0, ret+0(FP) RET // int32 runtime·kevent(int kq, Kevent *changelist, int nchanges, Kevent *eventlist, int nevents, Timespec *timeout) TEXT runtime·kevent(SB),NOSPLIT,$0 MOVW kq+0(FP), R0 // arg 1 - kq MOVD ch+8(FP), R1 // arg 2 - changelist MOVW nch+16(FP), R2 // arg 3 - nchanges MOVD ev+24(FP), R3 // arg 4 - eventlist MOVW nev+32(FP), R4 // arg 5 - nevents MOVD ts+40(FP), R5 // arg 6 - timeout SVC $SYS___kevent50 BCC ok NEG R0, R0 ok: MOVW R0, ret+48(FP) RET // void runtime·closeonexec(int32 fd) TEXT runtime·closeonexec(SB),NOSPLIT,$0 MOVW fd+0(FP), R0 // arg 1 - fd MOVW $F_SETFD, R1 MOVW $FD_CLOEXEC, R2 SVC $SYS_fcntl RET // func runtime·setNonblock(int32 fd) TEXT runtime·setNonblock(SB),NOSPLIT|NOFRAME,$0-4 MOVW fd+0(FP), R0 // arg 1 - fd MOVD $F_GETFL, R1 // arg 2 - cmd MOVD $0, R2 // arg 3 SVC $SYS_fcntl MOVD $O_NONBLOCK, R2 EOR R0, R2 // arg 3 - flags MOVW fd+0(FP), R0 // arg 1 - fd MOVD $F_SETFL, R1 // arg 2 - cmd SVC $SYS_fcntl RET