是否可以在Go代码中包含内联汇编?

时间:2010-06-01 15:26:33

标签: assembly go

是否可以在Go代码中包含内联汇编?

5 个答案:

答案 0 :(得分:36)

不支持内联汇编,但您可以使用C编译汇编代码,使用cgo进行编译并使用import "C",例如gmp.go。您也可以使用与Go直接兼容的汇编样式编写,例如在asm_linux_amd64.s中,它要求函数名称以“·”开头。

或者,你可以使用nasm和gccgo,这是我最喜欢的方式。 (注意,Nasm似乎不支持以“·”开头的函数。)

这是一个有效的“你好世界”的例子:

hello.asm:

; Based on hello.asm from nasm

    SECTION .data       ; data section
msg:    db "Hello World",10 ; the string to print, 10=cr
len:    equ $-msg       ; "$" means "here"
                ; len is a value, not an address

    SECTION .text       ; code section

global go.main.hello        ; make label available to linker (Go)
go.main.hello:

    ; --- setup stack frame
    push rbp            ; save old base pointer
    mov rbp,rsp   ; use stack pointer as new base pointer

    ; --- print message
    mov edx,len     ; arg3, length of string to print
    mov ecx,msg     ; arg2, pointer to string
    mov ebx,1       ; arg1, where to write, screen
    mov eax,4       ; write sysout command to int 80 hex
    int 0x80        ; interrupt 80 hex, call kernel

    ; --- takedown stack frame
    mov rsp,rbp  ; use base pointer as new stack pointer
    pop rbp      ; get the old base pointer

    ; --- return
    mov rax,0       ; error code 0, normal, no error
    ret         ; return

main.go:

package main

func hello();

func main() {
    hello()
    hello()
}

一个方便的Makefile:

main: main.go hello.o
    gccgo hello.o main.go -o main

hello.o: hello.asm
    nasm -f elf64 -o hello.o hello.asm

clean:
    rm -rf _obj *.o *~ *.6 *.gch a.out main

我在main.go中调用hello()两次,只是为了仔细检查hello()是否正确返回。

请注意,直接调用中断80h在Linux上不被认为是好的样式,并且写入的调用函数是C更“未来证明”。另请注意,这是专门针对64位Linux的程序集,并且不以任何方式,形状或形式与平台无关。

我知道这不是你问题的直接答案,但这是我所知道的最简单的使用Go组装的路线,缺乏内联。如果您确实需要内联,则可以编写一个脚本,从源文件中提取内联汇编,并按照上述模式进行准备。足够近? :)

Go,C和Nasm的快速示例:gonasm.tgz

更新: gccgo的更高版本需要-g标志,只需要“main.hello”而不是“go.main.hello”。以下是Go,C和Yasm的更新示例:goyasm.tgz

答案 1 :(得分:24)

Go编程语言中没有任何工具可以支持内联汇编语言代码,并且没有计划这样做。 Go确实支持链接到assemblerC中编写的例程。有一个实验性功能,可以将SWIG support添加到Go。

答案 2 :(得分:17)

事实证明这是可能的,请参阅:http://www.doxsey.net/blog/go-and-assembly

答案 3 :(得分:13)

不,你不能,但是通过使用go编译器很容易提供一个函数的程序集实现。没有必要使用"导入C"使用组装。

看看数学库中的一个例子:

http://golang.org/src/pkg/math/abs.go:在此go文件中声明了Abs函数。 (此文件中还有一个abs的实现,但由于它具有小写名称,因此不会导出。)

package math

// Abs returns the absolute value of x.
//
// Special cases are:
//  Abs(±Inf) = +Inf
//  Abs(NaN) = NaN
func Abs(x float64) float64

然后,在http://golang.org/src/pkg/math/abs_amd64.s中,在此文件中为英特尔64位实现了Abs:

// func Abs(x float64) float64
TEXT ·Abs(SB),NOSPLIT,$0
    MOVQ   $(1<<63), BX
    MOVQ   BX, X0 // movsd $(-0.0), x0
    MOVSD  x+0(FP), X1
    ANDNPD X1, X0
    MOVSD  X0, ret+8(FP)
    RET

答案 4 :(得分:3)

标准Go编译器中的优化传递(即:8g + 8l,而不是gccgo)基本上使用二进制形式的原始指令。目前没有办法(没有实现)编译器区分编译器生成的程序集和用户提供的内联汇编代码 - 这个是Go编译器不允许内联的主要原因部件。换句话说,由于编译器架构,编译器不支持内联汇编。

Go语言本身当然没有任何内容阻止其他Go语言实现(即:其他Go编译器)支持内联汇编。内联汇编是一种特定于编译器的决策 - 它与Go语言本身没什么关系。

在任何一种情况下,内联汇编都是不安全的,因为Go的类型系统无法检查它是否正确。似乎最好实现任何需要在C语言中使用内联汇编的函数,并从Go调用C函数。