2010-08-04 00:19:16 +08:00
|
|
|
//===- LowerAtomic.cpp - Lower atomic intrinsics --------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This pass lowers atomic intrinsics to non-atomic form for use in a known
|
|
|
|
// non-preemptible environment.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "loweratomic"
|
|
|
|
#include "llvm/Transforms/Scalar.h"
|
|
|
|
#include "llvm/Function.h"
|
2010-09-06 04:10:47 +08:00
|
|
|
#include "llvm/IntrinsicInst.h"
|
2010-08-04 00:19:16 +08:00
|
|
|
#include "llvm/Pass.h"
|
|
|
|
#include "llvm/Support/IRBuilder.h"
|
|
|
|
using namespace llvm;
|
|
|
|
|
2010-09-06 04:10:47 +08:00
|
|
|
static bool LowerAtomicIntrinsic(IntrinsicInst *II) {
|
|
|
|
IRBuilder<> Builder(II->getParent(), II);
|
|
|
|
unsigned IID = II->getIntrinsicID();
|
2010-08-04 00:19:16 +08:00
|
|
|
switch (IID) {
|
|
|
|
case Intrinsic::memory_barrier:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Intrinsic::atomic_load_add:
|
|
|
|
case Intrinsic::atomic_load_sub:
|
|
|
|
case Intrinsic::atomic_load_and:
|
|
|
|
case Intrinsic::atomic_load_nand:
|
|
|
|
case Intrinsic::atomic_load_or:
|
|
|
|
case Intrinsic::atomic_load_xor:
|
|
|
|
case Intrinsic::atomic_load_max:
|
|
|
|
case Intrinsic::atomic_load_min:
|
|
|
|
case Intrinsic::atomic_load_umax:
|
|
|
|
case Intrinsic::atomic_load_umin: {
|
2010-09-06 04:13:07 +08:00
|
|
|
Value *Ptr = II->getArgOperand(0), *Delta = II->getArgOperand(1);
|
2010-08-04 00:19:16 +08:00
|
|
|
|
|
|
|
LoadInst *Orig = Builder.CreateLoad(Ptr);
|
2010-08-06 15:43:46 +08:00
|
|
|
Value *Res = NULL;
|
2010-08-04 00:19:16 +08:00
|
|
|
switch (IID) {
|
2010-09-06 04:13:07 +08:00
|
|
|
default: assert(0 && "Unrecognized atomic modify operation");
|
|
|
|
case Intrinsic::atomic_load_add:
|
|
|
|
Res = Builder.CreateAdd(Orig, Delta);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_sub:
|
|
|
|
Res = Builder.CreateSub(Orig, Delta);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_and:
|
|
|
|
Res = Builder.CreateAnd(Orig, Delta);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_nand:
|
|
|
|
Res = Builder.CreateNot(Builder.CreateAnd(Orig, Delta));
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_or:
|
|
|
|
Res = Builder.CreateOr(Orig, Delta);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_xor:
|
|
|
|
Res = Builder.CreateXor(Orig, Delta);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_max:
|
|
|
|
Res = Builder.CreateSelect(Builder.CreateICmpSLT(Orig, Delta),
|
|
|
|
Delta, Orig);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_min:
|
|
|
|
Res = Builder.CreateSelect(Builder.CreateICmpSLT(Orig, Delta),
|
|
|
|
Orig, Delta);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_umax:
|
|
|
|
Res = Builder.CreateSelect(Builder.CreateICmpULT(Orig, Delta),
|
|
|
|
Delta, Orig);
|
|
|
|
break;
|
|
|
|
case Intrinsic::atomic_load_umin:
|
|
|
|
Res = Builder.CreateSelect(Builder.CreateICmpULT(Orig, Delta),
|
|
|
|
Orig, Delta);
|
|
|
|
break;
|
2010-08-04 00:19:16 +08:00
|
|
|
}
|
|
|
|
Builder.CreateStore(Res, Ptr);
|
|
|
|
|
2010-09-06 04:10:47 +08:00
|
|
|
II->replaceAllUsesWith(Orig);
|
2010-08-04 00:19:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Intrinsic::atomic_swap: {
|
2010-09-06 04:13:07 +08:00
|
|
|
Value *Ptr = II->getArgOperand(0), *Val = II->getArgOperand(1);
|
2010-08-04 00:19:16 +08:00
|
|
|
LoadInst *Orig = Builder.CreateLoad(Ptr);
|
|
|
|
Builder.CreateStore(Val, Ptr);
|
2010-09-06 04:10:47 +08:00
|
|
|
II->replaceAllUsesWith(Orig);
|
2010-08-04 00:19:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Intrinsic::atomic_cmp_swap: {
|
2010-09-06 04:13:07 +08:00
|
|
|
Value *Ptr = II->getArgOperand(0), *Cmp = II->getArgOperand(1);
|
2010-09-06 04:10:47 +08:00
|
|
|
Value *Val = II->getArgOperand(2);
|
2010-08-04 00:19:16 +08:00
|
|
|
|
|
|
|
LoadInst *Orig = Builder.CreateLoad(Ptr);
|
|
|
|
Value *Equal = Builder.CreateICmpEQ(Orig, Cmp);
|
|
|
|
Value *Res = Builder.CreateSelect(Equal, Val, Orig);
|
|
|
|
Builder.CreateStore(Res, Ptr);
|
2010-09-06 04:10:47 +08:00
|
|
|
II->replaceAllUsesWith(Orig);
|
2010-08-04 00:19:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-09-06 04:10:47 +08:00
|
|
|
assert(II->use_empty() &&
|
2010-08-04 00:19:16 +08:00
|
|
|
"Lowering should have eliminated any uses of the intrinsic call!");
|
2010-09-06 04:10:47 +08:00
|
|
|
II->eraseFromParent();
|
2010-08-04 00:19:16 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-09-06 04:10:47 +08:00
|
|
|
namespace {
|
|
|
|
struct LowerAtomic : public BasicBlockPass {
|
|
|
|
static char ID;
|
|
|
|
LowerAtomic() : BasicBlockPass(ID) {}
|
|
|
|
bool runOnBasicBlock(BasicBlock &BB) {
|
|
|
|
bool Changed = false;
|
|
|
|
for (BasicBlock::iterator DI = BB.begin(), DE = BB.end(); DI != DE; )
|
|
|
|
if (IntrinsicInst *II = dyn_cast<IntrinsicInst>(DI++))
|
|
|
|
Changed |= LowerAtomicIntrinsic(II);
|
|
|
|
return Changed;
|
2010-08-04 00:19:16 +08:00
|
|
|
}
|
2010-09-06 04:10:47 +08:00
|
|
|
};
|
2010-08-04 00:19:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
char LowerAtomic::ID = 0;
|
2010-08-24 01:52:01 +08:00
|
|
|
INITIALIZE_PASS(LowerAtomic, "loweratomic",
|
|
|
|
"Lower atomic intrinsics to non-atomic form",
|
2010-10-08 06:25:06 +08:00
|
|
|
false, false)
|
2010-08-04 00:19:16 +08:00
|
|
|
|
|
|
|
Pass *llvm::createLowerAtomicPass() { return new LowerAtomic(); }
|