cmake函数參数解析

近期在迁移公司的make系统到cmake上。发现cmake的function參数非常奇怪。比如,假设我们向一个function传递list作为參数,在function中,形參会变成例如以下状况:

set(SRC)
list(APPEND SRC a.cpp b.cpp)
list(APPEND SRC c.cpp d.cpp)

function(tst_arguments src_list)
	message("src_list = "${src_list})
endfunction()

message("SRC = "${SRC})
tst_arguments(${SRC})

==== output ====
SRC = a.cppb.cppc.cppd.cpp
src_list = a.cpp

非常奇怪的是,这里的${SRC}在function外是完整的4个元素。而在function却仅仅剩下了头一个元素(可能跟list的定长有关)。

假设我们要传给function以n个源文件组成的list,这样显然不行。

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

function(tst_arguments src_list)
	message("ARGC = "${ARGC})
	message("ARGV = "${ARGV})

        set(INDEX 0)

	while(INDEX LESS ${ARGC})
		message("ARG = "${ARGV${INDEX}})
		math(EXPR INDEX "${INDEX} + 1")
	endwhile()
endfunction()

tst_arguments(${SRC})

==== output ====
ARGC = 4
ARGV = a.cppb.cppc.cppd.cpp
ARG = a.cpp
ARG = b.cpp
ARG = c.cpp
ARG = d.cpp



当然,你也能够使用cmake的foreach循环遍历參数。这招对付仅仅有一个list的參数时十分有效。可是在出现多个參数的情况就非常麻烦。例如以下:

#
#如果函数link_lib将src_list中的源文件链接成库,依据type制定是链接静态库还是动态库
#
function(link_lib src_list type)

        message("ARGC = "${ARGC})
	message("ARGV = "${ARGV})

        #下面依据參数的实际情做了操作,手动处理。以保证正确获取src_list和type
        set(INDEX 0)
        math(EXPR MAX "${ARGC} - 1")
        while(INDEX LESS ${MAX})
             #do something to link
             math(EXPR INDEX "${INDEX} + 1")
        endwhile()
endfunction()

link_lib(${SRC} , so)

==== output ====
ARGC = 5
ARGV = a.cppb.cppc.cppd.cppso






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

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

include (CMakeParseArguments)  #必须包括这个cmake文件才干使用<span class="highlighted">cmake_parse_arguments</span>

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来解析的,所以在函数声明中就不须要定义參数了:

function(tst_arguments)
  CMAKE_PARSE_ARGUMENTS(
    TEST "" "NAME;COMMAND;BASELINE"
       "ARGSLIST"
       ${ARGN}
  )

  message("TEST_DEFAULT_ARGS is ${TEST_DEFAULT_ARGS} from ${ARGN}")
  message("TEST_NAME is ${TEST_NAME}")
  message("TEST_COMMAND is ${TEST_COMMAND}")
  message("TEST_ARGSLIST is ${TEST_ARGSLIST}")
  message("TEST_BASELINE is ${TEST_BASELINE}")

endfunction(tst_arguments)

这里的前缀是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

时间: 2024-08-07 00:13:07

cmake函数參数解析的相关文章

linux kernel的cmdline參数解析原理分析

利用工作之便,今天研究了kernel下cmdline參数解析过程.记录在此.与大家共享.转载请注明出处.谢谢. Kernel 版本:3.4.55 Kernel启动时会解析cmdline,然后依据这些參数如console root来进行配置执行. Cmdline是由bootloader传给kernel.如uboot.将须要传给kernel的參数做成一个tags链表放在ram中,将首地址传给kernel,kernel解析tags来获取cmdline等信息. Uboot传參给kernel以及kerne

Python命令行选项參数解析策略

概述 在Python的项目开发过程中,我们有时须要为程序提供一些能够通过命令行进行调用的接口.只是,并非直接使用 command + 当前文件 就ok的,我们须要对其设置可选的各种各样的操作类型.所以,这样的情况下我们就有必要对传入的參数进行解析操作. 以下就此问题提出几种不同的解决策略.希望于你故意. 版权说明 著作权归作者全部. 商业转载请联系作者获得授权,非商业转载请注明出处. 作者:Coding-Naga 发表日期: 2016年3月18日 链接:http://blog.csdn.net/

