/* Copyright (c) 2008, Avian Contributors Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. There is NO WARRANTY for this software. See license.txt for details. */ #include "process.h" using namespace vm; namespace { unsigned mangledSize(int8_t c) { switch (c) { case '_': case ';': case '[': return 2; case '$': return 6; default: return 1; } } unsigned mangle(int8_t c, char* dst) { switch (c) { case '/': dst[0] = '_'; return 1; case '_': dst[0] = '_'; dst[1] = '1'; return 2; case ';': dst[0] = '_'; dst[1] = '2'; return 2; case '[': dst[0] = '_'; dst[1] = '3'; return 2; case '$': memcpy(dst, "_00024", 6); return 6; default: dst[0] = c; return 1; } } unsigned jniNameLength(Thread* t, object method, bool decorate) { unsigned size = 5; object className = ::className(t, methodClass(t, method)); for (unsigned i = 0; i < byteArrayLength(t, className) - 1; ++i) { size += mangledSize(byteArrayBody(t, className, i)); } ++ size; object methodName = ::methodName(t, method); for (unsigned i = 0; i < byteArrayLength(t, methodName) - 1; ++i) { size += mangledSize(byteArrayBody(t, methodName, i)); } if (decorate) { size += 2; object methodSpec = ::methodSpec(t, method); for (unsigned i = 1; i < byteArrayLength(t, methodSpec) - 1 and byteArrayBody(t, methodSpec, i) != ')'; ++i) { size += mangledSize(byteArrayBody(t, methodSpec, i)); } } return size; } void makeJNIName(Thread* t, char* name, object method, bool decorate) { memcpy(name, "Java_", 5); name += 5; object className = ::className(t, methodClass(t, method)); for (unsigned i = 0; i < byteArrayLength(t, className) - 1; ++i) { name += mangle(byteArrayBody(t, className, i), name); } *(name++) = '_'; object methodName = ::methodName(t, method); for (unsigned i = 0; i < byteArrayLength(t, methodName) - 1; ++i) { name += mangle(byteArrayBody(t, methodName, i), name); } if (decorate) { *(name++) = '_'; *(name++) = '_'; object methodSpec = ::methodSpec(t, method); for (unsigned i = 1; i < byteArrayLength(t, methodSpec) - 1 and byteArrayBody(t, methodSpec, i) != ')'; ++i) { name += mangle(byteArrayBody(t, methodSpec, i), name); } } *(name++) = 0; } void* resolveNativeMethod(Thread* t, const char* undecorated, const char* decorated) { for (System::Library* lib = t->m->libraries; lib; lib = lib->next()) { void* p = lib->resolve(undecorated); if (p) { return p; } else { p = lib->resolve(decorated); if (p) { return p; } } } return 0; } } // namespace namespace vm { void* resolveNativeMethod2(Thread* t, object method) { unsigned undecoratedSize = jniNameLength(t, method, false); char undecorated[undecoratedSize + 1 + 6]; // extra 6 is for code below makeJNIName(t, undecorated + 1, method, false); unsigned decoratedSize = jniNameLength(t, method, true); char decorated[decoratedSize + 1 + 6]; // extra 6 is for code below makeJNIName(t, decorated + 1, method, true); void* p = ::resolveNativeMethod(t, undecorated + 1, decorated + 1); if (p) { return p; } #ifdef __MINGW32__ // on windows, we also try the _%s@%d variant, since the SWT // libraries use it. unsigned footprint = methodParameterFootprint(t, method) + 1; if (methodFlags(t, method) & ACC_STATIC) { ++ footprint; } *undecorated = '_'; snprintf(undecorated + undecoratedSize + 1, 5, "@%d", footprint * BytesPerWord); *decorated = '_'; snprintf(decorated + decoratedSize + 1, 5, "@%d", footprint * BytesPerWord); p = ::resolveNativeMethod(t, undecorated, decorated); if (p) { return p; } #endif return 0; } int findLineNumber(Thread* t, object method, unsigned ip) { if (methodFlags(t, method) & ACC_NATIVE) { return NativeLine; } // our parameter indicates the instruction following the one we care // about, so we back up first: -- ip; object code = methodCode(t, method); object lnt = codeLineNumberTable(t, code); if (lnt) { unsigned bottom = 0; unsigned top = lineNumberTableLength(t, lnt); for (unsigned span = top - bottom; span; span = top - bottom) { unsigned middle = bottom + (span / 2); LineNumber* ln = lineNumberTableBody(t, lnt, middle); if (ip >= lineNumberIp(ln) and (middle + 1 == lineNumberTableLength(t, lnt) or ip < lineNumberIp(lineNumberTableBody(t, lnt, middle + 1)))) { return lineNumberLine(ln); } else if (ip < lineNumberIp(ln)) { top = middle; } else if (ip > lineNumberIp(ln)) { bottom = middle + 1; } } abort(t); } else { return UnknownLine; } } } // namespace vm