2013-07-11 02:49:00 +08:00
|
|
|
//===- ObjCARCAliasAnalysis.cpp - ObjC ARC Optimization -------------------===//
|
2013-01-28 13:51:54 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// \file
|
|
|
|
/// This file defines a simple ARC-aware AliasAnalysis using special knowledge
|
|
|
|
/// of Objective C to enhance other optimization passes which rely on the Alias
|
|
|
|
/// Analysis infrastructure.
|
|
|
|
///
|
|
|
|
/// WARNING: This file knows about certain library functions. It recognizes them
|
|
|
|
/// by name, and hardwires knowledge of their semantics.
|
|
|
|
///
|
|
|
|
/// WARNING: This file knows about how certain Objective-C library functions are
|
|
|
|
/// used. Naive LLVM IR transformations which would otherwise be
|
|
|
|
/// behavior-preserving may break these assumptions.
|
|
|
|
///
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "ObjCARC.h"
|
|
|
|
#include "ObjCARCAliasAnalysis.h"
|
|
|
|
#include "llvm/IR/Instruction.h"
|
|
|
|
#include "llvm/InitializePasses.h"
|
|
|
|
#include "llvm/PassAnalysisSupport.h"
|
|
|
|
#include "llvm/PassSupport.h"
|
|
|
|
|
2014-04-22 10:55:47 +08:00
|
|
|
#define DEBUG_TYPE "objc-arc-aa"
|
|
|
|
|
2013-01-28 13:51:54 +08:00
|
|
|
namespace llvm {
|
|
|
|
class Function;
|
|
|
|
class Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace llvm::objcarc;
|
|
|
|
|
|
|
|
// Register this pass...
|
|
|
|
char ObjCARCAliasAnalysis::ID = 0;
|
|
|
|
INITIALIZE_AG_PASS(ObjCARCAliasAnalysis, AliasAnalysis, "objc-arc-aa",
|
|
|
|
"ObjC-ARC-Based Alias Analysis", false, true, false)
|
|
|
|
|
|
|
|
ImmutablePass *llvm::createObjCARCAliasAnalysisPass() {
|
|
|
|
return new ObjCARCAliasAnalysis();
|
|
|
|
}
|
|
|
|
|
2015-03-05 02:43:29 +08:00
|
|
|
bool ObjCARCAliasAnalysis::doInitialization(Module &M) {
|
|
|
|
InitializeAliasAnalysis(this, &M.getDataLayout());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-01-28 13:51:54 +08:00
|
|
|
void
|
|
|
|
ObjCARCAliasAnalysis::getAnalysisUsage(AnalysisUsage &AU) const {
|
|
|
|
AU.setPreservesAll();
|
|
|
|
AliasAnalysis::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
|
|
|
|
AliasAnalysis::AliasResult
|
|
|
|
ObjCARCAliasAnalysis::alias(const Location &LocA, const Location &LocB) {
|
|
|
|
if (!EnableARCOpts)
|
|
|
|
return AliasAnalysis::alias(LocA, LocB);
|
|
|
|
|
|
|
|
// First, strip off no-ops, including ObjC-specific no-ops, and try making a
|
|
|
|
// precise alias query.
|
2015-02-19 08:42:38 +08:00
|
|
|
const Value *SA = GetRCIdentityRoot(LocA.Ptr);
|
|
|
|
const Value *SB = GetRCIdentityRoot(LocB.Ptr);
|
2013-01-28 13:51:54 +08:00
|
|
|
AliasResult Result =
|
2014-07-24 20:16:19 +08:00
|
|
|
AliasAnalysis::alias(Location(SA, LocA.Size, LocA.AATags),
|
|
|
|
Location(SB, LocB.Size, LocB.AATags));
|
2013-01-28 13:51:54 +08:00
|
|
|
if (Result != MayAlias)
|
|
|
|
return Result;
|
|
|
|
|
|
|
|
// If that failed, climb to the underlying object, including climbing through
|
|
|
|
// ObjC-specific no-ops, and try making an imprecise alias query.
|
|
|
|
const Value *UA = GetUnderlyingObjCPtr(SA);
|
|
|
|
const Value *UB = GetUnderlyingObjCPtr(SB);
|
|
|
|
if (UA != SA || UB != SB) {
|
|
|
|
Result = AliasAnalysis::alias(Location(UA), Location(UB));
|
|
|
|
// We can't use MustAlias or PartialAlias results here because
|
|
|
|
// GetUnderlyingObjCPtr may return an offsetted pointer value.
|
|
|
|
if (Result == NoAlias)
|
|
|
|
return NoAlias;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If that failed, fail. We don't need to chain here, since that's covered
|
|
|
|
// by the earlier precise query.
|
|
|
|
return MayAlias;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ObjCARCAliasAnalysis::pointsToConstantMemory(const Location &Loc,
|
|
|
|
bool OrLocal) {
|
|
|
|
if (!EnableARCOpts)
|
|
|
|
return AliasAnalysis::pointsToConstantMemory(Loc, OrLocal);
|
|
|
|
|
|
|
|
// First, strip off no-ops, including ObjC-specific no-ops, and try making
|
|
|
|
// a precise alias query.
|
2015-02-19 08:42:38 +08:00
|
|
|
const Value *S = GetRCIdentityRoot(Loc.Ptr);
|
2014-07-24 20:16:19 +08:00
|
|
|
if (AliasAnalysis::pointsToConstantMemory(Location(S, Loc.Size, Loc.AATags),
|
2013-01-28 13:51:54 +08:00
|
|
|
OrLocal))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// If that failed, climb to the underlying object, including climbing through
|
|
|
|
// ObjC-specific no-ops, and try making an imprecise alias query.
|
|
|
|
const Value *U = GetUnderlyingObjCPtr(S);
|
|
|
|
if (U != S)
|
|
|
|
return AliasAnalysis::pointsToConstantMemory(Location(U), OrLocal);
|
|
|
|
|
|
|
|
// If that failed, fail. We don't need to chain here, since that's covered
|
|
|
|
// by the earlier precise query.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
AliasAnalysis::ModRefBehavior
|
|
|
|
ObjCARCAliasAnalysis::getModRefBehavior(ImmutableCallSite CS) {
|
|
|
|
// We have nothing to do. Just chain to the next AliasAnalysis.
|
|
|
|
return AliasAnalysis::getModRefBehavior(CS);
|
|
|
|
}
|
|
|
|
|
|
|
|
AliasAnalysis::ModRefBehavior
|
|
|
|
ObjCARCAliasAnalysis::getModRefBehavior(const Function *F) {
|
|
|
|
if (!EnableARCOpts)
|
|
|
|
return AliasAnalysis::getModRefBehavior(F);
|
|
|
|
|
|
|
|
switch (GetFunctionClass(F)) {
|
2015-02-20 03:51:32 +08:00
|
|
|
case ARCInstKind::NoopCast:
|
2013-01-28 13:51:54 +08:00
|
|
|
return DoesNotAccessMemory;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return AliasAnalysis::getModRefBehavior(F);
|
|
|
|
}
|
|
|
|
|
|
|
|
AliasAnalysis::ModRefResult
|
|
|
|
ObjCARCAliasAnalysis::getModRefInfo(ImmutableCallSite CS, const Location &Loc) {
|
|
|
|
if (!EnableARCOpts)
|
|
|
|
return AliasAnalysis::getModRefInfo(CS, Loc);
|
|
|
|
|
2015-02-20 03:51:32 +08:00
|
|
|
switch (GetBasicARCInstKind(CS.getInstruction())) {
|
|
|
|
case ARCInstKind::Retain:
|
|
|
|
case ARCInstKind::RetainRV:
|
|
|
|
case ARCInstKind::Autorelease:
|
|
|
|
case ARCInstKind::AutoreleaseRV:
|
|
|
|
case ARCInstKind::NoopCast:
|
|
|
|
case ARCInstKind::AutoreleasepoolPush:
|
|
|
|
case ARCInstKind::FusedRetainAutorelease:
|
|
|
|
case ARCInstKind::FusedRetainAutoreleaseRV:
|
2013-01-28 13:51:54 +08:00
|
|
|
// These functions don't access any memory visible to the compiler.
|
|
|
|
// Note that this doesn't include objc_retainBlock, because it updates
|
|
|
|
// pointers when it copies block data.
|
|
|
|
return NoModRef;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return AliasAnalysis::getModRefInfo(CS, Loc);
|
|
|
|
}
|
|
|
|
|
|
|
|
AliasAnalysis::ModRefResult
|
|
|
|
ObjCARCAliasAnalysis::getModRefInfo(ImmutableCallSite CS1,
|
|
|
|
ImmutableCallSite CS2) {
|
|
|
|
// TODO: Theoretically we could check for dependencies between objc_* calls
|
|
|
|
// and OnlyAccessesArgumentPointees calls or other well-behaved calls.
|
|
|
|
return AliasAnalysis::getModRefInfo(CS1, CS2);
|
|
|
|
}
|