diff options
| author | Dimitry Andric <dim@FreeBSD.org> | 2012-05-03 16:50:55 +0000 | 
|---|---|---|
| committer | Dimitry Andric <dim@FreeBSD.org> | 2012-05-03 16:50:55 +0000 | 
| commit | b61ab53cb789e568acbb2952fbead20ab853a696 (patch) | |
| tree | 8575c732129e272992ac5d7b4c2519238fff4735 /lib/Transforms/Scalar/ObjCARC.cpp | |
| parent | 63faed5b8e4f2755f127fcb8aa440480c0649327 (diff) | |
Diffstat (limited to 'lib/Transforms/Scalar/ObjCARC.cpp')
| -rw-r--r-- | lib/Transforms/Scalar/ObjCARC.cpp | 69 | 
1 files changed, 53 insertions, 16 deletions
| diff --git a/lib/Transforms/Scalar/ObjCARC.cpp b/lib/Transforms/Scalar/ObjCARC.cpp index 40b0b20616893..7e3e69b78d307 100644 --- a/lib/Transforms/Scalar/ObjCARC.cpp +++ b/lib/Transforms/Scalar/ObjCARC.cpp @@ -162,6 +162,7 @@ namespace {      IC_MoveWeak,            ///< objc_moveWeak (derived)      IC_CopyWeak,            ///< objc_copyWeak (derived)      IC_DestroyWeak,         ///< objc_destroyWeak (derived) +    IC_StoreStrong,         ///< objc_storeStrong (derived)      IC_CallOrUser,          ///< could call objc_release and/or "use" pointers      IC_Call,                ///< could call objc_release      IC_User,                ///< could "use" a pointer @@ -262,6 +263,7 @@ static InstructionClass GetFunctionClass(const Function *F) {                return StringSwitch<InstructionClass>(F->getName())                       .Case("objc_storeWeak",             IC_StoreWeak)                       .Case("objc_initWeak",              IC_InitWeak) +                     .Case("objc_storeStrong",           IC_StoreStrong)                       .Default(IC_CallOrUser);              // Second argument is i8**.              if (PointerType *Pte1 = dyn_cast<PointerType>(ETy1)) @@ -618,22 +620,35 @@ static bool DoesObjCBlockEscape(const Value *BlockPtr) {        const User *UUser = *UI;        // Special - Use by a call (callee or argument) is not considered        // to be an escape. -      if (isa<CallInst>(UUser) || isa<InvokeInst>(UUser)) -        continue; -      // Use by an instruction which copies the value is an escape if the -      // result is an escape. -      if (isa<BitCastInst>(UUser) || isa<GetElementPtrInst>(UUser) || -          isa<PHINode>(UUser) || isa<SelectInst>(UUser)) { -        Worklist.push_back(UUser); +      switch (GetBasicInstructionClass(UUser)) { +      case IC_StoreWeak: +      case IC_InitWeak: +      case IC_StoreStrong: +      case IC_Autorelease: +      case IC_AutoreleaseRV: +        // These special functions make copies of their pointer arguments. +        return true; +      case IC_User: +      case IC_None: +        // Use by an instruction which copies the value is an escape if the +        // result is an escape. +        if (isa<BitCastInst>(UUser) || isa<GetElementPtrInst>(UUser) || +            isa<PHINode>(UUser) || isa<SelectInst>(UUser)) { +          Worklist.push_back(UUser); +          continue; +        } +        // Use by a load is not an escape. +        if (isa<LoadInst>(UUser)) +          continue; +        // Use by a store is not an escape if the use is the address. +        if (const StoreInst *SI = dyn_cast<StoreInst>(UUser)) +          if (V != SI->getValueOperand()) +            continue; +        break; +      default: +        // Regular calls and other stuff are not considered escapes.          continue;        } -      // Use by a load is not an escape. -      if (isa<LoadInst>(UUser)) -        continue; -      // Use by a store is not an escape if the use is the address. -      if (const StoreInst *SI = dyn_cast<StoreInst>(UUser)) -        if (V != SI->getValueOperand()) -          continue;        // Otherwise, conservatively assume an escape.        return true;      } @@ -883,7 +898,7 @@ bool ObjCARCExpand::runOnFunction(Function &F) {        // These calls return their argument verbatim, as a low-level        // optimization. However, this makes high-level optimizations        // harder. Undo any uses of this optimization that the front-end -      // emitted here. We'll redo them in a later pass. +      // emitted here. We'll redo them in the contract pass.        Changed = true;        Inst->replaceAllUsesWith(cast<CallInst>(Inst)->getArgOperand(0));        break; @@ -997,7 +1012,11 @@ bool ObjCARCAPElim::runOnModule(Module &M) {      return false;    // Find the llvm.global_ctors variable, as the first step in -  // identifying the global constructors. +  // identifying the global constructors. In theory, unnecessary autorelease +  // pools could occur anywhere, but in practice it's pretty rare. Global +  // ctors are a place where autorelease pools get inserted automatically, +  // so it's pretty common for them to be unnecessary, and it's pretty +  // profitable to eliminate them.    GlobalVariable *GV = M.getGlobalVariable("llvm.global_ctors");    if (!GV)      return false; @@ -2263,6 +2282,7 @@ void ObjCARCOpt::OptimizeIndividualCalls(Function &F) {      case IC_DestroyWeak: {        CallInst *CI = cast<CallInst>(Inst);        if (isNullOrUndef(CI->getArgOperand(0))) { +        Changed = true;          Type *Ty = CI->getArgOperand(0)->getType();          new StoreInst(UndefValue::get(cast<PointerType>(Ty)->getElementType()),                        Constant::getNullValue(Ty), @@ -2278,6 +2298,7 @@ void ObjCARCOpt::OptimizeIndividualCalls(Function &F) {        CallInst *CI = cast<CallInst>(Inst);        if (isNullOrUndef(CI->getArgOperand(0)) ||            isNullOrUndef(CI->getArgOperand(1))) { +        Changed = true;          Type *Ty = CI->getArgOperand(0)->getType();          new StoreInst(UndefValue::get(cast<PointerType>(Ty)->getElementType()),                        Constant::getNullValue(Ty), @@ -3165,6 +3186,8 @@ void ObjCARCOpt::MoveCalls(Value *Arg,    }  } +/// PerformCodePlacement - Identify pairings between the retains and releases, +/// and delete and/or move them.  bool  ObjCARCOpt::PerformCodePlacement(DenseMap<const BasicBlock *, BBState>                                     &BBStates, @@ -3178,6 +3201,7 @@ ObjCARCOpt::PerformCodePlacement(DenseMap<const BasicBlock *, BBState>    SmallVector<Instruction *, 4> NewReleases;    SmallVector<Instruction *, 8> DeadInsts; +  // Visit each retain.    for (MapVector<Value *, RRInfo>::const_iterator I = Retains.begin(),         E = Retains.end(); I != E; ++I) {      Value *V = I->first; @@ -3651,6 +3675,7 @@ bool ObjCARCOpt::doInitialization(Module &M) {    if (!EnableARCOpts)      return false; +  // If nothing in the Module uses ARC, don't do anything.    Run = ModuleHasARC(M);    if (!Run)      return false; @@ -3985,6 +4010,7 @@ void ObjCARCContract::ContractRelease(Instruction *Release,  }  bool ObjCARCContract::doInitialization(Module &M) { +  // If nothing in the Module uses ARC, don't do anything.    Run = ModuleHasARC(M);    if (!Run)      return false; @@ -4060,6 +4086,7 @@ bool ObjCARCContract::runOnFunction(Function &F) {        --BBI;        while (isNoopInstruction(BBI)) --BBI;        if (&*BBI == GetObjCArg(Inst)) { +        Changed = true;          InlineAsm *IA =            InlineAsm::get(FunctionType::get(Type::getVoidTy(Inst->getContext()),                                             /*isVarArg=*/false), @@ -4109,6 +4136,13 @@ bool ObjCARCContract::runOnFunction(Function &F) {          Use &U = UI.getUse();          unsigned OperandNo = UI.getOperandNo();          ++UI; // Increment UI now, because we may unlink its element. + +        // If the call's return value dominates a use of the call's argument +        // value, rewrite the use to use the return value. We check for +        // reachability here because an unreachable call is considered to +        // trivially dominate itself, which would lead us to rewriting its +        // argument in terms of its return value, which would lead to +        // infinite loops in GetObjCArg.          if (DT->isReachableFromEntry(U) &&              DT->dominates(Inst, U)) {            Changed = true; @@ -4123,6 +4157,9 @@ bool ObjCARCContract::runOnFunction(Function &F) {              if (Replacement->getType() != UseTy)                Replacement = new BitCastInst(Replacement, UseTy, "",                                              &BB->back()); +            // While we're here, rewrite all edges for this PHI, rather +            // than just one use at a time, to minimize the number of +            // bitcasts we emit.              for (unsigned i = 0, e = PHI->getNumIncomingValues();                   i != e; ++i)                if (PHI->getIncomingBlock(i) == BB) { | 
