如何用llvmc+实现函数指针+;应用程序编程接口?

如何用llvmc+实现函数指针+;应用程序编程接口?,llvm,llvm-ir,Llvm,Llvm Ir,假设我想手动将下面的代码转换为IR代码: #include <stdio.h> int main() { int (*p)(const char *__s); // how to implement this? p = puts; // and this? p("Hello World!\n"); } 但是我不知道应该使用哪个api来生成这个 以下是我的部分实现: #include "llvm/Support/raw_ostream

假设我想手动将下面的代码转换为IR代码:

#include <stdio.h>
int main()
{
  int (*p)(const char *__s);  // how to implement this?
  p = puts;                   // and this?
  p("Hello World!\n");
}
但是我不知道应该使用哪个api来生成这个

以下是我的部分实现:

#include "llvm/Support/raw_ostream.h"

int main() {
  llvm::LLVMContext context;
  llvm::IRBuilder<> builder(context);
  llvm::Module *module = new llvm::Module("top", context);

  llvm::FunctionType *functionType = llvm::FunctionType::get(builder.getInt32Ty(), false);
  llvm::Function *mainFunction = llvm::Function::Create(functionType, llvm::Function::ExternalLinkage, "main", module);

  llvm::BasicBlock *entry = llvm::BasicBlock::Create(context, "entrypoint", mainFunction);
  builder.SetInsertPoint(entry);

  llvm::Value *helloWorld = builder.CreateGlobalStringPtr("hello world\n");

  std::vector<llvm::Type *> putArgs;
  putArgs.push_back(builder.getInt8Ty()->getPointerTo());
  llvm::ArrayRef<llvm::Type *> argsRef(putArgs);
  llvm::FunctionType *putsType = llvm::FunctionType::get(builder.getInt32Ty(), argsRef, false);
  llvm::Constant *putFunction = module->getOrInsertFunction("puts", putsType);

  // code to implement function pointer
  // code to assign puts() to function pointer

  builder.CreateCall(putFunction, helloWorld);  // call the function pointer instead of the function it self
  builder.CreateRet(llvm::ConstantInt::get(builder.getInt32Ty(), 0));

  module->print(llvm::errs(), nullptr);
}
#包括“llvm/Support/raw_ostream.h”
int main(){
llvm::LLVMContext上下文;
llvm::IRBuilder(上下文);
llvm::Module*Module=新的llvm::Module(“顶部”,上下文);
llvm::FunctionType*FunctionType=llvm::FunctionType::get(builder.getInt32Ty(),false);
llvm::Function*mainFunction=llvm::Function::Create(functionType,llvm::Function::ExternalLinkage,“main”,module);
llvm::BasicBlock*entry=llvm::BasicBlock::Create(上下文,“入口点”,main函数);
builder.SetInsertPoint(输入);
llvm::Value*helloWorld=builder.CreateGlobalStringPtr(“hello world\n”);
std::向量putArgs;
putArgs.push_back(builder.getInt8Ty()->getPointerTo());
llvm::ArrayRef argsRef(putArgs);
llvm::FunctionType*putsType=llvm::FunctionType::get(builder.getInt32Ty(),argsRef,false);
llvm::Constant*putFunction=module->getOrInsertFunction(“put”,putsype);
//实现函数指针的代码
//将puts()赋值给函数指针的代码
CreateCall(putFunction,helloWorld);//调用函数指针,而不是它本身的函数
CreateRet(llvm::ConstantInt::get(builder.getInt32Ty(),0));
模块->打印(llvm::errs(),nullptr);
}
我发现
llvm::Function
llvm::Value
的一个子类,所以我想
llvm::Constant*putFunction
本身就是我要寻找的函数指针,但是如何让这个值在IR代码中表示呢?更具体地说,如何使用生成器生成IR代码?

我已经解决了

缺少的难题是:

llvm::Value *p = builder.CreateAlloca(putFunction->getType(), nullptr, "p");
builder.CreateStore(putFunction, p, false);
llvm::Value *temp = builder.CreateLoad(p);
builder.CreateCall(temp, helloWorld);

关键概念是
putFunction->getType()
llvm::FunctionType::get()
是不同的类型,我错误地将
llvm::FunctionType::get()
作为函数指针。

@puts
已经是函数指针了。
alloca
代码只是将指针复制到堆栈上。根据您尝试执行的操作,您不需要这样做(例如,如果您尝试创建vtable,就像在您的另一个问题中一样,您可以直接在全局数组中使用
@put
,“直接”是指“附加了位广播”,因为通常vtable中的每个函数都具有相同的类型)@sepp2k很高兴再次见到你。这个问题的最初目的是我不知道如何将C风格的函数指针生成到ir代码中。我想你是对的,我必须走错方向。我很难在汇编语言视图中看到一段代码,你有什么建议吗?我只是想发明一种oop玩具语言,尽可能简单,但它似乎比我想象的要难。
llvm::Value *p = builder.CreateAlloca(putFunction->getType(), nullptr, "p");
builder.CreateStore(putFunction, p, false);
llvm::Value *temp = builder.CreateLoad(p);
builder.CreateCall(temp, helloWorld);