fix custom-classloader-related concurrency problems and other bugs

The main changes in this commit ensure that we don't hold the global
class lock when doing class resolution using application-defined
classloaders.  Such classloaders may do their own locking (in fact,
it's almost certain), making deadlock likely when mixed with VM-level
locking in various orders.

Other changes include a fix to avoid overflow when waiting for
extremely long intervals and a GC root stack mapping bug.
This commit is contained in:
Joel Dice 2010-09-16 19:43:27 -06:00
parent f485016637
commit d0d53e2e10
12 changed files with 337 additions and 232 deletions

View File

@ -15,7 +15,7 @@ import java.util.WeakHashMap;
public class Thread implements Runnable { public class Thread implements Runnable {
private long peer; private long peer;
private boolean interrupted; private volatile boolean interrupted;
private boolean daemon; private boolean daemon;
private byte state; private byte state;
private byte priority; private byte priority;
@ -141,25 +141,9 @@ public class Thread implements Runnable {
public static native Thread currentThread(); public static native Thread currentThread();
private static native void interrupt(long peer); public native void interrupt();
public synchronized void interrupt() { public native boolean interrupted();
if (peer != 0) {
interrupt(peer);
} else {
interrupted = true;
}
}
public static boolean interrupted() {
Thread t = currentThread();
synchronized (t) {
boolean v = t.interrupted;
t.interrupted = false;
return v;
}
}
public static boolean isInterrupted() { public static boolean isInterrupted() {
return currentThread().interrupted; return currentThread().interrupted;

View File

@ -340,10 +340,6 @@ vm-sources = \
vm-asm-sources = $(src)/$(asm).S vm-asm-sources = $(src)/$(asm).S
ifeq ($(process),compile) ifeq ($(process),compile)
vm-depends += \
$(src)/compiler.h \
$(src)/vector.h
vm-sources += \ vm-sources += \
$(src)/compiler.cpp \ $(src)/compiler.cpp \
$(src)/$(asm).cpp $(src)/$(asm).cpp

View File

@ -84,7 +84,12 @@ Avian_avian_SystemClassLoader_resourceExists
if (LIKELY(name)) { if (LIKELY(name)) {
RUNTIME_ARRAY(char, n, stringLength(t, name) + 1); RUNTIME_ARRAY(char, n, stringLength(t, name) + 1);
stringChars(t, name, RUNTIME_ARRAY_BODY(n)); stringChars(t, name, RUNTIME_ARRAY_BODY(n));
return getFinder(t, loader)->exists(RUNTIME_ARRAY_BODY(n));
bool r = getFinder(t, loader)->exists(RUNTIME_ARRAY_BODY(n));
fprintf(stderr, "resource %s exists? %d\n", n, r);
return r;
} else { } else {
t->exception = t->m->classpath->makeThrowable t->exception = t->m->classpath->makeThrowable
(t, Machine::NullPointerExceptionType); (t, Machine::NullPointerExceptionType);
@ -144,7 +149,11 @@ Avian_avian_resource_Handler_00024ResourceInputStream_getContentLength
RUNTIME_ARRAY(char, p, stringLength(t, path) + 1); RUNTIME_ARRAY(char, p, stringLength(t, path) + 1);
stringChars(t, path, RUNTIME_ARRAY_BODY(p)); stringChars(t, path, RUNTIME_ARRAY_BODY(p));
System::Region* r = t->m->appFinder->find(RUNTIME_ARRAY_BODY(p)); System::Region* r = t->m->bootFinder->find(RUNTIME_ARRAY_BODY(p));
if (r == 0) {
r = t->m->appFinder->find(RUNTIME_ARRAY_BODY(p));
}
if (r) { if (r) {
jint rSize = r->length(); jint rSize = r->length();
r->dispose(); r->dispose();
@ -164,8 +173,12 @@ Avian_avian_resource_Handler_00024ResourceInputStream_open
RUNTIME_ARRAY(char, p, stringLength(t, path) + 1); RUNTIME_ARRAY(char, p, stringLength(t, path) + 1);
stringChars(t, path, RUNTIME_ARRAY_BODY(p)); stringChars(t, path, RUNTIME_ARRAY_BODY(p));
return reinterpret_cast<int64_t> System::Region* r = t->m->bootFinder->find(RUNTIME_ARRAY_BODY(p));
(t->m->appFinder->find(RUNTIME_ARRAY_BODY(p))); if (r == 0) {
r = t->m->appFinder->find(RUNTIME_ARRAY_BODY(p));
}
return reinterpret_cast<int64_t>(r);
} else { } else {
t->exception = t->m->classpath->makeThrowable t->exception = t->m->classpath->makeThrowable
(t, Machine::NullPointerExceptionType); (t, Machine::NullPointerExceptionType);

View File

@ -50,7 +50,7 @@ class MyClasspath : public Classpath {
return vm::makeThread return vm::makeThread
(t, 0, 0, 0, NewState, NormalPriority, 0, 0, 0, (t, 0, 0, 0, NewState, NormalPriority, 0, 0, 0,
root(t, Machine::BootLoader), 0, 0, group); root(t, Machine::BootLoader), 0, 0, group, 0);
} }
virtual void virtual void
@ -70,6 +70,7 @@ class MyClasspath : public Classpath {
(Thread* t, Machine::Type type, object message, object trace, object cause) (Thread* t, Machine::Type type, object message, object trace, object cause)
{ {
PROTECT(t, message); PROTECT(t, message);
PROTECT(t, trace);
PROTECT(t, cause); PROTECT(t, cause);
if (trace == 0) { if (trace == 0) {

View File

@ -186,7 +186,7 @@ class MyClasspath : public Classpath {
return vm::makeThread return vm::makeThread
(t, 0, NormalPriority, 0, 0, false, false, false, 0, group, (t, 0, NormalPriority, 0, 0, false, false, false, 0, group,
root(t, Machine::BootLoader), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, false, 0, 0, root(t, Machine::BootLoader), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, false, 0, 0,
0, false); 0, 0, false);
} }
virtual void virtual void
@ -210,6 +210,7 @@ class MyClasspath : public Classpath {
(Thread* t, Machine::Type type, object message, object trace, object cause) (Thread* t, Machine::Type type, object message, object trace, object cause)
{ {
PROTECT(t, message); PROTECT(t, message);
PROTECT(t, trace);
PROTECT(t, cause); PROTECT(t, cause);
if (trace == 0) { if (trace == 0) {
@ -534,6 +535,23 @@ setProperty(Thread* t, object method, object properties,
t->m->processor->invoke(t, method, properties, n, v); t->m->processor->invoke(t, method, properties, n, v);
} }
object
interruptLock(Thread* t, object thread)
{
if (threadInterruptLock(t, thread) == 0) {
PROTECT(t, thread);
ACQUIRE(t, t->m->referenceLock);
if (threadInterruptLock(t, thread) == 0) {
object head = makeMonitorNode(t, 0, 0);
object lock = makeMonitor(t, 0, 0, 0, head, head, 0);
set(t, thread, ThreadInterruptLock, lock);
}
}
return threadInterruptLock(t, thread);
}
} // namespace local } // namespace local
} // namespace } // namespace
@ -1255,10 +1273,14 @@ JVM_Interrupt(Thread* t, jobject thread)
{ {
ENTER(t, Thread::ActiveState); ENTER(t, Thread::ActiveState);
monitorAcquire(t, local::interruptLock(t, *thread));
Thread* p = reinterpret_cast<Thread*>(threadPeer(t, *thread)); Thread* p = reinterpret_cast<Thread*>(threadPeer(t, *thread));
if (p) { if (p) {
interrupt(t, p); interrupt(t, p);
} else {
threadInterrupted(t, *thread) = true;
} }
monitorRelease(t, local::interruptLock(t, *thread));
} }
extern "C" JNIEXPORT jboolean JNICALL extern "C" JNIEXPORT jboolean JNICALL
@ -1266,12 +1288,12 @@ JVM_IsInterrupted(Thread* t, jobject thread, jboolean clear)
{ {
ENTER(t, Thread::ActiveState); ENTER(t, Thread::ActiveState);
acquire(t, *thread); monitorAcquire(t, local::interruptLock(t, *thread));
bool v = threadInterrupted(t, *thread); bool v = threadInterrupted(t, *thread);
if (clear) { if (clear) {
threadInterrupted(t, *thread) = false; threadInterrupted(t, *thread) = false;
} }
release(t, *thread); monitorRelease(t, local::interruptLock(t, *thread));
return v; return v;
} }
@ -2414,7 +2436,10 @@ extern "C" JNIEXPORT struct hostent* JNICALL
JVM_GetHostByName(char*) { abort(); } JVM_GetHostByName(char*) { abort(); }
extern "C" JNIEXPORT int JNICALL extern "C" JNIEXPORT int JNICALL
JVM_GetHostName(char*, int) { abort(); } JVM_GetHostName(char* name, int length)
{
return gethostname(name, length);
}
extern "C" JNIEXPORT void* JNICALL extern "C" JNIEXPORT void* JNICALL
JVM_RawMonitorCreate(void) JVM_RawMonitorCreate(void)

View File

@ -327,6 +327,12 @@ compiledSize(intptr_t address)
return reinterpret_cast<uintptr_t*>(address)[-1]; return reinterpret_cast<uintptr_t*>(address)[-1];
} }
intptr_t
methodCompiled(Thread* t, object method)
{
return codeCompiled(t, methodCode(t, method));
}
intptr_t intptr_t
compareIpToMethodBounds(Thread* t, intptr_t ip, object method) compareIpToMethodBounds(Thread* t, intptr_t ip, object method)
{ {
@ -2619,7 +2625,7 @@ throw_(MyThread* t, object o)
(t, Machine::NullPointerExceptionType); (t, Machine::NullPointerExceptionType);
} }
// printTrace(t, t->exception); printTrace(t, t->exception);
unwind(t); unwind(t);
} }
@ -2659,6 +2665,8 @@ makeNew64(Thread* t, object class_)
void void
gcIfNecessary(MyThread* t) gcIfNecessary(MyThread* t)
{ {
stress(t);
if (UNLIKELY(t->flags & Thread::UseBackupHeapFlag)) { if (UNLIKELY(t->flags & Thread::UseBackupHeapFlag)) {
collect(t, Heap::MinorCollection); collect(t, Heap::MinorCollection);
} }
@ -3219,6 +3227,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
if (inTryBlock(t, code, ip - 1)) { if (inTryBlock(t, code, ip - 1)) {
c->saveLocals(); c->saveLocals();
frame->trace(0, 0);
} }
if (CheckArrayBounds) { if (CheckArrayBounds) {
@ -3311,6 +3320,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
if (inTryBlock(t, code, ip - 1)) { if (inTryBlock(t, code, ip - 1)) {
c->saveLocals(); c->saveLocals();
frame->trace(0, 0);
} }
if (CheckArrayBounds) { if (CheckArrayBounds) {
@ -3741,6 +3751,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
if (inTryBlock(t, code, ip - 3)) { if (inTryBlock(t, code, ip - 3)) {
c->saveLocals(); c->saveLocals();
frame->trace(0, 0);
} }
} }
@ -4709,6 +4720,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip,
if (inTryBlock(t, code, ip - 3)) { if (inTryBlock(t, code, ip - 3)) {
c->saveLocals(); c->saveLocals();
frame->trace(0, 0);
} }
} }
@ -5006,7 +5018,7 @@ logCompile(MyThread* t, const void* code, unsigned size, const char* class_,
} }
} }
void object
translateExceptionHandlerTable(MyThread* t, Compiler* c, object method, translateExceptionHandlerTable(MyThread* t, Compiler* c, object method,
intptr_t start) intptr_t start)
{ {
@ -5043,7 +5055,7 @@ translateExceptionHandlerTable(MyThread* t, Compiler* c, object method,
if (exceptionHandlerCatchType(oldHandler)) { if (exceptionHandlerCatchType(oldHandler)) {
type = resolveClassInPool type = resolveClassInPool
(t, method, exceptionHandlerCatchType(oldHandler) - 1); (t, method, exceptionHandlerCatchType(oldHandler) - 1);
if (UNLIKELY(t->exception)) return; if (UNLIKELY(t->exception)) return 0;
} else { } else {
type = 0; type = 0;
} }
@ -5051,11 +5063,13 @@ translateExceptionHandlerTable(MyThread* t, Compiler* c, object method,
set(t, newTable, ArrayBody + ((i + 1) * BytesPerWord), type); set(t, newTable, ArrayBody + ((i + 1) * BytesPerWord), type);
} }
set(t, methodCode(t, method), CodeExceptionHandlerTable, newTable); return newTable;
} else {
return 0;
} }
} }
void object
translateLineNumberTable(MyThread* t, Compiler* c, object code, intptr_t start) translateLineNumberTable(MyThread* t, Compiler* c, object code, intptr_t start)
{ {
object oldTable = codeLineNumberTable(t, code); object oldTable = codeLineNumberTable(t, code);
@ -5075,7 +5089,9 @@ translateLineNumberTable(MyThread* t, Compiler* c, object code, intptr_t start)
lineNumberLine(newLine) = lineNumberLine(oldLine); lineNumberLine(newLine) = lineNumberLine(oldLine);
} }
set(t, code, CodeLineNumberTable, newTable); return newTable;
} else {
return 0;
} }
} }
@ -5527,23 +5543,23 @@ compareSubroutineTracePointers(const void* va, const void* vb)
object object
makeGeneralFrameMapTable(MyThread* t, Context* context, uint8_t* start, makeGeneralFrameMapTable(MyThread* t, Context* context, uint8_t* start,
TraceElement** elements, unsigned pathFootprint, TraceElement** elements, unsigned elementCount,
unsigned mapCount) unsigned pathFootprint, unsigned mapCount)
{ {
unsigned mapSize = frameMapSizeInBits(t, context->method); unsigned mapSize = frameMapSizeInBits(t, context->method);
unsigned indexOffset = sizeof(FrameMapTableHeader); unsigned indexOffset = sizeof(FrameMapTableHeader);
unsigned mapsOffset = indexOffset unsigned mapsOffset = indexOffset
+ (context->traceLogCount * sizeof(FrameMapTableIndexElement)); + (elementCount * sizeof(FrameMapTableIndexElement));
unsigned pathsOffset = mapsOffset + (ceiling(mapCount * mapSize, 32) * 4); unsigned pathsOffset = mapsOffset + (ceiling(mapCount * mapSize, 32) * 4);
object table = makeByteArray(t, pathsOffset + pathFootprint); object table = makeByteArray(t, pathsOffset + pathFootprint);
int8_t* body = &byteArrayBody(t, table, 0); int8_t* body = &byteArrayBody(t, table, 0);
new (body) FrameMapTableHeader(context->traceLogCount); new (body) FrameMapTableHeader(elementCount);
unsigned nextTableIndex = pathsOffset; unsigned nextTableIndex = pathsOffset;
unsigned nextMapIndex = 0; unsigned nextMapIndex = 0;
for (unsigned i = 0; i < context->traceLogCount; ++i) { for (unsigned i = 0; i < elementCount; ++i) {
TraceElement* p = elements[i]; TraceElement* p = elements[i];
unsigned mapBase = nextMapIndex; unsigned mapBase = nextMapIndex;
@ -5645,27 +5661,26 @@ makeGeneralFrameMapTable(MyThread* t, Context* context, uint8_t* start,
object object
makeSimpleFrameMapTable(MyThread* t, Context* context, uint8_t* start, makeSimpleFrameMapTable(MyThread* t, Context* context, uint8_t* start,
TraceElement** elements) TraceElement** elements, unsigned elementCount)
{ {
unsigned mapSize = frameMapSizeInBits(t, context->method); unsigned mapSize = frameMapSizeInBits(t, context->method);
object table = makeIntArray object table = makeIntArray
(t, context->traceLogCount (t, elementCount + ceiling(elementCount * mapSize, 32));
+ ceiling(context->traceLogCount * mapSize, 32));
assert(t, intArrayLength(t, table) == context->traceLogCount assert(t, intArrayLength(t, table) == elementCount
+ simpleFrameMapTableSize(t, context->method, table)); + simpleFrameMapTableSize(t, context->method, table));
for (unsigned i = 0; i < context->traceLogCount; ++i) { for (unsigned i = 0; i < elementCount; ++i) {
TraceElement* p = elements[i]; TraceElement* p = elements[i];
intArrayBody(t, table, i) = static_cast<intptr_t>(p->address->value()) intArrayBody(t, table, i) = static_cast<intptr_t>(p->address->value())
- reinterpret_cast<intptr_t>(start); - reinterpret_cast<intptr_t>(start);
assert(t, context->traceLogCount + ceiling((i + 1) * mapSize, 32) assert(t, elementCount + ceiling((i + 1) * mapSize, 32)
<= intArrayLength(t, table)); <= intArrayLength(t, table));
if (mapSize) { if (mapSize) {
copyFrameMap(&intArrayBody(t, table, context->traceLogCount), p->map, copyFrameMap(&intArrayBody(t, table, elementCount), p->map,
mapSize, i * mapSize, p, 0); mapSize, i * mapSize, p, 0);
} }
} }
@ -5673,7 +5688,7 @@ makeSimpleFrameMapTable(MyThread* t, Context* context, uint8_t* start,
return table; return table;
} }
uint8_t* void
finish(MyThread* t, Allocator* allocator, Context* context) finish(MyThread* t, Allocator* allocator, Context* context)
{ {
Compiler* c = context->compiler; Compiler* c = context->compiler;
@ -5703,7 +5718,13 @@ finish(MyThread* t, Allocator* allocator, Context* context)
trap(); trap();
} }
// todo: this is a CPU-intensive operation, so consider doing it
// earlier before we've acquired the global class lock to improve
// parallelism (the downside being that it may end up being a waste
// of cycles if another thread compiles the same method in parallel,
// which might be mitigated by fine-grained, per-method locking):
unsigned codeSize = c->compile(); unsigned codeSize = c->compile();
uintptr_t* code = static_cast<uintptr_t*> uintptr_t* code = static_cast<uintptr_t*>
(allocator->allocate(pad(codeSize) + pad(c->poolSize()) + BytesPerWord)); (allocator->allocate(pad(codeSize) + pad(c->poolSize()) + BytesPerWord));
code[0] = codeSize; code[0] = codeSize;
@ -5744,21 +5765,20 @@ finish(MyThread* t, Allocator* allocator, Context* context)
} }
} }
translateExceptionHandlerTable object newExceptionHandlerTable = translateExceptionHandlerTable
(t, c, context->method, reinterpret_cast<intptr_t>(start)); (t, c, context->method, reinterpret_cast<intptr_t>(start));
if (UNLIKELY(t->exception)) return 0; if (UNLIKELY(t->exception)) return;
PROTECT(t, newExceptionHandlerTable);
translateLineNumberTable(t, c, methodCode(t, context->method), object newLineNumberTable = translateLineNumberTable
reinterpret_cast<intptr_t>(start)); (t, c, methodCode(t, context->method), reinterpret_cast<intptr_t>(start));
{ object code = methodCode(t, context->method); { object code = methodCode(t, context->method);
code = makeCode(t, 0, code = makeCode
codeExceptionHandlerTable(t, code), (t, 0, newExceptionHandlerTable, newLineNumberTable,
codeLineNumberTable(t, code), reinterpret_cast<uintptr_t>(start), codeMaxStack(t, code),
codeMaxStack(t, code), codeMaxLocals(t, code), 0);
codeMaxLocals(t, code),
0);
set(t, context->method, MethodCode, code); set(t, context->method, MethodCode, code);
} }
@ -5771,44 +5791,46 @@ finish(MyThread* t, Allocator* allocator, Context* context)
for (TraceElement* p = context->traceLog; p; p = p->next) { for (TraceElement* p = context->traceLog; p; p = p->next) {
assert(t, index < context->traceLogCount); assert(t, index < context->traceLogCount);
SubroutineTrace* trace = p->subroutineTrace; if (p->address) {
unsigned myMapCount = 1; SubroutineTrace* trace = p->subroutineTrace;
if (trace) { unsigned myMapCount = 1;
for (Subroutine* s = trace->path->call->subroutine; if (trace) {
s; s = s->stackNext) for (Subroutine* s = trace->path->call->subroutine;
{ s; s = s->stackNext)
unsigned callCount = s->callCount; {
myMapCount *= callCount; unsigned callCount = s->callCount;
if (not s->visited) { myMapCount *= callCount;
s->visited = true; if (not s->visited) {
pathFootprint += sizeof(FrameMapTablePath) s->visited = true;
+ (sizeof(int32_t) * callCount); pathFootprint += sizeof(FrameMapTablePath)
+ (sizeof(int32_t) * callCount);
}
} }
} }
}
mapCount += myMapCount; mapCount += myMapCount;
RUNTIME_ARRAY_BODY(elements)[index++] = p; RUNTIME_ARRAY_BODY(elements)[index++] = p;
if (p->target) { if (p->target) {
insertCallNode insertCallNode
(t, makeCallNode (t, makeCallNode
(t, p->address->value(), p->target, p->flags, 0)); (t, p->address->value(), p->target, p->flags, 0));
}
} }
} }
qsort(RUNTIME_ARRAY_BODY(elements), context->traceLogCount, qsort(RUNTIME_ARRAY_BODY(elements), index,
sizeof(TraceElement*), compareTraceElementPointers); sizeof(TraceElement*), compareTraceElementPointers);
object map; object map;
if (pathFootprint) { if (pathFootprint) {
map = makeGeneralFrameMapTable map = makeGeneralFrameMapTable
(t, context, start, RUNTIME_ARRAY_BODY(elements), pathFootprint, (t, context, start, RUNTIME_ARRAY_BODY(elements), index, pathFootprint,
mapCount); mapCount);
} else { } else {
map = makeSimpleFrameMapTable map = makeSimpleFrameMapTable
(t, context, start, RUNTIME_ARRAY_BODY(elements)); (t, context, start, RUNTIME_ARRAY_BODY(elements), index);
} }
set(t, methodCode(t, context->method), CodePool, map); set(t, methodCode(t, context->method), CodePool, map);
@ -5838,12 +5860,10 @@ finish(MyThread* t, Allocator* allocator, Context* context)
} }
syncInstructionCache(start, codeSize); syncInstructionCache(start, codeSize);
return start;
} }
uint8_t* void
compile(MyThread* t, Allocator* allocator, Context* context) compile(MyThread* t, Context* context)
{ {
Compiler* c = context->compiler; Compiler* c = context->compiler;
@ -5908,7 +5928,7 @@ compile(MyThread* t, Allocator* allocator, Context* context)
Compiler::State* state = c->saveState(); Compiler::State* state = c->saveState();
compile(t, &frame, 0); compile(t, &frame, 0);
if (UNLIKELY(t->exception)) return 0; if (UNLIKELY(t->exception)) return;
context->dirtyRoots = false; context->dirtyRoots = false;
unsigned eventIndex = calculateFrameMaps(t, context, 0, 0); unsigned eventIndex = calculateFrameMaps(t, context, 0, 0);
@ -5961,7 +5981,7 @@ compile(MyThread* t, Allocator* allocator, Context* context)
} }
compile(t, &frame2, exceptionHandlerIp(eh), start); compile(t, &frame2, exceptionHandlerIp(eh), start);
if (UNLIKELY(t->exception)) return 0; if (UNLIKELY(t->exception)) return;
context->eventLog.append(PopContextEvent); context->eventLog.append(PopContextEvent);
@ -5977,8 +5997,6 @@ compile(MyThread* t, Allocator* allocator, Context* context)
context->dirtyRoots = false; context->dirtyRoots = false;
calculateFrameMaps(t, context, 0, 0); calculateFrameMaps(t, context, 0, 0);
} }
return finish(t, allocator, context);
} }
void void
@ -7169,7 +7187,7 @@ class SegFaultHandler: public System::SignalHandler {
t->exception = root(t, Machine::NullPointerException); t->exception = root(t, Machine::NullPointerException);
} }
// printTrace(t, t->exception); printTrace(t, t->exception);
object continuation; object continuation;
findUnwindTarget(t, ip, base, stack, &continuation); findUnwindTarget(t, ip, base, stack, &continuation);
@ -7283,10 +7301,13 @@ class MyProcessor: public Processor {
object class_, object class_,
object code) object code)
{ {
if (code) {
codeCompiled(t, code) = local::defaultThunk(static_cast<MyThread*>(t));
}
return vm::makeMethod return vm::makeMethod
(t, vmFlags, returnCode, parameterCount, parameterFootprint, flags, (t, vmFlags, returnCode, parameterCount, parameterFootprint, flags,
offset, 0, name, spec, addendum, class_, code, offset, 0, name, spec, addendum, class_, code);
local::defaultThunk(static_cast<MyThread*>(t)));
} }
virtual object virtual object
@ -8083,7 +8104,7 @@ fixupMethods(Thread* t, object map, BootImage* image, uint8_t* code)
assert(t, (methodCompiled(t, method) - image->codeBase) assert(t, (methodCompiled(t, method) - image->codeBase)
<= image->codeSize); <= image->codeSize);
methodCompiled(t, method) codeCompiled(t, methodCode(t, method))
= (methodCompiled(t, method) - image->codeBase) = (methodCompiled(t, method) - image->codeBase)
+ reinterpret_cast<uintptr_t>(code); + reinterpret_cast<uintptr_t>(code);
@ -8597,65 +8618,86 @@ compile(MyThread* t, Allocator* allocator, BootContext* bootContext,
if (UNLIKELY(t->exception)) return; if (UNLIKELY(t->exception)) return;
} }
if (methodAddress(t, method) == defaultThunk(t)) { if (methodAddress(t, method) != defaultThunk(t)) {
ACQUIRE(t, t->m->classLock); return;
if (methodAddress(t, method) == defaultThunk(t)) {
assert(t, (methodFlags(t, method) & ACC_NATIVE) == 0);
Context context(t, bootContext, method);
uint8_t* compiled = compile(t, allocator, &context);
if (UNLIKELY(t->exception)) return;
if (DebugMethodTree) {
fprintf(stderr, "insert method at %p\n", compiled);
}
// We can't set the MethodCompiled field on the original method
// before it is placed into the method tree, since another
// thread might call the method, from which stack unwinding
// would fail (since there is not yet an entry in the method
// tree). However, we can't insert the original method into the
// tree before setting the MethodCompiled field on it since we
// rely on that field to determine its position in the tree.
// Therefore, we insert a clone in its place. Later, we'll
// replace the clone with the original to save memory.
object clone = makeMethod
(t, methodVmFlags(t, method),
methodReturnCode(t, method),
methodParameterCount(t, method),
methodParameterFootprint(t, method),
methodFlags(t, method),
methodOffset(t, method),
methodNativeID(t, method),
methodName(t, method),
methodSpec(t, method),
methodAddendum(t, method),
methodClass(t, method),
methodCode(t, method),
reinterpret_cast<intptr_t>(compiled));
setRoot
(t, MethodTree, treeInsert
(t, &(context.zone), root(t, MethodTree),
reinterpret_cast<intptr_t>(compiled), clone,
root(t, MethodTreeSentinal),
compareIpToMethodBounds));
storeStoreMemoryBarrier();
methodCompiled(t, method) = reinterpret_cast<intptr_t>(compiled);
if (methodVirtual(t, method)) {
classVtable(t, methodClass(t, method), methodOffset(t, method))
= compiled;
}
treeUpdate(t, root(t, MethodTree), reinterpret_cast<intptr_t>(compiled),
method, root(t, MethodTreeSentinal), compareIpToMethodBounds);
}
} }
assert(t, (methodFlags(t, method) & ACC_NATIVE) == 0);
// We must avoid acquiring any locks until after the first pass of
// compilation, since this pass may trigger classloading operations
// involving application classloaders and thus the potential for
// deadlock. To make this safe, we use a private clone of the
// method so that we won't be confused if another thread updates the
// original while we're working.
object clone = makeMethod
(t, methodVmFlags(t, method),
methodReturnCode(t, method),
methodParameterCount(t, method),
methodParameterFootprint(t, method),
methodFlags(t, method),
methodOffset(t, method),
methodNativeID(t, method),
methodName(t, method),
methodSpec(t, method),
methodAddendum(t, method),
methodClass(t, method),
methodCode(t, method));
loadMemoryBarrier();
if (methodAddress(t, method) != defaultThunk(t)) {
return;
}
PROTECT(t, clone);
Context context(t, bootContext, clone);
compile(t, &context);
if (UNLIKELY(t->exception)) return;
ACQUIRE(t, t->m->classLock);
if (methodAddress(t, method) != defaultThunk(t)) {
return;
}
finish(t, allocator, &context);
if (UNLIKELY(t->exception)) return;
if (DebugMethodTree) {
fprintf(stderr, "insert method at %p\n",
reinterpret_cast<void*>(methodCompiled(t, clone)));
}
// We can't update the MethodCode field on the original method
// before it is placed into the method tree, since another thread
// might call the method, from which stack unwinding would fail
// (since there is not yet an entry in the method tree). However,
// we can't insert the original method into the tree before updating
// the MethodCode field on it since we rely on that field to
// determine its position in the tree. Therefore, we insert the
// clone in its place. Later, we'll replace the clone with the
// original to save memory.
setRoot
(t, MethodTree, treeInsert
(t, &(context.zone), root(t, MethodTree),
methodCompiled(t, clone), clone, root(t, MethodTreeSentinal),
compareIpToMethodBounds));
storeStoreMemoryBarrier();
set(t, method, MethodCode, methodCode(t, clone));
if (methodVirtual(t, method)) {
classVtable(t, methodClass(t, method), methodOffset(t, method))
= reinterpret_cast<void*>(methodCompiled(t, clone));
}
treeUpdate(t, root(t, MethodTree), methodCompiled(t, clone),
method, root(t, MethodTreeSentinal), compareIpToMethodBounds);
} }
object& object&

View File

@ -2007,11 +2007,11 @@ parseSize(const char* s)
return 0; return 0;
} else if (s[length - 1] == 'k') { } else if (s[length - 1] == 'k') {
memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1); memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1);
RUNTIME_ARRAY_BODY(buffer)[length] = 0; RUNTIME_ARRAY_BODY(buffer)[length - 1] = 0;
return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024; return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024;
} else if (s[length - 1] == 'm') { } else if (s[length - 1] == 'm') {
memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1); memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1);
RUNTIME_ARRAY_BODY(buffer)[length] = 0; RUNTIME_ARRAY_BODY(buffer)[length - 1] = 0;
return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024 * 1024; return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024 * 1024;
} else { } else {
return atoi(s); return atoi(s);

View File

@ -726,6 +726,8 @@ internByteArray(Thread* t, object array)
{ {
PROTECT(t, array); PROTECT(t, array);
ACQUIRE(t, t->m->referenceLock);
object n = hashMapFindNode object n = hashMapFindNode
(t, root(t, Machine::ByteArrayMap), array, byteArrayHash, byteArrayEqual); (t, root(t, Machine::ByteArrayMap), array, byteArrayHash, byteArrayEqual);
if (n) { if (n) {
@ -1199,7 +1201,7 @@ parseCode(Thread* t, Stream& s, object pool)
unsigned maxLocals = s.read2(); unsigned maxLocals = s.read2();
unsigned length = s.read4(); unsigned length = s.read4();
object code = makeCode(t, pool, 0, 0, maxStack, maxLocals, length); object code = makeCode(t, pool, 0, 0, 0, maxStack, maxLocals, length);
s.read(&codeBody(t, code, 0), length); s.read(&codeBody(t, code, 0), length);
PROTECT(t, code); PROTECT(t, code);
@ -1297,7 +1299,6 @@ addInterfaceMethods(Thread* t, object class_, object virtualMap,
methodSpec(t, method), methodSpec(t, method),
0, 0,
class_, class_,
0,
0); 0);
hashMapInsert(t, virtualMap, method, method, methodHash); hashMapInsert(t, virtualMap, method, method, methodHash);
@ -1705,6 +1706,7 @@ makeArrayClass(Thread* t, object loader, unsigned dimensions, object spec,
// todo: arrays should implement Cloneable and Serializable // todo: arrays should implement Cloneable and Serializable
if (classVmFlags(t, type(t, Machine::JobjectType)) & BootstrapFlag) { if (classVmFlags(t, type(t, Machine::JobjectType)) & BootstrapFlag) {
PROTECT(t, loader);
PROTECT(t, spec); PROTECT(t, spec);
PROTECT(t, elementClass); PROTECT(t, elementClass);
@ -2014,10 +2016,10 @@ boot(Thread* t)
setRoot(t, Machine::StringMap, makeWeakHashMap(t, 0, 0)); setRoot(t, Machine::StringMap, makeWeakHashMap(t, 0, 0));
m->processor->boot(t, 0); m->processor->boot(t, 0);
{ object bootCode = makeCode(t, 0, 0, 0, 0, 0, 1); { object bootCode = makeCode(t, 0, 0, 0, 0, 0, 0, 1);
codeBody(t, bootCode, 0) = impdep1; codeBody(t, bootCode, 0) = impdep1;
object bootMethod = makeMethod object bootMethod = makeMethod
(t, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, bootCode, 0); (t, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, bootCode);
PROTECT(t, bootMethod); PROTECT(t, bootMethod);
#include "type-java-initializations.cpp" #include "type-java-initializations.cpp"
@ -3217,59 +3219,65 @@ resolveClass(Thread* t, object loader, object spec, bool throw_)
{ {
return resolveSystemClass(t, loader, spec, throw_); return resolveSystemClass(t, loader, spec, throw_);
} else { } else {
expect(t, throw_);
PROTECT(t, loader); PROTECT(t, loader);
PROTECT(t, spec); PROTECT(t, spec);
ACQUIRE(t, t->m->classLock); { ACQUIRE(t, t->m->classLock);
if (classLoaderMap(t, loader) == 0) { if (classLoaderMap(t, loader) == 0) {
object map = makeHashMap(t, 0, 0); object map = makeHashMap(t, 0, 0);
set(t, loader, ClassLoaderMap, map); set(t, loader, ClassLoaderMap, map);
}
object class_ = hashMapFind
(t, classLoaderMap(t, loader), spec, byteArrayHash, byteArrayEqual);
if (class_) {
return class_;
}
} }
object class_ = hashMapFind object class_;
(t, classLoaderMap(t, loader), spec, byteArrayHash, byteArrayEqual); if (byteArrayBody(t, spec, 0) == '[') {
class_ = resolveArrayClass(t, loader, spec, throw_);
} else {
if (root(t, Machine::LoadClassMethod) == 0) {
object m = resolveMethod
(t, root(t, Machine::BootLoader), "java/lang/ClassLoader",
"loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
if (class_ == 0) { if (m) {
if (byteArrayBody(t, spec, 0) == '[') { setRoot(t, Machine::LoadClassMethod, m);
class_ = resolveArrayClass(t, loader, spec, throw_);
} else {
if (root(t, Machine::LoadClassMethod) == 0) {
object m = resolveMethod
(t, root(t, Machine::BootLoader), "java/lang/ClassLoader",
"loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
if (m) { object classLoaderClass = type(t, Machine::ClassLoaderType);
setRoot(t, Machine::LoadClassMethod, m);
object classLoaderClass = type(t, Machine::ClassLoaderType);
if (classVmFlags(t, classLoaderClass) & BootstrapFlag) { if (classVmFlags(t, classLoaderClass) & BootstrapFlag) {
resolveSystemClass resolveSystemClass
(t, root(t, Machine::BootLoader), (t, root(t, Machine::BootLoader),
vm::className(t, classLoaderClass)); vm::className(t, classLoaderClass));
} }
} }
} }
if (LIKELY(t->exception == 0)) {
object method = findVirtualMethod
(t, root(t, Machine::LoadClassMethod), objectClass(t, loader));
if (LIKELY(t->exception == 0)) { if (LIKELY(t->exception == 0)) {
object method = findVirtualMethod PROTECT(t, method);
(t, root(t, Machine::LoadClassMethod), objectClass(t, loader));
if (LIKELY(t->exception == 0)) { RUNTIME_ARRAY(char, s, byteArrayLength(t, spec));
PROTECT(t, method); replace('/', '.', RUNTIME_ARRAY_BODY(s), reinterpret_cast<char*>
(&byteArrayBody(t, spec, 0)));
RUNTIME_ARRAY(char, s, byteArrayLength(t, spec)); object specString = makeString(t, "%s", s);
replace('/', '.', RUNTIME_ARRAY_BODY(s), reinterpret_cast<char*>
(&byteArrayBody(t, spec, 0)));
object specString = makeString(t, "%s", s); class_ = t->m->processor->invoke(t, method, loader, specString);
class_ = t->m->processor->invoke(t, method, loader, specString); if (LIKELY(class_ and t->exception == 0)) {
class_ = jclassVmClass(t, class_);
if (LIKELY(class_ and t->exception == 0)) {
class_ = jclassVmClass(t, class_);
}
} }
} }
} }
@ -3278,6 +3286,8 @@ resolveClass(Thread* t, object loader, object spec, bool throw_)
if (class_) { if (class_) {
PROTECT(t, class_); PROTECT(t, class_);
ACQUIRE(t, t->m->classLock);
hashMapInsert(t, classLoaderMap(t, loader), spec, class_, hashMapInsert(t, classLoaderMap(t, loader), spec, class_,
byteArrayHash); byteArrayHash);
} else if (throw_ and t->exception == 0) { } else if (throw_ and t->exception == 0) {
@ -3659,8 +3669,8 @@ collect(Thread* t, Heap::CollectionType type)
ENTER(t, Thread::ExclusiveState); ENTER(t, Thread::ExclusiveState);
#ifdef VM_STRESS #ifdef VM_STRESS
bool stress = t->stress; bool stress = (t->flags |= Thread::StressFlag);
if (not stress) t->stress = true; if (not stress) atomicOr(&(t->flags), Thread::StressFlag);
#endif #endif
Machine* m = t->m; Machine* m = t->m;
@ -3681,7 +3691,7 @@ collect(Thread* t, Heap::CollectionType type)
m->fixedFootprint = 0; m->fixedFootprint = 0;
#ifdef VM_STRESS #ifdef VM_STRESS
if (not stress) t->stress = false; if (not stress) atomicAnd(&(t->flags), ~Thread::StressFlag);
#endif #endif
object f = t->m->finalizeQueue; object f = t->m->finalizeQueue;
@ -4005,6 +4015,8 @@ vmPrintTrace(Thread* t)
Thread* t; Thread* t;
} v(t); } v(t);
fprintf(stderr, "debug trace for thread %p\n", t);
t->m->processor->walkStack(t, &v); t->m->processor->walkStack(t, &v);
} }
@ -4012,24 +4024,24 @@ vmPrintTrace(Thread* t)
void* void*
vmAddressFromLine(Thread* t, object m, unsigned line) vmAddressFromLine(Thread* t, object m, unsigned line)
{ {
object code = methodCode(t, m); object code = methodCode(t, m);
printf("code: %p\n", code); printf("code: %p\n", code);
object lnt = codeLineNumberTable(t, code); object lnt = codeLineNumberTable(t, code);
printf("lnt: %p\n", lnt); printf("lnt: %p\n", lnt);
if (lnt) { if (lnt) {
unsigned last = 0; unsigned last = 0;
unsigned bottom = 0; unsigned bottom = 0;
unsigned top = lineNumberTableLength(t, lnt); unsigned top = lineNumberTableLength(t, lnt);
for(unsigned i = bottom; i < top; i++) for(unsigned i = bottom; i < top; i++)
{ {
LineNumber* ln = lineNumberTableBody(t, lnt, i); LineNumber* ln = lineNumberTableBody(t, lnt, i);
if(lineNumberLine(ln) == line) if(lineNumberLine(ln) == line)
return reinterpret_cast<void*>(lineNumberIp(ln)); return reinterpret_cast<void*>(lineNumberIp(ln));
else if(lineNumberLine(ln) > line) else if(lineNumberLine(ln) > line)
return reinterpret_cast<void*>(last); return reinterpret_cast<void*>(last);
last = lineNumberIp(ln); last = lineNumberIp(ln);
} }
} }
return 0; return 0;
} }

View File

@ -30,6 +30,9 @@
#define ACQUIRE(t, x) MonitorResource MAKE_NAME(monitorResource_) (t, x) #define ACQUIRE(t, x) MonitorResource MAKE_NAME(monitorResource_) (t, x)
#define ACQUIRE_OBJECT(t, x) \
ObjectMonitorResource MAKE_NAME(monitorResource_) (t, x)
#define ACQUIRE_RAW(t, x) RawMonitorResource MAKE_NAME(monitorResource_) (t, x) #define ACQUIRE_RAW(t, x) RawMonitorResource MAKE_NAME(monitorResource_) (t, x)
#define ENTER(t, state) StateResource MAKE_NAME(stateResource_) (t, state) #define ENTER(t, state) StateResource MAKE_NAME(stateResource_) (t, state)
@ -1507,20 +1510,19 @@ shutDown(Thread* t);
inline void inline void
stress(Thread* t) stress(Thread* t)
{ {
if ((not t->stress) if ((t->flags & (Thread::StressFlag | Thread::TracingFlag)) == 0
and (not t->tracing)
and t->state != Thread::NoState and t->state != Thread::NoState
and t->state != Thread::IdleState) and t->state != Thread::IdleState)
{ {
t->stress = true; atomicOr(&(t->flags), Thread::StressFlag);
# ifdef VM_STRESS_MAJOR # ifdef VM_STRESS_MAJOR
collect(t, Heap::MajorCollection); collect(t, Heap::MajorCollection);
# else // not VM_STRESS_MAJOR # else // not VM_STRESS_MAJOR
collect(t, Heap::MinorCollection); collect(t, Heap::MinorCollection);
# endif // not VM_STRESS_MAJOR # endif // not VM_STRESS_MAJOR
t->stress = false; atomicAnd(&(t->flags), ~Thread::StressFlag);
} }
} }
@ -2607,6 +2609,21 @@ monitorNotifyAll(Thread* t, object monitor)
while (monitorNotify(t, monitor)) { } while (monitorNotify(t, monitor)) { }
} }
class ObjectMonitorResource {
public:
ObjectMonitorResource(Thread* t, object o): o(o), protector(t, &(this->o)) {
monitorAcquire(protector.t, o);
}
~ObjectMonitorResource() {
monitorRelease(protector.t, o);
}
private:
object o;
Thread::SingleProtector protector;
};
object object
objectMonitor(Thread* t, object o, bool createNew); objectMonitor(Thread* t, object o, bool createNew);
@ -2621,8 +2638,7 @@ acquire(Thread* t, object o)
object m = objectMonitor(t, o, true); object m = objectMonitor(t, o, true);
if (DebugMonitors) { if (DebugMonitors) {
fprintf(stderr, "thread %p acquires %p for %x\n", fprintf(stderr, "thread %p acquires %p for %x\n", t, m, hash);
t, m, hash);
} }
monitorAcquire(t, m); monitorAcquire(t, m);
@ -2639,8 +2655,7 @@ release(Thread* t, object o)
object m = objectMonitor(t, o, false); object m = objectMonitor(t, o, false);
if (DebugMonitors) { if (DebugMonitors) {
fprintf(stderr, "thread %p releases %p for %x\n", fprintf(stderr, "thread %p releases %p for %x\n", t, m, hash);
t, m, hash);
} }
monitorRelease(t, m); monitorRelease(t, m);

View File

@ -14,6 +14,8 @@
# define _XOPEN_SOURCE # define _XOPEN_SOURCE
#endif #endif
#define __STDC_CONSTANT_MACROS
#include "sys/mman.h" #include "sys/mman.h"
#include "sys/types.h" #include "sys/types.h"
#include "sys/stat.h" #include "sys/stat.h"
@ -287,7 +289,9 @@ class MySystem: public System {
owner_ = 0; owner_ = 0;
pthread_mutex_unlock(&mutex); pthread_mutex_unlock(&mutex);
if (time) { // pretend anything greater than one million years (in
// milliseconds) is infinity so as to avoid overflow:
if (time and time < INT64_C(31536000000000000)) {
int64_t then = s->now() + time; int64_t then = s->now() + time;
timespec ts = { then / 1000, (then % 1000) * 1000 * 1000 }; timespec ts = { then / 1000, (then % 1000) * 1000 * 1000 };
int rv UNUSED = pthread_cond_timedwait int rv UNUSED = pthread_cond_timedwait

View File

@ -26,8 +26,7 @@
(type field avian/VMField) (type field avian/VMField)
(type method avian/VMMethod (type method avian/VMMethod)
(intptr_t compiled))
(type addendum avian/Addendum) (type addendum avian/Addendum)
@ -64,6 +63,7 @@
(object pool) (object pool)
(object exceptionHandlerTable) (object exceptionHandlerTable)
(object lineNumberTable) (object lineNumberTable)
(intptr_t compiled)
(uint16_t maxStack) (uint16_t maxStack)
(uint16_t maxLocals) (uint16_t maxLocals)
(array uint8_t body)) (array uint8_t body))
@ -163,6 +163,7 @@
(type thread java/lang/Thread (type thread java/lang/Thread
(require object sleepLock) (require object sleepLock)
(require object interruptLock)
(require uint8_t interrupted) (require uint8_t interrupted)
(alias peer uint64_t eetop)) (alias peer uint64_t eetop))

View File

@ -1,4 +1,14 @@
public class GC { public class GC {
private static final Integer cache[] = new Integer[100];
private static final Integer MAX_INT_OBJ = new Integer(Integer.MAX_VALUE);
private static Integer valueOf(int i) {
try {
return cache[i];
} catch (ArrayIndexOutOfBoundsException e) {
return (i == Integer.MAX_VALUE) ? MAX_INT_OBJ : new Integer(i);
}
}
private static void small() { private static void small() {
for (int i = 0; i < 1024; ++i) { for (int i = 0; i < 1024; ++i) {
@ -145,6 +155,8 @@ public class GC {
} }
public static void main(String[] args) { public static void main(String[] args) {
valueOf(1000);
Object[] array = new Object[1024 * 1024]; Object[] array = new Object[1024 * 1024];
array[0] = new Object(); array[0] = new Object();