zoukankan      html  css  js  c++  java
  • cmake多目录,生成so的模板

    什么是CMake

      你或许听过好几种 Make 工具,例如 GNU Make ,QT 的 qmake ,微软的 MS nmake,BSD Make(pmake),Makepp,等等。这些 Make 工具遵循着不同的规范和标准,所执行的 Makefile 格式也千差万别。这样就带来了一个严峻的问题:如果软件想跨平台,必须要保证能够在不同平台编译。而如果使用上面的 Make 工具,就得为每一种标准写一次 Makefile ,这将是一件让人抓狂的工作。

      CMake就是针对上面问题所设计的工具:它首先允许开发者编写一种平台无关的CMakeLists.txt文件来定制整个编译流程,然后再根据目标用户的平台进一步生成所需的本地化Makefile和工程文件,如Unix的Makefile或Windows的Visual Studio工程。从而做到"Write once, run everywhere"。显然,CMake是一个比上述几种make更高级的编译配置工具。

      在linux平台下使用CMake生成Makefile并编译的流程如下:

      1、编写CMake配置文件CMakeLists.txt

      2、执行命令cmake PATH,其中PATH是CMakeLists.txt所在的目录。

      3、使用make命令进行编译。

      一个通用的模板如下。

    源码目录结构

    ├── CMakeLists.txt     根目录cmakelists.t
    ├── math
    │   ├── inc
    │    │    ├── MathFunctions.h
    │   ├── src
    │    │    ├── MathFunctions.cc
    │   └── CMakeLists.txt 子目录cmakelists.txt
    ├── main.c  main函数
    └── README.md

    源码

    根目录CMakeLists.txt

    #-------------------------------------------------------------------------------
    # File: CMakeLists.txt
    # Description: CMake Configuration File
    #-------------------------------------------------------------------------------
    # CMake最低版本要求
    cmake_minimum_required(VERSION 2.8)
    # 项目名称
    project(Template)
    
    #-------------------------------------------------------------------------------
    # project setting
    #-------------------------------------------------------------------------------
    # setting for compiling in debug or release mode
    if(${CMAKE_BUILD_TYPE} MATCHES "Release")
      message(STATUS "Release版本")
      set(BuildType "Release")
    else()
      message(STATUS "Debug版本")
      set(BuildType "Debug")
    endif()
    
    # 设置bin和lib库目录
    # PROJECT_SOURCE_DIR为cmake内置变量,表示工程的根目录
    set(RELEASE_DIR ${PROJECT_SOURCE_DIR}/release)
    # debug和release版本目录不一样
    # LIBRARY_OUTPUT_PATH为cmake内置变量,表示目标链接库文件的存放位置
    set(LIBRARY_OUTPUT_PATH ${RELEASE_DIR}/linux/${BuildType})
    set(EXECUTABLE_OUTPUT_PATH ${RELEASE_DIR}/linux/${BuildType})
    
    #-----------------------------------------------------------------------------
    # compile setting
    #-----------------------------------------------------------------------------
    # 设置编译选项
    # -fPIC:标识编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的
    # add_compile_options是针对所有类型编译器的
    add_compile_options(-fPIC)
    # 设置c++编译器编译选项        
    set(CXX_FLAGS
      -std=c++11
      -Wno-overflow
      -Wno-conversion-null
      -Wno-maybe-uninitialized
      -Wno-uninitialized
      -Wno-sequence-point
      -Wno-write-strings
      -Wno-unused-label
      -Wno-unused-variable
      -Wno-unused-variable
      -Wno-cpp
      -Wno-deprecated
      -Wno-unused-but-set-variable -pthread)
    
    # string(REPLACE <match-string> <replace-string> <out-var> <input>...)
    # 替换CXX_FLAGS变量中的;字符为空格,然后写入CMAKE_CXX_FLAGS,CMAKE_CXX_FLAGS为cmake内置变量,用于设置C++编译选项
    string(REPLACE ";" " " CMAKE_CXX_FLAGS "${CXX_FLAGS}")
    # CMAKE_CXX_COMPILER为cmake内置变量,用于指定gcc,g++版本编译
    set(CMAKE_CXX_COMPILER "g++")
    # 指定Debug模式下编译选项,CMAKE_CXX_FLAGS_DEBUG为cmake内置变量
    set(CMAKE_CXX_FLAGS_DEBUG "-g -O0")
    # 指定Release模式下编译选项,CMAKE_CXX_FLAGS_RELEASE为cmake内置变量
    # -g表示添加gdb调试选项
    set(CMAKE_CXX_FLAGS_RELEASE "-g -O2 -finline-limit=1000")
    
    
    MESSAGE(STATUS "CMAKE_BUILD_TYPE : ${CMAKE_BUILD_TYPE}")
    IF( "${CMAKE_BUILD_TYPE}" STREQUAL "Release")
        MESSAGE(STATUS "CMAKE_CXX_FLAGS : ${CMAKE_CXX_FLAGS_RELEASE} ${CMAKE_CXX_FLAGS}")
    ELSE()
        MESSAGE(STATUS "CMAKE_CXX_FLAGS : ${CMAKE_CXX_FLAGS_DEBUG} ${CMAKE_CXX_FLAGS}")
    ENDIF()
    
    #-------------------------------------------------------------------------------
    # thirdparty dependencies
    #-------------------------------------------------------------------------------
    
    #-------------------------------------------------------------------------------
    # project generation
    #-------------------------------------------------------------------------------
    
    # 查找当前目录下的所有源文件
    # 并将名称保存到DIR_SRCS 变量中
    aux_source_directory(. DIR_SRCS)
    # 向当前工程添加存放源文件的子目录
    # 注意,该子目录中须包含CMakeList.txt文件
    add_subdirectory(math)
    # 添加当前编译目标使用到的头文件搜索路径,多个路径之间用空格分隔
    # 如果路径包含空格,可以使用双引号将它括起来
    include_directories("${PROJECT_SOURCE_DIR}/math/inc")
    # 指定生成目标
    add_executable(dynamic main.cc)
    # 添加链接库
    target_link_libraries(dynamic mathLib)

    子目录CMakeLists.txt

    # 设置release版本还是debug版本
    if(${CMAKE_BUILD_TYPE} MATCHES "Release")
        message(STATUS "Release版本")
        set(BuildType "Release")
    else()
        set(BuildType "Debug")
        message(STATUS "Debug版本")
    endif()
    
    #设置lib库目录
    set(RELEASE_DIR ${PROJECT_SOURCE_DIR}/release)
    # debug和release版本目录不一样
    #设置生成的so动态库最后输出的路径
    set(LIBRARY_OUTPUT_PATH ${RELEASE_DIR}/linux/${BuildType})
    add_compile_options(-fPIC)
    
    # 查找src目录下的所有源文件
    # 并将名称保存到 DIR_LIB_SRCS 变量
    aux_source_directory(src DIR_LIB_SRCS)
    # 添加当前编译目标使用到的头文件搜索路径,多个路径之间用空格分隔
    # 如果路径包含空格,可以使用双引号将它括起来
    include_directories("${PROJECT_SOURCE_DIR}/math/inc")
    # 生成动态库
    add_library (mathLib SHARED  ${DIR_LIB_SRCS})
    それでも私の大好きな人
  • 相关阅读:
    java编程思想概括(第二章)一切都是对象
    java编程思想概括(第六章)复用类
    .net程序在无.net环境下运行
    一个让人哭笑不得的触发器
    iReaper for android
    博易博客删除垃圾评论
    解决博易2.0版分页问题
    允许更新此预编译站点的作用
    java包的命名规则技巧
    简易拨号器iCall
  • 原文地址:https://www.cnblogs.com/codingmengmeng/p/14304875.html
Copyright © 2011-2022 走看看