blob: 6267d537f0618b8257a88aabbe0ffb2883ae2f64 [file] [log] [blame]
/*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package java.lang;
import java.lang.ref.Reference;
import java.util.Objects;
/**
* Functional test for WeakPairMap
*
* @author Peter Levart
*/
public class WeakPairMapTest {
public static void main(String[] args) {
WeakPairMap<Object, Object, String> pm = new WeakPairMap<>();
Object key1 = new Object();
Object key2 = new Object();
// check for emptiness
assertEquals(pm.containsKeyPair(key1, key2), false);
assertEquals(pm.get(key1, key2), null);
// check for NPE(s)
for (Object k1 : new Object[]{null, key1}) {
for (Object k2 : new Object[]{null, key2}) {
for (String v : new String[]{null, "abc"}) {
if (k1 != null && k2 != null && v != null) {
// skip non-null args
continue;
}
try {
pm.put(k1, k2, v);
throw new AssertionError("Unexpected code path, k1=" +
k1 + ", k2=" + k2 + ", v=" + v);
} catch (NullPointerException e) {
// expected
}
try {
pm.putIfAbsent(k1, k2, v);
throw new AssertionError("Unexpected code path, k1=" +
k1 + ", k2=" + k2 + ", v=" + v);
} catch (NullPointerException e) {
// expected
}
if (k1 != null && k2 != null) {
// skip non-null args
continue;
}
try {
pm.computeIfAbsent(k1, k2, (_k1, _k2) -> v);
throw new AssertionError("Unexpected code path, k1=" +
k1 + ", k2=" + k2 + ", v=" + v);
} catch (NullPointerException e) {
// expected
}
try {
pm.containsKeyPair(k1, k2);
throw new AssertionError("Unexpected code path, k1=" +
k1 + ", k2=" + k2);
} catch (NullPointerException e) {
// expected
}
try {
pm.get(k1, k2);
throw new AssertionError("Unexpected code path, k1=" +
k1 + ", k2=" + k2);
} catch (NullPointerException e) {
// expected
}
}
}
}
// how much to wait when it is expected for entry to be retained
final long retentionTimeout = 500L;
// how much to wait when it is expected for entry to be removed
final long cleanupTimeout = 30_000L;
// check insertion
assertEquals(pm.putIfAbsent(key1, key2, "abc"), null);
assertEquals(pm.get(key1, key2), "abc");
// check retention while both keys are still reachable
assertEquals(gcAndWaitRemoved(pm, "abc", retentionTimeout), false);
assertEquals(pm.get(key1, key2), "abc");
// check cleanup when both keys are unreachable
key1 = null;
key2 = null;
assertEquals(gcAndWaitRemoved(pm, "abc", cleanupTimeout), true);
// new insertion
key1 = new Object();
key2 = new Object();
assertEquals(pm.putIfAbsent(key1, key2, "abc"), null);
assertEquals(pm.get(key1, key2), "abc");
// check retention while both keys are still reachable
assertEquals(gcAndWaitRemoved(pm, "abc", retentionTimeout), false);
assertEquals(pm.get(key1, key2), "abc");
// check cleanup when 1st key is unreachable
key1 = null;
assertEquals(gcAndWaitRemoved(pm, "abc", cleanupTimeout), true);
Reference.reachabilityFence(key2);
// new insertion
key1 = new Object();
key2 = new Object();
assertEquals(pm.putIfAbsent(key1, key2, "abc"), null);
assertEquals(pm.get(key1, key2), "abc");
// check retention while both keys are still reachable
assertEquals(gcAndWaitRemoved(pm, "abc", retentionTimeout), false);
assertEquals(pm.get(key1, key2), "abc");
// check cleanup when 2nd key is unreachable
key2 = null;
assertEquals(gcAndWaitRemoved(pm, "abc", cleanupTimeout), true);
Reference.reachabilityFence(key1);
}
/**
* Trigger GC and wait for at most {@code millis} ms for given value to
* be removed from given WeakPairMap.
*
* @return true if element has been removed or false if not
*/
static <V> boolean gcAndWaitRemoved(WeakPairMap<?, ?, V> pm, V value,
long millis) {
System.gc();
for (int i = 0; i < (millis + 99) / 100 && pm.values().contains(value); i++) {
try {
Thread.sleep(100L);
} catch (InterruptedException e) {
throw new AssertionError("Interrupted");
}
}
return !pm.values().contains(value);
}
static void assertEquals(Object actual, Object expected) {
if (!Objects.equals(actual, expected)) {
throw new AssertionError("Expected: " + expected + ", actual: " + actual);
}
}
}