【通过操作指针,与指针做函数參数&amp;#39;实现字串在主串中出现的次数,然后将出现的部分依照要求进行替换 】

#include<stdio.h> #include<stdlib.h> int strTime(const char *str1, const char *str2, int *time) { int count = 0; char *p1 = str1; char *p2 = str2; //p1是第一次出现的位置 p1 = strstr(p1, p2); //注意这里不要写成*p1!=NULL 由于p1 是null的地址一旦读取*p1 会出错的!!!!不能读取操作系统的数据

C语言中的system函数參数具体解释

http://blog.csdn.net/pipisorry/article/details/33024727 函数名: system 功   能: 发出一个DOS命令 用   法: int system(char *command); system函数已经被收录在标准c库中,能够直接调用 system()函数用于向操作系统传递控制台命令行,以WINDOWS系统为例,通过system()函数运行命令和在DOS窗体中运行命令的效果是一样的,所以仅仅要在运行窗体中能够使用的命令都能够用SYSTEM(

JavaScript的最大函数參数长度和最大栈深度检測

一般代码也许不会涉及最大參数长度和最大栈深度,但某些特殊场合,检測这两个參数还是有必要的.比如:用递归计算斐波那契数列的第n个值,不了解最大栈深度,难免显得肤浅.又比如:将一串charCode转成String,不了解最大參数长度.採用字符串拼接的方式,效率提不上.特别是在串较长的情况下. 下面两个方法分别实现了JavaScript执行环境的 最大函数參数长度检測 和 最大栈深度检測. 不同浏览器的測试结果不同,同一款浏览器在不同机器測试结果不同.甚至某些浏览器连续測试时先后结果会有不同. fun

输入10个数,求出最大元素是第几个数(数组作为函数參数)

小结: 调用有參数的函数时,须要提供实參.实參能够是常量.变量.表达式. 数组元素的作用相当于变量. 数组元素能够作为函数实參,其方法与变量同样.向形參传递数组元素的值.数组名也可作为实參和形參,传递的是数组的首地址. 数组元素能够作函数的实參,可是不能作形參. 由于形參是在函数调用时暂时分配存储单元的.不可能为一个数组元素独立分配存储单元(数组是一个总体,在内存中占连续的一段存储单元).在用数组元素作为函数实參时,把实參传递给形參.是"值传递",数据传递方向是从实參传到形參.单向传递

函数指针作为函数參数,实现冒泡排序的升序排序和降序排序

#include<stdio.h> #define N 10//定义数组元素个数 int Ascending(int a,int b);//升序排列的函数声明 int Descending(int a,int b);//降序排列的函数声明 void swap(int*,int*);//交换数据的函数声明 void BubbleSort(int a[],int n,int (*compare)(int,int));//声明排序函数,通过函数指针作为函数调用 void Display(int a[

C语言函数參数传递原理

C语言中參数的传递方式一般存在两种方式:一种是通过栈的形式传递.还有一种是通过寄存器的方式传递的. 这次.我们仅仅是具体描写叙述一下第一种參数传递方式,第二种方式在这里不做具体介绍. 首先,我们看一下,以下一个简单的调用例程: int Add (int a, int b, int c) { return a+b+c; } void main() { int x =0 , y = 1, z = 2; int result = 0; result = Add(x, y, z); printf("Re

python 命令行參数解析

本文是从我还有一个博客转载过来的,欢迎大家点击进去看一下,帮我添加点人气^_^ ImPyy 选择模块 依据python參考手冊的提示,optparse 已经废弃,应使用 argparse 教程 概念 argparse 模块使用 add_argument 来加入可选的命令行參数,原型例如以下: ArgumentParser.add_argument(name or flags...[, action][, nargs][, const][, default][, type][, choices][