首先,我是LLVM传递的新手。
我试图在转换过程之后(使用C ++ API)向LLVM中的指令添加元数据。我打算存储此信息以供工具链中的其他工具使用。我有两个问题。
我希望我存储的信息作为元数据提供给另一个工作在LLVM IR上的工具。元数据是个好主意吗?我打算将字符串存储为带有一些指令的元数据。
如果元数据是正确的方式,我需要一些帮助来创建元数据节点。我打算使用setMedata()函数将它附加到指令。 setMetadata()的哪个变体是正确使用的变体。我不确定我的数据应该是哪个MDKind。我想创建一个MDString,将它附加到我的MDNode,然后用一条指令调用setMetadata()。如果我想将元数据附加到函数内的指令,我应该在setMedata()中使用什么上下文。上下文与元数据的相关性是什么?
我尝试在论坛和llvm doxygen文档中阅读了很多讨论,但我没有得到一个清晰而完整的答案来回答我的所有问题。感谢您的帮助或一些可以帮助我理解这一点的材料。
答案 0 :(得分:23)
在我看来:
如果你的“其他工具”本身不是通行证,那么是的,我认为元数据是最好的方法 - 将所有内容保存在IR中,易于通过眼睛识别,简单地手动添加以进行测试,以及 - 也许最多重要的是 - 只要您不重用现有的元数据类型,就不会与其他任何内容发生冲突。
但是,如果你的“其他工具”本身就是一个传递,那么还有一个选择:你可以让一个传递依赖于另一个,而不是直接在后面的传递中使用前面的信息。优点是您不必修改IR。
使用char*
的{{1}}变体,如下所示:
setMetadata
如果这是第一次在LLVMContext& C = Inst->getContext();
MDNode* N = MDNode::get(C, MDString::get(C, "my md string content"));
Inst->setMetadata("my.md.name", N);
中使用该字符串,它将自动注册setMetadata
作为模块中的新类型(我相信它在整个上下文中实际上是一致的)。您可以稍后使用以下方法检索字符串:
my.md.name
如果您想在同一范围内重复调用cast<MDString>(Inst->getMetadata("my.md.name")->getOperand(0))->getString();
或getMetadata
,您也可以使用setMetadata
来获取实际使用的类型,并使用这些变体使用种类值的方法。
最后,请注意,您可以将元数据节点范围限制在函数内部 - 使用Module::getMDKindID
变体 - 尽管我自己从未使用过它。