0

我正在构建“main.c”文件:

int sum(int x, int y) {
  return x + y;
}

int dbl(int x) {
  return x + x;
}

void call_fn(int (*fn)(int)) {
  (*fn)(42);
}

void erase_type(void *fn) {
  call_fn(fn);
}

int main() {
  erase_type(sum);
  return 0;
}

对于启用了“cfi”清理功能的主机“x86”平台,使用以下命令:

./clang-8 \
-fsanitize=cfi \
-fvisibility=hidden \
-fno-sanitize-trap=all \
-fuse-ld=gold \
-flto \
main.c \
-o \
main

现在我需要为“aarch64”处理器编译这个文件。我知道如何简单地编译这个文件(禁用 cfi 功能)。

rm -f main.o && \
./clang-8 \
-cc1 \
-triple \
aarch64-funnyos-unknown-gnueabi \
-emit-obj \
-target-cpu \
generic \
-O2 \
-Wall \
-o \
main.o \
-x \
c \
main.c \
&& \
file main.o

但我不知道如何在启用“cfi”的情况下为“aarch64”编译“main.c”文件。

有人知道怎么做吗?

UPD。假设 CPU 是 ARMv8 类型,并且 Linux 正在运行,例如,在 QEMU 模拟器上。假设,主机是 x86 并且在 Linux 上运行。我需要在主机 x86 Linux 上为 ARMv8 目标(在 QEMU、Linux 上运行)交叉编译代码。

4

2 回答 2

0

您的 x86 和 AArch64 命令非常不同。第一个运行支持编译器标志的编译器驱动程序(特别是)。-fsanitize=cfi

您的第二个命令虽然运行编译器正确(如-cc1开关所示)。这是一种内部模式:它由编译器驱动程序在后台运行,因此它不支持普通的编译器标志,也不打算由普通用户运行。您应该切换到在 AArch64 上使用编译器驱动程序。

于 2019-04-12T10:35:20.910 回答
0

首先,我们需要通过clang的编译器将main.c编译成llvm的中间代码:

./clang-8 \
--target=\
aarch64-funnyos-unknown-gnueabi \
-c \
-flto \
-fsanitize=cfi \
-fvisibility=default \
main.c \
-o \
main.o

下一步是通过黄金链接器(例如,来自 AndroidNDK)使用黄金插件(来自 LLVM 项目)将此代码翻译/链接成二进制文件,告诉链接器仿真模式(-m 选项,这意味着二进制文件适用于哪个架构应该生成):

/PATH/TO/aarch64-linux-android-4.9/prebuilt/linux-x86_64/aarch64-linux-android/bin/ld.gold \
-pie \
-m \
aarch64_elf64_le_vec \
-o \
main \
-plugin \
/PATH/TO/llvm-project/build/lib/LLVMgold.so \
-plugin-opt=mcpu=generic \
-plugin-opt=O2 \
-plugin-opt=-debugger-tune=gdb \
main.o

PS 感谢@yugr 的帮助和帮助!

于 2019-04-18T08:33:28.097 回答