2003-12-28 08:19:41 +00:00
|
|
|
//===-- IntrinsicLowering.cpp - Intrinsic Lowering default implementation -===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file was developed by the LLVM research group and is distributed under
|
|
|
|
// the University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the default intrinsic lowering implementation.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2003-12-28 08:30:20 +00:00
|
|
|
#include "llvm/IntrinsicLowering.h"
|
2004-02-14 02:47:17 +00:00
|
|
|
#include "llvm/Constants.h"
|
2004-02-12 17:01:09 +00:00
|
|
|
#include "llvm/DerivedTypes.h"
|
2003-12-28 08:19:41 +00:00
|
|
|
#include "llvm/Module.h"
|
|
|
|
#include "llvm/iOther.h"
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
void DefaultIntrinsicLowering::LowerIntrinsicCall(CallInst *CI) {
|
|
|
|
Function *Callee = CI->getCalledFunction();
|
|
|
|
assert(Callee && "Cannot lower an indirect call!");
|
|
|
|
|
|
|
|
Module *M = Callee->getParent();
|
|
|
|
|
|
|
|
switch (Callee->getIntrinsicID()) {
|
|
|
|
case Intrinsic::not_intrinsic:
|
|
|
|
std::cerr << "Cannot lower a call to a non-intrinsic function '"
|
|
|
|
<< Callee->getName() << "'!\n";
|
|
|
|
abort();
|
|
|
|
default:
|
|
|
|
std::cerr << "Error: Code generator does not support intrinsic function '"
|
|
|
|
<< Callee->getName() << "'!\n";
|
|
|
|
abort();
|
|
|
|
|
|
|
|
// The default implementation of setjmp/longjmp transforms setjmp into a
|
|
|
|
// noop that always returns zero and longjmp into a call to abort. This
|
|
|
|
// allows code that never longjmps to work correctly.
|
|
|
|
case Intrinsic::setjmp:
|
|
|
|
case Intrinsic::sigsetjmp:
|
|
|
|
if (CI->getType() != Type::VoidTy)
|
|
|
|
CI->replaceAllUsesWith(Constant::getNullValue(CI->getType()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Intrinsic::longjmp:
|
|
|
|
case Intrinsic::siglongjmp:
|
|
|
|
// Insert the call to abort
|
|
|
|
new CallInst(M->getOrInsertFunction("abort", Type::VoidTy, 0), "", CI);
|
|
|
|
break;
|
2004-01-05 05:36:30 +00:00
|
|
|
|
2004-02-14 02:47:17 +00:00
|
|
|
case Intrinsic::returnaddress:
|
|
|
|
case Intrinsic::frameaddress:
|
|
|
|
CI->replaceAllUsesWith(ConstantPointerNull::get(
|
|
|
|
cast<PointerType>(CI->getType())));
|
|
|
|
break;
|
|
|
|
|
2004-01-05 05:36:30 +00:00
|
|
|
case Intrinsic::dbg_stoppoint:
|
|
|
|
case Intrinsic::dbg_region_start:
|
|
|
|
case Intrinsic::dbg_region_end:
|
2004-01-14 20:41:29 +00:00
|
|
|
case Intrinsic::dbg_declare:
|
2004-01-05 05:36:30 +00:00
|
|
|
case Intrinsic::dbg_func_start:
|
|
|
|
if (CI->getType() != Type::VoidTy)
|
|
|
|
CI->replaceAllUsesWith(Constant::getNullValue(CI->getType()));
|
|
|
|
break; // Simply strip out debugging intrinsics
|
2004-02-12 17:01:09 +00:00
|
|
|
|
|
|
|
case Intrinsic::memcpy: {
|
|
|
|
// The memcpy intrinsic take an extra alignment argument that the memcpy
|
|
|
|
// libc function does not.
|
|
|
|
const FunctionType *CFT = Callee->getFunctionType();
|
|
|
|
FunctionType *FT =
|
|
|
|
FunctionType::get(*CFT->param_begin(),
|
|
|
|
std::vector<const Type*>(CFT->param_begin(), CFT->param_end()-1),
|
|
|
|
false);
|
|
|
|
Function *MemCpy = M->getOrInsertFunction("memcpy", FT);
|
|
|
|
new CallInst(MemCpy, std::vector<Value*>(CI->op_begin()+1, CI->op_end()-1),
|
|
|
|
CI->getName(), CI);
|
|
|
|
break;
|
|
|
|
}
|
2004-02-12 18:11:20 +00:00
|
|
|
case Intrinsic::memmove: {
|
2004-02-14 02:47:17 +00:00
|
|
|
// The memmove intrinsic take an extra alignment argument that the memmove
|
2004-02-12 18:11:20 +00:00
|
|
|
// libc function does not.
|
|
|
|
const FunctionType *CFT = Callee->getFunctionType();
|
|
|
|
FunctionType *FT =
|
|
|
|
FunctionType::get(*CFT->param_begin(),
|
|
|
|
std::vector<const Type*>(CFT->param_begin(), CFT->param_end()-1),
|
|
|
|
false);
|
|
|
|
Function *MemMove = M->getOrInsertFunction("memmove", FT);
|
|
|
|
new CallInst(MemMove, std::vector<Value*>(CI->op_begin()+1, CI->op_end()-1),
|
|
|
|
CI->getName(), CI);
|
|
|
|
break;
|
|
|
|
}
|
2004-02-14 02:47:17 +00:00
|
|
|
case Intrinsic::memset: {
|
|
|
|
// The memset intrinsic take an extra alignment argument that the memset
|
|
|
|
// libc function does not.
|
|
|
|
const FunctionType *CFT = Callee->getFunctionType();
|
|
|
|
FunctionType *FT =
|
|
|
|
FunctionType::get(*CFT->param_begin(),
|
|
|
|
std::vector<const Type*>(CFT->param_begin(), CFT->param_end()-1),
|
|
|
|
false);
|
|
|
|
Function *MemSet = M->getOrInsertFunction("memset", FT);
|
|
|
|
new CallInst(MemSet, std::vector<Value*>(CI->op_begin()+1, CI->op_end()-1),
|
|
|
|
CI->getName(), CI);
|
|
|
|
break;
|
|
|
|
}
|
2003-12-28 08:19:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(CI->use_empty() &&
|
|
|
|
"Lowering should have eliminated any uses of the intrinsic call!");
|
|
|
|
CI->getParent()->getInstList().erase(CI);
|
|
|
|
}
|