1. 程式人生 > >cmake 學習筆記(三)

cmake 學習筆記(三)

學習一下cmake的 finder。

finder是神馬東西?

當編譯一個需要使用第三方庫的軟體時,我們需要知道:

去哪兒找標頭檔案 .h

對比GCC的 -I 引數

去哪兒找庫檔案 (.so/.dll/.lib/.dylib/...)

對比GCC的 -L 引數

需要連結的庫檔案的名字

對比GCC的 -l 引數

這也是一個 finder 需要返回的最基本的資訊。

如何使用?

比如說,我們需要一個第三方庫 curl,那麼我們的 CMakeLists.txt 需要指定標頭檔案目錄,和庫檔案,類似:

include_directiories(/usr/include)
target_link_libraries(myprogram curl)

如果藉助於cmake提供的finder會怎麼樣呢?使用cmake的Modules目錄下的FindCURL.cmake,相應的 CMakeList.txt 檔案:

find_package(CURL REQUIRED)
include_directories(${CURL_INCLUDE_DIR})
target_link_libraries(curltest ${CURL_LIBRARY})

或者

find_package(CURL)
if(CURL_FOUND)
include_directories(${CURL_INCLUDE_DIR})
target_link_libraries(curltest ${CURL_LIBRARY})
else(CURL_FOUND)
message(FATAL_ERROR "curl not found!")
endif(CURL_FOUND)

如果我們使用的finder,不是cmake自帶的怎麼辦?

  • 放置位置:工程根目錄下的 cmake/Modules/
  • 然後在 CMakeList.txt 中新增
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/Modules/")

find_package如何工作

find_package 將會在module路徑下查詢 Find<name>.cmake。首先它搜尋 ${CMAKE_MODULE_PATH}中的所有路徑,然後搜尋 <CMAKE_ROOT>/share/cmake-x.y/Modules/

如果這個檔案未找到,它將會查詢 <Name>Config.cmake 或 <lower-case-name>-config.cmake 檔案。這兩個檔案是庫檔案安裝時自己安裝的,將自己的路徑硬編碼到其中。

前者稱為 module 模式,後者稱為 config 模式

每個模組一般都會提供一下幾個變數

  • <name>_FOUND

  • <name>_INCLUDE_DIR 或 <name>_INCLUDES

  • <name>_LIBRARY 或 <name>_LIBRARIES 或 <name>_LIBS

  • <name>_DEFINITIONS

編寫finder

  • 首先使用 find_package 探測本軟體包依賴的第三方庫(引數 QUIETLY 和 REQUIRED應該被傳遞)
  • 如果 pkg-config 可用,則可以用其去探測include/library路徑
  • 分別使用 find_path 和 find_library 查詢標頭檔案和庫檔案
    • pkg-config 提供的路徑僅作為參考
    • CMake 有很多硬編碼的路徑
    • 結果放到 <name>_INCLUDE_DIR 和 <name>_LIBRARY (注意:單數而不是複數)

  • 設定 <name>_INCLUDE_DIRS 為 <name>_INCLUDE_DIR <dependency1>_INCLUDE_DIRS ...

  • 設定 <name>_LIBRARIES 為 <name>_LIBRARY <dependency1>_LIBRARIES ...

    • 依賴使用複數,包自身使用單數形式(由find_path和find_library提供)
  • 呼叫巨集 find_package_handle_standard_args() 設定 <name>_FOUND 並列印或失敗資訊

參考