make中使用了目标和为伪目标相互依赖的方法将需要编译的项目和
依赖的目标链接起来。比如一个.bin文件依赖.o ,而.o文件依赖于
.s 或者.c。
对于编译项目,需要处理的基本问题就是如上的需求。
而scons的编译函数中都有一个编译目标 和编译来源的入参,因此scons
在编译过程中会自动推导依赖关系,而不用显示的去指明依赖链条.
pip install scons
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ echo Program\(\"main.cpp\"\) > Sconstruct
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
Program("main.cpp")
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons
scons: Reading SConscript files ...
scons: done reading SConscript files.
scons: Building targets ...
g++ -o main.o -c main.cpp
g++ -o main main.o
scons: done building targets.
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ ls
main main.cpp main.o Sconstruct
1.2 分析上边的过程Program函数是用来生成最终的执行目标,可以自动推导默认的编译方法编译出最终目标ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ ls
main main.cpp main.o Sconstruct
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -c
scons: Reading SConscript files ...
scons: done reading SConscript files.
scons: Cleaning targets ...
Removed main.o
Removed main
scons: done cleaning targets.
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ ls
main.cpp Sconstruct
2.2 scons 上不用去专门编写项目的清理方法,scons会自动由最终结果往前的这个链条上所有中间文件,并删除掉ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons --tree=all
scons: Reading SConscript files ...
scons: done reading SConscript files.
scons: Building targets ...
g++ -o main.o -c main.cpp
g++ -o main main.o
+-.
+-Sconstruct
+-main
| +-main.o
| | +-main.cpp
| | +-/usr/bin/g++
| +-/usr/bin/g++
+-main.cpp
+-main.o
+-main.cpp
+-/usr/bin/g++
scons: done building targets.
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q
g++ -o main.o -c main.cpp
g++ -o main main.o
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -c -Q
Removed main.o
Removed main
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ echo Program\(\"nice\",\"main.cpp\"\) > Sconstruct
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
Program("nice","main.cpp")
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q
g++ -o main.o -c main.cpp
g++ -o nice main.o
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ ls
main.cpp main.o nice Sconstruct
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cp main.cpp main2.cpp
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ echo Program\(\"main2.cpp\"\) >> Sconstruct
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q
g++ -o main.o -c main.cpp
g++ -o main main.o
g++ -o main2.o -c main2.cpp
g++ -o main2 main2.o
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -c -Q
Removed main.o
Removed main
Removed main2.o
Removed main2
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q main2
g++ -o main2.o -c main2.cpp
g++ -o main2 main2.o
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
Program("nice",["main.cpp","res/help.cpp"])
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q
g++ -o main.o -c main.cpp
g++ -o res/help.o -c res/help.cpp
g++ -o nice main.o res/help.o
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
file=Glob("*.cpp") + Glob("*/*.cpp")
print(file)
Program("nice",file)
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q
[<SCons.Node.FS.File object at 0x56510efe8698>, <SCons.Node.FS.File object at 0x56510efe9b08>]
g++ -o main.o -c main.cpp
g++ -o res/help.o -c res/help.cpp
g++ -o nice main.o res/help.o
这里看到Glob函数能做模糊匹配的方式匹配到源文件,并且搜到的源文件可以参与到编译中
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
file=Glob("*.cpp") + Glob("*/*.cpp")
tmp_file = []
for item in file:
tmp_file.append(str(item))
print(tmp_file)
Program("nice",file)
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons -Q
['main.cpp', 'res/help.cpp']
g++ -o main.o -c main.cpp
g++ -o res/help.o -c res/help.cpp
g++ -o nice main.o res/help.o
工程应用中经常会遇到各个阶段编译参数不一致的情况,需要拆分出单个步骤进行编译
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
env = Environment()
env2 = env.Clone()
env3 = env.Clone()
#1. 设置汇编阶段
env["CC"] = "g++"
env["CCFLAGS"] = ["-DCCFLAG","-S"]
env["OBJSUFFIX"] = [".s"]
#2. 设置编译阶段
env2["AS"] = "g++"
env2["ASFLAGS"] = ["-DASFLAGS","-c"]
env2["OBJSUFFIX"] = [".o"]
#3. 设置链接阶段
env3["LINK"] = "g++"
env3["LINKFLAGS"] = ["-DLINKFALGS",]
outfile1 = env.Object("main.s","main.cpp")
outfile2 = env2.Object("main.o","main.s")
env3.Program("aaa", "main.o")
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons
scons: Reading SConscript files ...
scons: done reading SConscript files.
scons: Building targets ...
g++ -o main.s -c -DCCFLAG -S main.cpp
g++ -DASFLAGS -c -o main.o main.s
g++ -o aaa -DLINKFALGS main.o
scons: done building targets.
Sconstruct 文件是一个工程的编译入口,一般在大型工程中不能将所有编译过程全部放到入口文件中,这样会导致工程可读性变差。
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ tree
.
├── build
│ └── file1
├── Sconstruct
└── source
└── pro1
└── main.cpp
4 directories, 6 files
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat Sconstruct
file1 = "build/file1"
SConscript(file1)
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ cat build/file1
import os
env = Environment()
#这里路径最多退到Sconstruct 这一层
soruce_dir = "../source/pro1"
srcfile = ["main.cpp",]
usefile = [os.path.join(soruce_dir,f) for f in srcfile]
env.Program("aaa", usefile)
ggg@ggg-X550JX ~/test_demo/test_demo/pro1 $ scons
scons: Reading SConscript files ...
scons: done reading SConscript files.
scons: Building targets ...
g++ -o source/pro1/main.o -c source/pro1/main.cpp
g++ -o build/aaa source/pro1/main.o
scons: done building targets.
根据命令输入目标缩小搜查范围,并只做部分编译(待完善)
comd="md5sum $SOURCE > $TAEGET"
target2 =env.Command("output.md5",srcfile,comd)
env.Clone()
#单个环境下如果重复给同一个环境进行环境变量替换,那么最终编译时候都会按最后一次赋值进行编译
pipe=os.open("%s 2>&1;echo ---;env" %cmd,"r")
text=pip.read()
sts=pipe.close()
最终解析出需要的环境变量添加到
os.environ[k] =v
# 在编译文件内进行区别的进行import 导入参数文件(python 格式的变量)
# 部分打印信息令人迷惑,实际上可能会被执行两次以上