Add a ConstantDataVector::getSplatValue() method, for parity with

ConstantVector.  Fix some outright bugs in the implementation of
ConstantArray and Constant struct, which would cause us to not make
one big UndefValue when asking for an array/struct with all undef
elements.  Enhance Constant::isAllOnesValue to work with
ConstantDataVector.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@149021 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Chris Lattner
2012-01-26 02:31:22 +00:00
parent b56900aa86
commit e150e2df42
2 changed files with 91 additions and 39 deletions

View File

@@ -766,6 +766,10 @@ public:
/// i32/i64/float/double) and must be a ConstantFP or ConstantInt. /// i32/i64/float/double) and must be a ConstantFP or ConstantInt.
static Constant *getSplat(unsigned NumElts, Constant *Elt); static Constant *getSplat(unsigned NumElts, Constant *Elt);
/// getSplatValue - If this is a splat constant, meaning that all of the
/// elements have the same value, return that value. Otherwise return NULL.
Constant *getSplatValue() const;
/// getType - Specialize the getType() method to always return a VectorType, /// getType - Specialize the getType() method to always return a VectorType,
/// which reduces the amount of casting needed in parts of the compiler. /// which reduces the amount of casting needed in parts of the compiler.
/// ///

View File

@@ -78,6 +78,11 @@ bool Constant::isAllOnesValue() const {
if (Constant *Splat = CV->getSplatValue()) if (Constant *Splat = CV->getSplatValue())
return Splat->isAllOnesValue(); return Splat->isAllOnesValue();
// Check for constant vectors which are splats of -1 values.
if (const ConstantDataVector *CV = dyn_cast<ConstantDataVector>(this))
if (Constant *Splat = CV->getSplatValue())
return Splat->isAllOnesValue();
return false; return false;
} }
@@ -711,17 +716,27 @@ Constant *ConstantArray::get(ArrayType *Ty, ArrayRef<Constant*> V) {
} }
LLVMContextImpl *pImpl = Ty->getContext().pImpl; LLVMContextImpl *pImpl = Ty->getContext().pImpl;
// If this is an all-zero array, return a ConstantAggregateZero object // If this is an all-zero array, return a ConstantAggregateZero object
bool isAllZero = true;
bool isUndef = false;
if (!V.empty()) { if (!V.empty()) {
Constant *C = V[0]; Constant *C = V[0];
if (!C->isNullValue()) isAllZero = C->isNullValue();
return pImpl->ArrayConstants.getOrCreate(Ty, V); isUndef = isa<UndefValue>(C);
if (isAllZero || isUndef)
for (unsigned i = 1, e = V.size(); i != e; ++i) for (unsigned i = 1, e = V.size(); i != e; ++i)
if (V[i] != C) if (V[i] != C) {
return pImpl->ArrayConstants.getOrCreate(Ty, V); isAllZero = false;
isUndef = false;
break;
}
} }
if (isAllZero)
return ConstantAggregateZero::get(Ty); return ConstantAggregateZero::get(Ty);
if (isUndef)
return UndefValue::get(Ty);
return pImpl->ArrayConstants.getOrCreate(Ty, V);
} }
/// ConstantArray::get(const string&) - Return an array that is initialized to /// ConstantArray::get(const string&) - Return an array that is initialized to
@@ -780,14 +795,31 @@ ConstantStruct::ConstantStruct(StructType *T, ArrayRef<Constant *> V)
// ConstantStruct accessors. // ConstantStruct accessors.
Constant *ConstantStruct::get(StructType *ST, ArrayRef<Constant*> V) { Constant *ConstantStruct::get(StructType *ST, ArrayRef<Constant*> V) {
// Create a ConstantAggregateZero value if all elements are zeros.
for (unsigned i = 0, e = V.size(); i != e; ++i)
if (!V[i]->isNullValue())
return ST->getContext().pImpl->StructConstants.getOrCreate(ST, V);
assert((ST->isOpaque() || ST->getNumElements() == V.size()) && assert((ST->isOpaque() || ST->getNumElements() == V.size()) &&
"Incorrect # elements specified to ConstantStruct::get"); "Incorrect # elements specified to ConstantStruct::get");
// Create a ConstantAggregateZero value if all elements are zeros.
bool isZero = true;
bool isUndef = false;
if (!V.empty()) {
isUndef = isa<UndefValue>(V[0]);
isZero = V[0]->isNullValue();
if (isUndef || isZero) {
for (unsigned i = 0, e = V.size(); i != e; ++i) {
if (!V[i]->isNullValue())
isZero = false;
if (!isa<UndefValue>(V[i]))
isUndef = false;
}
}
}
if (isZero)
return ConstantAggregateZero::get(ST); return ConstantAggregateZero::get(ST);
if (isUndef)
return UndefValue::get(ST);
return ST->getContext().pImpl->StructConstants.getOrCreate(ST, V);
} }
Constant *ConstantStruct::get(StructType *T, ...) { Constant *ConstantStruct::get(StructType *T, ...) {
@@ -2329,6 +2361,20 @@ bool ConstantDataSequential::isCString() const {
return Str.drop_back().find(0) == StringRef::npos; return Str.drop_back().find(0) == StringRef::npos;
} }
/// getSplatValue - If this is a splat constant, meaning that all of the
/// elements have the same value, return that value. Otherwise return NULL.
Constant *ConstantDataVector::getSplatValue() const {
const char *Base = getRawDataValues().data();
// Compare elements 1+ to the 0'th element.
unsigned EltSize = getElementByteSize();
for (unsigned i = 1, e = getNumElements(); i != e; ++i)
if (memcmp(Base, Base+i*EltSize, EltSize))
return 0;
// If they're all the same, return the 0th one as a representative.
return getElementAsConstant(0);
}
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
// replaceUsesOfWithOnConstant implementations // replaceUsesOfWithOnConstant implementations
@@ -2360,9 +2406,10 @@ void ConstantArray::replaceUsesOfWithOnConstant(Value *From, Value *To,
// Fill values with the modified operands of the constant array. Also, // Fill values with the modified operands of the constant array. Also,
// compute whether this turns into an all-zeros array. // compute whether this turns into an all-zeros array.
bool isAllZeros = false;
unsigned NumUpdated = 0; unsigned NumUpdated = 0;
if (!ToC->isNullValue()) {
// Keep track of whether all the values in the array are "ToC".
bool AllSame = true;
for (Use *O = OperandList, *E = OperandList+getNumOperands(); O != E; ++O) { for (Use *O = OperandList, *E = OperandList+getNumOperands(); O != E; ++O) {
Constant *Val = cast<Constant>(O->get()); Constant *Val = cast<Constant>(O->get());
if (Val == From) { if (Val == From) {
@@ -2370,23 +2417,14 @@ void ConstantArray::replaceUsesOfWithOnConstant(Value *From, Value *To,
++NumUpdated; ++NumUpdated;
} }
Values.push_back(Val); Values.push_back(Val);
} AllSame = Val == ToC;
} else {
isAllZeros = true;
for (Use *O = OperandList, *E = OperandList+getNumOperands();O != E; ++O) {
Constant *Val = cast<Constant>(O->get());
if (Val == From) {
Val = ToC;
++NumUpdated;
}
Values.push_back(Val);
if (isAllZeros) isAllZeros = Val->isNullValue();
}
} }
Constant *Replacement = 0; Constant *Replacement = 0;
if (isAllZeros) { if (AllSame && ToC->isNullValue()) {
Replacement = ConstantAggregateZero::get(getType()); Replacement = ConstantAggregateZero::get(getType());
} else if (AllSame && isa<UndefValue>(ToC)) {
Replacement = UndefValue::get(getType());
} else { } else {
// Check to see if we have this array type already. // Check to see if we have this array type already.
bool Exists; bool Exists;
@@ -2446,16 +2484,24 @@ void ConstantStruct::replaceUsesOfWithOnConstant(Value *From, Value *To,
// Fill values with the modified operands of the constant struct. Also, // Fill values with the modified operands of the constant struct. Also,
// compute whether this turns into an all-zeros struct. // compute whether this turns into an all-zeros struct.
bool isAllZeros = false; bool isAllZeros = false;
if (!ToC->isNullValue()) { bool isAllUndef = false;
for (Use *O = OperandList, *E = OperandList + getNumOperands(); O != E; ++O) if (ToC->isNullValue()) {
Values.push_back(cast<Constant>(O->get()));
} else {
isAllZeros = true; isAllZeros = true;
for (Use *O = OperandList, *E = OperandList+getNumOperands(); O != E; ++O) { for (Use *O = OperandList, *E = OperandList+getNumOperands(); O != E; ++O) {
Constant *Val = cast<Constant>(O->get()); Constant *Val = cast<Constant>(O->get());
Values.push_back(Val); Values.push_back(Val);
if (isAllZeros) isAllZeros = Val->isNullValue(); if (isAllZeros) isAllZeros = Val->isNullValue();
} }
} else if (isa<UndefValue>(ToC)) {
isAllUndef = true;
for (Use *O = OperandList, *E = OperandList+getNumOperands(); O != E; ++O) {
Constant *Val = cast<Constant>(O->get());
Values.push_back(Val);
if (isAllUndef) isAllUndef = isa<UndefValue>(Val);
}
} else {
for (Use *O = OperandList, *E = OperandList + getNumOperands(); O != E; ++O)
Values.push_back(cast<Constant>(O->get()));
} }
Values[OperandToUpdate] = ToC; Values[OperandToUpdate] = ToC;
@@ -2464,6 +2510,8 @@ void ConstantStruct::replaceUsesOfWithOnConstant(Value *From, Value *To,
Constant *Replacement = 0; Constant *Replacement = 0;
if (isAllZeros) { if (isAllZeros) {
Replacement = ConstantAggregateZero::get(getType()); Replacement = ConstantAggregateZero::get(getType());
} else if (isAllUndef) {
Replacement = UndefValue::get(getType());
} else { } else {
// Check to see if we have this struct type already. // Check to see if we have this struct type already.
bool Exists; bool Exists;