1.3 arm-linux交叉编译链

平常我们做的编译叫本地编译,也就是在当前平台编译,编译得到的程序也是在本地执行。相对而言的交叉编译指的是在一个平台上生成另一个平台的可执行代码。

常见的交叉编译有以下3种。

在Windows PC上,利用ADS(ARM 开发环境),使用armcc编译器,编译出针对ARM CPU的可执行代码。

在Linux PC上,利用arm-linux-gcc编译器,编译出针对Linux ARM平台的可执行代码。

在Windows PC上,利用cygwin环境,运行arm-elf-gcc编译器,编译出针对ARM CPU的可执行代码。

1.3.1 arm-linux交叉编译工具链的制作方法

由于一般嵌入式开发系统的存储大小是有限的,通常都要在性能优越的 PC 上建立一个用于目标机的交叉编译工具链,用该交叉编译工具链在PC上编译目标机上要运行的程序,比如在PC平台(X86 CPU)上编译出能运行在以ARM为内核的CPU平台上的程序。要生成在目标机上运行的程序,必须要用交叉编译工具链完成。交叉编译工具链是一个由编译器、连接器和解释器组成的综合开发环境,交叉编译工具链主要由binutils、gcc和glibc 3个部分组成。有时出于减小libc库大小的考虑,也可以用别的 c 库来代替 glibc,例如 uClibc、dietlibc 和 newlib。建立交叉编译工具链是一个相当复杂的过程,如果不想自己经历复杂繁琐的编译过程,网上有一些编译好的可用的交叉编译工具链可以下载,但就以学习为目的来说,读者有必要学习自己制作一个交叉编译工具链。本节通过具体的实例讲述基于ARM的嵌入式Linux交叉编译工具链的制作过程。

制作arm-linux交叉编译工具链的一般通过crosstool工具或者crosstool_NG,前者使用方便,但是制作会受到一些限制,使用crosstool最多只能编译gcc 4.1.1、glibc 2.x的版本。crosstool-NG是新的用来建立交叉工具链的工具,它是crosstool的替换者,crosstool_NG有更好的定制性,并且一直保持着更新,对新版本的编译工具链的支持比较好,当然也带来了一些麻烦,它并不是下载下来就可以使用的,必须先配置安装。我们这里选用crosstool_NG来制作我们的编译链。

1.安装crosstool_NG

在crosstool_NG官网上下载最新版本。

        zhuzhaoqi@zhuzhaoqi-desktop:~$ mkdir arm-linux-tools
        zhuzhaoqi@zhuzhaoqi-desktop:~$ cd arm-linux-tools/
        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$ ls

获取源码操作命令:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$ wget http://crosstool-ng.org/ download/
    crosstool-ng/crosstool-ng-1.18.0.tar.bz2
        --2013-03-26 21:34:34--  http://crosstool-ng.org/download/crosstool-ng/crosstool-ng-
    1.18.0.tar.bz2
        正在解析主机 crosstool-ng.org... 140.211.15.107
        正在连接 crosstool-ng.org|140.211.15.107|:80... 已连接。
        已发出 HTTP 请求,正在等待回应... 200 OK
        长度: 1884219 (1.8M) [application/x-bzip]
        正在保存至: “crosstool-ng-1.18.0.tar.bz2”
        100%[======================================>] 1,884,219    223K/s   花时 8.8s

下载源码成功之后解压源码:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$ tar jxvf crosstool-ng-1.18.0.tar.bz2
        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$ ls
        crosstool-ng-1.18.0  crosstool-ng-1.18.0.tar.bz2

考虑到后续将要使用到的各种目录,在这里先建立好后续所需目录。

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$ mkdir crosstool-build crosstool-install src
        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$ ls
        crosstool-build    crosstool-ng-1.18.0         src
        crosstool-install  crosstool-ng-1.18.0.tar.bz2

由于Ubuntu操作系统的很多开发软件都没有安装,因此要先安装一些制作交叉编译链必备的软件。在Ubuntu下安装软件的命令为: sudo apt-get install ***。

笔者建议arm-linux交叉编译工具链的制作最好在CentOS系统中完成,因为CentOS系统自带较为完善的开发软件,对于初学者不会造成不必要的麻烦。

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools$  sudo  apt-get  install  sed  bash  cut dpkg-dev patch texinfom4 libtool statwebsvn tar gzip bzip2 lzmabison flex texinfo automake libtool patchcvs cvsd gawk-y

配置整个工程并且进行依赖检测:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$      ./configure--prefix /home/zhuzhaoqi/arm-linux-tools/crosstool-install

