在下面的示例中,我试图提取每条指令引用的所有全局变量,但我一直遗漏其中一些。
... // inside a BB
for (Instruction * I : BB) {
for (Use &U : I->operands()) {
if(GlobalVariable * GV = dyn_cast<GlobalVariable >(U)){
// Do something with GV
GV->dump();
}
}
}
但是当我的目标是获得全球价值:
@end = global i8 0, align 1
@.str = private unnamed_addr constant [4 x i8] c"YES\00", align 1
@.str.2 = private unnamed_addr constant [3 x i8] c"NO\00", align 1
define void @TempF() {
entry:
%tmp8 = load i8, i8* @end, align 1
%tmp9 = trunc i8 %tmp8 to i1
%tmp10 = select i1 %tmp9, i8* getelementptr inbounds ([4 x i8], [4 x i8]* @.str, i32 0, i32 0), i8* getelementptr inbounds ([3 x i8], [3 x i8]* @.str.2, i32 0, i32 0)
ret void
}
在此IR上运行传递时,只会打印@end = global i8 0, align 1
。但不是......
@.str = private unnamed_addr constant [4 x i8] c"YES\00", align 1
@.str.2 = private unnamed_addr constant [3 x i8] c"NO\00", align 1
我理解@.str
不是@end
所属的操作数的一部分。 @.str
是其中一个操作数的引用。
为了能够在操作数中获取全局变量,我需要进行哪些修改?
答案 0 :(得分:2)
在您的情况下,select指令具有GetElementPtr operators。您将需要扩展您的代码以检测这种情况,然后遍历参数(指针和)到运算符。我建议在下面对你的for循环进行扩展,目前打印出其他这些情况。
for (Use &U : (&*I)->operands()) {
if(GlobalVariable * GV = dyn_cast<GlobalVariable>(U)){
// Do something with GV
GV->dump();
}
else if (GEPOperator* gepo = dyn_cast<GEPOperator>(&U))
{
errs() << "GEPO - " << *gepo << "\n";
if (GlobalVariable* gv = dyn_cast<GlobalVariable>(gepo->getPointerOperand()))
{
errs() << "GV - " << *gv << "\n";
}
for (auto it = gepo->idx_begin(), et = gepo->idx_end(); it != et; ++it)
{
if (GlobalVariable* gv = dyn_cast<GlobalVariable>(*it))
{
errs() << "GVi - " << *gv << "\n";
}
}
}
}