热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

交叉编译工具链介绍

基本概念什么是交叉编译交叉编译可以理解为,在当前编译平台下,编译出来的程序能运行在体系结构不同的另一种目标平台上,但是编译平台本身却不

基本概念


什么是交叉编译


  • 交叉编译可以理解为,在当前编译平台下,编译出来的程序能运行在体系结构不同的另一种目标平台上,但是编译平台本身却不能运行该程序。
  • 比如,我们在 x86 平台上,编写程序并编译成能运行在 ARM 平台的程序,编译得到的程序在 x86 平台上是不能运行的,必须放到 ARM 平台上才能运行。
  • 交叉编译链就是为了编译跨平台体系结构的程序代码而形成的由多个子工具构成的一套完整的工具集。同时,它隐藏了预处理、编译、汇编、链接等细节,当我们指定了源文件(.c)时,它会自动按照编译流程调用不同的子工具,自动生成最终的二进制程序映像(.bin)。

为什么要交叉编译


  • 速度(Speed): 目标平台的运行速度往往比主机慢得多,许多专用的嵌入式硬件被设计为低成本和低功耗,没有太高的性能。
  • 能力(Capability): 整个编译过程是非常消耗资源的,嵌入式系统往往没有足够的内存或磁盘空间
  • 可用性(Availability): 即使目标平台资源很充足,可以本地编译,但是第一个在目标平台上运行的本地编译器总需要通过交叉编译获得
  • 灵活性(Flexibility): 一个完整的Linux编译环境需要很多支持包,交叉编译使我们不需要花时间将各种支持包移植到目标板上
  • 便利性(Convenience ):无头盒子的用户界面有点局促。诊断构建中断已经足够令人沮丧了。从CD安装到没有CD-ROM驱动器的机器上是一件痛苦的事。在测试环境和开发环境之间来回重新启动很快就过时了,而且能够从意外的全局化测试系统中恢复过来是件好事。

为什么交叉编译比较困难


1.可移植的本机编译很困难

