MIR Serialization: Serialize the variable sized stack objects.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@242095 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Alex Lorenz 2015-07-14 00:26:26 +00:00
parent b53f724f91
commit dee03ee0f9
6 changed files with 101 additions and 7 deletions

View File

@ -129,9 +129,8 @@ template <> struct MappingTraits<MachineBasicBlock> {
/// ///
/// TODO: Determine isPreallocated flag by mapping between objects and local /// TODO: Determine isPreallocated flag by mapping between objects and local
/// objects (Serialize local objects). /// objects (Serialize local objects).
/// TODO: Serialize variable sized objects.
struct MachineStackObject { struct MachineStackObject {
enum ObjectType { DefaultType, SpillSlot }; enum ObjectType { DefaultType, SpillSlot, VariableSized };
// TODO: Serialize LLVM alloca reference. // TODO: Serialize LLVM alloca reference.
unsigned ID; unsigned ID;
ObjectType Type = DefaultType; ObjectType Type = DefaultType;
@ -144,6 +143,7 @@ template <> struct ScalarEnumerationTraits<MachineStackObject::ObjectType> {
static void enumeration(yaml::IO &IO, MachineStackObject::ObjectType &Type) { static void enumeration(yaml::IO &IO, MachineStackObject::ObjectType &Type) {
IO.enumCase(Type, "default", MachineStackObject::DefaultType); IO.enumCase(Type, "default", MachineStackObject::DefaultType);
IO.enumCase(Type, "spill-slot", MachineStackObject::SpillSlot); IO.enumCase(Type, "spill-slot", MachineStackObject::SpillSlot);
IO.enumCase(Type, "variable-sized", MachineStackObject::VariableSized);
} }
}; };
@ -154,7 +154,8 @@ template <> struct MappingTraits<MachineStackObject> {
"type", Object.Type, "type", Object.Type,
MachineStackObject::DefaultType); // Don't print the default type. MachineStackObject::DefaultType); // Don't print the default type.
YamlIO.mapOptional("offset", Object.Offset); YamlIO.mapOptional("offset", Object.Offset);
YamlIO.mapRequired("size", Object.Size); if (Object.Type != MachineStackObject::VariableSized)
YamlIO.mapRequired("size", Object.Size);
YamlIO.mapOptional("alignment", Object.Alignment); YamlIO.mapOptional("alignment", Object.Alignment);
} }

View File

