Skip to content

Commit

Permalink
[CGP] Make optimizeMemoryInst capable of handling multiple AddrModes
Browse files Browse the repository at this point in the history
Currently optimizeMemoryInst requires that all of the AddrModes it sees are
identical. This patch makes it capable of tracking multiple AddrModes, so long
as they differ in at most one field.

This patch does nothing by itself, but later patches will make use of it to
insert or reuse phi or select instructions for the differing fields.

Differential Revision: https://reviews.llvm.org/D38278


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@314795 91177308-0d34-0410-b5e6-96231b3b80d8
  • Loading branch information
john-brawn-arm committed Oct 3, 2017
1 parent dbfce9e commit 3db5064
Showing 1 changed file with 154 additions and 20 deletions.
174 changes: 154 additions & 20 deletions lib/CodeGen/CodeGenPrepare.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2676,16 +2676,68 @@ namespace {
struct ExtAddrMode : public TargetLowering::AddrMode {
Value *BaseReg = nullptr;
Value *ScaledReg = nullptr;
Value *OriginalValue = nullptr;

enum FieldName {
NoField = 0x00,
BaseRegField = 0x01,
BaseGVField = 0x02,
BaseOffsField = 0x04,
ScaledRegField = 0x08,
ScaleField = 0x10,
MultipleFields = 0xff
};

ExtAddrMode() = default;

void print(raw_ostream &OS) const;
void dump() const;

bool operator==(const ExtAddrMode& O) const {
return (BaseReg == O.BaseReg) && (ScaledReg == O.ScaledReg) &&
(BaseGV == O.BaseGV) && (BaseOffs == O.BaseOffs) &&
(HasBaseReg == O.HasBaseReg) && (Scale == O.Scale);
FieldName compare(const ExtAddrMode &other) {
// First check that the types are the same on each field, as differing types
// is something we can't cope with later on.
if (BaseReg && other.BaseReg &&
BaseReg->getType() != other.BaseReg->getType())
return MultipleFields;
if (BaseGV && other.BaseGV &&
BaseGV->getType() != other.BaseGV->getType())
return MultipleFields;
if (ScaledReg && other.ScaledReg &&
ScaledReg->getType() != other.ScaledReg->getType())
return MultipleFields;

// Check each field to see if it differs.
unsigned Result = NoField;
if (BaseReg != other.BaseReg)
Result |= BaseRegField;
if (BaseGV != other.BaseGV)
Result |= BaseGVField;
if (BaseOffs != other.BaseOffs)
Result |= BaseOffsField;
if (ScaledReg != other.ScaledReg)
Result |= ScaledRegField;
// Don't count 0 as being a different scale, because that actually means
// unscaled (which will already be counted by having no ScaledReg).
if (Scale && other.Scale && Scale != other.Scale)
Result |= ScaleField;

if (countPopulation(Result) > 1)
return MultipleFields;
else
return static_cast<FieldName>(Result);
}

// AddrModes with a base reg or gv where the reg/gv is just the original
// value are trivial.
bool isTrivial() {
bool Trivial = (BaseGV && BaseGV == OriginalValue) ||
(BaseReg && BaseReg == OriginalValue);
// If the AddrMode is trivial it shouldn't have an offset or be scaled.
if (Trivial) {
assert(BaseOffs == 0);
assert(Scale == 0);
}
return Trivial;
}
};

Expand Down Expand Up @@ -3302,6 +3354,92 @@ class AddressingModeMatcher {
Value *PromotedOperand) const;
};

/// \brief A helper class for combining addressing modes.
class AddressingModeCombiner {
private:
/// The addressing modes we've collected.
SmallVector<ExtAddrMode, 16> AddrModes;

/// The field in which the AddrModes differ, when we have more than one.
ExtAddrMode::FieldName DifferentField = ExtAddrMode::NoField;

/// Are the AddrModes that we have all just equal to their original values?
bool AllAddrModesTrivial = true;

public:
/// \brief Get the combined AddrMode
const ExtAddrMode &getAddrMode() const {
return AddrModes[0];
}

/// \brief Add a new AddrMode if it's compatible with the AddrModes we already
/// have.
/// \return True iff we succeeded in doing so.
bool addNewAddrMode(ExtAddrMode &NewAddrMode) {
// Take note of if we have any non-trivial AddrModes, as we need to detect
// when all AddrModes are trivial as then we would introduce a phi or select
// which just duplicates what's already there.
AllAddrModesTrivial = AllAddrModesTrivial && NewAddrMode.isTrivial();

// If this is the first addrmode then everything is fine.
if (AddrModes.empty()) {
AddrModes.emplace_back(NewAddrMode);
return true;
}

// Figure out how different this is from the other address modes, which we
// can do just by comparing against the first one given that we only care
// about the cumulative difference.
ExtAddrMode::FieldName ThisDifferentField =
AddrModes[0].compare(NewAddrMode);
if (DifferentField == ExtAddrMode::NoField)
DifferentField = ThisDifferentField;
else if (DifferentField != ThisDifferentField)
DifferentField = ExtAddrMode::MultipleFields;

// If this AddrMode is the same as all the others then everything is fine
// (which should only happen when there is actually only one AddrMode).
if (DifferentField == ExtAddrMode::NoField) {
assert(AddrModes.size() == 1);
return true;
}

// If NewAddrMode differs in only one dimension then we can handle it by
// inserting a phi/select later on.
if (DifferentField != ExtAddrMode::MultipleFields) {
AddrModes.emplace_back(NewAddrMode);
return true;
}

// We couldn't combine NewAddrMode with the rest, so return failure.
AddrModes.clear();
return false;
}

/// \brief Combine the addressing modes we've collected into a single
/// addressing mode.
/// \return True iff we successfully combined them or we only had one so
/// didn't need to combine them anyway.
bool combineAddrModes() {
// If we have no AddrModes then they can't be combined.
if (AddrModes.size() == 0)
return false;

// A single AddrMode can trivially be combined.
if (AddrModes.size() == 1)
return true;

// If the AddrModes we collected are all just equal to the value they are
// derived from then combining them wouldn't do anything useful.
if (AllAddrModesTrivial)
return false;

// TODO: Combine multiple AddrModes by inserting a select or phi for the
// field in which the AddrModes differ.
return false;
}
};

} // end anonymous namespace

