Skip to content

Commit e41f892

Browse files
committed
Implement Reload
Implemented a default Reload. It will simply reload all the same values as Commit had saved. This version conservatively reloads everything which is probably ok for garbage collected systems where objects may move. We should consider some enhancements at some point. - option to reload some or all of the elements via selective reload to ensure you don't refetch from memory if not needed. - option to take into account stack size changes before reload. This is be useful in VMs where the virtual machine changes the stack with pops of arguments , and pushed results. In that case, you would adjust the stack for pops, and the reload the number of new elements pushed by the VM. Test for VirtualMachineOperandStack reload which uses the operand stack, passes stack elements which are modfied in an external C call, then ensures those modified values are correct when returned and reloaded. Signed-off-by: John Duimovich <jduimovich@gmail.com>
1 parent b6fd872 commit e41f892

File tree

3 files changed

+76
-2
lines changed

3 files changed

+76
-2
lines changed

compiler/ilgen/VirtualMachineOperandStack.cpp

+19-2
Original file line numberDiff line numberDiff line change
@@ -87,8 +87,25 @@ VirtualMachineOperandStack::Commit(TR::IlBuilder *b)
8787
}
8888
}
8989

90-
void
91-
VirtualMachineOperandStack::MergeInto(OMR::VirtualMachineOperandStack *other, TR::IlBuilder *b)
90+
void
91+
VirtualMachineOperandStack::Reload(TR::IlBuilder* b)
92+
{
93+
TR::IlType* Element = _elementType;
94+
TR::IlType* pElement = _mb->typeDictionary()->PointerTo(Element);
95+
// reload the elements back into the simulated operand stack
96+
// If the # of stack element has changed, the user should adjust the # of elements
97+
// using Drop beforehand to add/delete stack elements.
98+
TR::IlValue* stack = b->Load("OperandStack_base");
99+
for (int32_t i = _stackTop; i >= 0; i--) {
100+
_stack[i] = b->LoadAt(pElement,
101+
b->IndexAt(pElement,
102+
stack,
103+
b->ConstInt32(i - _stackOffset)));
104+
}
105+
}
106+
107+
void
108+
VirtualMachineOperandStack::MergeInto(OMR::VirtualMachineOperandStack* other, TR::IlBuilder* b)
92109
{
93110
TR_ASSERT(_stackTop == other->_stackTop, "stacks are not same size");
94111
for (int32_t i=_stackTop;i >= 0;i--)

compiler/ilgen/VirtualMachineOperandStack.hpp

+8
Original file line numberDiff line numberDiff line change
@@ -106,6 +106,14 @@ class VirtualMachineOperandStack : public VirtualMachineState
106106
* @param b the builder where the operations will be placed to recreate the virtual machine operand stack
107107
*/
108108
virtual void Commit(TR::IlBuilder *b);
109+
110+
/**
111+
* @brief read the virtual machine stack back into the simulated operand stack
112+
* @param b the builder where the operations will be placed to recreate the simulated operand stack
113+
* Users can call Drop beforehand with the appropriate positive or negative count to ensure the simulated
114+
* stack accounts for new or dropped virtual machine stack elements.
115+
*/
116+
virtual void Reload(TR::IlBuilder *b);
109117

110118
/**
111119
* @brief create an identical copy of the current object.

jitbuilder/release/src/OperandStackTests.cpp

+49
Original file line numberDiff line numberDiff line change
@@ -311,6 +311,30 @@ verifyResult12(STACKVALUETYPE top)
311311
OperandStackTestMethod::verifyStack("11", 3, 3, 5, 4, expectedResult12Top);
312312
}
313313

314+
// used to compare expected values and report fail it not equal
315+
void
316+
verifyValuesEqual(STACKVALUETYPE v1, STACKVALUETYPE v2)
317+
{
318+
REPORT2(v1 == v2, "verifyValuesEqual v1", v1, "verifyValuesEqual v2", v2);
319+
}
320+
321+
// take the arguments from the stack and modify them
322+
void
323+
modifyTop3Elements(int32_t amountToAdd)
324+
{
325+
if (verbose) cout << "Push();Push();Push() - modify elements passed in real stack and return";
326+
STACKVALUETYPE *realSP = *verifySP;
327+
REPORT1(realSP[0]== 3, "modifyTop3Elements realSP[0]", realSP[0]);
328+
REPORT1(realSP[-1]== 2, "modifyTop3Elements realSP[-1]", realSP[-1]);
329+
REPORT1(realSP[-2]== 1, "modifyTop3Elements realSP[-2]", realSP[-2]);
330+
realSP[0] += amountToAdd;
331+
realSP[-1] += amountToAdd;
332+
realSP[-2] += amountToAdd;
333+
}
334+
335+
336+
337+
314338
bool
315339
OperandStackTestMethod::verifyUntouched(int32_t maxTouched)
316340
{
@@ -372,12 +396,15 @@ OperandStackTestMethod::OperandStackTestMethod(TR::TypeDictionary *d)
372396
DefineFunction("verifyResult10", "0", "0", (void *)&verifyResult10, NoType, 1, _valueType);
373397
DefineFunction("verifyResult11", "0", "0", (void *)&verifyResult11, NoType, 0);
374398
DefineFunction("verifyResult12", "0", "0", (void *)&verifyResult12, NoType, 1, _valueType);
399+
DefineFunction("verifyValuesEqual", "0", "0", (void *)&verifyValuesEqual, NoType, 2, _valueType, _valueType);
400+
DefineFunction("modifyTop3Elements", "0", "0", (void *)&modifyTop3Elements, NoType, 1, _valueType);
375401
}
376402

377403
// convenience macros
378404
#define STACK(b) (((TestState *)(b)->vmState())->_stack)
379405
#define STACKTOP(b) (((TestState *)(b)->vmState())->_stackTop)
380406
#define COMMIT(b) ((b)->vmState()->Commit(b))
407+
#define RELOAD(b) ((b)->vmState()->Reload(b))
381408
#define PUSH(b,v) (STACK(b)->Push(b,v))
382409
#define POP(b) (STACK(b)->Pop(b))
383410
#define TOP(b) (STACK(b)->Top())
@@ -431,6 +458,9 @@ OperandStackTestMethod::testStack(TR::BytecodeBuilder *b, bool useEqual)
431458

432459
COMMIT(b);
433460
b->Call("verifyResult11", 0);
461+
462+
463+
434464

435465
TR::BytecodeBuilder *thenBB = OrphanBytecodeBuilder(0, (char*)"BCI_then");
436466
TR::BytecodeBuilder *elseBB = OrphanBytecodeBuilder(1, (char*)"BCI_else");
@@ -452,6 +482,25 @@ OperandStackTestMethod::testStack(TR::BytecodeBuilder *b, bool useEqual)
452482

453483
COMMIT(mergeBB);
454484
mergeBB->Call("verifyResult12", 1, TOP(mergeBB));
485+
486+
int amountToAdd = 10;
487+
// Reload test. Call a routine that modifies stack elements passed to it.
488+
// Test by reloading and test the popped values
489+
PUSH(mergeBB, mergeBB->ConstInteger(_valueType, 1));
490+
PUSH(mergeBB, mergeBB->ConstInteger(_valueType, 2));
491+
PUSH(mergeBB, mergeBB->ConstInteger(_valueType, 3));
492+
COMMIT(mergeBB);
493+
mergeBB->Call("modifyTop3Elements",1, mergeBB->ConstInteger(_valueType, amountToAdd));
494+
RELOAD(mergeBB);
495+
TR::IlValue *modifiedStackElement = POP(mergeBB);
496+
TR::IlValue *expected = mergeBB->ConstInteger(_valueType, 3+amountToAdd);
497+
mergeBB->Call("verifyValuesEqual", 2, modifiedStackElement, expected);
498+
modifiedStackElement = POP(mergeBB);
499+
expected = mergeBB->ConstInteger(_valueType, 2+amountToAdd);
500+
mergeBB->Call("verifyValuesEqual",2, modifiedStackElement, expected);
501+
modifiedStackElement = POP(mergeBB);
502+
expected = mergeBB->ConstInteger(_valueType, 1+amountToAdd);
503+
mergeBB->Call("verifyValuesEqual",2, modifiedStackElement, expected);
455504

456505
mergeBB->Return();
457506

0 commit comments

Comments
 (0)