Skip to content

Commit 7acb0d0

Browse files
committed
runtime: fix syscall9 on darwin/arm64
The aarch64 ABI says that only the first 8 arguments should be passed as registers, subsequent arguments should be put on the stack. Syscall9 is not putting the 9th argument on the stack, and it should. The standard library hasn't hit this issue because it uses Syscall9 for functions that only require 7 or 8 parameters. Change-Id: I1fafca5b16f977ea856e3f08b4ff3d0a2a7a4dfe Reviewed-on: https://go-review.googlesource.com/c/go/+/702297 Reviewed-by: Michael Pratt <[email protected]> Reviewed-by: Cherry Mui <[email protected]> LUCI-TryBot-Result: Go LUCI <[email protected]>
1 parent 60c1ee9 commit 7acb0d0

File tree

5 files changed

+197
-5
lines changed

5 files changed

+197
-5
lines changed

src/runtime/sys_darwin_arm64.s

Lines changed: 1 addition & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -724,13 +724,9 @@ TEXT runtime·syscall9(SB),NOSPLIT,$0
724724
MOVD 56(R0), R6 // a7
725725
MOVD 64(R0), R7 // a8
726726
MOVD 72(R0), R8 // a9
727+
MOVD R8, 0(RSP) // the 9th arg and onwards must be passed on the stack
727728
MOVD 8(R0), R0 // a1
728729

729-
// If fn is declared as vararg, we have to pass the vararg arguments on the stack.
730-
// See syscall above. The only function this applies to is openat, for which the 4th
731-
// arg must be on the stack.
732-
MOVD R3, (RSP)
733-
734730
BL (R12)
735731

736732
MOVD 8(RSP), R2 // pop structure pointer

src/runtime/syscall_test.go

