SCons构建系统

    在的网站上可以找到详细的SCons用户手册,本章节讲述SCons的基本用法,以及如何在RT-Thread中用好SCons工具。

    构建工具是一种软件,它可以根据一定的规则或指令,将源代码编译成可执行的二进制程序。这是构建工具最基本也是最重要的功能。实际上,构建工具的功能不止于此,通常这些规则有一定的语法,并组织成文件。这些文件用于来控制构建工具的行为,在完成软件构建之外,也可以做其他事情。

    目前最流行的构建工具是GNU Make。很多知名开源软件,如Linux内核就采用Make构建。Make通过读取Makefile文件来检测文件的组织结构和依赖关系,并完成Makefile中所指定的命令。

    由于历史原因,Makefile的语法比较混乱,不利于初学者学习。此外,在Windows平台上使用Make也不方便,需要安装Cygwin环境。为了克服Make的种种缺点,人们开发了其他构建工具,如CMake和SCons等。

    RT-Thread早期使用Make/Makefile构建。从0.3.x开始,RT-Thread开发团队逐渐引入了SCons构建系统,引入SCons唯一的目是:使大家从复杂的Makefile配置、IDE配置中脱离出来,把精力集中在RT-Thread功能开发上。

    有些读者可能会有些疑惑,这里介绍的构建工具与IDE有什么不同。

    通常IDE有自己的管理源码的方式,一些IDE使用XML来组织文件,并解决依赖关系。大部分IDE会根据用户所添加的源码生成类似Makefile或SConscript的脚本文件,在底层调用类似Make与SCons的工具来构建源码。IDE通过图形化的操作来完成构建。

    本节介绍RT-Thread中常用的SCons命令。SCons不仅完成基本的编译,还可以生成MDK/IAR/VS工程。

    scons

    这个命令用于直接编译目标。如果执行过scons后修改一些文件,再次执行scons时,则SCons会进行增量编译,仅编译修改过的文件并链接。

    注:

    如果在Windows上执行scons输出以下的警告信息,

    1. scons: warning: No version of Visual Studio compiler found - C/C++ compilers most likely not set correctly

    说明scons并没在你的机器上找到Visual Studio编译器,但实际上我们主要是针对设备开发,和Windows本地没什么关系。请直接忽略掉它。

    scons -jN

    多线程编译目标,在多核计算机上可以加快编译速度。一般来说,一颗cpu核心可以支持2个线程。双核机器上使用-j4即可。

    1. scons -j4

    注:如果你只是想看看编译错误或警告,最好是不使用-j参数,这样错误信息不会因为多个文件并行编译而导致出错信息夹杂在一起 *

    scons -c

    清除编译目标。这个命令会清除执行scons时生成的临时文件和目标文件。

    scons —target=XXX -s

    1. scons --target=mdk4 -s

    可以在当前目录生成一个新的名为project.uvproj文件。双击它打开,就可以使用MDK来编译、调试。不习惯SCons的同学可以使用这种方式。

    当修改了rtconfig.h打开或者关闭某些组件时,也需要使用这个命令重新生成对应的定制化的工程。

    注意:

    要生成Keil MDK的工程文件,前提条件是当前目录存在一个工程模版文件,然后scons才会根据这份模版文件加入相关的源码,头文件搜索路径,编译参数,链接参数等。而至于这个工程是针对哪颗芯片的,则直接由这份工程模版文件指定。所以大多数情况下,这个模版文件是一份空的工程文件,用于辅助SCons生成project.uvproj。

    如果打开project.uvproj失败,请删除project.uvopt后,重新生成工程。

    1. scons --target=iar -s

    自动生成IAR工程;

    1. scons --target=vs2012 -s
    2. Scons --target=vs2005 -s

    在bsp/simulator下,可以使用这个命令生成vs2012的工程或vs2005的工程。

    scons —verbose

    默认情况下,scons编译的输出不会显示编译参数,如下所示:

    1. F:\Project\git\rt-thread\bsp\stm32f10x>scons
    2. scons: Reading SConscript files ...
    3. scons: done reading SConscript files.
    4. scons: Building targets ...
    5. scons: building associated VariantDir targets: build
    6. CC build\applications\application.o
    7. CC build\applications\startup.o
    8. CC build\components\drivers\serial\serial.o
    9. ...
    1. armcc -o build\src\mempool.o -c --device DARMSTM --apcs=interwork -ID:/Keil/ARM/
    2. RV31/INC -g -O0 -DUSE_STDPERIPH_DRIVER -DSTM32F10X_HD -Iapplications -IF:\Projec
    3. t\git\rt-thread\applications -I. -IF:\Project\git\rt-thread -Idrivers -IF:\Proje
    4. ct\git\rt-thread\drivers -ILibraries\STM32F10x_StdPeriph_Driver\inc -IF:\Project
    5. \git\rt-thread\Libraries\STM32F10x_StdPeriph_Driver\inc -ILibraries\STM32_USB-FS
    6. -Device_Driver\inc -IF:\Project\git\rt-thread\Libraries\STM32_USB-FS-Device_Driv
    7. er\inc -ILibraries\CMSIS\CM3\DeviceSupport\ST\STM32F10x -IF:\Project\git\rt-thre
    8. ad\Libraries\CMSIS\CM3\DeviceSupport\ST\STM32F10x -IF:\Project\git\rt-thread\com
    9. ponents\CMSIS\Include -Iusb -IF:\Project\git\rt-thread\usb -I. -IF:\Project\git\
    10. rt-thread -IF:\Project\git\rt-thread\include -IF:\Project\git\rt-thread\libcpu\a
    11. rm\cortex-m3 -IF:\Project\git\rt-thread\libcpu\arm\common -IF:\Project\git\rt-th
    12. read\components\drivers\include -IF:\Project\git\rt-thread\components\drivers\in
    13. clude -IF:\Project\git\rt-thread\components\finsh -IF:\Project\git\rt-thread\com
    14. ponents\init F:\Project\git\rt-thread\src\mempool.c
    15. ...

    SCons使用SConscript和SConstruct文件来组织源码结构,通常来说一个项目只有一个SConstruct,但是会有多个SConscript。一般情况下,每个存放有源代码的子目录下都会放置一个SConscript,这些SCons的脚本文件组成如下所示的等级结构。

    [图片待补充]

    为了使RT-Thread更好的支持多种编译器,以及方便的调整编译参数,RT-Thread为每个bsp单独创建了一个名为rtconfig.py的文件。因此每一个RT-Thread bsp目录下都会存在下面三个文件,它们具体控制BSP的编译。

    1. rtconfig.py
    2. SConstruct
    3. SConscript

    大部分组件源码文件夹下存在SConscript文件,这些文件会被BSP目录下的SConscript文件“找到”从而将rtconfig.h中定义的组件加入编译器来。一个BSP中只有一个SConstruct文件,但是却会有多个SConscript文件,可以说SConscript文件是组织源码的主力军。

    修改编译器选项

    在rtconfig.py中控制了大部分编译选项。下面以stm32f10x/rtconfig.py为例(部分)

    1. elif PLATFORM == 'armcc':
    2. # toolchains
    3. CC = 'armcc'
    4. AS = 'armasm'
    5. AR = 'armar'
    6. LINK = 'armlink'
    7. TARGET_EXT = 'axf'
    8.  
    9. DEVICE = ' --device DARMSTM'
    10. CFLAGS = DEVICE + ' --apcs=interwork'
    11. AFLAGS = DEVICE
    12. LFLAGS = DEVICE + ' --info sizes --info totals --info unused --info veneers --list rtthread-stm32.map --scatter stm32_rom.sct'
    13.  
    14. LFLAGS += ' --libpath ' + EXEC_PATH + '/ARM/RV31/LIB'
    15.  
    16. EXEC_PATH += '/arm/bin40/'
    17.  
    18. if BUILD == 'debug':
    19. CFLAGS += ' -g -O0'
    20. AFLAGS += ' -g'
    21. else:
    22. CFLAGS += ' -O2'
    23.  
    24. POST_ACTION = 'fromelf --bin $TARGET --output rtthread.bin \nfromelf -z $TARGET'

    其中CFLAGS存储C文件的编译选项,AFLAGS 则是汇编文件的编译选项,LFLAGS 是链接选项。BUILD 变量控制代码优化的级别。默认 BUILD 变量取值为,即使用debug方式编译,优化级别0。如果将这个变量修改为其他值,就会使用优化级别2编译。下面几种都是可行的写法(总之只要不是'debug'就可以了)。

    建议在开发阶段都使用debug方式编译,不开优化,等产品稳定之后再考虑优化。

    关于这些选项的具体含义需要参考编译器手册,如上面使用的armcc是MDK的底层编译器。其编译选项的含义在MDK help中有详细说明。

    内置函数

    如果想要将自己的一些源代码加入到SCons编译环境中,一般可以创建或修改已有SConscript文件。SConscript文件可以控制源码文件的加入,并且可以指定文件的Group(与MDK/IAR等IDE中的Group的概念类似)。

    SCons提供了很多内置函数可以帮助我们快速添加源码程序。简单介绍一些常用函数。

    1. GetCurrentDir()

    获取当前路径

    1. Glob('*.c')

    获取当前目录下的所有C文件。修改参数的值为其他后缀就可以匹配当前目录下的所有某类型的文件。

    1. GetDepend(macro)

    在tools/目录下的脚本文件中定义,它会从rtconfig.h文件读取组件配置信息,其参数为rtconfig.h中的宏名。如果rtconfig.h打开了某个宏,则这个方法(函数)返回真,否则返回假。

    1. Split(str)

    将字符串str分割成一个list

    1. DefineGroup(name, src, depend, **parameters)

    这是RT-Thread基于SCons扩展的一个方法(函数)。DefineGroup用于定义一个组件。组件可以是一个目录(下的文件或子目录),也是后续一些IDE工程文件中的一个Group或文件夹。

    • name来定义这个group的名字
    • src用于定义这个Group中包含的文件,一般指的是C/C++源文件。方便起见,也能够通过Glob函数采用通配符的方式列出SConscript文件所在目录中匹配的文件。
    • depend 用于定义这个Group编译时所依赖的选项(例如finsh组件依赖于RT_USING_FINSH宏定义)。编译选项一般指rtconfig.h中定义的RT_USING_xxx宏。当在rtconfig.h配置文件中定义了相应宏时,那么这个Group才会被加入到编译环境中进行编译。如果依赖的宏并没在rtconfig.h中被定义,那么这个Group将不会被加入编译。相类似的,在使用scons生成为IDE工程文件时,如果依赖的宏未被定义,相应的Group也不会在工程文件中出现。
    • parameters则可以输入一组字符串,后面还可以加入的参数包括:
      • CCFLAGS – C源文件编译参数;
      • CPPPATH – 头文件路径;
      • CPPDEFINES – 添加预定义宏;
      • LINKFLAGS – 链接时参数。
      • LIBRARY – 包含此参数,则会将组件生成的目标文件打包成库文件
        可见DefineGroup的功能十分强大,实际使用时不需要配置所有参数。
    1. SConscript(dirs, variant_dir, duplicate)

    SCons内置函数。其参数包括三个:

    • dirs指明SConscript文件路径,
    • variant_dir指定生成的目标文件的存放路径,
    • duiplicate的作用是设定是否拷贝或链接源文件到variant_dir
      利用这些函数,再配合一些简单的Python语句我们就能随心所欲向项目中添加或者删除源码了。下一节我们将介绍几个典型的SConscript示例文件来学习,并达到举一反三的目的。

    bsp/stm32f10x/application/SConcript

    1. Import('RTT_ROOT')
    2. Import('rtconfig')
    3. from building import *
    4.  
    5. src = Glob('*.c')
    6. cwd = GetCurrentDir()
    7. include_path = [cwd]
    8.  
    9. group = DefineGroup('Applications', src, depend = [''], CPPPATH = include_path)
    10.  
    11. Return('group')

    上面这个脚本完成如下功能:

    src = Glob('*.c')得到当前目录下所有的C文件,cwd = GetCurrentDir()将当前路径赋值给cwd,注意cwd是一个字符串;include_path = [cwd]将当前头文件路径保存为一个list变量。最后一行使用DefineGroup创建一个组。组名为Applications。depend为空,表示该组不依赖任何rtconfig.h的任何宏。CPPPATH = include_path表示将当前目录添加到系统的头文件路径中。

    总结:这个源程序会将当前目录下的所有c程序加入到组Applications中,并将这个目录添加到系统头文件搜索路径中。因此,如果在这个目录下增加或者删除文件,就可以将文件加入工程或者从工程中删除。

    它适用于批量添加源码文件。

    SConscript示例2

    component/finsh/SConscript

    1. Import('rtconfig')
    2. from building import *
    3.  
    4. cwd = GetCurrentDir()
    5. src = Glob('*.c')
    6. CPPPATH = [cwd]
    7. if rtconfig.CROSS_TOOL == 'keil':
    8. LINKFLAGS = ' --keep __fsym_* --keep __vsym_* '
    9. else:
    10. LINKFLAGS = ''
    11.  
    12. group = DefineGroup('finsh', src, depend = ['RT_USING_FINSH'], CPPPATH = CPPPATH,
    13. LINKFLAGS = LINKFLAGS)
    14.  
    15. Return('group')

    从第7行开始,与示例1有些区别。

    1. if rtconfig.CROSS_TOOL == 'keil':
    2. LINKFLAGS = ' --keep __fsym_* --keep __vsym_* '
    3. else:
    4. LINKFLAGS = ''

    DefinGroup同样将finsh目录下的所有文件创建为finsh组。

    depend = ['RT_USING_FINSH']表示这个组依赖rtconfig.h中的RT_USING_FINSH。即,当rtconfig.h中打开宏RT_USING_FINSH时,finsh组内的源码才会被实际编译,否则SCons不会编译。

    CPPPATH = CPPPATH,左边的CPPPATH是DefineGroup中内置参数,右边的CPPPATH是本文件第6行定义的,意思是将finsh目录加入到系统头文件目录中。这样我们就可以在其他源码中引用finsh目录下的头文件了,如finsh.h。

    LINKFLAGS = LINKFLAGS的含义与类似。左边的LINKFLAGS表示链接参数,右边的LINKFLAGS则是前面if else语句所设定的值。

    SConscript示例3

    bsp/stm32f10x/SConscript

    cwd = str(Dir('#') 获取工程的顶级目录,也就是工程的SConstruct所在的目录,在这里它的效果与 cwd = GetCurrentDir()相同。随后定义了一个空的list型变量objs。第6行list = os.listdir(cwd)得到当前目录下的所有子目录,并保存到变量list中。随后是一个python的for循环,其含义是取出一个当前目录的子目录,利用os.path.join(cwd,d)拼接成一个完整路径,然后判断这个子目录是否存在一个名为SConscript的文件,若存在,则执行

    1. objs = objs + SConscript(os.path.join(d, 'SConscript'))

    上面这一句中使用了SCons提供的一个内置函数SConscript,它可以读入一个新的SConscript文件,并将SConscript文件中所指明的源码加入编译列表中来。

    stm32f10x/drivers/SConscript

    1. Import('RTT_ROOT')
    2. Import('rtconfig')
    3. from building import *
    4.  
    5. cwd = GetCurrentDir()
    6.  
    7. src = Split('''
    8. board.c
    9. stm32f10x_it.c
    10. led.c
    11. usart.c
    12. ''')
    13.  
    14. # add Ethernet drvers.
    15. if GetDepend('RT_USING_LWIP'):
    16. src += ['dm9000a.c']
    17.  
    18. # add Ethernet drvers.
    19. if GetDepend('RT_USING_DFS'):
    20. src += ['sdcard.c']
    21.  
    22. # add Ethernet drvers.
    23. if GetDepend('RT_USING_RTC'):
    24. src += ['rtc.c']
    25.  
    26. # add Ethernet drvers.
    27. if GetDepend('RT_USING_RTGUI'):
    28. src += ['touch.c']
    29. if rtconfig.RT_USING_LCD_TYPE == 'ILI932X':
    30. src += ['ili_lcd_general.c']
    31. elif rtconfig.RT_USING_LCD_TYPE == 'SSD1289':
    32. src += ['ssd1289.c']
    33.  
    34. CPPPATH = [cwd]
    35.  
    36. group = DefineGroup('Drivers', src, depend = [''], CPPPATH = CPPPATH)
    37.  
    38. Return('group')

    第8行使用Split方法来将一个文件字符串分割成成一个list,其效果等价于

    1. src = ['board.c', 'stm32f10x_it.c', 'led.c', 'usart.c']

    第15行到第33行使用了GetDepend方法检查rtconfig.h中的某个宏是否打开,如果打开,则使用src += [src_name]来添加源码。最后使用DefineGroup创建组。

    添加库

    在进行编译时添加一个额外的库,需要注意不同的工具链对二进制库的命名。例如GCC工具链,它识别的是libabc.a这样的库名称,在指定库时是指定abc,而不是libabc。所以在链接额外库时需要在SConscript文件中特别注意。另外,在指定额外库时,也最好指定相应的库搜索路径,以下是一个示例:

    1. # RT-Thread building script for component
    2.  
    3. Import('rtconfig')
    4. from building import *
    5.  
    6. cwd = GetCurrentDir()
    7. src = Split('''
    8. ''')
    9.  
    10. LIBPATH = [cwd + '/libs']
    11. LIBS = ['abc']
    12.  
    13. group = DefineGroup('ABC', src, depend = [''], LIBS = LIBS, LIBPATH=LIBPATH)

    如果工具链是GCC,则库的名称应该是libabc.a;如果工具链是armcc,则库的名称应该是abc.lib。库的搜索路径是当前目录下的'libs'目录。

    增加一个SCons命令

    在RT-Thread tools目录下存放有RT-Thread自己定义的一些辅助building的脚本,例如用于自动生成RT-Thread针对一些IDE集成开发环境的工程文件。其中最主要的是building.py脚本。

    例如针对一个hello world的简单程序,假设它的源文件是:

    1. /* file: hello.c */
    2. #include <stdio.h>
    3.  
    4. int main(int argc, char** argv)
    5. {
    6. printf("Hello, world!\n");
    7. }

    只需要在这个文件目录下添加一个如下内容的SConstruct文件:

    1. Program('hello.c')

    然后在这个目录下执行命令:

    1. % scons
    2. scons: Reading SConscript files ...
    3. scons: done reading SConscript files.
    4. scons: Building targets ...
    5. cc -o hello.o -c hello.c
    6. cc -o hello hello.o
    7. scons: done building targets.

    将会在当前目录下生成hello的应用程序。所以相比于Makefile,一个简单的hello.c到hello的转换,只需要一句话。如果hello是由两个文件编译而成,也只需要把SConstruct文件修改成:

    1. Program(['hello.c', 'file1.c'])

    同时也可以指定编译出的目标文件名称:

    1. Program('program', ['hello.c', 'file1.c'])

    有的时候也可以偷偷懒,例如把当面目录下的所有C文件都作为源文件来编译:

    Glob函数就是用于使用当前目录下的所有C文件。除了Glob函数以外,也有Split函数。Split函数写的脚本具备更好的可读性以及更精确的可定制性:

    1. src = Split('''
    2. hello.c
    3. file1.c
    4. Program('program', src)

    它的效果与 是一致的,但具有更清晰的可读性。

    对于复杂、大型的系统,显然不仅仅是一个目录下的几个文件就可以搞定的,很可能是由数个文件夹一级级组合而成。

    在 SCons中,可以编写SConscript脚本文件来编译这些相对独立目录中的文件,同时也可以使用SCons中的Export和Import函数在SConstruct与SConscript文件之间共享数据(也就是Python中的一个对象数据)。