gcc & clang

是的,既然三套代码在同一台电脑上编译却出现了不同的编译器选择(有的用 GCC,有的用 Clang),​​可以排除电脑本身的环境问题​ ​(如系统 PATH、默认工具链等全局配置),因为如果是电脑环境的问题,三套代码的行为应该一致。问题的根源更可能是 ​​代码版本或构建配置的差异​​。以下是详细分析和排查步骤:


​1. 排除电脑环境的干扰​

  • ​验证电脑的全局编译器默认值​ ​:

    在终端中直接运行以下命令,确认全局默认的 gccclang 指向的版本:

    复制代码
    which gcc clang
    gcc --version | head -n1
    clang --version | head -n1
    • 如果输出显示同一编译器(例如均为 GCC),则说明电脑环境本身​不是​导致差异的原因。
  • ​检查环境变量 CC ​:

    运行 env | grep CC,确认没有全局的 CC=clangCC=gcc 覆盖。

    • 如果为空,说明编译器的选择是由代码或构建脚本控制的。

​2. 聚焦代码版本和构建配置的差异​

​(1) 检查内核配置(.configdefconfig)​
  • ​关键配置项​ ​:

    在三个版本的 .config 中搜索以下选项:

    复制代码
    grep -E "CONFIG_CC_IS_|CONFIG_LTO|CONFIG_CLANG_VERSION" output/.config
    • 如果某个版本的 .config 中显式设置了 CONFIG_CC_IS_CLANG=y,而其他版本为 CONFIG_CC_IS_GCC=y,说明配置本身存在差异。
  • ​可能的原因​​:

    • 华为在 5.10.184 版本中​默认启用了 Clang 支持​ (例如通过 defconfig 预设)。
    • 某些版本可能通过 Kconfig 条件自动选择了 Clang(如检测到工具链优化需求)。
​(2) 检查构建脚本或 Makefile 覆盖​
  • ​搜索硬编码的 CC= 赋值​ ​:

    在代码根目录运行以下命令,查找强制指定编译器的脚本:

    复制代码
    grep -r "CC\s*=\s*clang" .
    grep -r "CC\s*=\s*gcc" .
    • 如果某个版本的 build.sh 或顶层 Makefile 中显式指定了 CC=clang,则会覆盖默认行为。
  • ​华为定制化的可能​ ​:

    某些厂商会通过 build.sh 或 SDK 工具链强制指定编译器,例如:

    复制代码
    # 华为可能在某版本的 build.sh 中添加了:
    export CC=clang
    make ...
​(3) 检查工具链路径的差异​
  • ​不同版本可能引用了不同的工具链​
    在三个版本的构建日志中搜索 CROSS_COMPILECC 的实际调用路径:

    复制代码
    grep -r "CROSS_COMPILE=" output/build.log
    grep -r "aarch64-none-linux-gnu-gcc" output/build.log
    • 如果 5.10.184 版本使用了华为提供的 Clang 工具链(如 aarch64-linux-gnu-clang),而其他版本使用 GCC,则会导致差异。

​3. 复现和验证方法​

​(1) 统一构建命令​

对三套代码​​使用完全相同的命令​​编译,排除脚本干扰:

复制代码
# 强制使用 GCC
make CC=gcc clean all

# 强制使用 Clang
make CC=clang clean all
  • 观察是否仍然存在差异。如果强制指定后行为一致,则说明问题出在构建配置中。
​(2) 对比内核的 Kconfig 差异​

检查三个版本中 scripts/Kconfig.includeinit/Kconfig 的差异:

复制代码
diff -u linux-5.10.86/scripts/Kconfig.include linux-5.10.184/scripts/Kconfig.include
  • 可能发现某些版本修改了编译器检测逻辑(例如放宽 Clang 的判定条件)。
​(3) 检查构建日志​

在三个版本的 build.log 中搜索以下关键信息:

复制代码
grep -i "checking for .* compiler" build.log
grep -i "selected compiler" build.log
  • 内核在配置阶段会打印编译器检测结果,例如:

    复制代码
    checking for gcc... aarch64-none-linux-gnu-gcc
    checking for clang... not found

​4. 根本原因推测​

根据描述,最可能的原因是:

  1. 5.10.184 版本的代码或配置显式启用了 Clang​
    • 可能修改了 defconfig 或添加了 CONFIG_LTO_CLANG=y
    • 华为可能在该版本中测试了 Clang 的兼容性,因此默认切换。
  2. ​构建脚本的差异​
    某些版本的 build.sh 可能包含 export CC=clang,而其他版本没有。

​5. 解决方案​

​(1) 统一编译器(推荐)​

如果希望三套代码均使用 GCC:

复制代码
# 在编译时显式指定 GCC
make CC=gcc CROSS_COMPILE=aarch64-none-linux-gnu- clean all
​(2) 检查华为的构建文档​

联系华为或查阅 SDK 文档,确认 5.10.184 版本是否​​官方要求使用 Clang​​。如果是,则需要:

  • 使用配套的 Clang 工具链。
  • 接受 plat_1105.lto.o 的生成行为(LTO 是 Clang 的常见优化手段)。
​(3) 修复调试信息(如需)​

如果目标是让 objdump 输出带注释的反汇编,参考前文的 Clang 调试选项:

复制代码
llvm-objdump -d --source plat_1105.lto.o  # 显示源码和汇编

​总结​

  • ​同一台电脑​:排除全局环境问题,差异来自代码或配置。
  • ​根本原因​ :不同版本的 .config、构建脚本或 Kconfig 逻辑导致编译器选择不同。
  • ​解决方向​
    1. 检查华为的版本间差异(如 defconfigbuild.sh)。
    2. 显式指定 CC= 以统一编译器。
    3. 如需 Clang,适配 LTO 和调试信息的生成方式。
相关推荐
XiaoCCCcCCccCcccC1 小时前
MySQL 表内容的增删查改 -- CRUD操作,聚合函数,group by 子句
数据库·mysql
码农捻旧1 小时前
MySQL 9.3 超详细下载安装教程(Windows版)附图文说明
数据库·windows·mysql·adb·程序员创富
SofterICer1 小时前
8.7 基于EAP-AKA的订阅转移
linux·服务器·数据库
Lz__Heng2 小时前
如何在使用kickstart安装物理机操作系统的过程中核对服务器的SN
运维·服务器·自动化·kickstart
wanhengidc2 小时前
网站服务器出现异常的原因是什么?
运维·服务器
stormsha2 小时前
GO语言进阶:掌握进程OS操作与高效编码数据转换
开发语言·数据库·后端·golang·go语言·源代码管理
尘埃不入你眼眸3 小时前
MySQL的基础操作
数据库·mysql
老神在在0013 小时前
javaEE1
java·开发语言·学习·java-ee
魔道不误砍柴功3 小时前
《接口和抽象类到底怎么选?设计原则与经典误区解析》
java·开发语言
yzlAurora3 小时前
MySQL问题:MVCC是什么?
数据库·oracle