Lines changed: 28 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
// Copyright 2025 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package runtime_test
6+
7+
import (
8+
"internal/testenv"
9+
"runtime"
10+
"testing"
11+
)
12+
13+
func TestSyscallArgs(t *testing.T) {
14+
if runtime.GOOS != "darwin" {
15+
t.Skipf("skipping test: GOARCH=%s", runtime.GOARCH)
16+
}
17+
testenv.MustHaveCGO(t)
18+
19+
exe, err := buildTestProg(t, "testsyscall")
20+
if err != nil {
21+
t.Fatal(err)
22+
}
23+
24+
cmd := testenv.Command(t, exe)
25+
if out, err := cmd.CombinedOutput(); err != nil {
26+
t.Fatalf("test program failed: %v\n%s", err, out)
27+
}
28+
}
Lines changed: 65 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
// Copyright 2025 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package main
6+
7+
import (
8+
_ "runtime/testdata/testsyscall/testsyscallc" // unfortunately, we can't put C and assembly in the package
9+
_ "unsafe" // for go:linkname
10+
)
11+
12+
//go:linkname syscall_syscall syscall.syscall
13+
func syscall_syscall(fn, a1, a2, a3 uintptr) (r1, r2, err uintptr)
14+
15+
//go:linkname syscall_syscall6 syscall.syscall6
16+
func syscall_syscall6(fn, a1, a2, a3, a4, a5, a6 uintptr) (r1, r2, err uintptr)
17+
18+
//go:linkname syscall_syscall9 syscall.syscall9
19+
func syscall_syscall9(fn, a1, a2, a3, a4, a5, a6, a7, a8, a9 uintptr) (r1, r2, err uintptr)
20+
21+
var (
22+
syscall_check0_trampoline_addr uintptr
23+
syscall_check1_trampoline_addr uintptr
24+
syscall_check2_trampoline_addr uintptr
25+
syscall_check3_trampoline_addr uintptr
26+
syscall_check4_trampoline_addr uintptr
27+
syscall_check5_trampoline_addr uintptr
28+
syscall_check6_trampoline_addr uintptr
29+
syscall_check7_trampoline_addr uintptr
30+
syscall_check8_trampoline_addr uintptr
31+
syscall_check9_trampoline_addr uintptr
32+
)
33+
34+
func main() {
35+
if ret, _, _ := syscall_syscall(syscall_check0_trampoline_addr, 0, 0, 0); ret != 1 {
36+
panic("hello0")
37+
}
38+
if ret, _, _ := syscall_syscall(syscall_check1_trampoline_addr, 1, 0, 0); ret != 1 {
39+
panic("hello1")
40+
}
41+
if ret, _, _ := syscall_syscall(syscall_check2_trampoline_addr, 1, 2, 0); ret != 1 {
42+
panic("hello2")
43+
}
44+
if ret, _, _ := syscall_syscall(syscall_check3_trampoline_addr, 1, 2, 3); ret != 1 {
45+
panic("hello3")
46+
}
47+
if ret, _, _ := syscall_syscall6(syscall_check4_trampoline_addr, 1, 2, 3, 4, 0, 0); ret != 1 {
48+
panic("hello4")
49+
}
50+
if ret, _, _ := syscall_syscall6(syscall_check5_trampoline_addr, 1, 2, 3, 4, 5, 0); ret != 1 {
51+
panic("hello5")
52+
}
53+
if ret, _, _ := syscall_syscall6(syscall_check6_trampoline_addr, 1, 2, 3, 4, 5, 6); ret != 1 {
54+
panic("hello6")
55+
}
56+
if ret, _, _ := syscall_syscall9(syscall_check7_trampoline_addr, 1, 2, 3, 4, 5, 6, 7, 0, 0); ret != 1 {
57+
panic("hello7")
58+
}
59+
if ret, _, _ := syscall_syscall9(syscall_check8_trampoline_addr, 1, 2, 3, 4, 5, 6, 7, 8, 0); ret != 1 {
60+
panic("hello8")
61+
}
62+
if ret, _, _ := syscall_syscall9(syscall_check9_trampoline_addr, 1, 2, 3, 4, 5, 6, 7, 8, 9); ret != 1 {
63+
panic("hello9")
64+
}
65+
}
Lines changed: 55 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
// Copyright 2025 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
#include "textflag.h"
6+
7+
TEXT syscall_check0_trampoline<>(SB),NOSPLIT,$0-0
8+
JMP syscall_check0(SB)
9+
GLOBL ·syscall_check0_trampoline_addr(SB), RODATA, $8
10+
DATA ·syscall_check0_trampoline_addr(SB)/8, $syscall_check0_trampoline<>(SB)
11+
12+
TEXT syscall_check1_trampoline<>(SB),NOSPLIT,$0-0
13+
JMP syscall_check1(SB)
14+
GLOBL ·syscall_check1_trampoline_addr(SB), RODATA, $8
15+
DATA ·syscall_check1_trampoline_addr(SB)/8, $syscall_check1_trampoline<>(SB)
16+
17+
TEXT syscall_check2_trampoline<>(SB),NOSPLIT,$0-0
18+
JMP syscall_check2(SB)
19+
GLOBL ·syscall_check2_trampoline_addr(SB), RODATA, $8
20+
DATA ·syscall_check2_trampoline_addr(SB)/8, $syscall_check2_trampoline<>(SB)
21+
22+
TEXT syscall_check3_trampoline<>(SB),NOSPLIT,$0-0
23+
JMP syscall_check3(SB)
24+
GLOBL ·syscall_check3_trampoline_addr(SB), RODATA, $8
25+
DATA ·syscall_check3_trampoline_addr(SB)/8, $syscall_check3_trampoline<>(SB)
26+
27+
TEXT syscall_check4_trampoline<>(SB),NOSPLIT,$0-0
28+
JMP syscall_check4(SB)
29+
GLOBL ·syscall_check4_trampoline_addr(SB), RODATA, $8
30+
DATA ·syscall_check4_trampoline_addr(SB)/8, $syscall_check4_trampoline<>(SB)
31+
32+
TEXT syscall_check5_trampoline<>(SB),NOSPLIT,$0-0
33+
JMP syscall_check5(SB)
34+
GLOBL ·syscall_check5_trampoline_addr(SB), RODATA, $8
35+
DATA ·syscall_check5_trampoline_addr(SB)/8, $syscall_check5_trampoline<>(SB)
36+
37+
TEXT syscall_check6_trampoline<>(SB),NOSPLIT,$0-0
38+
JMP syscall_check6(SB)
39+
GLOBL ·syscall_check6_trampoline_addr(SB), RODATA, $8
40+
DATA ·syscall_check6_trampoline_addr(SB)/8, $syscall_check6_trampoline<>(SB)
41+
42+
TEXT syscall_check7_trampoline<>(SB),NOSPLIT,$0-0
43+
JMP syscall_check7(SB)
44+
GLOBL ·syscall_check7_trampoline_addr(SB), RODATA, $8
45+
DATA ·syscall_check7_trampoline_addr(SB)/8, $syscall_check7_trampoline<>(SB)
46+
47+
TEXT syscall_check8_trampoline<>(SB),NOSPLIT,$0-0
48+
JMP syscall_check8(SB)
49+
GLOBL ·syscall_check8_trampoline_addr(SB), RODATA, $8
50+
DATA ·syscall_check8_trampoline_addr(SB)/8, $syscall_check8_trampoline<>(SB)
51+
52+
TEXT syscall_check9_trampoline<>(SB),NOSPLIT,$0-0
53+
JMP syscall_check9(SB)
54+
GLOBL ·syscall_check9_trampoline_addr(SB), RODATA, $8
55+
DATA ·syscall_check9_trampoline_addr(SB)/8, $syscall_check9_trampoline<>(SB)
Lines changed: 48 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
// Copyright 2025 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package testsyscallc
6+
7+
/*
8+
int syscall_check0(void) {
9+
return 1;
10+
}
11+
12+
int syscall_check1(int a1) {
13+
return a1 == 1;
14+
}
15+
16+
int syscall_check2(int a1, int a2) {
17+
return a1 == 1 && a2 == 2;
18+
}
19+
20+
int syscall_check3(int a1, int a2, int a3) {
21+
return a1 == 1 && a2 == 2 && a3 == 3;
22+
}
23+
24+
int syscall_check4(int a1, int a2, int a3, int a4) {
25+
return a1 == 1 && a2 == 2 && a3 == 3 && a4 == 4;
26+
}
27+
28+
int syscall_check5(int a1, int a2, int a3, int a4, int a5) {
29+
return a1 == 1 && a2 == 2 && a3 == 3 && a4 == 4 && a5 == 5;
30+
}
31+
32+
int syscall_check6(int a1, int a2, int a3, int a4, int a5, int a6) {
33+
return a1 == 1 && a2 == 2 && a3 == 3 && a4 == 4 && a5 == 5 && a6 == 6;
34+
}
35+
36+
int syscall_check7(int a1, int a2, int a3, int a4, int a5, int a6, int a7) {
37+
return a1 == 1 && a2 == 2 && a3 == 3 && a4 == 4 && a5 == 5 && a6 == 6 && a7 == 7;
38+
}
39+
40+
int syscall_check8(int a1, int a2, int a3, int a4, int a5, int a6, int a7, int a8) {
41+
return a1 == 1 && a2 == 2 && a3 == 3 && a4 == 4 && a5 == 5 && a6 == 6 && a7 == 7 && a8 == 8;
42+
}
43+
44+
int syscall_check9(int a1, int a2, int a3, int a4, int a5, int a6, int a7, int a8, int a9) {
45+
return a1 == 1 && a2 == 2 && a3 == 3 && a4 == 4 && a5 == 5 && a6 == 6 && a7 == 7 && a8 == 8 && a9 == 9;
46+
}
47+
*/
48+
import "C"

0 commit comments

Comments
 (0)