在安装过程中,提示如下错误:

        ……
        checking how to run the C preprocessor... gcc -E
        checking for ranlib... ranlib
        checking for objcopy... objcopy
        checking for absolute path to objcopy... /usr/bin/objcopy
        checking for objdump... objdump
        checking for absolute path to objdump... /usr/bin/objdump
        checking for readelf... readelf
        checking for absolute path to readelf... /usr/bin/readelf
        checking for bison... no
        configure: error: missing required tool: bison

输出错误提示缺失bison这个软件,安装:

            zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$   sudo   apt-get install bison

安装完成之后,再次进行配置:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$      ./configure
    --prefix /home/zhuzhaoqi/arm-linux-tools/crosstool-install

又一次输出错误:

        ……
        checking for bison... bison
        checking for flex... no
        configure: error: missing required tool: flex

提示缺失flex这个软件,进行安装:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$   sudo   apt-get install flex

安装完成之后,再一次进行配置:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$      ./configure--prefix /home/zhuzhaoqi/arm-linux-tools/crosstool-install

又一次提示错误:

        checking for bison... bison
        checking for flex... flex
        checking for gperf... no
        configure: error: missing required tool: gperf

提示缺失gperf这个软件,进行安装:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$   sudo   apt-get install gperf

安装完成之后,再一次进行配置:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$      ./configure--prefix /home/zhuzhaoqi/arm-linux-tools/crosstool-install

再一次提示出错:

        ……
        checking for bison... bison
        checking for flex... flex
        checking for gperf... gperf
        checking for makeinfo... no
        configure: error: missing required tool: makeinfo

缺失makeinfo软件,进行安装,如果安装的是makeinfo,则会有如下提示:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$   sudo   apt-get install makeinfo
        正在读取软件包列表... 完成
        正在分析软件包的依赖关系树
        E: 无法找到软件包makeinfo

此时应该安装texinfo软件:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$   sudo   apt-get install makeinfo

安装完成之后,再一次进行配置:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$      ./configure
    --prefix /home/zhuzhaoqi/arm-linux-tools/crosstool-install

这次的配置成功,如果读者操作还会报错的话,依照上面方法找出其根源进行改正即可。成功配置之后会自动创建我们需要的Makefile文件。

        checking for library containing initscr... -lncursesw
        configure: creating ./config.status
        config.status: creating Makefile
        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$ ls
        bootstrap     configure    ct-ng.comp  LICENSES    patches  steps.mk
        config        configure.ac  ct-ng.in    licenses.d   README   TODO
        config.log        contrib        docs        Makefile         samples
        config.status  COPYING      kconfig    Makefile.in  scripts

执行Makefile文件:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$ make
          SED    'ct-ng'
          SED    'scripts/crosstool-NG.sh'
          SED    'scripts/saveSample.sh'
          SED    'scripts/showTuple.sh'
          GEN    'config/configure.in'
          GEN    'paths.mk'
          GEN    'paths.sh'
          DEP    'nconf.gui.dep'
          DEP    'nconf.dep'
          DEP    'lxdialog/yesno.dep'
          DEP    'lxdialog/util.dep'
          DEP    'lxdialog/textbox.dep'
          DEP    'lxdialog/menubox.dep'
          DEP    'lxdialog/inputbox.dep'
          DEP    'lxdialog/checklist.dep'
          DEP    'mconf.dep'
          DEP    'conf.dep'
          BISON  'zconf.tab.c'
          GPERF  'zconf.hash.c'
          LEX    'lex.zconf.c'
          DEP    'zconf.tab.dep'
          CC    'zconf.tab.o'
          CC    'conf.o'
          LD    'conf'
          CC    'lxdialog/checklist.o'
          CC    'lxdialog/inputbox.o'
          CC    'lxdialog/menubox.o'
          CC    'lxdialog/textbox.o'
          CC    'lxdialog/util.o'
          CC    'lxdialog/yesno.o'
          CC    'mconf.o'
          LD    'mconf'
          CC    'nconf.o'
          CC    'nconf.gui.o'
          LD    'nconf'
          SED    'docs/ct-ng.1'
          GZIP   'docs/ct-ng.1.gz'

编译成功之后进行安装:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$ make install

成功安装之后,可以看到已经安装到指定的目录下,最后输出这么一句话:

        ……
        For auto-completion, do not forget to install 'ct-ng.comp' into
        your bash completion directory (usually /etc/bash_completion.d)

这是在提醒我们不要忘记了配置环境变量,多么人性化的提示。接下来配置环境变量。

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0$     sudo     echo"PATH=$PATH:/home/zhuzhaoqi/arm-linux-tools/crosstool-install/bin" >> ~/.bashrc

执行使其生效:

        zhuzhaoqi@zhuzhaoqi-desktop:~$ source /home/zhuzhaoqi/.bashrc

