]> Cypherpunks repositories - gostls13.git/commitdiff
syscall: implement rawVforkSyscall for linux/arm64
authorHoward Zhang <howard.zhang@arm.com>
Tue, 6 Aug 2019 09:42:37 +0000 (02:42 -0700)
committerTobias Klauser <tobias.klauser@gmail.com>
Wed, 11 Sep 2019 07:19:55 +0000 (07:19 +0000)
This allows the use of CLONE_VFORK and CLONE_VM for fork/exec, preventing
"fork/exec ...: cannot allocate memory" failures from occuring when attempting
to execute commands from a Go process that has a large memory footprint.
Additionally, this should reduce the latency of fork/exec on linux/arm64.

With CLONE_VM the child process shares the same memory with the parent
process. On its own this would lead to conflicting use of the same
memory, so CLONE_VFORK is used to suspend the parent process until the
child releases the memory when switching to the new program binary
via the exec syscall. When the parent process continues to run, one
has to consider the changes to memory that the child process did,
namely the return address of the syscall function needs to be restored
from a register.

exec.Command() callers can start in a faster manner, as child process who
do exec commands job can be cloned faster via vfork than via fork on arm64.

The same problem was addressed on linux/amd64 via issue #5838.

Updates #31936
Contributed by Howard Zhang <howard.zhang@arm.com> and Bin Lu <bin.lu@arm.com>

Change-Id: Ia99d81d877f564ec60d19f17e596276836576eaf
Reviewed-on: https://go-review.googlesource.com/c/go/+/189418
Run-TryBot: Tobias Klauser <tobias.klauser@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Tobias Klauser <tobias.klauser@gmail.com>
Reviewed-by: Cherry Zhang <cherryyz@google.com>
src/syscall/asm_linux_arm64.s
src/syscall/exec_linux.go
src/syscall/syscall_linux_arm64.go

index 7edeafca81ddbc871d9c1c73121973570d12456f..fb22f8d547d78e6e30ef771982a22069480af40e 100644 (file)
@@ -103,6 +103,29 @@ ok:
        MOVD    ZR, err+72(FP)  // errno
        RET
 
+// func rawVforkSyscall(trap, a1 uintptr) (r1, err uintptr)
+TEXT ·rawVforkSyscall(SB),NOSPLIT,$0-32
+       MOVD    a1+8(FP), R0
+       MOVD    $0, R1
+       MOVD    $0, R2
+       MOVD    $0, R3
+       MOVD    $0, R4
+       MOVD    $0, R5
+       MOVD    trap+0(FP), R8  // syscall entry
+       SVC
+       CMN     $4095, R0
+       BCC     ok
+       MOVD    $-1, R4
+       MOVD    R4, r1+16(FP)   // r1
+       NEG     R0, R0
+       MOVD    R0, err+24(FP)  // errno
+       RET
+ok:
+       MOVD    R0, r1+16(FP)   // r1
+       MOVD    ZR, err+24(FP)  // errno
+       RET
+
+
 // func rawSyscallNoError(trap uintptr, a1, a2, a3 uintptr) (r1, r2 uintptr);
 TEXT ·rawSyscallNoError(SB),NOSPLIT,$0-48
        MOVD    a1+8(FP), R0
index a2242b2057cc3cc9d21865fe4282f957bb089ca5..3540d511bf96c4fe5ca0907f6569a943793d9bab 100644 (file)
@@ -196,7 +196,7 @@ func forkAndExecInChild1(argv0 *byte, argv, envv []*byte, chroot, dir *byte, att
                }
        }
 
-       hasRawVforkSyscall := runtime.GOARCH == "amd64" || runtime.GOARCH == "ppc64" || runtime.GOARCH == "s390x"
+       hasRawVforkSyscall := runtime.GOARCH == "amd64" || runtime.GOARCH == "ppc64" || runtime.GOARCH == "s390x" || runtime.GOARCH == "arm64"
 
        // About to call fork.
        // No more allocation or calls of non-assembly functions.
index 1ea48892ba1e529f391cdf6c13fca5422e89d031..95065bfe2d8011e98da93919521cefb489413753 100644 (file)
@@ -155,6 +155,4 @@ const (
        SYS_EPOLL_WAIT   = 1069
 )
 
-func rawVforkSyscall(trap, a1 uintptr) (r1 uintptr, err Errno) {
-       panic("not implemented")
-}
+func rawVforkSyscall(trap, a1 uintptr) (r1 uintptr, err Errno)