Я пытаюсь написать немного SIMD в основном для учебных целей. Я знаю, что Go может связывать сборку, но я не могу заставить его работать правильно.
Вот самый простой пример, который я могу сделать (поэлементное умножение векторов):
vec_amd64.s (примечание: под фактическим файлом стоит пробел, RET
иначе он вызывает ошибки)
// func mul(v1, v2 Vec4) Vec4
TEXT .mul(SB),4,$0-48
MOVUPS v1+0(FP), X0
MOVUPS v2+16(FP), X1
MULPS X1, X0
// also tried ret+32 since I've seen some places do that
MOVUPS X0, toReturn+32(FP)
RET
vec.go
package simd
type Vec4 [4]float32
func (v1 Vec4) Mul(v2 Vec4) Vec4 {
return Vec4{v1[0] * v2[0], v1[1] * v2[1], v1[2] * v2[2], v1[3] * v2[3]}
}
func mul(v1, v2 Vec4) Vec4
simd_test.go
package simd
import (
"testing"
)
func TestMul(t *testing.T) {
v1 := Vec4{1, 2, 3, 4}
v2 := Vec4{5, 6, 7, 8}
res := v1.Mul(v2)
res2 := mul(v1, v2)
// Placeholder until I get it to compile
if res != res2 {
t.Fatalf("Expected %v; got %v", res, res2)
}
}
При попытке запустить go test
появляется сообщение об ошибке:
# testmain
simd.TestMul: call to external function simd.mul
simd.TestMul: undefined: simd.mul
Команда go env
сообщает, GOHOSTARCH
что amd64
моя версия Go - 1.3. Чтобы убедиться, что проблема не в архитектуре, я нашел другой пакет, который использует сборку, и удалил все файлы сборки, кроме _amd64.s
одного, и его тесты прошли нормально.
Я также попытался изменить его на экспортированный идентификатор на случай, если это вызывало странности, но без кубиков. Я думаю, что я довольно внимательно следил за шаблоном в таких пакетах, как math/big
, надеюсь, это что-то простое и очевидное, что мне не хватает.
Я знаю, что Go, по крайней мере, пытается использовать сборку, потому что, если я внесу синтаксическую ошибку в файл .s, инструмент сборки пожалуется на это.
Редактировать:
Для ясности, go build
компилируется чисто, но go test
вызывает появление ошибки.
go build
закончил чисто,go test
не удалось.