go 提供了 `syscall.syscall` 和 `syscall.syscall6` 等底层函数,允许开发者绕过标准库封装,直接通过系统调用号触发 linux 内核中的自定义系统调用,无需修改 go 源码或生成脚本。
在 Go 中调用自定义 Linux 系统调用,核心在于使用 syscall 包提供的裸系统调用接口。Go 的 syscall.Syscall 系列函数(如 Syscall, Syscall6, RawSyscall)直接封装了 syscall 指令(x86-64 下为 syscall,ARM64 下为 svc),
可传入系统调用号及最多 6 个参数,与 C 中的 syscall() 行为高度一致。
假设你的自定义系统调用号为 384(需确保该号在内核中已正确定义并启用),且其原型为:
long my_syscall(int arg1, char *arg2, size_t len);
对应 Go 调用方式如下:
package main
import (
"fmt"
"syscall"
"unsafe"
)
const (
SYS_my_syscall = 384 // 替换为实际分配的 syscall number
)
func main() {
arg1 := int32(42)
msg := []byte("hello from Go")
var msgPtr uintptr
if len(msg) > 0 {
msgPtr = uintptr(unsafe.Pointer(&msg[0]))
}
// 使用 Syscall6:sysno, a1, a2, a3, a4, a5, a6
// 前三个参数对应 arg1、msgPtr、len(msg),其余填 0
ret, _, errno := syscall.Syscall6(
SYS_my_syscall,
uintptr(arg1),
msgPtr,
uintptr(len(msg)),
0, 0, 0,
)
if errno != 0 {
fmt.Printf("System call failed: %v\n", errno)
return
}
fmt.Printf("System call returned: %d\n", ret)
}⚠️ 注意事项:
总之,syscall.Syscall6 是 Go 中调用任意 Linux 系统调用(包括自定义)最简洁、标准且无需侵入 Go 工具链的方式——它正是为这类场景而设计的底层桥梁。