mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-03 13:31:05 +00:00
Allow constants of different types to share constant pool entries
if they have compatible encodings. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@85359 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
c5a75523ef
commit
83f6120c9a
@ -581,6 +581,47 @@ MachineConstantPool::~MachineConstantPool() {
|
|||||||
delete Constants[i].Val.MachineCPVal;
|
delete Constants[i].Val.MachineCPVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// CanShareConstantPoolEntry - Test whether the given two constants
|
||||||
|
/// can be allocated the same constant pool entry.
|
||||||
|
static bool CanShareConstantPoolEntry(Constant *A, Constant *B,
|
||||||
|
const TargetData *TD) {
|
||||||
|
// Handle the trivial case quickly.
|
||||||
|
if (A == B) return true;
|
||||||
|
|
||||||
|
// If they have the same type but weren't the same constant, quickly
|
||||||
|
// reject them.
|
||||||
|
if (A->getType() == B->getType()) return false;
|
||||||
|
|
||||||
|
// For now, only support constants with the same size.
|
||||||
|
if (TD->getTypeStoreSize(A->getType()) != TD->getTypeStoreSize(B->getType()))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// If a floating-point value and an integer value have the same encoding,
|
||||||
|
// they can share a constant-pool entry.
|
||||||
|
if (ConstantFP *AFP = dyn_cast<ConstantFP>(A))
|
||||||
|
if (ConstantInt *BI = dyn_cast<ConstantInt>(B))
|
||||||
|
return AFP->getValueAPF().bitcastToAPInt() == BI->getValue();
|
||||||
|
if (ConstantFP *BFP = dyn_cast<ConstantFP>(B))
|
||||||
|
if (ConstantInt *AI = dyn_cast<ConstantInt>(A))
|
||||||
|
return BFP->getValueAPF().bitcastToAPInt() == AI->getValue();
|
||||||
|
|
||||||
|
// Two vectors can share an entry if each pair of corresponding
|
||||||
|
// elements could.
|
||||||
|
if (ConstantVector *AV = dyn_cast<ConstantVector>(A))
|
||||||
|
if (ConstantVector *BV = dyn_cast<ConstantVector>(B)) {
|
||||||
|
if (AV->getType()->getNumElements() != BV->getType()->getNumElements())
|
||||||
|
return false;
|
||||||
|
for (unsigned i = 0, e = AV->getType()->getNumElements(); i != e; ++i)
|
||||||
|
if (!CanShareConstantPoolEntry(AV->getOperand(i), BV->getOperand(i), TD))
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Handle other cases.
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/// getConstantPoolIndex - Create a new entry in the constant pool or return
|
/// getConstantPoolIndex - Create a new entry in the constant pool or return
|
||||||
/// an existing one. User must specify the log2 of the minimum required
|
/// an existing one. User must specify the log2 of the minimum required
|
||||||
/// alignment for the object.
|
/// alignment for the object.
|
||||||
@ -594,9 +635,12 @@ unsigned MachineConstantPool::getConstantPoolIndex(Constant *C,
|
|||||||
//
|
//
|
||||||
// FIXME, this could be made much more efficient for large constant pools.
|
// FIXME, this could be made much more efficient for large constant pools.
|
||||||
for (unsigned i = 0, e = Constants.size(); i != e; ++i)
|
for (unsigned i = 0, e = Constants.size(); i != e; ++i)
|
||||||
if (Constants[i].Val.ConstVal == C &&
|
if (!Constants[i].isMachineConstantPoolEntry() &&
|
||||||
(Constants[i].getAlignment() & (Alignment - 1)) == 0)
|
CanShareConstantPoolEntry(Constants[i].Val.ConstVal, C, TD)) {
|
||||||
|
if ((unsigned)Constants[i].getAlignment() < Alignment)
|
||||||
|
Constants[i].Alignment = Alignment;
|
||||||
return i;
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
Constants.push_back(MachineConstantPoolEntry(C, Alignment));
|
Constants.push_back(MachineConstantPoolEntry(C, Alignment));
|
||||||
return Constants.size()-1;
|
return Constants.size()-1;
|
||||||
|
19
test/CodeGen/X86/constant-pool-sharing.ll
Normal file
19
test/CodeGen/X86/constant-pool-sharing.ll
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
; RUN: llc < %s -march=x86-64 | FileCheck %s
|
||||||
|
|
||||||
|
; llc should share constant pool entries between this integer vector
|
||||||
|
; and this floating-point vector since they have the same encoding.
|
||||||
|
|
||||||
|
; CHECK: LCPI1_0(%rip), %xmm0
|
||||||
|
; CHECK: movaps %xmm0, (%rdi)
|
||||||
|
; CHECK: movaps %xmm0, (%rsi)
|
||||||
|
|
||||||
|
define void @foo(<4 x i32>* %p, <4 x float>* %q, i1 %t) nounwind {
|
||||||
|
entry:
|
||||||
|
br label %loop
|
||||||
|
loop:
|
||||||
|
store <4 x i32><i32 1073741824, i32 1073741824, i32 1073741824, i32 1073741824>, <4 x i32>* %p
|
||||||
|
store <4 x float><float 2.0, float 2.0, float 2.0, float 2.0>, <4 x float>* %q
|
||||||
|
br i1 %t, label %loop, label %ret
|
||||||
|
ret:
|
||||||
|
ret void
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user