Go コードにインラインアセンブリを含めることは可能ですか?
このブログ投稿コンパイルは別のファイルに移動して.s
編集しますが、列をなして多くの C コンパイラがサポートしているように、Go 関数の一部として asm を使用します。
ベストアンサー1
インラインアセンブリはサポートされていませんが、Cを介してアセンブリで書かれたコードをリンクし、cgoでコンパイルして、import "C"
次のように使用することはできます。gmp.go代わりに、Goと直接互換性のあるアセンブリスタイルで記述することもできます。asm_linux_amd64.s関数名は「·」で始まる必要があります。
または、nasm と gccgo を使用することもできます。これは、これまでのところ私のお気に入りの方法です。(Nasm は「·」で始まる関数をサポートしていないようです)。
実際に動作する「hello world」の例を次に示します。
こんにちは。
; 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
メイン.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
hello()
hello() が正しく返されることを再確認するため、main.go で 2 回呼び出します。
割り込み 80h を直接呼び出すことは Linux では良いスタイルとはみなされず、C で記述された関数を呼び出す方が「将来性」が高いことに注意してください。また、これは 64 ビット Linux 専用のアセンブリであり、いかなる形においてもプラットフォームに依存しないことに注意してください。
これは質問への直接的な回答ではないことは承知していますが、インライン化なしでアセンブリを Go で使用するための、私が知る限り最も簡単な方法です。インライン化が本当に必要な場合は、ソース ファイルからインライン アセンブリを抽出し、上記のパターンに従って準備するスクリプトを作成することができます。十分に近いでしょうか? :)
Go、C、Nasm の簡単な例:ゴナズム
アップデート:gccgo の以降のバージョンでは -g フラグが必要になり、「go.main.hello」の代わりに「main.hello」のみが必要になります。以下は Go、C、Yasm の更新された例です。ゴヤスム