cmake 使用
# Copyright 2016 The Cartographer Authors # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. cmake_minimum_required(VERSION 2.8.12) # Ships with Ubuntu 14.04 (Trusty) project(cartographer_ros) set(PACKAGE_DEPENDENCIES cartographer_ros_msgs eigen_conversions geometry_msgs message_runtime nav_msgs pcl_conversions rosbag roscpp roslib sensor_msgs std_msgs tf2 tf2_eigen tf2_ros urdf visualization_msgs ) find_package(cartographer REQUIRED) include("${CARTOGRAPHER_CMAKE_DIR}/functions.cmake") set(BUILD_SHARED_LIBS OFF) option(BUILD_GRPC "build features that require Cartographer gRPC support" false) google_initialize_cartographer_project() google_enable_testing() find_package(catkin REQUIRED COMPONENTS ${PACKAGE_DEPENDENCIES}) include(FindPkgConfig) find_package(LuaGoogle REQUIRED) find_package(PCL REQUIRED COMPONENTS common) find_package(Eigen3 REQUIRED) find_package(Boost REQUIRED COMPONENTS system iostreams) find_package(urdfdom_headers REQUIRED) if(DEFINED urdfdom_headers_VERSION) if(${urdfdom_headers_VERSION} GREATER 0.4.1) add_definitions(-DURDFDOM_HEADERS_HAS_SHARED_PTR_DEFS) endif() endif() include_directories( ${urdfdom_headers_INCLUDE_DIRS} ) # Override Catkin's GTest configuration to use GMock. set(GTEST_FOUND TRUE) set(GTEST_INCLUDE_DIRS ${GMOCK_INCLUDE_DIRS}) set(GTEST_LIBRARIES ${GMOCK_LIBRARIES}) catkin_package( CATKIN_DEPENDS ${PACKAGE_DEPENDENCIES} DEPENDS # TODO(damonkohler): This should be here but causes Catkin to abort because # protobuf specifies a library '-lpthread' instead of just 'pthread'. # CARTOGRAPHER PCL EIGEN3 Boost urdfdom_headers INCLUDE_DIRS "." LIBRARIES ${PROJECT_NAME} ) file(GLOB_RECURSE ALL_SRCS "*.cc" "*.h") file(GLOB_RECURSE ALL_TESTS "*_test.cc") file(GLOB_RECURSE ALL_EXECUTABLES "*_main.cc") file(GLOB_RECURSE ALL_GRPC_FILES "cartographer_ros/cartographer_grpc/*") list(REMOVE_ITEM ALL_SRCS ${ALL_TESTS}) list(REMOVE_ITEM ALL_SRCS ${ALL_EXECUTABLES}) if (NOT ${BUILD_GRPC}) list(REMOVE_ITEM ALL_SRCS ${ALL_GRPC_FILES}) list(REMOVE_ITEM ALL_TESTS ${ALL_GRPC_FILES}) list(REMOVE_ITEM ALL_EXECUTABLES ${ALL_GRPC_FILES}) endif() add_library(${PROJECT_NAME} ${ALL_SRCS}) add_subdirectory("cartographer_ros") target_link_libraries(${PROJECT_NAME} PUBLIC cartographer) # Lua target_include_directories(${PROJECT_NAME} SYSTEM PUBLIC ${LUA_INCLUDE_DIR}) # PCL target_include_directories(${PROJECT_NAME} SYSTEM PUBLIC ${PCL_INCLUDE_DIRS}) target_link_libraries(${PROJECT_NAME} PUBLIC ${PCL_LIBRARIES}) set(BLACKLISTED_PCL_DEFINITIONS " -march=native -msse4.2 -mfpmath=sse ") foreach(DEFINITION ${PCL_DEFINITIONS}) list (FIND BLACKLISTED_PCL_DEFINITIONS "${DEFINITION}" DEFINITIONS_INDEX) if (${DEFINITIONS_INDEX} GREATER -1) continue() endif() set(TARGET_COMPILE_FLAGS "${TARGET_COMPILE_FLAGS} ${DEFINITION}") endforeach() # Eigen target_include_directories(${PROJECT_NAME} SYSTEM PUBLIC "${EIGEN3_INCLUDE_DIR}") # Boost target_include_directories(${PROJECT_NAME} SYSTEM PUBLIC "${Boost_INCLUDE_DIRS}") target_link_libraries(${PROJECT_NAME} PUBLIC ${Boost_LIBRARIES}) # Catkin target_include_directories(${PROJECT_NAME} SYSTEM PUBLIC ${catkin_INCLUDE_DIRS}) target_link_libraries(${PROJECT_NAME} PUBLIC ${catkin_LIBRARIES}) add_dependencies(${PROJECT_NAME} ${catkin_EXPORTED_TARGETS}) # Add the binary directory first, so that port.h is included after it has # been generated. target_include_directories(${PROJECT_NAME} PUBLIC $<BUILD_INTERFACE:${PROJECT_BINARY_DIR}> $<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}> $<INSTALL_INTERFACE:include> ) set(TARGET_COMPILE_FLAGS "${TARGET_COMPILE_FLAGS} ${GOOG_CXX_FLAGS}") set_target_properties(${PROJECT_NAME} PROPERTIES COMPILE_FLAGS ${TARGET_COMPILE_FLAGS}) if (CATKIN_ENABLE_TESTING) foreach(TEST_SOURCE_FILENAME ${ALL_TESTS}) get_filename_component(TEST_NAME ${TEST_SOURCE_FILENAME} NAME_WE) catkin_add_gtest(${TEST_NAME} ${TEST_SOURCE_FILENAME}) # catkin_add_gtest uses a plain (i.e. no PUBLIC/PRIVATE/INTERFACE) call to # target_link_libraries. That forces us to do the same. target_include_directories(${TEST_NAME} SYSTEM PUBLIC ${LUA_INCLUDE_DIR}) target_link_libraries(${TEST_NAME} ${LUA_LIBRARIES}) target_include_directories(${TEST_NAME} SYSTEM PUBLIC ${catkin_INCLUDE_DIRS}) target_link_libraries(${TEST_NAME} ${catkin_LIBRARIES}) add_dependencies(${TEST_NAME} ${catkin_EXPORTED_TARGETS}) target_link_libraries(${TEST_NAME} cartographer) target_link_libraries(${TEST_NAME} ${PROJECT_NAME}) set_target_properties(${TEST_NAME} PROPERTIES COMPILE_FLAGS ${TARGET_COMPILE_FLAGS}) endforeach() endif() install(DIRECTORY launch urdf configuration_files DESTINATION ${CATKIN_PACKAGE_SHARE_DESTINATION} ) install(PROGRAMS scripts/tf_remove_frames.py DESTINATION ${CATKIN_PACKAGE_BIN_DESTINATION} ) install(TARGETS ${PROJECT_NAME} ARCHIVE DESTINATION ${CATKIN_PACKAGE_LIB_DESTINATION} LIBRARY DESTINATION ${CATKIN_PACKAGE_LIB_DESTINATION} RUNTIME DESTINATION ${CATKIN_GLOBAL_BIN_DESTINATION} ) # Install source headers. file(GLOB_RECURSE HDRS "*.h") foreach(HDR ${HDRS}) file(RELATIVE_PATH REL_FIL ${PROJECT_SOURCE_DIR} ${HDR}) get_filename_component(INSTALL_DIR ${REL_FIL} DIRECTORY) install( FILES ${HDR} DESTINATION include/${INSTALL_DIR} ) endforeach()
find_package是CMake中用于查找并加载外部项目或库的命令。它尝试找到指定的包,并提供该包的详细信息,如头文件的位置、库文件的位置等,以便在项目中使用。
find_package的工作模式
find_package有两种工作模式:Module模式和Config模式。
-
Module模式:这是find_package的基础工作模式,也是默认模式。在这种模式下,CMake会寻找一个名为Find<PackageName>.cmake的模块文件。这个文件通常由CMake、操作系统或项目本身提供,而不是由包本身提供。这些模块文件通常位于CMake的模块路径或CMake安装目录下的Modules文件夹中。
-
Config模式:这是find_package的高级工作模式。只有在find_package中指定了CONFIG、NO_MODULE等关键字,或者在Module模式查找失败后,才会进入Config模式。在Config模式下,CMake会寻找名为<PackageName>Config.cmake或<lowercase-package-name>-config.cmake的配置文件。这些文件通常由库的开发者提供,并安装在系统目录中。
find_package的基本用法
一个典型的find_package调用可能如下所示:
-
<PackageName>:要查找的包名。
-
<version>:可选参数,指定要查找的包的版本。
-
REQUIRED:如果指定,表示包是必需的,如果找不到包,CMake配置将失败。
-
COMPONENTS:可选参数,指定要查找的包的特定组件。
find_package的搜索顺序
在Module模式下,find_package首先在CMAKE_MODULE_PATH变量指定的路径中查找,如果没有找到,它会在CMake的安装目录下的Modules文件夹中查找。
在Config模式下,搜索顺序更为复杂,包括但不限于以下路径:
-
CMAKE_PREFIX_PATH、CMAKE_FRAMEWORK_PATH、CMAKE_APPBUNDLE_PATH等CMake变量或环境变量指定的路径。
-
系统环境PATH变量指定的路径。
-
CMAKE_SYSTEM_PREFIX_PATH、CMAKE_SYSTEM_FRAMEWORK_PATH、CMAKE_SYSTEM_APPBUNDLE_PATH等CMake系统变量指定的路径。
如何使用find_package
为了使用find_package,你需要确保CMake可以找到相应的模块文件。如果你是包的开发者,你需要提供<PackageName>Config.cmake或Find<PackageName>.cmake文件,并确保它们被安装到CMake可以找到的位置。
如果你是包的使用者,你需要确保这些文件存在于你的系统中,并且CMAKE_MODULE_PATH或其他相关变量已经正确设置,以便CMake可以找到它们。
总结
find_package是CMake中一个强大的命令,它简化了查找和使用外部依赖的过程。理解它的工作原理和使用方法对于CMake项目的构建至关重要。通过正确使用find_package,你可以轻松地将外部库集成到你的项目中,无论这些库是使用CMake构建的还是非CMake构建的。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】