Loading... (original) (raw)

FULL PRODUCT VERSION :
java version "1.7.0-ea"
Java(TM) SE Runtime Environment (build 1.7.0-ea-b23)
Java HotSpot(TM) Server VM (build 12.0-b01, mixed mode)

ADDITIONAL OS VERSION INFORMATION :
Linux localhost.localdomain 2.6.20.1 #3 SMP Sun Mar 25 22:45:33 CEST 2007 i686 i686 i386 GNU/Linux

A DESCRIPTION OF THE PROBLEM :
Direct-ByteBuffers use sun.misc.Cleaner to get notified when they have been GCed and java.nio.Bits.reserveMemory() to trigger a GC run when too many dead but uncollected references to DirectByteBuffer-Memory is laying on the heap.

The test-code below stresses the following problem:
The object which references the Direct-ByteBuffer has a finalzer, therefor the DirectByteBuffer cannot be reclamed with only a single GC run. So reserveMemory only triggers a single GC which is enough to discover that the anoynmous instance of runnable can be collected - however the buffer still can be referenced inside of the finalizer - so the GC cannot dereference it.

I would be happy if a fix for the problem would be accepted via OpenJDK, if so I've a working prototype.

STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
Let the test below run for a small amount of time.

EXPECTED VERSUS ACTUAL BEHAVIOR :
EXPECTED -
No Exceptions thrown
ACTUAL -
java.lang.OutOfMemoryError: Direct buffer memory

ERROR MESSAGES/STACK TRACES THAT OCCUR :
java.lang.OutOfMemoryError: Direct buffer memory

REPRODUCIBILITY :
This bug can be reproduced always.

---------- BEGIN SOURCE ----------

public static int n=0; //Just global to avoid hotspot optimizations
public static void main(String[] args) throws Exception {
while(true) {
Runnable r = new Runnable() {
ByteBuffer buffer = null;

public void run() {
buffer = ByteBuffer.allocateDirect(32768);
}

public void finalize() {
n++;
}
};
r.run();
}
}
---------- END SOURCE ----------

CUSTOMER SUBMITTED WORKAROUND :
Almost not possible in application code, because you can never know wether some legacy-code uses finalizers (even jdk7's source has plenty of them) and has a reference to the objects created by you.