improve handling of unusual bytecode in JIT compiler

Bytecode generated by compilers other than javac or ecj (such as
jython's dynamically generated classes) can contain unreachable code
and exception handlers which apply to more than one try/catch scope.
Previously, the VM's JIT compiler did not handle either of these cases
well, hence this commit.
This commit is contained in:
Joel Dice 2011-04-08 18:50:22 -06:00
parent 60db8cc047
commit 36aa0d6792

View File

@ -5883,36 +5883,75 @@ logCompile(MyThread* t, const void* code, unsigned size, const char* class_,
} }
} }
unsigned int
resolveIp(MyThread* t, Context* context, object code, unsigned ip) resolveIpForwards(Context* context, int start, int end)
{ {
if (context->visitTable[ip]) { while (start < end and context->visitTable[start] == 0) {
return ip; ++ start;
}
if (start >= end) {
return -1;
} else { } else {
// Very rarely, we'll encounter an unreachable goto which is return start;
// referred to by the exception handler table or line number }
// table. This can be a problem if we need to determine the
// corresponding machine address for the bytecode instruction
// since we never generated any machine code for it. For now, we
// just follow the goto. The "proper" fix is probably to go ahead
// and generate machine code even though it's unreachable; that
// will protect us against unreachable instructions which aren't
// gotos as well.
switch (codeBody(t, code, ip)) {
case goto_: {
unsigned tmp = ip + 1;
return ip + codeReadInt16(t, code, tmp);
} }
case goto_w: { int
unsigned tmp = ip + 1; resolveIpBackwards(Context* context, int start, int end)
return ip + codeReadInt32(t, code, tmp); {
while (start >= end and context->visitTable[start] == 0) {
-- start;
} }
default: if (start < end) {
return ip; return -1;
} else {
return start;
} }
} }
object
truncateIntArray(Thread* t, object array, unsigned length)
{
expect(t, intArrayLength(t, array) > length);
PROTECT(t, array);
object newArray = makeIntArray(t, length);
memcpy(&intArrayBody(t, newArray, 0), &intArrayBody(t, array, 0),
length * 4);
return newArray;
}
object
truncateArray(Thread* t, object array, unsigned length)
{
expect(t, arrayLength(t, array) > length);
PROTECT(t, array);
object newArray = makeArray(t, length);
memcpy(&arrayBody(t, newArray, 0), &arrayBody(t, array, 0),
length * BytesPerWord);
return newArray;
}
object
truncateLineNumberTable(Thread* t, object table, unsigned length)
{
expect(t, lineNumberTableLength(t, table) > length);
PROTECT(t, table);
object newTable = makeLineNumberTable(t, length);
memcpy(lineNumberTableBody(t, newTable, 0),
lineNumberTableBody(t, table, 0),
length * sizeof(LineNumber));
return newTable;
} }
object object
@ -5934,22 +5973,29 @@ translateExceptionHandlerTable(MyThread* t, Context* context, intptr_t start)
object newTable = makeArray(t, length + 1); object newTable = makeArray(t, length + 1);
PROTECT(t, newTable); PROTECT(t, newTable);
set(t, newTable, ArrayBody, newIndex); unsigned ni = 0;
for (unsigned oi = 0; oi < length; ++ oi) {
for (unsigned i = 0; i < length; ++i) {
ExceptionHandler* oldHandler = exceptionHandlerTableBody ExceptionHandler* oldHandler = exceptionHandlerTableBody
(t, oldTable, i); (t, oldTable, oi);
intArrayBody(t, newIndex, i * 3) int handlerStart = resolveIpForwards
= c->machineIp (context, exceptionHandlerStart(oldHandler),
(resolveIp exceptionHandlerEnd(oldHandler));
(t, context, methodCode(t, context->method),
exceptionHandlerStart(oldHandler)))->value() - start;
intArrayBody(t, newIndex, (i * 3) + 1) if (LIKELY(handlerStart >= 0)) {
= c->machineIp(exceptionHandlerEnd(oldHandler))->value() - start; int handlerEnd = resolveIpBackwards
(context, exceptionHandlerEnd(oldHandler),
exceptionHandlerStart(oldHandler));
intArrayBody(t, newIndex, (i * 3) + 2) assert(t, handlerEnd >= 0);
intArrayBody(t, newIndex, ni * 3)
= c->machineIp(handlerStart)->value() - start;
intArrayBody(t, newIndex, (ni * 3) + 1)
= c->machineIp(handlerEnd)->value() - start;
intArrayBody(t, newIndex, (ni * 3) + 2)
= c->machineIp(exceptionHandlerIp(oldHandler))->value() - start; = c->machineIp(exceptionHandlerIp(oldHandler))->value() - start;
object type; object type;
@ -5960,8 +6006,18 @@ translateExceptionHandlerTable(MyThread* t, Context* context, intptr_t start)
type = 0; type = 0;
} }
set(t, newTable, ArrayBody + ((i + 1) * BytesPerWord), type); set(t, newTable, ArrayBody + ((ni + 1) * BytesPerWord), type);
++ ni;
} }
}
if (UNLIKELY(ni < length)) {
newIndex = truncateIntArray(t, newIndex, ni * 3);
newTable = truncateArray(t, newTable, ni + 1);
}
set(t, newTable, ArrayBody, newIndex);
return newTable; return newTable;
} else { } else {
@ -5978,18 +6034,28 @@ translateLineNumberTable(MyThread* t, Context* context, intptr_t start)
unsigned length = lineNumberTableLength(t, oldTable); unsigned length = lineNumberTableLength(t, oldTable);
object newTable = makeLineNumberTable(t, length); object newTable = makeLineNumberTable(t, length);
for (unsigned i = 0; i < length; ++i) { unsigned ni = 0;
LineNumber* oldLine = lineNumberTableBody(t, oldTable, i); for (unsigned oi = 0; oi < length; ++oi) {
LineNumber* newLine = lineNumberTableBody(t, newTable, i); LineNumber* oldLine = lineNumberTableBody(t, oldTable, oi);
LineNumber* newLine = lineNumberTableBody(t, newTable, ni);
int ip = resolveIpForwards
(context, lineNumberIp(oldLine), oi + 1 < length
? lineNumberIp(lineNumberTableBody(t, oldTable, oi + 1)) - 1
: lineNumberIp(oldLine) + 1);
if (LIKELY(ip >= 0)) {
lineNumberIp(newLine) lineNumberIp(newLine)
= context->compiler->machineIp = context->compiler->machineIp(ip)->value() - start;
(resolveIp
(t, context, methodCode(t, context->method), lineNumberIp(oldLine)))
->value()
- start;
lineNumberLine(newLine) = lineNumberLine(oldLine); lineNumberLine(newLine) = lineNumberLine(oldLine);
++ ni;
}
}
if (UNLIKELY(ni < length)) {
newTable = truncateLineNumberTable(t, newTable, ni);
} }
return newTable; return newTable;
@ -6881,25 +6947,27 @@ compile(MyThread* t, Context* context)
THREAD_RUNTIME_ARRAY(t, bool, visited, visitCount); THREAD_RUNTIME_ARRAY(t, bool, visited, visitCount);
memset(RUNTIME_ARRAY_BODY(visited), 0, visitCount * sizeof(bool)); memset(RUNTIME_ARRAY_BODY(visited), 0, visitCount * sizeof(bool));
while (visitCount) { bool progress = true;
bool progress UNUSED = false; while (progress) {
progress = false;
for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) { for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) {
c->restoreState(state); c->restoreState(state);
ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i); ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i);
unsigned start = resolveIp int start = resolveIpForwards
(t, context, methodCode(t, context->method), (context, exceptionHandlerStart(eh), exceptionHandlerEnd(eh));
exceptionHandlerStart(eh));
if ((not RUNTIME_ARRAY_BODY(visited)[i]) if ((not RUNTIME_ARRAY_BODY(visited)[i])
and start >= 0
and context->visitTable[start]) and context->visitTable[start])
{ {
-- visitCount;
RUNTIME_ARRAY_BODY(visited)[i] = true; RUNTIME_ARRAY_BODY(visited)[i] = true;
progress = true; progress = true;
THREAD_RUNTIME_ARRAY(t, uint8_t, stackMap, if (context->visitTable[exceptionHandlerIp(eh)] == 0) {
THREAD_RUNTIME_ARRAY
(t, uint8_t, stackMap,
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));
@ -6928,8 +6996,7 @@ compile(MyThread* t, Context* context)
eventIndex = calculateFrameMaps(t, context, 0, eventIndex); eventIndex = calculateFrameMaps(t, context, 0, eventIndex);
} }
} }
}
assert(t, progress);
} }
} }