cmake函数参数解析

最近在迁移公司的make系统到cmake上,发现cmake的function参数很奇怪。例如,如果我们向一个function传递list作为参数,在function中,形参会变成如下状况:

[plain]  view plain copy
  1. set(SRC)  
  2. list(APPEND SRC a.cpp b.cpp)  
  3. list(APPEND SRC c.cpp d.cpp)  
  4.   
  5. function(tst_arguments src_list)  
  6.     message("src_list = "${src_list})  
  7. endfunction()  
  8.   
  9. message("SRC = "${SRC})  
  10. tst_arguments(${SRC})  
  11.   
  12. ==== output ====  
  13. SRC = a.cppb.cppc.cppd.cpp  
  14. src_list = a.cpp  

 
很奇怪的是,这里的${SRC}在function外是完整的4个元素,而在function却只剩下了头一个元素(可能跟list的定长有关)。如果我们要传给function以n个源文件组成的list,这样显然不行。 

          一种简单的解决方法是使用ARGV,ARGC配合,他们的含义如同C/C++中main的argv和argc,分别代表参数和参数个数,使用如下方法解析参数:

[plain]  view plain copy
  1. function(tst_arguments src_list)  
  2.     message("ARGC = "${ARGC})  
  3.     message("ARGV = "${ARGV})  
  4.   
  5.         set(INDEX 0)  
  6.       
  7.     while(INDEX LESS ${ARGC})  
  8.         message("ARG = "${ARGV${INDEX}})  
  9.         math(EXPR INDEX "${INDEX} + 1")  
  10.     endwhile()  
  11. endfunction()  
  12.   
  13. tst_arguments(${SRC})  
  14.   
  15. ==== output ====  
  16. ARGC = 4  
  17. ARGV = a.cppb.cppc.cppd.cpp  
  18. ARG = a.cpp  
  19. ARG = b.cpp  
  20. ARG = c.cpp  
  21. ARG = d.cpp   
 

 
 
当然,你也可以使用cmake的foreach循环遍历参数。这招对付只有一个list的参数时十分有效,但是在出现多个参数的情况就很麻烦,如下: 

[plain]  view plain copy
  1. #  
  2. #假设函数link_lib将src_list中的源文件链接成库,根据type制定是链接静态库还是动态库  
  3. #  
  4. function(link_lib src_list type)  
  5.   
  6.         message("ARGC = "${ARGC})  
  7.     message("ARGV = "${ARGV})  
  8.           
  9.           
  10.         #以下根据参数的实际情做了操作,手动处理,以保证正确获取src_list和type  
  11.         set(INDEX 0)  
  12.         math(EXPR MAX "${ARGC} - 1")  
  13.         while(INDEX LESS ${MAX})  
  14.              #do something to link  
  15.              math(EXPR INDEX "${INDEX} + 1")  
  16.         endwhile()  
  17. endfunction()  
  18.   
  19. link_lib(${SRC} , so)  
  20.   
  21. ==== output ====  
  22. ARGC = 5  
  23. ARGV = a.cppb.cppc.cppd.cppso  
 
 

 
 
 
 
原来,ARG把两个参数混在了一起,虽然后面我们使用while进行了特殊处理,但是这对于cmake的函数不具备普遍性,移植起来很麻烦。 

决定版的solution是使用cmake的cmake_parse_arguments来解析函数参数,它有点像解析一个map键值对,首先看下它的函数原型:

include (CMakeParseArguments)  #必须包含这个cmake文件才能使用cmake_parse_arguments 
CMAKE_PARSE_ARGUMENTS(<prefix> <options> <one_value_keywords> <multi_value_keywords> args...)  

prefix是一个前缀,等会儿在引用参数的时候会提到。

<option>是一个列表,里面可以包含一些你感兴趣的KeyWord,随后可以通过它来看看你所需要的KeyWord是否被设置。

<one_value_keywords>是一个单值参数的KeyWord列表。

<multi_value_keywords>是一个多值参数的KeyWord列表(如list)。

下面举个例子,看看如何使用它们,首先定义所需要的函数,由于参数是由CMAKE_PARSE_ARGUMENTS来解析的,所以在函数声明中就不需要定义参数了:

[plain]  view plain copy
  1. function(tst_arguments)  
  2.   CMAKE_PARSE_ARGUMENTS(  
  3.     TEST "" "NAME;COMMAND;BASELINE"  
  4.        "ARGSLIST"  
  5.        ${ARGN}  
  6.   )  
  7.   
  8.   message("TEST_DEFAULT_ARGS is ${TEST_DEFAULT_ARGS} from ${ARGN}")  
  9.   message("TEST_NAME is ${TEST_NAME}")  
  10.   message("TEST_COMMAND is ${TEST_COMMAND}")  
  11.   message("TEST_ARGSLIST is ${TEST_ARGSLIST}")  
  12.   message("TEST_BASELINE is ${TEST_BASELINE}")  
  13.   
  14. endfunction(tst_arguments)  

"${ARGN}"是CMake中的一个变量,指代宏和函数中传入的多余参数。跟C语言函数参数中的...类似。

这里的前缀是TEST,

<one_value_keywords>我们设置单值参数的KeyWord(NAME;COMMAND;BASELINE),这将在随后的函数调用中注明KeyWord和Value的关系,

<multi_value_keywords>我们设置多值参数的KeyWord("ARGSLIST"),调用函数:

TEST_ARGUMENT(  
    NAME  
      testiso  
    COMMAND  
      "RunMe"  
    ARGSLIST  
      ${SRC}  
    BASELINE  
      "/home/sakaue/iWork"  
)  
  
==== output ====  
TEST_DEFAULT_ARGS is  from NAME;testiso;COMMAND;RunMe;ARGSLIST;a.cpp;b.cpp;c.cpp;d.cpp;BASELINE;/home/sakaue/iWork  
TEST_NAME is testiso  
TEST_COMMAND is RunMe  
TEST_ARGSLIST is a.cpp;b.cpp;c.cpp;d.cpp  
TEST_BASELINE is /home/sakaue/iWork  

可以看见,这里调用时的参数传递如同map一样<NAME ,testiso_${datafile} >,<COMMAND , "RunMe">,<ARGSLIST , ${SRC}>等等,在函数中,使用 前缀+KeyWord 来调用Value,这样比自己解析参数方便许多,而且也不会在还有list参数时和其他类型函数混在一起的情况。


更多讯息参考:http://www.cmake.org/cmake/help/v3.0/module/CMakeParseArguments.html?highlight=cmake_parse_arguments


http://blog.csdn.net/sakaue/article/details/38377661?utm_source=tuicool



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值