@ -541,6 +541,14 @@ public:
return Objects[ObjectIdx+NumFixedObjects].Size == ~0ULL; return Objects[ObjectIdx+NumFixedObjects].Size == ~0ULL;
} }
/// Returns true if the specified index corresponds to a variable sized
/// object.
bool isVariableSizedObjectIndex(int ObjectIdx) const {
assert(unsigned(ObjectIdx + NumFixedObjects) < Objects.size() &&
"Invalid Object Idx!");
return Objects[ObjectIdx + NumFixedObjects].Size == 0;
}
/// Create a new statically sized stack object, returning /// Create a new statically sized stack object, returning
/// a nonnegative identifier to represent it. /// a nonnegative identifier to represent it.
int CreateStackObject(uint64_t Size, unsigned Alignment, bool isSS, int CreateStackObject(uint64_t Size, unsigned Alignment, bool isSS,

View File

@ -391,9 +391,14 @@ bool MIRParserImpl::initializeFrameInfo(MachineFrameInfo &MFI,
// Initialize the ordinary frame objects. // Initialize the ordinary frame objects.
for (const auto &Object : YamlMF.StackObjects) { for (const auto &Object : YamlMF.StackObjects) {
int ObjectIdx = MFI.CreateStackObject( int ObjectIdx;
Object.Size, Object.Alignment, if (Object.Type == yaml::MachineStackObject::VariableSized)
Object.Type == yaml::MachineStackObject::SpillSlot); ObjectIdx =
MFI.CreateVariableSizedObject(Object.Alignment, /*Alloca=*/nullptr);
else
ObjectIdx = MFI.CreateStackObject(
Object.Size, Object.Alignment,
Object.Type == yaml::MachineStackObject::SpillSlot);
MFI.setObjectOffset(ObjectIdx, Object.Offset); MFI.setObjectOffset(ObjectIdx, Object.Offset);
// TODO: Store the mapping between object IDs and object indices to parse // TODO: Store the mapping between object IDs and object indices to parse
// stack object references correctly. // stack object references correctly.

View File

@ -188,7 +188,9 @@ void MIRPrinter::convertStackObjects(yaml::MachineFunction &MF,
YamlObject.ID = ID++; YamlObject.ID = ID++;
YamlObject.Type = MFI.isSpillSlotObjectIndex(I) YamlObject.Type = MFI.isSpillSlotObjectIndex(I)
? yaml::MachineStackObject::SpillSlot ? yaml::MachineStackObject::SpillSlot
: yaml::MachineStackObject::DefaultType; : MFI.isVariableSizedObjectIndex(I)
? yaml::MachineStackObject::VariableSized
: yaml::MachineStackObject::DefaultType;
YamlObject.Offset = MFI.getObjectOffset(I); YamlObject.Offset = MFI.getObjectOffset(I);
YamlObject.Size = MFI.getObjectSize(I); YamlObject.Size = MFI.getObjectSize(I);
YamlObject.Alignment = MFI.getObjectAlignment(I); YamlObject.Alignment = MFI.getObjectAlignment(I);

View File

@ -0,0 +1,36 @@
# RUN: not llc -march=x86-64 -start-after branch-folder -stop-after branch-folder -o /dev/null %s 2>&1 | FileCheck %s
--- |
define i32 @test(i32 %a) {
entry:
%b = alloca i32
%x = alloca i64
%y = alloca i32, i32 %a
store i32 %a, i32* %b
store i64 2, i64* %x
%c = load i32, i32* %b
ret i32 %c
}
...
---
name: test
frameInfo:
stackSize: 24
offsetAdjustment: -16
maxAlignment: 8
adjustsStack: true
stack:
- { id: 0, offset: -20, size: 4, alignment: 4 }
- { id: 1, offset: -32, size: 8, alignment: 8 }
# CHECK: [[@LINE+1]]:55: unknown key 'size'
- { id: 2, type: variable-sized, offset: -32, size: 42, alignment: 1 }
body:
- id: 0
name: entry
instructions:
- 'MOV32mr %rsp, 1, _, -4, _, %edi'
- 'MOV64mi32 %rsp, 1, _, -16, _, 2'
- '%eax = MOV32rm %rsp, 1, _, -4, _'
- 'RETQ %eax'
...

View File

@ -0,0 +1,42 @@
# RUN: llc -march=x86-64 -start-after branch-folder -stop-after branch-folder -o /dev/null %s | FileCheck %s
# This test ensures that the MIR parser parses variable sized stack objects
# correctly.
--- |
define i32 @test(i32 %a) {
entry:
%b = alloca i32
%x = alloca i64
%y = alloca i32, i32 %a
store i32 %a, i32* %b
store i64 2, i64* %x
%c = load i32, i32* %b
ret i32 %c
}
...
---
name: test
frameInfo:
stackSize: 24
offsetAdjustment: -16
maxAlignment: 8
adjustsStack: true
# CHECK: stack:
# CHECK-NEXT: - { id: 0, offset: -20, size: 4, alignment: 4 }
# CHECK-NEXT: - { id: 1, offset: -32, size: 8, alignment: 8 }
# CHECK-NEXT: - { id: 2, type: variable-sized, offset: -32, alignment: 1 }
stack:
- { id: 0, offset: -20, size: 4, alignment: 4 }
- { id: 1, offset: -32, size: 8, alignment: 8 }
- { id: 2, type: variable-sized, offset: -32, alignment: 1 }
body:
- id: 0
name: entry
instructions:
- 'MOV32mr %rsp, 1, _, -4, _, %edi'
- 'MOV64mi32 %rsp, 1, _, -16, _, 2'
- '%eax = MOV32rm %rsp, 1, _, -4, _'
- 'RETQ %eax'
...