• CMake Tutorial 巡礼(5)_添加系统自察


    CMake Tutorial巡礼(5)_ 添加系统自察

    这是本系列的第六篇。
    上一篇我们学习了如何生成安装包,以及如何使用ctest进行简单的测试。
    这一篇我们继续学习,如何添加系统自察。什么是“系统自察”呢?具体到本例,就是对系统是否具有某些函数或功能进行预先的测试,通过判断系统的自身属性,在编译时决定采用不同的分支。

    本章导读

    在这里插入图片描述

    第五步:添加系统自察

    Let us consider adding some code to our project that depends on features the target platform may not have. For this example, we will add some code that depends on whether or not the target platform has the log and exp functions. Of course almost every platform has these functions but for this tutorial assume that they are not common.

    让我们考虑向项目中添加一些代码,这些代码依赖于目标平台可能没有的功能。对于此示例,我们将添加一些代码,这些代码将依赖于平台是否具有logexp函数,当然,几乎每个平台都具有这些功能,但在本Tutorial中,假设它们并不常见。

    If the platform has log and exp then we will use them to compute the square root in the mysqrt function. We first test for the availability of these functions using the CheckCXXSourceCompiles module in MathFunctions/CMakeLists.txt.

    如果平台有logexp函数,那么我们将使用它们来计算mysqrt函数中的平方根。我们首先使用MathFunctions/CMakeLists.txt中的CheckCXXSourceCompiles模块测试这些函数的可用性。

    Add the checks for log and exp to MathFunctions/CMakeLists.txt, after the call to target_include_directories():

    MathFunctions/CMakeLists.txt中添加针对logexp的测试,位于target_include_directories()调用之后。

    MathFunctions/CMakeLists.txt

    target_include_directories(MathFunctions
              INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}
              )
    
    # does this system provide the log and exp functions?
    include(CheckCXXSourceCompiles)
    check_cxx_source_compiles("
      #include 
      int main() {
        std::log(1.0);
        return 0;
      }
    " HAVE_LOG)
    check_cxx_source_compiles("
      #include 
      int main() {
        std::exp(1.0);
        return 0;
      }
    " HAVE_EXP)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    If available, use target_compile_definitions() to specify HAVE_LOG and HAVE_EXP as PRIVATE compile definitions.

    如果得到的验证结果是“可行”,使用target_compile_definitions()来指定HAVE_LOGHAVE_EXP,作为私有,即PRIVATE编译定义。

    MathFunctions/CMakeLists.txt

    if(HAVE_LOG AND HAVE_EXP)
      target_compile_definitions(MathFunctions
                                 PRIVATE "HAVE_LOG" "HAVE_EXP")
    endif()
    
    • 1
    • 2
    • 3
    • 4

    小白按:以上修改完成后,最终修改的结果为

    add_library(MathFunctions mysqrt.cxx)
    
    # state that anybody linking to us needs to include the current source dir
    # to find MathFunctions.h, while we don't.
    target_include_directories(MathFunctions
              INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}
              )
    
    # does this system provide the log and exp functions?
    include(CheckCXXSourceCompiles)
    check_cxx_source_compiles("
      #include 
      int main() {
        std::log(1.0);
        return 0;
      }
    " HAVE_LOG)
    check_cxx_source_compiles("
      #include 
      int main() {
        std::exp(1.0);
        return 0;
      }
    " HAVE_EXP)
    
    if(HAVE_LOG AND HAVE_EXP)
      target_compile_definitions(MathFunctions
                                 PRIVATE "HAVE_LOG" "HAVE_EXP")
    endif()
    
    # install rules
    install(TARGETS MathFunctions DESTINATION lib)
    install(FILES MathFunctions.h DESTINATION include)
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34

    If log and exp are available on the system, then we will use them to compute the square root in the mysqrt function. Add the following code to the mysqrt function in MathFunctions/mysqrt.cxx (don’t forget the #endif before returning the result!):

    如果logexp在系统上可用,那么我们将使用它们来计算mysqrt函数中的平方根。将以下代码添加到MathFunctions/mysqrt.cxx中的mysqrt函数(在返回结果之前不要忘记#endif):

    MathFunctions/mysqrt.cxx

    #if defined(HAVE_LOG) && defined(HAVE_EXP)
      double result = std::exp(std::log(x) * 0.5);
      std::cout << "Computing sqrt of " << x << " to be " << result
                << " using log and exp" << std::endl;
    #else
      double result = x;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    We will also need to modify mysqrt.cxx to include cmath.

    我们也需要修改mysqrt.cxx,使其包含cmath

    MathFunctions/mysqrt.cxx

    #include 
    
    • 1

    小白按:以上修改全部完成后,mysqrt.cxx的全貌为

    #include 
    #include   
    
    #include "MathFunctions.h" 
    // a hack square root calculation using simple operations
    double mysqrt(double x)
    {
      if (x <= 0) {
        return 0;
      }  
    
    #if defined(HAVE_LOG) && defined(HAVE_EXP)
      double result = std::exp(std::log(x) * 0.5);
      std::cout << "Computing sqrt of " << x << " to be " << result
                << " using log and exp" << std::endl;
    #else
      double result = x; 
    
      // do ten iterations
      for (int i = 0; i < 10; ++i) {
        if (result <= 0) {
          result = 0.1;
        }
        double delta = x - (result * result);
        result = result + 0.5 * delta / result;
        std::cout << "Computing sqrt of " << x << " to be " << result << std::endl;
      }
    #endif
      return result;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30

    Run the cmake executable or the cmake-gui to configure the project and then build it with your chosen build tool and run the Tutorial executable.

    执行 cmakecmake-gui,指定项目,并且用你选定的编译工具进行编译,最后运行Tutorial可执行文件。

    Which function gives better results now, sqrt or mysqrt?

    哪一个函数给出了更优的结果?sqrt还是mysqrt?

    小白按:给出所有的编译代码:

    mkdir Step5_build
    cd Step5_build
    cmake ../Step5
    cmake --build .
    cd Debug
    Tutorial 4294967206
    cd ..
    cmake ../Step5 -DUSE_MYMATH=OFF
    cmake --build .
    cd Debug
    Tutorial 4294967206
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    两个结果分别是:

    Computing sqrt of 4.29497e+09 to be 65536 using log and exp
    The square root of 4.29497e+09 is 65536
    
    • 1
    • 2
    The square root of 4.29497e+09 is 65536
    
    • 1

    可以看出,使用了logexp之后,平方根运算的精度已经和系统自带的sqrt函数不相上下了。

    这一小节还算是比较容易+平顺的,下一回合我们将要学习“添加自定义命令和生成文件”。

    【水平所限,错漏难免,创作不易,轻喷勿骂】

    在这里插入图片描述

  • 相关阅读:
    linux postgresql 常用指令
    C++标准模板(STL)- 类型支持 (数值极限,quiet_NaN,signaling_NaN,denorm_min)
    【校招VIP】前端专业课考点之CSMA/CD协议
    【k8s管理操作】
    VTK数据流程模型及基本对象的作用、关系介绍
    基于java《数据结构与算法》网上教学系统计算机毕业设计源码+系统+lw文档+mysql数据库+调试部署
    Linux应用开发基础知识——输入系统应用编程(七)
    12个微服务架构模式最佳实践
    Spring5 框架 ---- Spring5的新功能
    排序第三节——交换排序(冒泡排序+快速排序+快排的优化)(5个视频讲解)
  • 原文地址:https://blog.csdn.net/horsee/article/details/126793173