如何在 Ubuntu 上交叉编译 ARM 架构的 CGO 程序
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何在 Ubuntu 上交叉编译 ARM 架构的 CGO 程序相关的知识,希望对你有一定的参考价值。
我们都知道在不涉及 CGO 的时候,Go 的交叉编译非常简单,只需要设置对应的 GOOS 和 GOARCH 即可,但当涉及到 CGO 时,问题就变得有点复杂了,因为你需要指定特定的 GCC。例如,想在 Ubuntu 上交叉编译一个带有 CGO 的动态库,目标 CPU 架构为 arm,我们该如何操作呢?
示例代码
# shared.go
package main
import "C"
//export Sum
func Sum(a, b int) int {
return a + b
}
func main(){}
这个代码使用到了 CGO,然后暴露一个 Sum 方法,实现两个整数相加。
编译 32 位的 arm
因为通过设置 CGO_ENABLED=1 开启 CGO 编译,执行命令如下:
CGO_ENABLED=1 GOOS=linux GOARCH=arm go build -buildmode=c-shared -o share.so
但不幸,命令报错: gcc: error: unrecognized command line option ‘-marm‘。
正如一开始我提到,交叉编译 CGO 需要选择特定的 arm 交叉编译工具,而 Ubuntu 上编译 32 位的 arm 可以使用 gcc-arm-linux-gnueabihf,安装命令如下:
sudo apt-get update
sudo apt-get install gcc-arm-linux-gnueabihf
安装完成后,指定 CC 重新编译:
CGO_ENABLED=1 GOOS=linux GOARCH=arm CC=arm-linux-gnueabihf-gcc go build -buildmode=c-shared -o share.so
命令成功运行,此时目录下已经生成了一个叫做 share.so 的文件,通过 file 命令查看其属性,可以确认确实为 arm 32 版本。
share.so: ELF 32-bit LSB shared object, ARM, EABI5 version 1 (SYSV), dynamically linked, BuildID[sha1]=7b23579ddcbebdfc8f4b68512859661a45d66996, not stripped
编译 64 位的 arm
交叉编译的时候,不仅要针对平台选择 GCC,而且还要根据操作系统的位数来选,所以 64 位需要选择不同的 GCC,这里推荐 gcc-linaro-5.3-2016.02-x86_64_aarch64-linux-gnu.tar.xz。
安装命令:
wget https://releases.linaro.org/components/toolchain/binaries/5.3-2016.02/aarch64-linux-gnu/gcc-linaro-5.3-2016.02-x86_64_aarch64-linux-gnu.tar.xz
tar xvf gcc-linaro-5.3-2016.02-x86_64_aarch64-linux-gnu.tar.xz -C /usr/lib/
echo ‘export PATH="$PATH:/usr/lib/gcc-linaro-5.3-2016.02-x86_64_aarch64-linux-gnu/bin"‘ >> ~/.bashrc
source ~/.bashrc
安装完成,重新执行编译命令:
CGO_ENABLED=1 GOOS=linux GOARCH=arm64 CC=aarch64-linux-gnu-gcc-5.3.1 go build -buildmode=c-shared -o share.so
编译成功,并产生一个 share.so 文件,同样我们使用 file share.so 查看其元信息为:
share.so: ELF 64-bit LSB shared object, ARM aarch64, version 1 (SYSV), dynamically linked, BuildID[sha1]=5b0e7ff7c3af178039a7b934df8ca3e7572ab5b5, not stripped
到目前为止,我们已经成功在 Ubuntu 系统上交叉编译出了 CGO 程序的 arm 和 arm64 两个版本。
总结
当 Go 交叉编译涉及到 CGO 时,只指定 GOOS 和 GOARCH 是不够的,还需要通过 CC 参数指定相应的 GCC 版本,而 GCC 的选择又与当前系统以及目标架构有关:
- 交叉编译目标 CPU 架构(包括 32位 还是 64位)。
- 交叉编译所在操作系统。
以上是关于如何在 Ubuntu 上交叉编译 ARM 架构的 CGO 程序的主要内容,如果未能解决你的问题,请参考以下文章
使用 arm-apple-darwin10-llvm-gcc-4.2 在 MacOS 上交叉编译问题
在linux上交叉编译wxPython的时候遇到的一个找不到链接库的问题,求解答