From a69c366d07b6562e9a79ede4aa2e74c0266101e5 Mon Sep 17 00:00:00 2001 From: Joel Dice Date: Sun, 20 Jan 2008 11:55:08 -0700 Subject: [PATCH] save 'this' pointer on entrance to synchronized, non-static, non-native methods so we can release the monitor on exit, reguardless of whether the pointer at local index zero is overwritten --- src/compile.cpp | 55 +++++++++++++++++++++++++++++++++++-------------- 1 file changed, 39 insertions(+), 16 deletions(-) diff --git a/src/compile.cpp b/src/compile.cpp index 584bb6d5c5..66deb86e62 100644 --- a/src/compile.cpp +++ b/src/compile.cpp @@ -376,11 +376,20 @@ enum Event { TraceEvent }; +unsigned +localSize(MyThread* t, object method) +{ + unsigned size = codeMaxLocals(t, methodCode(t, method)); + if (methodFlags(t, method) & ACC_SYNCHRONIZED) { + ++ size; + } + return size; +} + unsigned frameSize(MyThread* t, object method) { - return codeMaxLocals(t, methodCode(t, method)) - + codeMaxStack(t, methodCode(t, method)); + return localSize(t, method) + codeMaxStack(t, methodCode(t, method)); } unsigned @@ -528,7 +537,7 @@ class Frame { } unsigned localSize() { - return codeMaxLocals(t, methodCode(t, context->method)); + return ::localSize(t, context->method); } unsigned stackSize() { @@ -1067,6 +1076,12 @@ class Frame { unsigned level; }; +unsigned +savedTargetIndex(MyThread* t, object method) +{ + return codeMaxLocals(t, methodCode(t, method)); +} + void findUnwindTarget(MyThread* t, void** targetIp, void** targetBase, void** targetStack) @@ -1095,7 +1110,7 @@ findUnwindTarget(MyThread* t, void** targetIp, void** targetBase, if (handler) { unsigned parameterFootprint = methodParameterFootprint(t, method); - unsigned localFootprint = codeMaxLocals(t, methodCode(t, method)); + unsigned localFootprint = localSize(t, method); stack = static_cast(base) - (localFootprint - parameterFootprint); @@ -1112,7 +1127,7 @@ findUnwindTarget(MyThread* t, void** targetIp, void** targetBase, if (methodFlags(t, method) & ACC_STATIC) { lock = methodClass(t, method); } else { - lock = *localObject(t, base, method, 0); + lock = *localObject(t, base, method, savedTargetIndex(t, method)); } release(t, lock); @@ -1572,7 +1587,8 @@ handleMonitorEvent(MyThread* t, Frame* frame, intptr_t function) if (methodFlags(t, method) & ACC_STATIC) { lock = frame->append(methodClass(t, method)); } else { - lock = c->memory(c->base(), localOffset(t, 0, method)); + lock = c->memory + (c->base(), localOffset(t, savedTargetIndex(t, method), method)); } c->indirectCall @@ -1585,6 +1601,20 @@ handleMonitorEvent(MyThread* t, Frame* frame, intptr_t function) void handleEntrance(MyThread* t, Frame* frame) { + object method = frame->context->method; + + if ((methodFlags(t, method) & ACC_SYNCHRONIZED) + and ((methodFlags(t, method) & ACC_STATIC) == 0)) + { + Compiler* c = frame->c; + + // save 'this' pointer in case it is overwritten. + unsigned index = savedTargetIndex(t, method); + mov(c, c->memory(c->base(), localOffset(t, 0, method)), + c->memory(c->base(), localOffset(t, index, method))); + frame->mark(index); + } + handleMonitorEvent (t, frame, reinterpret_cast(acquireMonitorForObject)); } @@ -3733,11 +3763,8 @@ compile(MyThread* t, Context* context) c->prologue(); - object code = methodCode(t, context->method); - PROTECT(t, code); - unsigned footprint = methodParameterFootprint(t, context->method); - unsigned locals = codeMaxLocals(t, code); + unsigned locals = localSize(t, context->method); c->reserve(locals - footprint); uintptr_t stackMap[stackMapSizeInWords(t, context->method)]; @@ -3794,10 +3821,7 @@ compile(MyThread* t, Context* context) uintptr_t* roots = context->rootTable + (start * frameMapSizeInWords(t, context->method)); - for (unsigned i = 0; - i < codeMaxLocals(t, methodCode(t, context->method)); - ++ i) - { + for (unsigned i = 0; i < localSize(t, context->method); ++ i) { if (getBit(roots, i)) { frame2.mark(i); } @@ -4059,8 +4083,7 @@ visitStackAndLocals(MyThread* t, Heap::Visitor* v, void* base, object node, count = parameterFootprint + height - argumentFootprint; } else { - count = codeMaxStack(t, methodCode(t, method)) - + codeMaxLocals(t, methodCode(t, method)); + count = frameSize(t, method); } if (count) {