mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-05-20 20:38:48 +00:00
Continue improving support for ConstantDataAggregate, and use the
new methods recently added to (sometimes greatly!) simplify code. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@149024 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
e150e2df42
commit
d59ae907ee
@ -2176,21 +2176,9 @@ static Constant *EvaluateStoreInto(Constant *Init, Constant *Val,
|
|||||||
|
|
||||||
std::vector<Constant*> Elts;
|
std::vector<Constant*> Elts;
|
||||||
if (StructType *STy = dyn_cast<StructType>(Init->getType())) {
|
if (StructType *STy = dyn_cast<StructType>(Init->getType())) {
|
||||||
|
|
||||||
// Break up the constant into its elements.
|
// Break up the constant into its elements.
|
||||||
if (ConstantStruct *CS = dyn_cast<ConstantStruct>(Init)) {
|
for (unsigned i = 0, e = STy->getNumElements(); i != e; ++i)
|
||||||
for (User::op_iterator i = CS->op_begin(), e = CS->op_end(); i != e; ++i)
|
Elts.push_back(Init->getAggregateElement(i));
|
||||||
Elts.push_back(cast<Constant>(*i));
|
|
||||||
} else if (isa<ConstantAggregateZero>(Init)) {
|
|
||||||
for (unsigned i = 0, e = STy->getNumElements(); i != e; ++i)
|
|
||||||
Elts.push_back(Constant::getNullValue(STy->getElementType(i)));
|
|
||||||
} else if (isa<UndefValue>(Init)) {
|
|
||||||
for (unsigned i = 0, e = STy->getNumElements(); i != e; ++i)
|
|
||||||
Elts.push_back(UndefValue::get(STy->getElementType(i)));
|
|
||||||
} else {
|
|
||||||
llvm_unreachable("This code is out of sync with "
|
|
||||||
" ConstantFoldLoadThroughGEPConstantExpr");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Replace the element that we are supposed to.
|
// Replace the element that we are supposed to.
|
||||||
ConstantInt *CU = cast<ConstantInt>(Addr->getOperand(OpNo));
|
ConstantInt *CU = cast<ConstantInt>(Addr->getOperand(OpNo));
|
||||||
@ -2209,22 +2197,11 @@ static Constant *EvaluateStoreInto(Constant *Init, Constant *Val,
|
|||||||
if (ArrayType *ATy = dyn_cast<ArrayType>(InitTy))
|
if (ArrayType *ATy = dyn_cast<ArrayType>(InitTy))
|
||||||
NumElts = ATy->getNumElements();
|
NumElts = ATy->getNumElements();
|
||||||
else
|
else
|
||||||
NumElts = cast<VectorType>(InitTy)->getNumElements();
|
NumElts = InitTy->getVectorNumElements();
|
||||||
|
|
||||||
// Break up the array into elements.
|
// Break up the array into elements.
|
||||||
if (ConstantArray *CA = dyn_cast<ConstantArray>(Init)) {
|
for (uint64_t i = 0, e = NumElts; i != e; ++i)
|
||||||
for (User::op_iterator i = CA->op_begin(), e = CA->op_end(); i != e; ++i)
|
Elts.push_back(Init->getAggregateElement(i));
|
||||||
Elts.push_back(cast<Constant>(*i));
|
|
||||||
} else if (ConstantVector *CV = dyn_cast<ConstantVector>(Init)) {
|
|
||||||
for (User::op_iterator i = CV->op_begin(), e = CV->op_end(); i != e; ++i)
|
|
||||||
Elts.push_back(cast<Constant>(*i));
|
|
||||||
} else if (isa<ConstantAggregateZero>(Init)) {
|
|
||||||
Elts.assign(NumElts, Constant::getNullValue(InitTy->getElementType()));
|
|
||||||
} else {
|
|
||||||
assert(isa<UndefValue>(Init) && "This code is out of sync with "
|
|
||||||
" ConstantFoldLoadThroughGEPConstantExpr");
|
|
||||||
Elts.assign(NumElts, UndefValue::get(InitTy->getElementType()));
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(CI->getZExtValue() < NumElts);
|
assert(CI->getZExtValue() < NumElts);
|
||||||
Elts[CI->getZExtValue()] =
|
Elts[CI->getZExtValue()] =
|
||||||
|
@ -1270,24 +1270,16 @@ Instruction *InstCombiner::visitExtractValueInst(ExtractValueInst &EV) {
|
|||||||
return ReplaceInstUsesWith(EV, Agg);
|
return ReplaceInstUsesWith(EV, Agg);
|
||||||
|
|
||||||
if (Constant *C = dyn_cast<Constant>(Agg)) {
|
if (Constant *C = dyn_cast<Constant>(Agg)) {
|
||||||
if (isa<UndefValue>(C))
|
if (Constant *C2 = C->getAggregateElement(*EV.idx_begin())) {
|
||||||
return ReplaceInstUsesWith(EV, UndefValue::get(EV.getType()));
|
if (EV.getNumIndices() == 0)
|
||||||
|
return ReplaceInstUsesWith(EV, C2);
|
||||||
if (isa<ConstantAggregateZero>(C))
|
// Extract the remaining indices out of the constant indexed by the
|
||||||
return ReplaceInstUsesWith(EV, Constant::getNullValue(EV.getType()));
|
// first index
|
||||||
|
return ExtractValueInst::Create(C2, EV.getIndices().slice(1));
|
||||||
if (isa<ConstantArray>(C) || isa<ConstantStruct>(C)) {
|
|
||||||
// Extract the element indexed by the first index out of the constant
|
|
||||||
Value *V = C->getOperand(*EV.idx_begin());
|
|
||||||
if (EV.getNumIndices() > 1)
|
|
||||||
// Extract the remaining indices out of the constant indexed by the
|
|
||||||
// first index
|
|
||||||
return ExtractValueInst::Create(V, EV.getIndices().slice(1));
|
|
||||||
else
|
|
||||||
return ReplaceInstUsesWith(EV, V);
|
|
||||||
}
|
}
|
||||||
return 0; // Can't handle other constants
|
return 0; // Can't handle other constants
|
||||||
}
|
}
|
||||||
|
|
||||||
if (InsertValueInst *IV = dyn_cast<InsertValueInst>(Agg)) {
|
if (InsertValueInst *IV = dyn_cast<InsertValueInst>(Agg)) {
|
||||||
// We're extracting from an insertvalue instruction, compare the indices
|
// We're extracting from an insertvalue instruction, compare the indices
|
||||||
const unsigned *exti, *exte, *insi, *inse;
|
const unsigned *exti, *exte, *insi, *inse;
|
||||||
|
@ -408,15 +408,14 @@ private:
|
|||||||
return LV; // Common case, already in the map.
|
return LV; // Common case, already in the map.
|
||||||
|
|
||||||
if (Constant *C = dyn_cast<Constant>(V)) {
|
if (Constant *C = dyn_cast<Constant>(V)) {
|
||||||
if (isa<UndefValue>(C))
|
Constant *Elt = C->getAggregateElement(i);
|
||||||
; // Undef values remain undefined.
|
|
||||||
else if (ConstantStruct *CS = dyn_cast<ConstantStruct>(C))
|
if (Elt == 0)
|
||||||
LV.markConstant(CS->getOperand(i)); // Constants are constant.
|
|
||||||
else if (isa<ConstantAggregateZero>(C)) {
|
|
||||||
Type *FieldTy = cast<StructType>(V->getType())->getElementType(i);
|
|
||||||
LV.markConstant(Constant::getNullValue(FieldTy));
|
|
||||||
} else
|
|
||||||
LV.markOverdefined(); // Unknown sort of constant.
|
LV.markOverdefined(); // Unknown sort of constant.
|
||||||
|
else if (isa<UndefValue>(Elt))
|
||||||
|
; // Undef values remain undefined.
|
||||||
|
else
|
||||||
|
LV.markConstant(Elt); // Constants are constant.
|
||||||
}
|
}
|
||||||
|
|
||||||
// All others are underdefined by default.
|
// All others are underdefined by default.
|
||||||
|
@ -856,6 +856,37 @@ static void WriteConstantInternal(raw_ostream &Out, const Constant *CV,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (const ConstantDataArray *CA = dyn_cast<ConstantDataArray>(CV)) {
|
||||||
|
// As a special case, print the array as a string if it is an array of
|
||||||
|
// i8 with ConstantInt values.
|
||||||
|
if (CA->isString()) {
|
||||||
|
Out << "c\"";
|
||||||
|
PrintEscapedString(CA->getAsString(), Out);
|
||||||
|
Out << '"';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Type *ETy = CA->getType()->getElementType();
|
||||||
|
Out << '[';
|
||||||
|
if (CA->getNumOperands()) {
|
||||||
|
TypePrinter.print(ETy, Out);
|
||||||
|
Out << ' ';
|
||||||
|
WriteAsOperandInternal(Out, CA->getElementAsConstant(0),
|
||||||
|
&TypePrinter, Machine,
|
||||||
|
Context);
|
||||||
|
for (unsigned i = 1, e = CA->getNumOperands(); i != e; ++i) {
|
||||||
|
Out << ", ";
|
||||||
|
TypePrinter.print(ETy, Out);
|
||||||
|
Out << ' ';
|
||||||
|
WriteAsOperandInternal(Out, CA->getElementAsConstant(i), &TypePrinter,
|
||||||
|
Machine, Context);
|
||||||
|
}
|
||||||
|
Out << ']';
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if (const ConstantStruct *CS = dyn_cast<ConstantStruct>(CV)) {
|
if (const ConstantStruct *CS = dyn_cast<ConstantStruct>(CV)) {
|
||||||
if (CS->getType()->isPacked())
|
if (CS->getType()->isPacked())
|
||||||
Out << '<';
|
Out << '<';
|
||||||
@ -886,21 +917,19 @@ static void WriteConstantInternal(raw_ostream &Out, const Constant *CV,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (const ConstantVector *CP = dyn_cast<ConstantVector>(CV)) {
|
if (isa<ConstantVector>(CV) || isa<ConstantDataVector>(CV)) {
|
||||||
Type *ETy = CP->getType()->getElementType();
|
Type *ETy = CV->getType()->getVectorElementType();
|
||||||
assert(CP->getNumOperands() > 0 &&
|
|
||||||
"Number of operands for a PackedConst must be > 0");
|
|
||||||
Out << '<';
|
Out << '<';
|
||||||
TypePrinter.print(ETy, Out);
|
TypePrinter.print(ETy, Out);
|
||||||
Out << ' ';
|
Out << ' ';
|
||||||
WriteAsOperandInternal(Out, CP->getOperand(0), &TypePrinter, Machine,
|
WriteAsOperandInternal(Out, CV->getAggregateElement(0U), &TypePrinter,
|
||||||
Context);
|
Machine, Context);
|
||||||
for (unsigned i = 1, e = CP->getNumOperands(); i != e; ++i) {
|
for (unsigned i = 1, e = CV->getType()->getVectorNumElements(); i != e;++i){
|
||||||
Out << ", ";
|
Out << ", ";
|
||||||
TypePrinter.print(ETy, Out);
|
TypePrinter.print(ETy, Out);
|
||||||
Out << ' ';
|
Out << ' ';
|
||||||
WriteAsOperandInternal(Out, CP->getOperand(i), &TypePrinter, Machine,
|
WriteAsOperandInternal(Out, CV->getAggregateElement(i), &TypePrinter,
|
||||||
Context);
|
Machine, Context);
|
||||||
}
|
}
|
||||||
Out << '>';
|
Out << '>';
|
||||||
return;
|
return;
|
||||||
|
@ -38,11 +38,10 @@ using namespace llvm;
|
|||||||
// ConstantFold*Instruction Implementations
|
// ConstantFold*Instruction Implementations
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
/// BitCastConstantVector - Convert the specified ConstantVector node to the
|
/// BitCastConstantVector - Convert the specified vector Constant node to the
|
||||||
/// specified vector type. At this point, we know that the elements of the
|
/// specified vector type. At this point, we know that the elements of the
|
||||||
/// input vector constant are all simple integer or FP values.
|
/// input vector constant are all simple integer or FP values.
|
||||||
static Constant *BitCastConstantVector(ConstantVector *CV,
|
static Constant *BitCastConstantVector(Constant *CV, VectorType *DstTy) {
|
||||||
VectorType *DstTy) {
|
|
||||||
|
|
||||||
if (CV->isAllOnesValue()) return Constant::getAllOnesValue(DstTy);
|
if (CV->isAllOnesValue()) return Constant::getAllOnesValue(DstTy);
|
||||||
if (CV->isNullValue()) return Constant::getNullValue(DstTy);
|
if (CV->isNullValue()) return Constant::getNullValue(DstTy);
|
||||||
@ -51,22 +50,21 @@ static Constant *BitCastConstantVector(ConstantVector *CV,
|
|||||||
// doing so requires endianness information. This should be handled by
|
// doing so requires endianness information. This should be handled by
|
||||||
// Analysis/ConstantFolding.cpp
|
// Analysis/ConstantFolding.cpp
|
||||||
unsigned NumElts = DstTy->getNumElements();
|
unsigned NumElts = DstTy->getNumElements();
|
||||||
if (NumElts != CV->getNumOperands())
|
if (NumElts != CV->getType()->getVectorNumElements())
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
Type *DstEltTy = DstTy->getElementType();
|
||||||
|
|
||||||
// Check to verify that all elements of the input are simple.
|
// Check to verify that all elements of the input are simple.
|
||||||
|
SmallVector<Constant*, 16> Result;
|
||||||
for (unsigned i = 0; i != NumElts; ++i) {
|
for (unsigned i = 0; i != NumElts; ++i) {
|
||||||
if (!isa<ConstantInt>(CV->getOperand(i)) &&
|
Constant *C = CV->getAggregateElement(i);
|
||||||
!isa<ConstantFP>(CV->getOperand(i)))
|
if (C == 0) return 0;
|
||||||
return 0;
|
C = ConstantExpr::getBitCast(C, DstEltTy);
|
||||||
|
if (isa<ConstantExpr>(C)) return 0;
|
||||||
|
Result.push_back(C);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bitcast each element now.
|
|
||||||
std::vector<Constant*> Result;
|
|
||||||
Type *DstEltTy = DstTy->getElementType();
|
|
||||||
for (unsigned i = 0; i != NumElts; ++i)
|
|
||||||
Result.push_back(ConstantExpr::getBitCast(CV->getOperand(i),
|
|
||||||
DstEltTy));
|
|
||||||
return ConstantVector::get(Result);
|
return ConstantVector::get(Result);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -142,8 +140,8 @@ static Constant *FoldBitCast(Constant *V, Type *DestTy) {
|
|||||||
if (isa<ConstantAggregateZero>(V))
|
if (isa<ConstantAggregateZero>(V))
|
||||||
return Constant::getNullValue(DestTy);
|
return Constant::getNullValue(DestTy);
|
||||||
|
|
||||||
if (ConstantVector *CV = dyn_cast<ConstantVector>(V))
|
// Handle ConstantVector and ConstantAggregateVector.
|
||||||
return BitCastConstantVector(CV, DestPTy);
|
return BitCastConstantVector(V, DestPTy);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Canonicalize scalar-to-vector bitcasts into vector-to-vector bitcasts
|
// Canonicalize scalar-to-vector bitcasts into vector-to-vector bitcasts
|
||||||
@ -692,42 +690,26 @@ Constant *llvm::ConstantFoldCastInstruction(unsigned opc, Constant *V,
|
|||||||
|
|
||||||
Constant *llvm::ConstantFoldSelectInstruction(Constant *Cond,
|
Constant *llvm::ConstantFoldSelectInstruction(Constant *Cond,
|
||||||
Constant *V1, Constant *V2) {
|
Constant *V1, Constant *V2) {
|
||||||
if (ConstantInt *CB = dyn_cast<ConstantInt>(Cond))
|
// Check for i1 and vector true/false conditions.
|
||||||
return CB->getZExtValue() ? V1 : V2;
|
|
||||||
|
|
||||||
// Check for zero aggregate and ConstantVector of zeros
|
|
||||||
if (Cond->isNullValue()) return V2;
|
if (Cond->isNullValue()) return V2;
|
||||||
|
if (Cond->isAllOnesValue()) return V1;
|
||||||
|
|
||||||
if (ConstantVector* CondV = dyn_cast<ConstantVector>(Cond)) {
|
// FIXME: CDV Condition.
|
||||||
|
// If the condition is a vector constant, fold the result elementwise.
|
||||||
|
if (ConstantVector *CondV = dyn_cast<ConstantVector>(Cond)) {
|
||||||
|
SmallVector<Constant*, 16> Result;
|
||||||
|
for (unsigned i = 0, e = V1->getType()->getVectorNumElements(); i != e;++i){
|
||||||
|
ConstantInt *Cond = dyn_cast<ConstantInt>(CondV->getOperand(i));
|
||||||
|
if (Cond == 0) break;
|
||||||
|
|
||||||
if (CondV->isAllOnesValue()) return V1;
|
Constant *Res = (Cond->getZExtValue() ? V2 : V1)->getAggregateElement(i);
|
||||||
|
if (Res == 0) break;
|
||||||
VectorType *VTy = cast<VectorType>(V1->getType());
|
Result.push_back(Res);
|
||||||
ConstantVector *CP1 = dyn_cast<ConstantVector>(V1);
|
|
||||||
ConstantVector *CP2 = dyn_cast<ConstantVector>(V2);
|
|
||||||
|
|
||||||
if ((CP1 || isa<ConstantAggregateZero>(V1)) &&
|
|
||||||
(CP2 || isa<ConstantAggregateZero>(V2))) {
|
|
||||||
|
|
||||||
// Find the element type of the returned vector
|
|
||||||
Type *EltTy = VTy->getElementType();
|
|
||||||
unsigned NumElem = VTy->getNumElements();
|
|
||||||
std::vector<Constant*> Res(NumElem);
|
|
||||||
|
|
||||||
bool Valid = true;
|
|
||||||
for (unsigned i = 0; i < NumElem; ++i) {
|
|
||||||
ConstantInt* c = dyn_cast<ConstantInt>(CondV->getOperand(i));
|
|
||||||
if (!c) {
|
|
||||||
Valid = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
Constant *C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Constant *C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res[i] = c->getZExtValue() ? C1 : C2;
|
|
||||||
}
|
|
||||||
// If we were able to build the vector, return it
|
|
||||||
if (Valid) return ConstantVector::get(Res);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If we were able to build the vector, return it.
|
||||||
|
if (Result.size() == V1->getType()->getVectorNumElements())
|
||||||
|
return ConstantVector::get(Result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -781,70 +763,22 @@ Constant *llvm::ConstantFoldInsertElementInstruction(Constant *Val,
|
|||||||
Constant *Idx) {
|
Constant *Idx) {
|
||||||
ConstantInt *CIdx = dyn_cast<ConstantInt>(Idx);
|
ConstantInt *CIdx = dyn_cast<ConstantInt>(Idx);
|
||||||
if (!CIdx) return 0;
|
if (!CIdx) return 0;
|
||||||
APInt idxVal = CIdx->getValue();
|
const APInt &IdxVal = CIdx->getValue();
|
||||||
if (isa<UndefValue>(Val)) {
|
|
||||||
// Insertion of scalar constant into vector undef
|
SmallVector<Constant*, 16> Result;
|
||||||
// Optimize away insertion of undef
|
for (unsigned i = 0, e = Val->getType()->getVectorNumElements(); i != e; ++i){
|
||||||
if (isa<UndefValue>(Elt))
|
if (i == IdxVal) {
|
||||||
return Val;
|
Result.push_back(Elt);
|
||||||
// Otherwise break the aggregate undef into multiple undefs and do
|
continue;
|
||||||
// the insertion
|
|
||||||
unsigned numOps =
|
|
||||||
cast<VectorType>(Val->getType())->getNumElements();
|
|
||||||
std::vector<Constant*> Ops;
|
|
||||||
Ops.reserve(numOps);
|
|
||||||
for (unsigned i = 0; i < numOps; ++i) {
|
|
||||||
Constant *Op =
|
|
||||||
(idxVal == i) ? Elt : UndefValue::get(Elt->getType());
|
|
||||||
Ops.push_back(Op);
|
|
||||||
}
|
}
|
||||||
return ConstantVector::get(Ops);
|
|
||||||
}
|
if (Constant *C = Val->getAggregateElement(i))
|
||||||
if (isa<ConstantAggregateZero>(Val)) {
|
Result.push_back(C);
|
||||||
// Insertion of scalar constant into vector aggregate zero
|
else
|
||||||
// Optimize away insertion of zero
|
return 0;
|
||||||
if (Elt->isNullValue())
|
|
||||||
return Val;
|
|
||||||
// Otherwise break the aggregate zero into multiple zeros and do
|
|
||||||
// the insertion
|
|
||||||
unsigned numOps =
|
|
||||||
cast<VectorType>(Val->getType())->getNumElements();
|
|
||||||
std::vector<Constant*> Ops;
|
|
||||||
Ops.reserve(numOps);
|
|
||||||
for (unsigned i = 0; i < numOps; ++i) {
|
|
||||||
Constant *Op =
|
|
||||||
(idxVal == i) ? Elt : Constant::getNullValue(Elt->getType());
|
|
||||||
Ops.push_back(Op);
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Ops);
|
|
||||||
}
|
|
||||||
if (ConstantVector *CVal = dyn_cast<ConstantVector>(Val)) {
|
|
||||||
// Insertion of scalar constant into vector constant
|
|
||||||
std::vector<Constant*> Ops;
|
|
||||||
Ops.reserve(CVal->getNumOperands());
|
|
||||||
for (unsigned i = 0; i < CVal->getNumOperands(); ++i) {
|
|
||||||
Constant *Op =
|
|
||||||
(idxVal == i) ? Elt : cast<Constant>(CVal->getOperand(i));
|
|
||||||
Ops.push_back(Op);
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Ops);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return ConstantVector::get(Result);
|
||||||
}
|
|
||||||
|
|
||||||
/// GetVectorElement - If C is a ConstantVector, ConstantAggregateZero or Undef
|
|
||||||
/// return the specified element value. Otherwise return null.
|
|
||||||
static Constant *GetVectorElement(Constant *C, unsigned EltNo) {
|
|
||||||
if (ConstantVector *CV = dyn_cast<ConstantVector>(C))
|
|
||||||
return CV->getOperand(EltNo);
|
|
||||||
|
|
||||||
Type *EltTy = cast<VectorType>(C->getType())->getElementType();
|
|
||||||
if (isa<ConstantAggregateZero>(C))
|
|
||||||
return Constant::getNullValue(EltTy);
|
|
||||||
if (isa<UndefValue>(C))
|
|
||||||
return UndefValue::get(EltTy);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Constant *llvm::ConstantFoldShuffleVectorInstruction(Constant *V1,
|
Constant *llvm::ConstantFoldShuffleVectorInstruction(Constant *V1,
|
||||||
@ -860,24 +794,21 @@ Constant *llvm::ConstantFoldShuffleVectorInstruction(Constant *V1,
|
|||||||
// Loop over the shuffle mask, evaluating each element.
|
// Loop over the shuffle mask, evaluating each element.
|
||||||
SmallVector<Constant*, 32> Result;
|
SmallVector<Constant*, 32> Result;
|
||||||
for (unsigned i = 0; i != MaskNumElts; ++i) {
|
for (unsigned i = 0; i != MaskNumElts; ++i) {
|
||||||
Constant *InElt = GetVectorElement(Mask, i);
|
Constant *InElt = Mask->getAggregateElement(i);
|
||||||
if (InElt == 0) return 0;
|
if (InElt == 0) return 0;
|
||||||
|
|
||||||
if (isa<UndefValue>(InElt))
|
if (isa<UndefValue>(InElt)) {
|
||||||
InElt = UndefValue::get(EltTy);
|
Result.push_back(UndefValue::get(EltTy));
|
||||||
else if (ConstantInt *CI = dyn_cast<ConstantInt>(InElt)) {
|
continue;
|
||||||
unsigned Elt = CI->getZExtValue();
|
|
||||||
if (Elt >= SrcNumElts*2)
|
|
||||||
InElt = UndefValue::get(EltTy);
|
|
||||||
else if (Elt >= SrcNumElts)
|
|
||||||
InElt = GetVectorElement(V2, Elt - SrcNumElts);
|
|
||||||
else
|
|
||||||
InElt = GetVectorElement(V1, Elt);
|
|
||||||
if (InElt == 0) return 0;
|
|
||||||
} else {
|
|
||||||
// Unknown value.
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
unsigned Elt = cast<ConstantInt>(InElt)->getZExtValue();
|
||||||
|
if (Elt >= SrcNumElts*2)
|
||||||
|
InElt = UndefValue::get(EltTy);
|
||||||
|
else if (Elt >= SrcNumElts)
|
||||||
|
InElt = V2->getAggregateElement(Elt - SrcNumElts);
|
||||||
|
else
|
||||||
|
InElt = V1->getAggregateElement(Elt);
|
||||||
|
if (InElt == 0) return 0;
|
||||||
Result.push_back(InElt);
|
Result.push_back(InElt);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -890,26 +821,10 @@ Constant *llvm::ConstantFoldExtractValueInstruction(Constant *Agg,
|
|||||||
if (Idxs.empty())
|
if (Idxs.empty())
|
||||||
return Agg;
|
return Agg;
|
||||||
|
|
||||||
if (isa<UndefValue>(Agg)) // ev(undef, x) -> undef
|
if (Constant *C = Agg->getAggregateElement(Idxs[0]))
|
||||||
return UndefValue::get(ExtractValueInst::getIndexedType(Agg->getType(),
|
return ConstantFoldExtractValueInstruction(C, Idxs.slice(1));
|
||||||
Idxs));
|
|
||||||
|
|
||||||
if (isa<ConstantAggregateZero>(Agg)) // ev(0, x) -> 0
|
return 0;
|
||||||
return
|
|
||||||
Constant::getNullValue(ExtractValueInst::getIndexedType(Agg->getType(),
|
|
||||||
Idxs));
|
|
||||||
|
|
||||||
// Otherwise recurse.
|
|
||||||
if (ConstantStruct *CS = dyn_cast<ConstantStruct>(Agg))
|
|
||||||
return ConstantFoldExtractValueInstruction(CS->getOperand(Idxs[0]),
|
|
||||||
Idxs.slice(1));
|
|
||||||
|
|
||||||
if (ConstantArray *CA = dyn_cast<ConstantArray>(Agg))
|
|
||||||
return ConstantFoldExtractValueInstruction(CA->getOperand(Idxs[0]),
|
|
||||||
Idxs.slice(1));
|
|
||||||
ConstantVector *CV = cast<ConstantVector>(Agg);
|
|
||||||
return ConstantFoldExtractValueInstruction(CV->getOperand(Idxs[0]),
|
|
||||||
Idxs.slice(1));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Constant *llvm::ConstantFoldInsertValueInstruction(Constant *Agg,
|
Constant *llvm::ConstantFoldInsertValueInstruction(Constant *Agg,
|
||||||
@ -919,84 +834,30 @@ Constant *llvm::ConstantFoldInsertValueInstruction(Constant *Agg,
|
|||||||
if (Idxs.empty())
|
if (Idxs.empty())
|
||||||
return Val;
|
return Val;
|
||||||
|
|
||||||
if (isa<UndefValue>(Agg)) {
|
unsigned NumElts;
|
||||||
// Insertion of constant into aggregate undef
|
if (StructType *ST = dyn_cast<StructType>(Agg->getType()))
|
||||||
// Optimize away insertion of undef.
|
NumElts = ST->getNumElements();
|
||||||
if (isa<UndefValue>(Val))
|
else if (ArrayType *AT = dyn_cast<ArrayType>(Agg->getType()))
|
||||||
return Agg;
|
NumElts = AT->getNumElements();
|
||||||
|
else
|
||||||
|
NumElts = AT->getVectorNumElements();
|
||||||
|
|
||||||
// Otherwise break the aggregate undef into multiple undefs and do
|
SmallVector<Constant*, 32> Result;
|
||||||
// the insertion.
|
for (unsigned i = 0; i != NumElts; ++i) {
|
||||||
CompositeType *AggTy = cast<CompositeType>(Agg->getType());
|
Constant *C = Agg->getAggregateElement(i);
|
||||||
unsigned numOps;
|
if (C == 0) return 0;
|
||||||
if (ArrayType *AR = dyn_cast<ArrayType>(AggTy))
|
|
||||||
numOps = AR->getNumElements();
|
|
||||||
else
|
|
||||||
numOps = cast<StructType>(AggTy)->getNumElements();
|
|
||||||
|
|
||||||
std::vector<Constant*> Ops(numOps);
|
if (Idxs[0] == i)
|
||||||
for (unsigned i = 0; i < numOps; ++i) {
|
C = ConstantFoldInsertValueInstruction(C, Val, Idxs.slice(1));
|
||||||
Type *MemberTy = AggTy->getTypeAtIndex(i);
|
|
||||||
Constant *Op =
|
|
||||||
(Idxs[0] == i) ?
|
|
||||||
ConstantFoldInsertValueInstruction(UndefValue::get(MemberTy),
|
|
||||||
Val, Idxs.slice(1)) :
|
|
||||||
UndefValue::get(MemberTy);
|
|
||||||
Ops[i] = Op;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (StructType* ST = dyn_cast<StructType>(AggTy))
|
Result.push_back(C);
|
||||||
return ConstantStruct::get(ST, Ops);
|
|
||||||
return ConstantArray::get(cast<ArrayType>(AggTy), Ops);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isa<ConstantAggregateZero>(Agg)) {
|
if (StructType *ST = dyn_cast<StructType>(Agg->getType()))
|
||||||
// Insertion of constant into aggregate zero
|
return ConstantStruct::get(ST, Result);
|
||||||
// Optimize away insertion of zero.
|
if (ArrayType *AT = dyn_cast<ArrayType>(Agg->getType()))
|
||||||
if (Val->isNullValue())
|
return ConstantArray::get(AT, Result);
|
||||||
return Agg;
|
return ConstantVector::get(Result);
|
||||||
|
|
||||||
// Otherwise break the aggregate zero into multiple zeros and do
|
|
||||||
// the insertion.
|
|
||||||
CompositeType *AggTy = cast<CompositeType>(Agg->getType());
|
|
||||||
unsigned numOps;
|
|
||||||
if (ArrayType *AR = dyn_cast<ArrayType>(AggTy))
|
|
||||||
numOps = AR->getNumElements();
|
|
||||||
else
|
|
||||||
numOps = cast<StructType>(AggTy)->getNumElements();
|
|
||||||
|
|
||||||
std::vector<Constant*> Ops(numOps);
|
|
||||||
for (unsigned i = 0; i < numOps; ++i) {
|
|
||||||
Type *MemberTy = AggTy->getTypeAtIndex(i);
|
|
||||||
Constant *Op =
|
|
||||||
(Idxs[0] == i) ?
|
|
||||||
ConstantFoldInsertValueInstruction(Constant::getNullValue(MemberTy),
|
|
||||||
Val, Idxs.slice(1)) :
|
|
||||||
Constant::getNullValue(MemberTy);
|
|
||||||
Ops[i] = Op;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (StructType *ST = dyn_cast<StructType>(AggTy))
|
|
||||||
return ConstantStruct::get(ST, Ops);
|
|
||||||
return ConstantArray::get(cast<ArrayType>(AggTy), Ops);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isa<ConstantStruct>(Agg) || isa<ConstantArray>(Agg)) {
|
|
||||||
// Insertion of constant into aggregate constant.
|
|
||||||
std::vector<Constant*> Ops(Agg->getNumOperands());
|
|
||||||
for (unsigned i = 0; i < Agg->getNumOperands(); ++i) {
|
|
||||||
Constant *Op = cast<Constant>(Agg->getOperand(i));
|
|
||||||
if (Idxs[0] == i)
|
|
||||||
Op = ConstantFoldInsertValueInstruction(Op, Val, Idxs.slice(1));
|
|
||||||
Ops[i] = Op;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (StructType* ST = dyn_cast<StructType>(Agg->getType()))
|
|
||||||
return ConstantStruct::get(ST, Ops);
|
|
||||||
return ConstantArray::get(cast<ArrayType>(Agg->getType()), Ops);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1174,7 +1035,6 @@ Constant *llvm::ConstantFoldBinaryInstruction(unsigned Opcode,
|
|||||||
// At this point we know neither constant is an UndefValue.
|
// At this point we know neither constant is an UndefValue.
|
||||||
if (ConstantInt *CI1 = dyn_cast<ConstantInt>(C1)) {
|
if (ConstantInt *CI1 = dyn_cast<ConstantInt>(C1)) {
|
||||||
if (ConstantInt *CI2 = dyn_cast<ConstantInt>(C2)) {
|
if (ConstantInt *CI2 = dyn_cast<ConstantInt>(C2)) {
|
||||||
using namespace APIntOps;
|
|
||||||
const APInt &C1V = CI1->getValue();
|
const APInt &C1V = CI1->getValue();
|
||||||
const APInt &C2V = CI2->getValue();
|
const APInt &C2V = CI2->getValue();
|
||||||
switch (Opcode) {
|
switch (Opcode) {
|
||||||
@ -1271,145 +1131,18 @@ Constant *llvm::ConstantFoldBinaryInstruction(unsigned Opcode,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (VectorType *VTy = dyn_cast<VectorType>(C1->getType())) {
|
} else if (VectorType *VTy = dyn_cast<VectorType>(C1->getType())) {
|
||||||
ConstantVector *CP1 = dyn_cast<ConstantVector>(C1);
|
// Perform elementwise folding.
|
||||||
ConstantVector *CP2 = dyn_cast<ConstantVector>(C2);
|
SmallVector<Constant*, 16> Result;
|
||||||
if ((CP1 != NULL || isa<ConstantAggregateZero>(C1)) &&
|
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
||||||
(CP2 != NULL || isa<ConstantAggregateZero>(C2))) {
|
Constant *LHS = C1->getAggregateElement(i);
|
||||||
std::vector<Constant*> Res;
|
Constant *RHS = C2->getAggregateElement(i);
|
||||||
Type* EltTy = VTy->getElementType();
|
if (LHS == 0 || RHS == 0) break;
|
||||||
Constant *C1 = 0;
|
|
||||||
Constant *C2 = 0;
|
Result.push_back(ConstantExpr::get(Opcode, LHS, RHS));
|
||||||
switch (Opcode) {
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
case Instruction::Add:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getAdd(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::FAdd:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getFAdd(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::Sub:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getSub(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::FSub:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getFSub(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::Mul:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getMul(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::FMul:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getFMul(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::UDiv:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getUDiv(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::SDiv:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getSDiv(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::FDiv:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getFDiv(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::URem:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getURem(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::SRem:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getSRem(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::FRem:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getFRem(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::And:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getAnd(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::Or:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getOr(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::Xor:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getXor(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::LShr:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getLShr(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::AShr:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getAShr(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
case Instruction::Shl:
|
|
||||||
for (unsigned i = 0, e = VTy->getNumElements(); i != e; ++i) {
|
|
||||||
C1 = CP1 ? CP1->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
C2 = CP2 ? CP2->getOperand(i) : Constant::getNullValue(EltTy);
|
|
||||||
Res.push_back(ConstantExpr::getShl(C1, C2));
|
|
||||||
}
|
|
||||||
return ConstantVector::get(Res);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (Result.size() == VTy->getNumElements())
|
||||||
|
return ConstantVector::get(Result);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ConstantExpr *CE1 = dyn_cast<ConstantExpr>(C1)) {
|
if (ConstantExpr *CE1 = dyn_cast<ConstantExpr>(C1)) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user