大多数的程序是在x86硬件上开发的,它们是在本地编译的。这意味着进行交叉编译会遇到两种类型的问题:程序本身的问题和构建系统的问题。
第一种类型的问题会影响所有非x86的目标,无论是针对本机还是针对交叉构建。大多数程序都会对其运行的机器类型做出配置,这些配置必须与所讨论的平台相匹配,否则该程序将无法运行。常见的配置包括:


  • 字长(Word size): 将指针复制到int可能会在64位平台上丢失数据,并且通过乘以4而不是sizeof(long)来确定malloc的大小也不是很好。也可能由于整数溢出而产生细微的安全漏洞,例如“ if(x(y + x
  • 字节序(Endianness): 不同的系统以不同的方式迭代地存储二进制数据,这意味着从磁盘或网络中读取整数或浮点数据的块可能需要转换。键入“ man字节序”以获取详细信息。
  • 对齐方式(Alignment):某些平台(例如arm)只能从4字节的偶数倍的地址读取或写入int,否则会出现段错误。即使是能够处理任意对齐方式的数据,处理未对齐数据的速度也较慢(它们必须取两次才能得到一半),因此编译器通常会填充结构以对齐变量。因此,将结构视为可以发送到磁盘或通过网络发送的大量数据,需要额外的工作以确保表示的一致性。
  • 默认参数(Default signedness) :不同平台之间“ char”数据类型默认为带符号还是无符号(在某些情况下,随编译器的不同而不同),这可能会导致一些令人惊讶的错误。一个简单的解决方法是提供一个编译器参数,例如“ -funsigned-char”,以将默认值强制为已知值。
  • 没有内存管理单元(NOMMU) :如果您的目标平台没有内存管理单元,则需要进行一些更改。您需要vfork()而不是fork(),只有某些类型的mmap()可以工作(共享或只读,但不能在写入时进行复制),并且堆栈不会动态增长。


2.交叉编译很困难

  • 配置问题(Configuration issues):具有单独配置步骤(标准configure / make / make安装的“ ./configure”部分)的软件包通常会测试字节序或页面大小等内容,以便在本地编译时可移植。交叉编译时,主机系统和目标系统之间的这些值会有所不同,因此在主机系统上运行测试会给出错误的答案。当目标没有该软件包或版本不兼容时,配置还可以检测主机上是否存在软件包,并包括对此软件包的支持。
  • 主机vs目标(HOSTCC vs TARGETCC):许多构建过程需要编译内容才能在主机系统上运行,例如上述配置测试或生成代码的程序(例如创建.h文件的C程序,然后在主构建过程中#include )。仅用目标编译器替换主机编译器就会破坏需要构建在构建本身中运行的事物的软件包。这样的软件包需要访问主机和目标编译器,并且需要教导何时使用每个软件包。
  • 工具链泄漏(Toolchain Leaks):配置不正确的交叉编译工具链可能会将主机系统的某些位泄漏到已编译的程序中,从而导致通常易于检测但难以诊断和纠正的故障。工具链可能#include错误的头文件,或在链接时搜索错误的库路径。共享库通常依赖于其他共享库,这些共享库也可能潜入对主机系统的意外链接时引用。
  • 库(Libraries): 动态链接的程序必须在编译时访问适当的共享库。需要将与目标系统共享的库添加到交叉编译工具链中,以便程序可以针对它们进行链接。
  • 测试(Testing):在本机版本上,开发系统提供了便利的测试环境。交叉编译时,确认成功构建的“ hello world”可能需要配置(至少)引导加载程序,内核,根文件系统和共享库。


工具链介绍


交叉编译工具Linaro简介

Linaro,一间非营利性质的开放源代码软件工程公司,主要的目标在于开发不同半导体公司系统单芯片(SoC)平台的共通软件,以促进消费者及厂商的福祉。针对于各个成员推出的 ARM系统单芯片(SoC),它开发了ARM开发工具、Linux内核以及Linux发行版(包括 Android 及 Ubuntu)的主要自动建构系统。


工具链下载地址

gcc-linaro:在binaries目录下包含了不同版本以及不同类型的工具链,要根据自己已有的芯片类型进行正确的选择下载。


交叉编译链的一般命名规则

arch-core-kernel-system


  • arch: 用于哪个目标平台。
  • core: 使用的是哪个CPU Core,如Cortex A8,但是这一组命名好像比较灵活,在其它厂家提供的交叉编译链中,有以厂家名称命名的,也有以开发板命名的,或者直接是none或cross的。
  • kernel: 所运行的OS,见过的有Linux,uclinux,bare(无OS)。
  • systen:交叉编译链所选择的库函数和目标映像的规范,如gnu,gnueabi等。其中gnu等价于glibc+oabi;gnueabi等价于glibc+eabi。


GNU Binutils

一般在解压后的交叉编译工具链bin目录下会有很多可执行程序工具,以不同的后缀进行区分,这些不同的后缀代表的意义如下:


  • ld :GNU链接器。
  • as :GNU汇编器。
  • gold:一个新的,更快的,仅ELF的链接器。
  • addr2line:将地址转换为文件名和行号。
  • ar:用于创建,修改和提取档案的实用程序。
  • c++filt:过滤以解编码编码的C ++符号。
  • dlltool:创建用于构建和使用DLL的文件。
  • elfedit:允许更改ELF格式文件。
  • gprof:显示分析信息。
  • nlmconv:将目标代码转换为NLM。
  • nm:列出目标文件中的符号。
  • objcopy:复制并转换目标文件。
  • objdump:显示目标文件中的信息。
  • ranlib:生成指向档案内容的索引。
  • readelf:显示来自任何ELF格式对象文件的信息。
  • size:列出的对象或归档文件的部分的尺寸。
  • strings:列出文件中的可打印字符串。
  • strip:丢弃的符号。
  • windmc:Windows兼容的消息编译器。
  • windres:Windows资源文件的编译器。


不同交叉编译工具链代表的意义


  • aarch64-elf:64位Armv8 Cortex-A,小端
  • arrch64-linux-gnu:64位Armv8 Cortex-A,小端
  • aarch64_be-elf: 64位Armv8 Cortex-A,大端
  • aarch64_be-linux-gnu:64位Armv8 Cortex-A,大端
  • arm-eabi:32位Armv7 Cortex-A,软浮点,低端
  • arm-linux-gnueabi:32位Armv7 Cortex-A,软浮点,小端
  • arm-linux-gnueabihf:32位Armv7 Cortex-A,硬浮点,小端
  • armeb-eabi:32位Armv7 Cortex-A,软浮点,大端
  • armeb-linux-gnueabi:32位Armv7 Cortex-A,软浮点,大端
  • armeb-linux-gnueabihf:32位Armv7 Cortex-A,硬浮点,大端
  • armv8l-linux-gnueabihf:32位Armv8 Cortex-A,硬浮点,小端


参考资料:


  • 交叉编译详解 一 概念篇
  • Linux交叉编译简介
  • GNU Binutils
  • ArmHardFloatPort

推荐阅读
  • Google在I/O开发者大会详细介绍Android N系统的更新和安全性提升
    Google在2016年的I/O开发者大会上详细介绍了Android N系统的更新和安全性提升。Android N系统在安全方面支持无缝升级更新和修补漏洞,引入了基于文件的数据加密系统和移动版本的Chrome浏览器可以识别恶意网站等新的安全机制。在性能方面,Android N内置了先进的图形处理系统Vulkan,加入了JIT编译器以提高安装效率和减少应用程序的占用空间。此外,Android N还具有自动关闭长时间未使用的后台应用程序来释放系统资源的机制。 ... [详细]
  • ShiftLeft:将静态防护与运行时防护结合的持续性安全防护解决方案
    ShiftLeft公司是一家致力于将应用的静态防护和运行时防护与应用开发自动化工作流相结合以提升软件开发生命周期中的安全性的公司。传统的安全防护方式存在误报率高、人工成本高、耗时长等问题,而ShiftLeft提供的持续性安全防护解决方案能够解决这些问题。通过将下一代静态代码分析与应用开发自动化工作流中涉及的安全工具相结合,ShiftLeft帮助企业实现DevSecOps的安全部分,提供高效、准确的安全能力。 ... [详细]
  • C语言注释工具及快捷键,删除C语言注释工具的实现思路
    本文介绍了C语言中注释的两种方式以及注释的作用,提供了删除C语言注释的工具实现思路,并分享了C语言中注释的快捷键操作方法。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了在Windows环境下如何配置php+apache环境,包括下载php7和apache2.4、安装vc2015运行时环境、启动php7和apache2.4等步骤。希望对需要搭建php7环境的读者有一定的参考价值。摘要长度为169字。 ... [详细]
  • C++字符字符串处理及字符集编码方案
    本文介绍了C++中字符字符串处理的问题,并详细解释了字符集编码方案,包括UNICODE、Windows apps采用的UTF-16编码、ASCII、SBCS和DBCS编码方案。同时说明了ANSI C标准和Windows中的字符/字符串数据类型实现。文章还提到了在编译时需要定义UNICODE宏以支持unicode编码,否则将使用windows code page编译。最后,给出了相关的头文件和数据类型定义。 ... [详细]
  • 本文记录了作者对x265开源代码的实现与框架进行学习与探索的过程,包括x265的下载地址与参考资料,以及在Win7 32 bit PC、VS2010平台上的安装与配置步骤。 ... [详细]
  • 本文介绍了Windows Vista操作系统中的用户账户保护功能,该功能是为了增强系统的安全性而设计的。通过对Vista测试版的体验,可以看到系统在安全性方面的进步。该功能的引入,为用户的账户安全提供了更好的保障。 ... [详细]
  • 目录浏览漏洞与目录遍历漏洞的危害及修复方法
    本文讨论了目录浏览漏洞与目录遍历漏洞的危害,包括网站结构暴露、隐秘文件访问等。同时介绍了检测方法,如使用漏洞扫描器和搜索关键词。最后提供了针对常见中间件的修复方式,包括关闭目录浏览功能。对于保护网站安全具有一定的参考价值。 ... [详细]
  • Apache Shiro 身份验证绕过漏洞 (CVE202011989) 详细解析及防范措施
    本文详细解析了Apache Shiro 身份验证绕过漏洞 (CVE202011989) 的原理和影响,并提供了相应的防范措施。Apache Shiro 是一个强大且易用的Java安全框架,常用于执行身份验证、授权、密码和会话管理。在Apache Shiro 1.5.3之前的版本中,与Spring控制器一起使用时,存在特制请求可能导致身份验证绕过的漏洞。本文还介绍了该漏洞的具体细节,并给出了防范该漏洞的建议措施。 ... [详细]
  • 本文概述了JNI的原理以及常用方法。JNI提供了一种Java字节码调用C/C++的解决方案,但引用类型不能直接在Native层使用,需要进行类型转化。多维数组(包括二维数组)都是引用类型,需要使用jobjectArray类型来存取其值。此外,由于Java支持函数重载,根据函数名无法找到对应的JNI函数,因此介绍了JNI函数签名信息的解决方案。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • MyBatis错题分析解析及注意事项
    本文对MyBatis的错题进行了分析和解析,同时介绍了使用MyBatis时需要注意的一些事项,如resultMap的使用、SqlSession和SqlSessionFactory的获取方式、动态SQL中的else元素和when元素的使用、resource属性和url属性的配置方式、typeAliases的使用方法等。同时还指出了在属性名与查询字段名不一致时需要使用resultMap进行结果映射,而不能使用resultType。 ... [详细]
  • Vagrant虚拟化工具的安装和使用教程
    本文介绍了Vagrant虚拟化工具的安装和使用教程。首先介绍了安装virtualBox和Vagrant的步骤。然后详细说明了Vagrant的安装和使用方法,包括如何检查安装是否成功。最后介绍了下载虚拟机镜像的步骤,以及Vagrant镜像网站的相关信息。 ... [详细]
  • 本文介绍了在Web应用系统中,数据库性能是导致系统性能瓶颈最主要的原因之一,尤其是在大规模系统中,数据库集群已经成为必备的配置之一。文章详细介绍了主从数据库架构的好处和实验环境的搭建方法,包括主数据库的配置文件修改和设置需要同步的数据库等内容。MySQL的主从复制功能在国内外大型网站架构体系中被广泛采用,本文总结了作者在实际的Web项目中的实践经验。 ... [详细]
author-avatar
40740
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有