我想将可变数量的参数传递给我的LLVM opt pass。
为此,我做了类似的事情:
static cl::list<std::string> Files(cl::Positional, cl::OneOrMore);
static cl::list<std::string> Libraries("l", cl::ZeroOrMore);
但是,如果我现在调用选择:
foo@foo-Ubuntu:~/llvm-ir-obfuscation$ opt -load cmake-build-debug/water/libMapInstWMPass.so -mapiWM programs/ll/sum100.ll -S 2 3 4 -o foo.ll
opt: Too many positional arguments specified!
Can specify at most 2 positional arguments: See: opt -help
,然后我得到错误,选择最多接受2个位置参数。
我做错了什么?
答案 0 :(得分:3)
我认为问题是opt
已经解析了自己的参数,并且已经将bitcode文件作为位置参数进行处理,因此具有多个位置参数会产生歧义。
该文档解释了API,就像它在独立应用程序中使用一样。所以,例如,如果你做这样的事情:
int main(int argc, char *argv[]) {
cl::list<std::string> Files(cl::Positional, cl::OneOrMore);
cl::list<std::string> Files2(cl::Positional, cl::OneOrMore);
cl::list<std::string> Libraries("l", cl::ZeroOrMore);
cl::ParseCommandLineOptions(argc, argv);
for(auto &e : Libraries) outs() << e << "\n";
outs() << "....\n";
for(auto &e : Files) outs() << e << "\n";
outs() << "....\n";
for(auto &e : Files2) outs() << e << "\n";
outs() << "....\n";
}
你得到这样的东西:
$ foo -l one two three four five six
one
....
two
three
four
five
....
six
....
现在,如果您交换两个位置参数定义,或者甚至将cl::OneOrMore
Files2
选项更改为cl::ZeroOrMore
,您将收到错误
$ option: error - option can never match, because another positional argument will match an unbounded number of values, and this option does not require a value!
就个人而言,当我使用opt
时,我放弃了positiontal参数选项并执行以下操作:
cl::list<std::string> Lists("lists", cl::desc("Specify names"), cl::OneOrMore);
允许我这样做:
opt -load ./fooPass.so -foo -o out.bc -lists one ./in.bc -lists two
并按照我获得的方式迭代std::string
列表:
one
two
答案 1 :(得分:2)
正如@compor建议的那样,这可能与交织opt
的参数和你自己的传递有关。 CommandLine库主要是为LLVM框架内的独立应用程序编写的。
但是,您可以执行以下操作:
static cl::list<std::string> Args1("args1", cl::Positional, cl::CommaSeparated);
static cl::list<std::string> Args2("args2", cl::ZeroOrMore);
这样做的好处是,您可以使用逗号(例如arg1,arg2,...
或使用标识符-args1 arg1 arg2 ...
)在命令行上输入多个参数;并将这些插入到Args1
列表中。如果您只在命令行上提供单个位置参数arg
,则Args1
将仅包含此参数。
此外,您可以在命令行中指定-args2 arg
,无论您身在何处(命名为非位置选项)。这些将进入Args2
列表。