do not omit calls to empty methods which may trigger class initialization

There's a small optimization in compileDirectInvoke which tries to
avoid generating calls to empty methods.  However, this causes
problems for code which uses such a call to ensure a class is
initialized -- if we omit that call, the class may not be
initialized and any side effects of that initialization may not
happen when the program expects them to.

This commit ensures that the compiler only omits empty method calls
when the target class does not need initialization.  It also removes
commented-out code in classpath-openjdk.cpp which was responsible for
loading libmawt proactively; that was a hack to get JogAmp to work
before we understood what the real problem was.
This commit is contained in:
Joel Dice 2014-02-09 14:58:47 -07:00
parent 07abed9c65
commit 1735a7976a
3 changed files with 19 additions and 10 deletions

View File

@ -698,15 +698,6 @@ class MyClasspath : public Classpath {
#else // not AVIAN_OPENJDK_SRC #else // not AVIAN_OPENJDK_SRC
expect(t, loadLibrary(t, libraryPath, "verify", true, true)); expect(t, loadLibrary(t, libraryPath, "verify", true, true));
expect(t, loadLibrary(t, libraryPath, "java", true, true)); expect(t, loadLibrary(t, libraryPath, "java", true, true));
# ifndef PLATFORM_WINDOWS
// at one point, loading libmawt ahead of time was necessary to
// make AWT work, but recent versions of OpenJDK seem to take care
// of this from Java code, in which case loading it ahead of time
// actually causes trouble, so we comment it out for now until we
// know exactly when it's needed:
//loadLibrary(t, libraryPath, "mawt", true, true, false);
# endif
#endif // not AVIAN_OPENJDK_SRC #endif // not AVIAN_OPENJDK_SRC
{ object assertionLock = resolveField { object assertionLock = resolveField

View File

@ -3234,7 +3234,11 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall)
Compiler::Operand* result = 0; Compiler::Operand* result = 0;
if (emptyMethod(t, target)) { // don't bother calling an empty method unless calling it might
// cause the class to be initialized, which may have side effects
if (emptyMethod(t, target)
and (not classNeedsInit(t, methodClass(t, target))))
{
tailCall = false; tailCall = false;
} else { } else {
BootContext* bc = frame->context->bootContext; BootContext* bc = frame->context->bootContext;

View File

@ -23,6 +23,16 @@ public class Misc {
} }
} }
private static class Static {
static {
staticRan = true;
}
public static void run() { }
}
private static boolean staticRan;
private static int alpha; private static int alpha;
private static int beta; private static int beta;
private static byte byte1, byte2, byte3; private static byte byte1, byte2, byte3;
@ -299,6 +309,10 @@ public class Misc {
} catch (java.net.UnknownHostException e) { } catch (java.net.UnknownHostException e) {
// cool // cool
} }
expect(! staticRan);
Static.run();
expect(staticRan);
} }
protected class Protected { } protected class Protected { }