如何验证 LLVM“ret”指令是不是返回 void?

Posted

技术标签:

【中文标题】如何验证 LLVM“ret”指令是不是返回 void?【英文标题】:How to verify if LLVM "ret" instruction returns void?如何验证 LLVM“ret”指令是否返回 void? 【发布时间】:2017-09-29 21:48:02 【问题描述】:

我有以下sn-p的代码:

static LLVMContext TempContext;
Type * RetTy = Type::getVoidTy(TempContext)
for (Instruction *I : ListOfInstructions) 
  if (isa<ReturnInst>(I)) 
    RetTy = I->getOperand(0)->getType();
    break
  

我试图在哪里捕获指令的RetTy,无论是否无效,所以我可以在

上使用它
getOrInsertFunction("TempF", FunctionType::get(RetTy, ArgsTys,false));

只要 ret 指令不是ret void,此代码就可以工作。

我尝试添加第二个if 来检查void 的情况,但这似乎不起作用,并且在FunctionType::get(...) 函数中的执行停止,打印回溯。

for (Instruction *I : ListOfInstructions) 
  if (isa<ReturnInst>(I)) 
    if ( I->getOperand(0)->getType() != Type::getVoidTy(TempContext)) 
      RetTy = I->getOperand(0)->getType();
      break
    
  
                                            

请注意,同时删除 for 循环并继续执行,因为函数 FunctionType::get(...) 可以很好地处理 RetTy 的初始化 Type * RetTy = Type::getVoidTy(TempContext) "void" 值。但是当 llvm 函数返回非 void 值时,我无法捕获。


我如何知道 I 指令何时是返回指令并且它在 LLVM IR 中返回 Void?

【问题讨论】:

【参考方案1】:

您当前代码的问题是ret void 没有操作数,因此调用getOperand(0) 会访问无效数据。

将你的 if 替换为:

if (ReturnInst *ri = dyn_cast<ReturnInst>(I))

  if (ri->getNumOperands() == 0)
  
    errs() << "VOID: " << *ri << "\n";
  
  else
  
    errs() << "NON-VOID: " << *ri << "\n";
  

现在,此代码将输出VOID: ret void,正确检测到指令。

作为替代方案,您可以使用任何指令检索函数的返回类型,方法是依赖函数I-&gt;getFunction()-&gt;getReturnType() 包含的指令;但是,这会假设函数格式正确且 ReturnInst 与其类型匹配,并且指令是函数的一部分。

【讨论】:

我在阅读文档时没有看到这一点。现在,您突出显示了它,我看到IRBuilder 调用ReturnInst::Create 方法初始化TerminatorInst(在InstrTypes.h 中),NumOps 参数设置为0。谢谢您的回答。

以上是关于如何验证 LLVM“ret”指令是不是返回 void?的主要内容,如果未能解决你的问题,请参考以下文章

X86 'cmpps' 指令和 LLVM IR 'fcmp' 指令是不是可以互换?

关于汇编指令call和ret的具体细节操作!

如何获得 LLVM 中两条指令之间的距离?

如何将第一条指令插入llvm中的空基本块

Android 逆向x86 汇编 ( push / pop 入栈 / 出栈 指令 | ret / retn 函数调用返回指令 | set 设置目标值指令 )

Android 逆向x86 汇编 ( push / pop 入栈 / 出栈 指令 | ret / retn 函数调用返回指令 | set 设置目标值指令 )