mirror of
https://github.com/corda/corda.git
synced 2025-01-07 13:38:47 +00:00
fix stack frame mapping code for exception handlers
Previously, the stack frame mapping code (responsible for statically calculating the map of GC roots for a method's stack frame during JIT compilation) would assume that the map of GC roots on entry to an exception handler is the same as on entry to the "try" block which the handler is attached to. Technically, this is true, but the algorithm we use does not consider whether a local variable is still "live" (i.e. will be read later) when calculating the map - only whether we can expect to find a reference there via normal (non-exceptional) control flow. This can backfire if, within a "try" block, the stack location which held an object reference on entry to the block gets overwritten with a non-reference (i.e. a primitive). If an exception is later thrown from such a block, we might end up trying to treat that non-reference as a reference during GC, which will crash the VM. The ideal way to fix this is to calculate the true interval for which each value is live and use that to produce the stack frame maps. This would provide the added benefit of ensuring that the garbage collector does not visit references which, although still present on the stack, will not be used again. However, this commit uses the less invasive strategy of ANDing together the root maps at each GC point within a "try" block and using the result as the map on entry to the corresponding exception handler(s). This should give us safe, if not optimal, results. Later on, we can refine it as described above.
This commit is contained in:
parent
c9b9db1621
commit
99bb7924b0
159
src/compile.cpp
159
src/compile.cpp
@ -588,8 +588,8 @@ print(SubroutinePath* path)
|
|||||||
if (path) {
|
if (path) {
|
||||||
fprintf(stderr, " (");
|
fprintf(stderr, " (");
|
||||||
while (true) {
|
while (true) {
|
||||||
fprintf(stderr, "%p", reinterpret_cast<void*>
|
fprintf(stderr, "%p", path->call->returnAddress->resolved() ?
|
||||||
(path->call->returnAddress->value()));
|
reinterpret_cast<void*>(path->call->returnAddress->value()) : 0);
|
||||||
path = path->stackNext;
|
path = path->stackNext;
|
||||||
if (path) {
|
if (path) {
|
||||||
fprintf(stderr, ", ");
|
fprintf(stderr, ", ");
|
||||||
@ -603,13 +603,18 @@ print(SubroutinePath* path)
|
|||||||
|
|
||||||
class SubroutineTrace {
|
class SubroutineTrace {
|
||||||
public:
|
public:
|
||||||
SubroutineTrace(SubroutinePath* path, SubroutineTrace* next):
|
SubroutineTrace(SubroutinePath* path, SubroutineTrace* next,
|
||||||
|
unsigned mapSize):
|
||||||
path(path),
|
path(path),
|
||||||
next(next)
|
next(next),
|
||||||
{ }
|
watch(false)
|
||||||
|
{
|
||||||
|
memset(map, 0, mapSize * BytesPerWord);
|
||||||
|
}
|
||||||
|
|
||||||
SubroutinePath* path;
|
SubroutinePath* path;
|
||||||
SubroutineTrace* next;
|
SubroutineTrace* next;
|
||||||
|
bool watch;
|
||||||
uintptr_t map[0];
|
uintptr_t map[0];
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -619,17 +624,21 @@ class TraceElement: public TraceHandler {
|
|||||||
static const unsigned TailCall = 1 << 1;
|
static const unsigned TailCall = 1 << 1;
|
||||||
static const unsigned LongCall = 1 << 2;
|
static const unsigned LongCall = 1 << 2;
|
||||||
|
|
||||||
TraceElement(Context* context, object target, unsigned flags,
|
TraceElement(Context* context, unsigned ip, object target, unsigned flags,
|
||||||
TraceElement* next):
|
TraceElement* next, unsigned mapSize):
|
||||||
context(context),
|
context(context),
|
||||||
address(0),
|
address(0),
|
||||||
next(next),
|
next(next),
|
||||||
subroutineTrace(0),
|
subroutineTrace(0),
|
||||||
subroutineTraceCount(0),
|
|
||||||
target(target),
|
target(target),
|
||||||
|
ip(ip),
|
||||||
|
subroutineTraceCount(0),
|
||||||
argumentIndex(0),
|
argumentIndex(0),
|
||||||
flags(flags)
|
flags(flags),
|
||||||
{ }
|
watch(false)
|
||||||
|
{
|
||||||
|
memset(map, 0, mapSize * BytesPerWord);
|
||||||
|
}
|
||||||
|
|
||||||
virtual void handleTrace(Promise* address, unsigned argumentIndex) {
|
virtual void handleTrace(Promise* address, unsigned argumentIndex) {
|
||||||
if (this->address == 0) {
|
if (this->address == 0) {
|
||||||
@ -642,10 +651,12 @@ class TraceElement: public TraceHandler {
|
|||||||
Promise* address;
|
Promise* address;
|
||||||
TraceElement* next;
|
TraceElement* next;
|
||||||
SubroutineTrace* subroutineTrace;
|
SubroutineTrace* subroutineTrace;
|
||||||
unsigned subroutineTraceCount;
|
|
||||||
object target;
|
object target;
|
||||||
|
unsigned ip;
|
||||||
|
unsigned subroutineTraceCount;
|
||||||
unsigned argumentIndex;
|
unsigned argumentIndex;
|
||||||
unsigned flags;
|
unsigned flags;
|
||||||
|
bool watch;
|
||||||
uintptr_t map[0];
|
uintptr_t map[0];
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1587,7 +1598,7 @@ class Frame {
|
|||||||
|
|
||||||
TraceElement* e = context->traceLog = new
|
TraceElement* e = context->traceLog = new
|
||||||
(context->zone.allocate(sizeof(TraceElement) + (mapSize * BytesPerWord)))
|
(context->zone.allocate(sizeof(TraceElement) + (mapSize * BytesPerWord)))
|
||||||
TraceElement(context, target, flags, context->traceLog);
|
TraceElement(context, ip, target, flags, context->traceLog, mapSize);
|
||||||
|
|
||||||
++ context->traceLogCount;
|
++ context->traceLogCount;
|
||||||
|
|
||||||
@ -4923,6 +4934,63 @@ printSet(uintptr_t m, unsigned limit)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
calculateTryCatchRoots(Context* context, SubroutinePath* subroutinePath,
|
||||||
|
uintptr_t* roots, unsigned mapSize, unsigned start,
|
||||||
|
unsigned end)
|
||||||
|
{
|
||||||
|
memset(roots, 0xFF, mapSize * BytesPerWord);
|
||||||
|
|
||||||
|
if (DebugFrameMaps) {
|
||||||
|
fprintf(stderr, "calculate try/catch roots from %d to %d", start, end);
|
||||||
|
if (subroutinePath) {
|
||||||
|
fprintf(stderr, " ");
|
||||||
|
print(subroutinePath);
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
for (TraceElement* te = context->traceLog; te; te = te->next) {
|
||||||
|
if (te->ip >= start and te->ip < end) {
|
||||||
|
uintptr_t* traceRoots = 0;
|
||||||
|
if (subroutinePath == 0) {
|
||||||
|
traceRoots = te->map;
|
||||||
|
te->watch = true;
|
||||||
|
} else {
|
||||||
|
for (SubroutineTrace* t = te->subroutineTrace; t; t = t->next) {
|
||||||
|
if (t->path == subroutinePath) {
|
||||||
|
traceRoots = t->map;
|
||||||
|
t->watch = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (traceRoots) {
|
||||||
|
if (DebugFrameMaps) {
|
||||||
|
fprintf(stderr, " use roots at ip %3d: ", te->ip);
|
||||||
|
printSet(*traceRoots, mapSize);
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
for (unsigned wi = 0; wi < mapSize; ++wi) {
|
||||||
|
roots[wi] &= traceRoots[wi];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (DebugFrameMaps) {
|
||||||
|
fprintf(stderr, " skip roots at ip %3d\n", te->ip);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (DebugFrameMaps) {
|
||||||
|
fprintf(stderr, "result roots : ");
|
||||||
|
printSet(*roots, mapSize);
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
unsigned
|
unsigned
|
||||||
calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
||||||
unsigned eventIndex, SubroutinePath* subroutinePath = 0)
|
unsigned eventIndex, SubroutinePath* subroutinePath = 0)
|
||||||
@ -4998,7 +5066,7 @@ calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (DebugFrameMaps) {
|
if (DebugFrameMaps) {
|
||||||
fprintf(stderr, "table roots at ip %3d: ", ip);
|
fprintf(stderr, " table roots at ip %3d: ", ip);
|
||||||
printSet(*tableRoots, mapSize);
|
printSet(*tableRoots, mapSize);
|
||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
}
|
}
|
||||||
@ -5022,27 +5090,27 @@ calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
|||||||
} break;
|
} break;
|
||||||
|
|
||||||
case PushExceptionHandlerEvent: {
|
case PushExceptionHandlerEvent: {
|
||||||
unsigned reference = context->eventLog.get2(eventIndex);
|
unsigned start = context->eventLog.get2(eventIndex);
|
||||||
|
eventIndex += 2;
|
||||||
|
unsigned end = context->eventLog.get2(eventIndex);
|
||||||
eventIndex += 2;
|
eventIndex += 2;
|
||||||
|
|
||||||
if (context->subroutineTable and context->subroutineTable[reference]) {
|
if (context->subroutineTable and context->subroutineTable[start]) {
|
||||||
Subroutine* s = context->subroutineTable[reference];
|
Subroutine* s = context->subroutineTable[start];
|
||||||
unsigned originalEventIndex = eventIndex;
|
unsigned originalEventIndex = eventIndex;
|
||||||
|
|
||||||
for (SubroutineCall* c = s->calls; c; c = c->next) {
|
for (SubroutineCall* c = s->calls; c; c = c->next) {
|
||||||
for (SubroutinePath* p = c->paths; p; p = p->listNext) {
|
for (SubroutinePath* p = c->paths; p; p = p->listNext) {
|
||||||
memcpy(RUNTIME_ARRAY_BODY(roots),
|
calculateTryCatchRoots
|
||||||
p->rootTable + (reference * mapSize),
|
(context, p, RUNTIME_ARRAY_BODY(roots), mapSize, start, end);
|
||||||
mapSize * BytesPerWord);
|
|
||||||
|
|
||||||
eventIndex = calculateFrameMaps
|
eventIndex = calculateFrameMaps
|
||||||
(t, context, RUNTIME_ARRAY_BODY(roots), originalEventIndex, p);
|
(t, context, RUNTIME_ARRAY_BODY(roots), originalEventIndex, p);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
memcpy(RUNTIME_ARRAY_BODY(roots),
|
calculateTryCatchRoots
|
||||||
context->rootTable + (reference * mapSize),
|
(context, 0, RUNTIME_ARRAY_BODY(roots), mapSize, start, end);
|
||||||
mapSize * BytesPerWord);
|
|
||||||
|
|
||||||
eventIndex = calculateFrameMaps
|
eventIndex = calculateFrameMaps
|
||||||
(t, context, RUNTIME_ARRAY_BODY(roots), eventIndex, 0);
|
(t, context, RUNTIME_ARRAY_BODY(roots), eventIndex, 0);
|
||||||
@ -5052,7 +5120,7 @@ calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
|||||||
case TraceEvent: {
|
case TraceEvent: {
|
||||||
TraceElement* te; context->eventLog.get(eventIndex, &te, BytesPerWord);
|
TraceElement* te; context->eventLog.get(eventIndex, &te, BytesPerWord);
|
||||||
if (DebugFrameMaps) {
|
if (DebugFrameMaps) {
|
||||||
fprintf(stderr, "trace roots at ip %3d: ", ip);
|
fprintf(stderr, " trace roots at ip %3d: ", ip);
|
||||||
printSet(*RUNTIME_ARRAY_BODY(roots), mapSize);
|
printSet(*RUNTIME_ARRAY_BODY(roots), mapSize);
|
||||||
if (subroutinePath) {
|
if (subroutinePath) {
|
||||||
fprintf(stderr, " ");
|
fprintf(stderr, " ");
|
||||||
@ -5061,13 +5129,17 @@ calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
|||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uintptr_t* map;
|
||||||
|
bool watch;
|
||||||
if (subroutinePath == 0) {
|
if (subroutinePath == 0) {
|
||||||
memcpy(te->map, RUNTIME_ARRAY_BODY(roots), mapSize * BytesPerWord);
|
map = te->map;
|
||||||
|
watch = te->watch;
|
||||||
} else {
|
} else {
|
||||||
SubroutineTrace* trace = 0;
|
SubroutineTrace* trace = 0;
|
||||||
for (SubroutineTrace* t = te->subroutineTrace; t; t = t->next) {
|
for (SubroutineTrace* t = te->subroutineTrace; t; t = t->next) {
|
||||||
if (t->path == subroutinePath) {
|
if (t->path == subroutinePath) {
|
||||||
trace = t;
|
trace = t;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5075,12 +5147,27 @@ calculateFrameMaps(MyThread* t, Context* context, uintptr_t* originalRoots,
|
|||||||
te->subroutineTrace = trace = new
|
te->subroutineTrace = trace = new
|
||||||
(context->zone.allocate
|
(context->zone.allocate
|
||||||
(sizeof(SubroutineTrace) + (mapSize * BytesPerWord)))
|
(sizeof(SubroutineTrace) + (mapSize * BytesPerWord)))
|
||||||
SubroutineTrace(subroutinePath, te->subroutineTrace);
|
SubroutineTrace(subroutinePath, te->subroutineTrace, mapSize);
|
||||||
|
|
||||||
++ te->subroutineTraceCount;
|
++ te->subroutineTraceCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(trace->map, RUNTIME_ARRAY_BODY(roots), mapSize * BytesPerWord);
|
map = trace->map;
|
||||||
|
watch = trace->watch;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (unsigned wi = 0; wi < mapSize; ++wi) {
|
||||||
|
uintptr_t v = RUNTIME_ARRAY_BODY(roots)[wi];
|
||||||
|
|
||||||
|
if (watch and map[wi] != v) {
|
||||||
|
if (DebugFrameMaps) {
|
||||||
|
fprintf(stderr, "dirty roots due to trace watch!\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
context->dirtyRoots = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
map[wi] = v;
|
||||||
}
|
}
|
||||||
|
|
||||||
eventIndex += BytesPerWord;
|
eventIndex += BytesPerWord;
|
||||||
@ -5184,14 +5271,12 @@ copyFrameMap(int32_t* dst, uintptr_t* src, unsigned mapSizeInBits,
|
|||||||
SubroutinePath* subroutinePath)
|
SubroutinePath* subroutinePath)
|
||||||
{
|
{
|
||||||
if (DebugFrameMaps) {
|
if (DebugFrameMaps) {
|
||||||
fprintf(stderr, " orig roots at ip %p: ", reinterpret_cast<void*>
|
fprintf(stderr, " orig roots at ip %3d: ", p->ip);
|
||||||
(p->address->value()));
|
|
||||||
printSet(src[0], ceiling(mapSizeInBits, BitsPerWord));
|
printSet(src[0], ceiling(mapSizeInBits, BitsPerWord));
|
||||||
print(subroutinePath);
|
print(subroutinePath);
|
||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
|
|
||||||
fprintf(stderr, "final roots at ip %p: ", reinterpret_cast<void*>
|
fprintf(stderr, " final roots at ip %3d: ", p->ip);
|
||||||
(p->address->value()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (unsigned j = 0; j < p->argumentIndex; ++j) {
|
for (unsigned j = 0; j < p->argumentIndex; ++j) {
|
||||||
@ -5698,8 +5783,16 @@ compile(MyThread* t, Allocator* allocator, Context* context)
|
|||||||
codeMaxStack(t, methodCode(t, context->method)));
|
codeMaxStack(t, methodCode(t, context->method)));
|
||||||
Frame frame2(&frame, RUNTIME_ARRAY_BODY(stackMap));
|
Frame frame2(&frame, RUNTIME_ARRAY_BODY(stackMap));
|
||||||
|
|
||||||
|
unsigned end = exceptionHandlerEnd(eh);
|
||||||
|
if (exceptionHandlerIp(eh) >= start
|
||||||
|
and exceptionHandlerIp(eh) < end)
|
||||||
|
{
|
||||||
|
end = exceptionHandlerIp(eh);
|
||||||
|
}
|
||||||
|
|
||||||
context->eventLog.append(PushExceptionHandlerEvent);
|
context->eventLog.append(PushExceptionHandlerEvent);
|
||||||
context->eventLog.append2(start);
|
context->eventLog.append2(start);
|
||||||
|
context->eventLog.append2(end);
|
||||||
|
|
||||||
for (unsigned i = 1;
|
for (unsigned i = 1;
|
||||||
i < codeMaxStack(t, methodCode(t, context->method));
|
i < codeMaxStack(t, methodCode(t, context->method));
|
||||||
@ -7211,9 +7304,9 @@ class MyProcessor: public Processor {
|
|||||||
compile(static_cast<MyThread*>(t),
|
compile(static_cast<MyThread*>(t),
|
||||||
local::codeAllocator(static_cast<MyThread*>(t)), 0,
|
local::codeAllocator(static_cast<MyThread*>(t)), 0,
|
||||||
resolveMethod(t, t->m->loader,
|
resolveMethod(t, t->m->loader,
|
||||||
"java/beans/PropertyChangeSupport",
|
"com/ecovate/nat/logic/Cache",
|
||||||
"firePropertyChange",
|
"findInCache",
|
||||||
"(Ljava/beans/PropertyChangeEvent;)V"));
|
"(Ljava/lang/String;Ljava/lang/String;JZ)Lcom/ecovate/shared/xmlrpc/Resource;"));
|
||||||
trap();
|
trap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
47
test/GC.java
47
test/GC.java
@ -104,6 +104,46 @@ public class GC {
|
|||||||
System.gc();
|
System.gc();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void stackMap7(boolean predicate) {
|
||||||
|
try {
|
||||||
|
if (predicate) {
|
||||||
|
Object a = null;
|
||||||
|
} else {
|
||||||
|
Object a = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
int a = 42;
|
||||||
|
throw new DummyException();
|
||||||
|
} finally {
|
||||||
|
System.gc();
|
||||||
|
}
|
||||||
|
} catch (DummyException e) {
|
||||||
|
e.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void stackMap8(boolean predicate) {
|
||||||
|
try {
|
||||||
|
Object x = new Object();
|
||||||
|
if (predicate) {
|
||||||
|
Object a = null;
|
||||||
|
} else {
|
||||||
|
Object a = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
int a = 42;
|
||||||
|
throw new DummyException();
|
||||||
|
} finally {
|
||||||
|
System.gc();
|
||||||
|
x.toString();
|
||||||
|
}
|
||||||
|
} catch (DummyException e) {
|
||||||
|
e.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
Object[] array = new Object[1024 * 1024];
|
Object[] array = new Object[1024 * 1024];
|
||||||
array[0] = new Object();
|
array[0] = new Object();
|
||||||
@ -139,6 +179,13 @@ public class GC {
|
|||||||
|
|
||||||
stackMap6(true);
|
stackMap6(true);
|
||||||
stackMap6(false);
|
stackMap6(false);
|
||||||
|
|
||||||
|
stackMap7(true);
|
||||||
|
stackMap7(false);
|
||||||
|
|
||||||
|
stackMap8(true);
|
||||||
|
stackMap8(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class DummyException extends RuntimeException { }
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user