Ответ 1
Нет поддержки встроенной сборки, но вы можете ссылаться на код, написанный в сборке через 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
Я вызываю hello()
дважды в main.go, просто чтобы проверить, правильно ли hello() возвращается.
Обратите внимание, что вызов прерывания 80h напрямую не считается хорошим стилем в Linux, а вызывающие функции, написанные, являются C более "будущим доказательством". Также обратите внимание, что это сборка специально для 64-битного Linux и не имеет никакой зависимости от платформы, формы или формы.
Я знаю, что это не прямой ответ на ваш вопрос, но это самый простой маршрут, который я знаю для использования сборки с Go, при отсутствии инкрустирования. Если вам действительно нужна инкрустация, можно написать script, который извлекает встроенную сборку из исходных файлов и подготавливает ее таким образом, который следует за шаблоном выше. Достаточно близко?:)
Быстрый пример для Go, C и Nasm: gonasm.tgz
Обновление:. В более поздних версиях gccgo нужен флаг -g, а вместо "go.main.hello" требуется только "main.hello". Вот обновленный пример для Go, C и Yasm: goyasm.tgz