使用ct-ng -v命令查看安装结果:

        zhuzhaoqi@zhuzhaoqi-desktop:~$ ct-ng -v
        GNU Make 3.81
        Copyright (C) 2006  Free Software Foundation, Inc.
        This is free software; see the source for copying conditions.
        There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A
        PARTICULAR PURPOSE.
        这个程序创建为 i486-pc-linux-gnu

OK!ct-ng环境变量添加成功,也就意味着整个crosstool-NG安装成功。

2.配置交叉编译链

现在需要去做的就是配置要编译的交叉编译工具链,在crosstool-NG中有很多已经做好的默认配置(位于crosstool-ng- X.Y.Z(crosstool-ng-1.18.0)/samples目录下),这里只需要进行修改就好了。对于编译器组件部分的版本最好不要修改,因为那个应该是经过测试后的最高版本了,但内核版本可以修改。

可以看到samples目录下的一些默认配置,如下所示:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0/samples$ ls
        alphaev56-unknown-linux-gnu          mips64el-n64-linux-uclibc
        alphaev67-unknown-linux-gnu          mips-ar2315-linux-gnu
        arm-bare_newlib_cortex_m3_nommu-eabi  mipsel-sde-elf
        arm-cortex_a15-linux-gnueabi         mipsel-unknown-linux-gnu
        arm-cortex_a8-linux-gnueabi          mips-malta-linux-gnu
        arm-davinci-linux-gnueabi           mips-unknown-elf
        armeb-unknown-eabi                  mips-unknown-linux-uclibc
        armeb-unknown-linux-gnueabi          powerpc-405-linux-gnu
        armeb-unknown-linux-uclibcgnueabi    powerpc64-unknown-linux-gnu
        arm-unknown-eabi                   powerpc-860-linux-gnu
        arm-unknown-linux-gnueabi           powerpc-e300c3-linux-gnu
        arm-unknown-linux-uclibcgnueabi      powerpc-e500v2-linux-gnuspe
        armv6-rpi-linux-gnueabi             powerpc-unknown-linux-gnu
        avr32-unknown-none                  powerpc-unknown-linux-uclibc
        bfin-unknown-linux-uclibc   powerpc-unknown_nofpu-linux-gnu
        i586-geode-linux-uclibc             s390-ibm-linux-gnu
        i586-mingw32msvc,i686-none-linux-gnu  s390x-ibm-linux-gnu
        i686-nptl-linux-gnu                 samples.mk
        i686-unknown-mingw32                sh4-unknown-linux-gnu
        m68k-unknown-elf                   x86_64-unknown-linux-gnu
        m68k-unknown-uclinux-uclibc          x86_64-unknown-linux-uclibc
        mips64el-n32-linux-uclibc           x86_64-unknown-mingw32

里面有很多默认配置,有arm、avr32、mips、powerpc等硬件平台,而arm平台有如下几个:

        arm-unknown-eabi是基于裸板,也就是无操作系统。
        arm-unknown-linux-gnueabi 是基于Linux。
        arm-unknown-linux-uclibcgnueabi 这个应该能看出来了,是为uclinux用的。
        arm-cortex_a15-linux-gnueabi可从名字上看是为cortex-a15用的。
        arm-cortex_a8-linux-gnueabi 这个也可从名字上看是为cortex-a8用的。
        arm-xxx$&#*&还有几个,这些暂且不去理会。

因为是制作 arm-linux 交叉编译链,所以选择 arm-unknown-linux-gnueabi 进行配置。将arm-unknown-linux-gnueabi文件夹复制到crosstool-build/目录下:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-ng-1.18.0/samples$   cp   -r arm-unknown-linux-gnueabi/ ../../crosstool-build/
        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-build$ ls
        arm-unknown-linux-gnueabi

将默认配置文件拷贝到工作目录(crosstool-build)下并改名为.config,因为默认的配置文件为.config,完成之后可以加载需要的配置。

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-build$                    cp arm-unknown-linux-gnueabi/crosstool.config .config

执行ct-ng menuconfig进入配置界面进行配置:

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-build$ ct-ng menuconfig
          LN    config
          MKDIR config.gen
          IN    config.gen/arch.in
          IN    config.gen/kernel.in
          IN    config.gen/cc.in
          IN    config.gen/binutils.in
          IN    config.gen/libc.in
          IN    config.gen/debug.in
          CONF  config/config.in
        #
        # configuration saved
        #

进入配置界面,如图1.1所示。

图1.1 ct-ng图形配置界面

下面设置源码目录和安装目录,这需要读者依据自己实际设定的情况来进行配置。

