在 Intel 机器上构建 Apple Silicon 二进制文件

问题描述

如何在 macOS 11 (Intel) 上编译 C 项目以在 Silicon 上工作?

我当前的构建脚本非常简单:

./configure
make
sudo make install

我尝试将 --host--target 标志与 aarch64-apple-darwinarm-apple-darwin 一起使用,但没有任何运气。

二进制文件始终默认为 x86_64

> file foobar.so
foobar.so: Mach-O 64-bit bundle x86_64

更新: 指定 --host 时似乎找不到 cc 和 gcc。

checking for arm-apple-darwin-cc... no
checking for arm-apple-darwin-gcc... no

解决方法

我在 this page 上找到了一个使用它的提示:

-target arm64-apple-macos11

当我从我的 mac 上运行它时:

clang++ main.cpp -target arm64-apple-macos11

生成的 a.out 二进制文件被列为:

% file a.out
a.out: Mach-O 64-bit executable arm64

我安装了 XCode 12.2。

我面前没有 Arm Mac,所以我假设这可行。

,

先生Curious 对 M1 的交叉编译也很好奇。一种意想不到的解决方案是Zig。它旨在成为交叉编译 C 等的最佳方式;它很容易从 Linux 中针对 M1。

几周前有一系列关于交叉编译到 M1 的流:Part 1 展示了如何在现有的 makefile 中使用 Zig 作为交叉编译器,并且在 Part 3 中他们成功地演示了编译适用于 M1 的 Linux 上的 Redis。

强烈推荐。

,

我们最终解决了这个问题,并且能够在 GitHub Actions 的 x86-64 机器上编译 darwin-arm64debian-aarch64 二进制文件。

我们为 arm64 预编译了所有依赖项,并以静态和动态方式链接它们。

export RELAY_DEPS_PATH=./build-deps/arm64
export PKG_CONFIG_PATH=./build-deps/arm64/lib/pkgconfig

cd ./relay-deps
TARGET=./build-deps make install

cd ./relay
phpize
./configure CFLAGS='-target arm64-apple-macos' \
  --host=aarch64-apple-darwin \
  --enable-relay-jemalloc-prefix
  [snip...]

make

# Dynamically linked binary
cc --target=arm64-apple-darwin \
  ${wl}-flat_namespace ${wl}-undefined ${wl}suppress \
  -o .libs/relay.so -bundle .libs/*.o \
  -L$RELAY_DEPS_PATH/lib -lhiredis -ljemalloc_pic [snip...]

# re-link to standard paths
./relay-deps/utils/macos/relink.sh .libs/relay.so /usr/local/lib
cp .libs/relay.so modules/relay.so

# Build a statically linked shared object
cc --target=arm64-apple-darwin \
  ${wl}-flat_namespace ${wl}-undefined ${wl}suppress \
  -o .libs/relay-static.so -bundle .libs/*.o \
  $RELAY_DEPS_PATH/lib/libhiredis.a \
  $RELAY_DEPS_PATH/lib/libjemalloc_pic.a \
  [snip...]

relink.sh

#!/bin/bash
set -e

printUsage() {
    echo "$0 <shared-object> <prefix>"
    exit 1
}

if [[ ! -f "$1" || -z "$2" ]]; then
    printUsage
    exit 1
fi

INFILE=$1
PREFIX=$2

links=(libjemalloc libhiredis [snip...])

if [ -z "$PREFIX" ]; then
    PREFIX=libs
fi

for link in ${links[@]}; do
    FROM=$(otool -L "$INFILE"|grep $link|awk '{print $1}')
    FILE=$(basename -- "$FROM")
    TO="$PREFIX/$FILE"

    echo "$FROM -> $TO"
    install_name_tool -change "$FROM" "$TO" "$1"
done

相关问答

错误1:Request method ‘DELETE‘ not supported 错误还原:...
错误1:启动docker镜像时报错:Error response from daemon:...
错误1:private field ‘xxx‘ is never assigned 按Alt...
报错如下,通过源不能下载,最后警告pip需升级版本 Requirem...