/// Try adding ScaleReg*Scale to the current addressing mode.
Expand Down Expand Up @@ -4391,11 +4529,10 @@ bool CodeGenPrepare::optimizeMemoryInst(Instruction *MemoryInst, Value *Addr,

// Use a worklist to iteratively look through PHI and select nodes, and
// ensure that the addressing mode obtained from the non-PHI/select roots of
// the graph are equivalent.
bool AddrModeFound = false;
// the graph are compatible.
bool PhiOrSelectSeen = false;
SmallVector<Instruction*, 16> AddrModeInsts;
ExtAddrMode AddrMode;
AddressingModeCombiner AddrModes;
TypePromotionTransaction TPT(RemovedInsts);
TypePromotionTransaction::ConstRestorationPt LastKnownGood =
TPT.getRestorationPoint();
Expand Down Expand Up @@ -4437,27 +4574,24 @@ bool CodeGenPrepare::optimizeMemoryInst(Instruction *MemoryInst, Value *Addr,
ExtAddrMode NewAddrMode = AddressingModeMatcher::Match(
V, AccessTy, AddrSpace, MemoryInst, AddrModeInsts, *TLI, *TRI,
InsertedInsts, PromotedInsts, TPT);
NewAddrMode.OriginalValue = V;

if (!AddrModeFound) {
AddrModeFound = true;
AddrMode = NewAddrMode;
continue;
}
if (NewAddrMode == AddrMode)
continue;

AddrModeFound = false;
break;
if (!AddrModes.addNewAddrMode(NewAddrMode))
break;
}

// If the addressing mode couldn't be determined, or if multiple different
// ones were determined, bail out now.
if (!AddrModeFound) {
// Try to combine the AddrModes we've collected. If we couldn't collect any,
// or we have multiple but either couldn't combine them or combining them
// wouldn't do anything useful, bail out now.
if (!AddrModes.combineAddrModes()) {
TPT.rollback(LastKnownGood);
return false;
}
TPT.commit();

// Get the combined AddrMode (or the only AddrMode, if we only had one).
ExtAddrMode AddrMode = AddrModes.getAddrMode();

// If all the instructions matched are already in this BB, don't do anything.
// If we saw a Phi node then it is not local definitely, and if we saw a select
// then we want to push the address calculation past it even if it's already
Expand Down

0 comments on commit 3db5064

Please sign in to comment.