mirror of
https://github.com/corda/corda.git
synced 2025-01-22 12:28:11 +00:00
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:
parent
60db8cc047
commit
36aa0d6792
251
src/compile.cpp
251
src/compile.cpp
@ -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;
|
||||||
} else {
|
|
||||||
// Very rarely, we'll encounter an unreachable goto which is
|
|
||||||
// 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: {
|
|
||||||
unsigned tmp = ip + 1;
|
|
||||||
return ip + codeReadInt32(t, code, tmp);
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
return ip;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (start >= end) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
return start;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
resolveIpBackwards(Context* context, int start, int end)
|
||||||
|
{
|
||||||
|
while (start >= end and context->visitTable[start] == 0) {
|
||||||
|
-- start;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (start < end) {
|
||||||
|
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,35 +5973,52 @@ 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);
|
||||||
= c->machineIp(exceptionHandlerIp(oldHandler))->value() - start;
|
|
||||||
|
|
||||||
object type;
|
intArrayBody(t, newIndex, ni * 3)
|
||||||
if (exceptionHandlerCatchType(oldHandler)) {
|
= c->machineIp(handlerStart)->value() - start;
|
||||||
type = resolveClassInPool
|
|
||||||
(t, context->method, exceptionHandlerCatchType(oldHandler) - 1);
|
intArrayBody(t, newIndex, (ni * 3) + 1)
|
||||||
} else {
|
= c->machineIp(handlerEnd)->value() - start;
|
||||||
type = 0;
|
|
||||||
|
intArrayBody(t, newIndex, (ni * 3) + 2)
|
||||||
|
= c->machineIp(exceptionHandlerIp(oldHandler))->value() - start;
|
||||||
|
|
||||||
|
object type;
|
||||||
|
if (exceptionHandlerCatchType(oldHandler)) {
|
||||||
|
type = resolveClassInPool
|
||||||
|
(t, context->method, exceptionHandlerCatchType(oldHandler) - 1);
|
||||||
|
} else {
|
||||||
|
type = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
set(t, newTable, ArrayBody + ((ni + 1) * BytesPerWord), type);
|
||||||
|
|
||||||
|
++ ni;
|
||||||
}
|
}
|
||||||
|
|
||||||
set(t, newTable, ArrayBody + ((i + 1) * BytesPerWord), type);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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 {
|
||||||
return 0;
|
return 0;
|
||||||
@ -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);
|
||||||
|
|
||||||
lineNumberIp(newLine)
|
int ip = resolveIpForwards
|
||||||
= context->compiler->machineIp
|
(context, lineNumberIp(oldLine), oi + 1 < length
|
||||||
(resolveIp
|
? lineNumberIp(lineNumberTableBody(t, oldTable, oi + 1)) - 1
|
||||||
(t, context, methodCode(t, context->method), lineNumberIp(oldLine)))
|
: lineNumberIp(oldLine) + 1);
|
||||||
->value()
|
|
||||||
- start;
|
|
||||||
|
|
||||||
lineNumberLine(newLine) = lineNumberLine(oldLine);
|
if (LIKELY(ip >= 0)) {
|
||||||
|
lineNumberIp(newLine)
|
||||||
|
= context->compiler->machineIp(ip)->value() - start;
|
||||||
|
|
||||||
|
lineNumberLine(newLine) = lineNumberLine(oldLine);
|
||||||
|
|
||||||
|
++ ni;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (UNLIKELY(ni < length)) {
|
||||||
|
newTable = truncateLineNumberTable(t, newTable, ni);
|
||||||
}
|
}
|
||||||
|
|
||||||
return newTable;
|
return newTable;
|
||||||
@ -6881,55 +6947,56 @@ 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) {
|
||||||
codeMaxStack(t, methodCode(t, context->method)));
|
THREAD_RUNTIME_ARRAY
|
||||||
Frame frame2(&frame, RUNTIME_ARRAY_BODY(stackMap));
|
(t, uint8_t, stackMap,
|
||||||
|
codeMaxStack(t, methodCode(t, context->method)));
|
||||||
|
Frame frame2(&frame, RUNTIME_ARRAY_BODY(stackMap));
|
||||||
|
|
||||||
unsigned end = exceptionHandlerEnd(eh);
|
unsigned end = exceptionHandlerEnd(eh);
|
||||||
if (exceptionHandlerIp(eh) >= start
|
if (exceptionHandlerIp(eh) >= start
|
||||||
and exceptionHandlerIp(eh) < end)
|
and exceptionHandlerIp(eh) < end)
|
||||||
{
|
{
|
||||||
end = exceptionHandlerIp(eh);
|
end = exceptionHandlerIp(eh);
|
||||||
|
}
|
||||||
|
|
||||||
|
context->eventLog.append(PushExceptionHandlerEvent);
|
||||||
|
context->eventLog.append2(start);
|
||||||
|
context->eventLog.append2(end);
|
||||||
|
|
||||||
|
for (unsigned i = 1;
|
||||||
|
i < codeMaxStack(t, methodCode(t, context->method));
|
||||||
|
++i)
|
||||||
|
{
|
||||||
|
frame2.set(localSize(t, context->method) + i, Frame::Integer);
|
||||||
|
}
|
||||||
|
|
||||||
|
compile(t, &frame2, exceptionHandlerIp(eh), start);
|
||||||
|
|
||||||
|
context->eventLog.append(PopContextEvent);
|
||||||
|
|
||||||
|
eventIndex = calculateFrameMaps(t, context, 0, eventIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
context->eventLog.append(PushExceptionHandlerEvent);
|
|
||||||
context->eventLog.append2(start);
|
|
||||||
context->eventLog.append2(end);
|
|
||||||
|
|
||||||
for (unsigned i = 1;
|
|
||||||
i < codeMaxStack(t, methodCode(t, context->method));
|
|
||||||
++i)
|
|
||||||
{
|
|
||||||
frame2.set(localSize(t, context->method) + i, Frame::Integer);
|
|
||||||
}
|
|
||||||
|
|
||||||
compile(t, &frame2, exceptionHandlerIp(eh), start);
|
|
||||||
|
|
||||||
context->eventLog.append(PopContextEvent);
|
|
||||||
|
|
||||||
eventIndex = calculateFrameMaps(t, context, 0, eventIndex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
assert(t, progress);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user