openide.util/test/unit/src/org/openide/util/WeakSetTest.java
author Jesse Glick <jglick@netbeans.org>
Fri, 19 Dec 2008 10:24:30 -0500
changeset 490 9ca5b3d4a70a
parent 361 b8bf023cf23e
child 829 3b2ed3e1f01b
permissions -rw-r--r--
Random failure, and some warnings.
     1 /*
     2  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
     3  *
     4  * Copyright 1997-2007 Sun Microsystems, Inc. All rights reserved.
     5  *
     6  * The contents of this file are subject to the terms of either the GNU
     7  * General Public License Version 2 only ("GPL") or the Common
     8  * Development and Distribution License("CDDL") (collectively, the
     9  * "License"). You may not use this file except in compliance with the
    10  * License. You can obtain a copy of the License at
    11  * http://www.netbeans.org/cddl-gplv2.html
    12  * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
    13  * specific language governing permissions and limitations under the
    14  * License.  When distributing the software, include this License Header
    15  * Notice in each file and include the License file at
    16  * nbbuild/licenses/CDDL-GPL-2-CP.  Sun designates this
    17  * particular file as subject to the "Classpath" exception as provided
    18  * by Sun in the GPL Version 2 section of the License file that
    19  * accompanied this code. If applicable, add the following below the
    20  * License Header, with the fields enclosed by brackets [] replaced by
    21  * your own identifying information:
    22  * "Portions Copyrighted [year] [name of copyright owner]"
    23  *
    24  * Contributor(s):
    25  *
    26  * The Original Software is NetBeans. The Initial Developer of the Original
    27  * Software is Sun Microsystems, Inc. Portions Copyright 1997-2006 Sun
    28  * Microsystems, Inc. All Rights Reserved.
    29  *
    30  * If you wish your version of this file to be governed by only the CDDL
    31  * or only the GPL Version 2, indicate your decision by adding
    32  * "[Contributor] elects to include this software in this distribution
    33  * under the [CDDL or GPL Version 2] license." If you do not indicate a
    34  * single choice of license, a recipient has the option to distribute
    35  * your version of this file under either the CDDL, the GPL Version 2 or
    36  * to extend the choice of license to its licensees as provided above.
    37  * However, if you add GPL Version 2 code and therefore, elected the GPL
    38  * Version 2 license, then the option applies only if the new code is
    39  * made subject to such option by the copyright holder.
    40  */
    41 
    42 package org.openide.util;
    43 
    44 import java.lang.ref.Reference;
    45 import java.lang.ref.WeakReference;
    46 import java.util.ArrayList;
    47 import java.util.Iterator;
    48 import java.util.List;
    49 import java.util.Set;
    50 import java.util.concurrent.ExecutorService;
    51 import java.util.concurrent.Executors;
    52 import org.netbeans.junit.NbTestCase;
    53 import org.netbeans.junit.RandomlyFails;
    54 
    55 public class WeakSetTest extends NbTestCase {
    56 
    57     public WeakSetTest(String testName) {
    58         super(testName);
    59     }
    60 
    61     public void testToArrayMayContainNullsIssue42271 () {
    62         class R implements Runnable {
    63             Object[] arr;
    64             Object last;
    65             
    66             public R () {
    67                 int cnt = 10;
    68                 arr = new Object[cnt];
    69                 for (int i = 0; i < cnt; i++) {
    70                     arr[i] = new Integer(i); // autoboxing makes test fail!
    71                 }
    72             }
    73             
    74             
    75             public void run () {
    76                 
    77                 Reference<?> r = new WeakReference<Object>(last);
    78                 for (int i = 0; i < arr.length; i++) {
    79                     arr[i] = null;
    80                 }
    81                 arr = null;
    82                 last = null;
    83                 assertGC ("Last item has to disappear", r);
    84             }
    85             
    86             public void putToSet (NotifyWhenIteratedSet s) {
    87                 for (int i = 0; i < arr.length; i++) {
    88                     s.add (arr[i]);
    89                 }
    90                 assertEquals (arr.length, s.size ());
    91                 Iterator<Object> it = s.superIterator();
    92                 Object prev = it.next ();
    93                 while (it.hasNext ()) {
    94                     prev = it.next ();
    95                 }
    96                 last = prev;
    97             }
    98         }
    99         R r = new R ();
   100         
   101         
   102         
   103         NotifyWhenIteratedSet ws = new NotifyWhenIteratedSet (r, 1);
   104         
   105         r.putToSet (ws);
   106         
   107         Object[] arr = ws.toArray ();
   108         for (int i = 0; i < arr.length; i++) {
   109             assertNotNull (i + "th index should not be null", arr[i]);
   110         }
   111     }
   112     
   113     private static final class NotifyWhenIteratedSet extends WeakSet<Object> {
   114         private Runnable run;
   115         private int cnt;
   116         
   117         public NotifyWhenIteratedSet (Runnable run, int cnt) {
   118             this.run = run;
   119             this.cnt = cnt;
   120         }
   121         
   122         public Iterator<Object> superIterator() {
   123             return super.iterator ();
   124         }
   125         
   126         @Override
   127         public Iterator<Object> iterator() {
   128             final Iterator<Object> it = super.iterator();
   129             class I implements Iterator<Object> {
   130                 public boolean hasNext() {
   131                     return it.hasNext ();
   132                 }
   133 
   134                 public Object next() {
   135                     if (--cnt == 0) {
   136                         run.run ();
   137                     }
   138                     return it.next ();
   139                 }
   140 
   141                 public void remove() {
   142                     it.remove();
   143                 }
   144             }
   145             return new I ();
   146         }
   147     }
   148     
   149     private final static class TestObj {
   150         static final Set<TestObj> testObjs = new WeakSet<TestObj>();
   151         private final String name;
   152         TestObj(String name) {
   153             this.name = name;
   154             synchronized (testObjs) {
   155                 testObjs.add(this);
   156             }
   157         }
   158 
   159         @Override
   160         public String toString() {
   161             return name;
   162         }
   163         
   164     }
   165     
   166     private final static class GC implements Runnable {
   167 
   168         public void run() {
   169             try {
   170                 for (int i = 0; i < 5; i++) {
   171                     gc();
   172                 }
   173             } catch (InterruptedException ex) {
   174                 // ignore
   175             }
   176         }
   177 
   178         static void gc() throws InterruptedException {
   179             List<byte[]> alloc = new ArrayList<byte[]>();
   180             int size = 100000;
   181             for (int i = 0; i < 50; i++) {
   182                 System.gc();
   183                 System.runFinalization();
   184                 try {
   185                     alloc.add(new byte[size]);
   186                     size = (int) (((double) size) * 1.3);
   187                 } catch (OutOfMemoryError error) {
   188                     size = size / 2;
   189                 }
   190                 if (i % 3 == 0) {
   191                     Thread.sleep(100);
   192                 }
   193             }
   194         }
   195     }
   196 
   197     /**
   198      * test for issue #106218
   199      * @throws java.lang.Exception
   200      */
   201     @RandomlyFails // OOME in NB-Core-Build #1908
   202     public void testWeakSetIntegrity() throws Exception {
   203         //CharSequence log = Log.enable(WeakSet.class.getName(), Level.FINE);
   204         ArrayList<WeakReference<TestObj>> awr = new ArrayList<WeakReference<TestObj>>();
   205         ExecutorService exec = Executors.newFixedThreadPool(5);
   206         exec.execute(new GC());
   207         for (int i = 0; i < 1000; i++) {
   208             TestObj to = new TestObj("T" + i);
   209             awr.add(new WeakReference<TestObj>(to));
   210             Thread.yield();
   211             for (WeakReference<TestObj> wro : awr) {
   212                 TestObj wroo = wro.get();
   213                 if (wroo != null) {
   214                     synchronized (TestObj.testObjs) {
   215                         boolean found = false;
   216                         for (TestObj o : TestObj.testObjs) {
   217                             if (o == wroo) {
   218                                 found = true;
   219                             }
   220                         }
   221                         if (found != TestObj.testObjs.contains(wroo)) {
   222                             //System.out.println(log.toString());
   223                             fail("Inconsistency of iterator chain and hash map");
   224                         }
   225                     }
   226                 }
   227             }
   228         }
   229         exec.shutdownNow();
   230     }    
   231 }