在ROS的编程过程中,如果CMakeLists.txt如果写不好,编译就很难成功。如果看不懂CMakeLists.txt那么很多错误你也不知道时什么回事。所以深入了解它是很右必要的。现在我们就来看看它。
我们使用cmake进行程序编译的时候,会根据CMakeLists.txt这个文件进行一步一步的处理,然后形成一个MakeFile文件,系统再通过这个文件的设置进行程序的编译。
我们可以先寻找一些cmake方面的东西进行一定的了解。ROS中的CMakeLists.txt也是基于普通的cmake的。
(1). RequiredCMakeVersion(cmake_minimum_required)
(2). PackageName(project())
(3). FindotherCMake/Catkinpackagesneededforbuild(find_package())
(4). Message/Service/ActionGenerators(add_message_files(),add_service_files(),add_action_files())
(5). Invokemessage/service/actiongeneration(generate_messages())
(6). Specifypackagebuildinfoexport(catkin_package())
(7). Libraries/Executablestobuild(add_library()/add_executable()/target_link_libraries())
(8). Teststobuild(catkin_add_gtest())
(9). Installrules(install())
下面就进行一一的解释。
catkinCMakeLists.txt都要以此开始,catkin编译需要2.8.3版本以上的cmake。
通过project()这个函数指定包的名字,在CMake中指定后,你可在其他地方通过使用变量${PROJECT_NAME}来引用它
这里指明构建这个package需要依赖的package,我们使用catkin_make的编译方式,至少需要catkin这个包。
find_package(catkinREQUIRED)
一个包被被find_package,那么就会导致一些CMake变量的产生,这些变量后面将在CMake的脚本中用到,这些变量描述了所依赖的包输出的头文件、源文件、库文件在哪里。这些变量的名字依照的惯例是
需要的所有包我们都可用这种方式包含进来,比如我们还需要roscpp,rospy,std_msgs。我们可以写成:
find_package(roscppREQUIRED)
find_package(rospyREQUIRED)
find_package(std_msgsREQUIRED)
这样的话,每个依赖的package都会产生几个变量,这样很不方便。所以还有另外一种方式:
find_package(catkinREQUIREDCOMPONENTS
roscpp
rospy
std_msgs
message_generation
)
这样,它会把所有pacakge里面的头文件和库文件等等目录加到一组变量上,比如:catkin_INCLUDE_DIRS,这样,我们就可以用这个变量查找需要的文件了。最终就只产生一组变量了。
当我们需要使用.msg.srv.action形式的文件时,我们需要特殊的预处理器把他们转化为系统可以识别特定编程语言(.h.cpp)。系统会用里面所有的(一些编程语言)生成器(比如gencpp,genpy,genlisp,etc)生成相应的.cpp.py文件。这就需要三个宏:add_message_files,add_service_files,add_action_files来相应的控制.msg.srv.action。这些宏后面必须跟着一个调用generate_messages()。他们的运用要注意一下几点:
1)这些宏必须在catkin_package()宏前面,即:
find_package(catkinREQUIREDCOMPONENTS...)
add_message_files(...)
add_service_files(...)
add_action_files(...)
generate_messages(...)
catkin_package(...)
...
2)宏catkin_package()中必须有CATKIN_DEPENDS依赖于message_runtime,即:
catkin_package(
...
CATKIN_DEPENDS message_runtime...
...)
3)find_package()必须依赖包message_generation
find_package(catkinREQUIRED COMPONENTS message_generation)
4)package.xml文件build_depend必须包含message_generation,run_depend必须包含message_runtime。
5)如果你有一个包编译.msg.srv,并且可执行文件要使用他们,那么你就需要创建一个显式的依赖项,自动生成message的target。这样才能按顺序来进行编译:
add_dependencies(some_target${PROJECT_NAME}_generate_messages_cpp)
这里的some_target是add_executable()设置的target的名字。
这是一个catkin提供的cmake宏,当我们要给构建系统指定catkin的特定的信息时就需要了,或者反过来利用他产生pkg-config和CMake文件。这个函数必须在声明add_library()或者add_executable()生成target之前使用。
编译产生的target有多种形式,通常有两种:程序可以运行的可执行文件以及在可执行文件编译和运行时要用到的库。
target的命名很重要,在catkin中target的名字必须是唯一的,和你之前构建产生的和安装的都不能相同。这只是cmake内部的需要。可以利用set_target_properties()函数将这个target进行重命名。例如:
set_target_properties(rviz_image_view
PROPERTIESOUTPUT_NAMEimage_view
PREFIX"")
这样就可将那个target rviz_image_view改为image_view。
ROS中的输出路径时默认的,但是也可通过下面进行修改:
set_target_properties(python_module_library
PROPERTIESLIBRARY_OUTPUT_DIRECTORY${CATKIN_DEVEL_PREFIX}/${CATKIN_PACKAGE_PYTHON_DESTINATION})
在我们指定target之前,我们需要指明对target而言,在哪里找源文件,特别是在哪里找头文件,在哪里找库文件。
IncludePaths:指明编译代码时在哪里寻找头文件;
LibraryPaths:指明可执行文件需要的库文件在哪里。
函数:
include_directories:他的参数是通过find_package产生的*_INCLUDE_DIRS变量和其他所有额外的头文件路径。例如:
include_directories(include${Boost_INCLUDE_DIRS}${catkin_INCLUDE_DIRS})
这里"include"表示你的pacakge里面的include这个路径也包含在里面。
link_directories():这个函数用来添加额外的库的路径,然而,这并不鼓励使用,因为所有的catkin和cmake的package在使用find_package时就已经自动的有他们的链接信息。简单的连接可以通过target_link_libraries()来进行。
例如:
add_executable(myProgramsrc/main.cppsrc/some_file.cppsrc/another_file.cpp)
add_library()用来指定编译产生的库。默认的catkin编译产生共享库:
add_library(${PROJECT_NAME}${${PROJECT_NAME}_SRCS})
使用target_link_libraries函数来指定可执行文件链接的库。这个要用在add_executable()后面。形如:
target_link_libraries(
比如,将可执行文件foo链接到库文件libmoo.so。
add_executable(foosrc/foo.cpp)
add_library(moosrc/moo.cpp)
target_link_libraries(foomoo)