我正在尝试从AST生成llvm-ir。
用于显示我添加的整数输出
Constant *CalleeF = TheModule->getOrInsertFunction("printf",FunctionType::get(IntegerType::getInt32Ty(Context), PointerType::get(Type::getInt8Ty(Context), 0), true);`
在调用我写的打印函数时,
Value* PrintStmt::codegen(){
Value* V,*val,*to_print;
vector<Value *> ArgsV;
for (unsigned int i = 0, e = outs.size(); i != e; ++i){
to_print = outs[i]->codegen();
if(outs[i]->type=="int"){
val=Builder.CreateGlobalStringPtr("%d");
}
ArgsV.push_back(val);
ArgsV.push_back(to_print);
V = Builder.CreateCall(CalleeF, ArgsV, "printfCall");
}
return V;
}
我应该写什么类似的代码来获取用户的输入,即scanf调用?
答案 0 :(得分:0)
对于scanf
调用,您可以先声明其原型
llvm::FunctionType *readFnType = llvm::FunctionType::get(builder.getInt32Ty(), true);
llvm::Function* readfn = llvm::Function::Create(readFnType, llvm::GlobalValue::ExternalLinkage, "scanf", TheModule));
并这样称呼
std::vector<Value*> ArgsV; // holds codegen IR for each argument
std::string StringFormat; // holds string formatting for all arguments
for(auto& arg : Args){
if(auto v = arg->codegen()){
if(v->getType()->isDoubleTy())
StringFormat += "%lf "; //
else if(v->getType()->isIntegerTy())
StringFormat += "%d ";
ArgsV.push_back(symbolTable[arg.name]);
}else return nullptr;
}
ArgsV.insert(ArgsV.begin(), builder.CreateGlobalStringPtr(StringFormat));
return builder.CreateCall(TheModule->getFunction("scanf"), ArgsV, "scanfCall");
其中symbolTable
是变量/参数名称到保存变量的堆栈分配地址的Value*
的映射。回想一下scanf
采用要写入的变量的地址,它说明了符号表查找。
值得一提的是,这使scanf
本质上不安全。您应该考虑改为使用fgets
和gets
函数。