Uploaded image for project: 'JDK'
  1. JDK
  2. JDK-6698009

CMS is not collecting weak references when -XX:+ParallelRefProcEnabled

XMLWordPrintable

    • Icon: Bug Bug
    • Resolution: Duplicate
    • Icon: P3 P3
    • None
    • 6u4
    • hotspot
    • gc
    • x86
    • solaris_10

      FULL PRODUCT VERSION :
      java version "1.6.0_04"
      Java(TM) SE Runtime Environment (build 1.6.0_04-b12)
      Java HotSpot(TM) Server VM (build 10.0-b19, mixed mode)


      FULL OS VERSION :
      SunOS gold01 5.10 Generic_125101-07 i86pc i386 i86pc Solaris

      A DESCRIPTION OF THE PROBLEM :
      We are using WeakReferences to manage distributed object cache. When program no longer references given object, we rely on WeakReference getting enqueued at some point so we can send remote system information that client is not longer interested in object updates. Unfortunately, there seems to be a big change of behaviour in CMS between 1.6.0_02 and 1.6.0_04 - with _02, weak references were cleared and enqueued within one or two CMS runs after they stopped being referenced, with _04 it seems we have to wait till we run out of memory (and full gc) for this to happen.

      THE PROBLEM WAS REPRODUCIBLE WITH -Xint FLAG: Did not try

      THE PROBLEM WAS REPRODUCIBLE WITH -server FLAG: Yes

      STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
      Run the program attached below with following parameters

      java -server -Xmx300m -XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode -XX:CMSIncrementalDutyCycleMin=5 -XX:+CMSIncrementalPacing -XX:+PrintGCDetails -XX:+UseParNewGC -XX:+ParallelRefProcEnabled -cp . WeakTest

      with jdk 1.6.0_02 and 1.6.0_04.


      EXPECTED VERSUS ACTUAL BEHAVIOR :
      With 1.6.0_02, as soon as first CMS is finished, some of "Removed X nnn" messages start to appear and then they continue to appear quite frequently. On my machine, it takes around 700 adds to reach the stage where references are starting to be removed almost in realtime.

      With 1.6.0_04, I'm not able to spot even single removal of weak reference for a long, long time (after that I have stopped testing - I suppose that on reaching out of memory condition, full gc will clear them finally).

      Please note that I do not mean refList growing - it is obvious leak (very small) to hold the weak references referenced. Problem is with queue removal thread never being woken up by queue.


      REPRODUCIBILITY :
      This bug can be reproduced always.

      ---------- BEGIN SOURCE ----------
      import java.lang.ref.*;
      import java.util.ArrayList;

      public class WeakTest {

        static Object[] refArray;
        
        public static void main(String[] args) throws Exception {
          
          Object[][][] start= new Object[1000][1000][2];
          
          final ReferenceQueue queue = new ReferenceQueue();
          
          ArrayList<Reference> refList = new ArrayList<Reference>();
          
          
          int counter = 0;
          
          new Thread() {
            @Override public void run() {
              while ( true ) {
                try {
                  Reference r = queue.remove();
                  System.out.println("Removed " + r);
                } catch (Exception exc) {
                  exc.printStackTrace();
                }
              }
            }
          }.start();
          
          refArray = new Object[100];
          
          
          while ( true ) {
            Integer[][] arr = new Integer[1000][];
            for ( int i =0; i < arr.length; i++) {
              arr[i] = new Integer[1000];
            }

            final int fcounter = counter++;

            
            refArray[fcounter%refArray.length] = arr[0];
            
            WeakReference reference = new WeakReference(arr[0],queue) {
              @Override public String toString() {
                return "X " + fcounter;
              }
            };
            
            
            refList.add(reference);
            
            
            if ( fcounter % 100 == 99 ) {
              System.out.println("Added " + fcounter + " references");
              Thread.sleep(10000);
            }
            
          }
        }
      }

      ---------- END SOURCE ----------

      CUSTOMER SUBMITTED WORKAROUND :
      Invoke full gc with System.gc() once per some time [not acceptable]
      Stay with 1.6.0_02

      Release Regression From : 6u2
      The above release value was the last known release where this
      bug was not reproducible. Since then there has been a regression.

      Release Regression From : 6u2
      The above release value was the last known release where this
      bug was not reproducible. Since then there has been a regression.

            ysr Y. Ramakrishna
            ndcosta Nelson Dcosta (Inactive)
            Votes:
            0 Vote for this issue
            Watchers:
            1 Start watching this issue

              Created:
              Updated:
              Resolved:
              Imported:
              Indexed: