mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-06-26 07:24:25 +00:00
IR: Properly return nullptr when getAggregateElement is out-of-bounds
We didn't properly handle the out-of-bounds case for ConstantAggregateZero and UndefValue. This would manifest as a crash when the constant folder was asked to fold a load of a constant global whose struct type has no operands. This fixes PR22595. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@229352 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@ -325,6 +325,9 @@ public:
|
|||||||
/// index.
|
/// index.
|
||||||
Constant *getElementValue(unsigned Idx) const;
|
Constant *getElementValue(unsigned Idx) const;
|
||||||
|
|
||||||
|
/// \brief Return the number of elements in the array, vector, or struct.
|
||||||
|
unsigned getNumElements() const;
|
||||||
|
|
||||||
/// Methods for support type inquiry through isa, cast, and dyn_cast:
|
/// Methods for support type inquiry through isa, cast, and dyn_cast:
|
||||||
///
|
///
|
||||||
static bool classof(const Value *V) {
|
static bool classof(const Value *V) {
|
||||||
@ -1196,6 +1199,9 @@ public:
|
|||||||
/// index.
|
/// index.
|
||||||
UndefValue *getElementValue(unsigned Idx) const;
|
UndefValue *getElementValue(unsigned Idx) const;
|
||||||
|
|
||||||
|
/// \brief Return the number of elements in the array, vector, or struct.
|
||||||
|
unsigned getNumElements() const;
|
||||||
|
|
||||||
void destroyConstant() override;
|
void destroyConstant() override;
|
||||||
|
|
||||||
/// Methods for support type inquiry through isa, cast, and dyn_cast:
|
/// Methods for support type inquiry through isa, cast, and dyn_cast:
|
||||||
|
@ -257,11 +257,11 @@ Constant *Constant::getAggregateElement(unsigned Elt) const {
|
|||||||
if (const ConstantVector *CV = dyn_cast<ConstantVector>(this))
|
if (const ConstantVector *CV = dyn_cast<ConstantVector>(this))
|
||||||
return Elt < CV->getNumOperands() ? CV->getOperand(Elt) : nullptr;
|
return Elt < CV->getNumOperands() ? CV->getOperand(Elt) : nullptr;
|
||||||
|
|
||||||
if (const ConstantAggregateZero *CAZ =dyn_cast<ConstantAggregateZero>(this))
|
if (const ConstantAggregateZero *CAZ = dyn_cast<ConstantAggregateZero>(this))
|
||||||
return CAZ->getElementValue(Elt);
|
return Elt < CAZ->getNumElements() ? CAZ->getElementValue(Elt) : nullptr;
|
||||||
|
|
||||||
if (const UndefValue *UV = dyn_cast<UndefValue>(this))
|
if (const UndefValue *UV = dyn_cast<UndefValue>(this))
|
||||||
return UV->getElementValue(Elt);
|
return Elt < UV->getNumElements() ? UV->getElementValue(Elt) : nullptr;
|
||||||
|
|
||||||
if (const ConstantDataSequential *CDS =dyn_cast<ConstantDataSequential>(this))
|
if (const ConstantDataSequential *CDS =dyn_cast<ConstantDataSequential>(this))
|
||||||
return Elt < CDS->getNumElements() ? CDS->getElementAsConstant(Elt)
|
return Elt < CDS->getNumElements() ? CDS->getElementAsConstant(Elt)
|
||||||
@ -764,6 +764,14 @@ Constant *ConstantAggregateZero::getElementValue(unsigned Idx) const {
|
|||||||
return getStructElement(Idx);
|
return getStructElement(Idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsigned ConstantAggregateZero::getNumElements() const {
|
||||||
|
const Type *Ty = getType();
|
||||||
|
if (const auto *AT = dyn_cast<ArrayType>(Ty))
|
||||||
|
return AT->getNumElements();
|
||||||
|
if (const auto *VT = dyn_cast<VectorType>(Ty))
|
||||||
|
return VT->getNumElements();
|
||||||
|
return Ty->getStructNumElements();
|
||||||
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// UndefValue Implementation
|
// UndefValue Implementation
|
||||||
@ -797,7 +805,14 @@ UndefValue *UndefValue::getElementValue(unsigned Idx) const {
|
|||||||
return getStructElement(Idx);
|
return getStructElement(Idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsigned UndefValue::getNumElements() const {
|
||||||
|
const Type *Ty = getType();
|
||||||
|
if (const auto *AT = dyn_cast<ArrayType>(Ty))
|
||||||
|
return AT->getNumElements();
|
||||||
|
if (const auto *VT = dyn_cast<VectorType>(Ty))
|
||||||
|
return VT->getNumElements();
|
||||||
|
return Ty->getStructNumElements();
|
||||||
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// ConstantXXX Classes
|
// ConstantXXX Classes
|
||||||
|
19
test/Transforms/InstSimplify/load.ll
Normal file
19
test/Transforms/InstSimplify/load.ll
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
; RUN: opt < %s -instsimplify -S | FileCheck %s
|
||||||
|
|
||||||
|
@zeroinit = constant {} zeroinitializer
|
||||||
|
@undef = constant {} undef
|
||||||
|
|
||||||
|
define i32 @crash_on_zeroinit() {
|
||||||
|
; CHECK-LABEL: @crash_on_zeroinit
|
||||||
|
; CHECK: ret i32 0
|
||||||
|
%load = load i32* bitcast ({}* @zeroinit to i32*)
|
||||||
|
ret i32 %load
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @crash_on_undef() {
|
||||||
|
; CHECK-LABEL: @crash_on_undef
|
||||||
|
; CHECK: ret i32 undef
|
||||||
|
%load = load i32* bitcast ({}* @undef to i32*)
|
||||||
|
ret i32 %load
|
||||||
|
}
|
||||||
|
|
Reference in New Issue
Block a user