第一步,设定源码包路径和交叉编译器的安装路径。

        Paths and misc options --->
          (/home/zhuzhaoqi/arm-linux-tools/src) Local tarballs directory 保存源码包路径
            (/home/zhuzhaoqi/arm-linux-tools/tools) Prefix directory交叉编译器的安装路径

配置之后的结构如图1.2所示。

图1.2 添加源码包路径和交叉编译器的安装路径

第二步,修改交叉编译器针对的构架。

因为本次是针对OK6410制作编译链,那就依据s3c6410的硬件特性来制作。

Target options是重点要修改的地方(以下配置均是基于已拷贝过来的配置)。

                Target Architecture(arm) 这个不用管,已经是arm了。
                Default instruction set mode (arm) 这个也不管,也已经是arm了。

Architecture level()需要进行修改。

通过查找资料,这个应该是指令集的架构,对于 S3C6410 ARM1176JZF-S 核心使用的是armv6zk架构,就选armv6zk。那么,具体都支持哪些架构呢?可以用man gcc来查询,搜索arm,再搜索-march=就可以找到本gcc支持的处理器核心列表了:

        -march=name
        This specifies the name of the target ARM architecture.  GCC uses
        this name to determine what kind of instructions it can emit when
        generating assembly code.  This option can be used in conjunction
        with or instead of the -mcpu= option.  Permissible names are:
        armv2, armv2a, armv3, armv3m, armv4, armv4t, armv5, armv5t, armv5e,
        armv5te, armv6, armv6j, armv6t2, armv6z, armv6zk, armv6-m, armv7,
        armv7-a, armv7-r, armv7-m, iwmmxt, iwmmxt2, ep9312.

Emit assembly for CPU()需要进行修改。

这个对应的是CPU的核心类型。同样,也和上面的选项一样,对应一个GCC选项。GCC的描述如下。

        -mcpu=name
        This specifies the name of the target ARM processor.  GCC uses this
        name to determine what kind of instructions it can emit when
        generating assembly code.  Permissible names are: arm2, arm250,
        arm3, arm6, arm60, arm600, arm610, arm620, arm7, arm7m, arm7d,
        arm7dm, arm7di, arm7dmi, arm70, arm700, arm700i, arm710, arm710c,
        arm7100, arm720, arm7500, arm7500fe, arm7tdmi,arm7tdmi-s, arm710t,
        arm720t, arm740t, strongarm, strongarm110, strongarm1100,
        strongarm1110, arm8, arm810, arm9, arm9e, arm920, arm920t, arm922t,
        arm946e-s, arm966e-s, arm968e-s, arm926ej-s, arm940t, arm9tdmi,
        arm10tdmi, arm1020t, arm1026ej-s, arm10e, arm1020e, arm1022e,
        arm1136j-s, arm1136jf-s, mpcore, mpcorenovfp, arm1156t2-s,
        arm1176jz-s, arm1176jzf-s, cortex-a8, cortex-a9, cortex-r4,
        cortex-r4f, cortex-m3, cortex-m1, xscale, iwmmxt, iwmmxt2, ep9312.

这样看简单一些了,如果是 S3C2410/S3C2440 就选 arm920t,如果是 s3c6410 就选arm1176jzf-s。

Tune for CPU() ,对应的GCC描述如下:

        -mtune=name
        This option is very similar to the -mcpu= option, except that
        instead of specifying the actual target processor type, and hence
        restricting which instructions can be used, it specifies that GCC
        should tune the performance of the code as if the target were of
        the type specified in this option, but still choosing the
        instructions that it will generate based on the cpu specified by a
        -mcpu= option.  For some ARM implementations better performance can
        be obtained by using this option.

意思是说这个选项和-mcpu 很类似,这里是指真实的CPU型号。不过有读者是编译2440的工具链,这里选择的是arm9tdmi,如果不是,那就空着。这里的作用是如果arm920t处理不了,就用arm9tdmi的方式来编译。

与Floating point()浮点相关的选项s3c6410 有硬件VFP,所以这里选的是hardware FPU。这个是给有硬浮点的处理器强行选软浮点用的。

Use specific FPU()跟浮点有关,这里不选任何内容。至于怎么组合,读者可以跟据自己的CPU的实际情况进行相应的配置。

        C compiler  --->
            *** Additional supported languages: ***
            [ ] Java //不用这个编译器来编译java

当然,如果读者需要用它来编译java那就不用去除。

其他选项采用默认设置存盘然后退出,这样就配置完了。

        zhuzhaoqi@zhuzhaoqi-desktop:~/arm-linux-tools/crosstool-build$ ct-ng build

开始编译,此编译过程需要花费大约两个小时,最终编译出arm-linux-gcc-4.4.1编译链。