1 /* 2 * Written by Doug Lea with assistance from members of JCP JSR-166 3 * Expert Group and released to the public domain, as explained at 4 * http://creativecommons.org/publicdomain/zero/1.0/ 5 * Other contributors include Andrew Wright, Jeffrey Hayes, 6 * Pat Fisher, Mike Judd. 7 */ 8 9 package jsr166; 10 11 import java.util.concurrent.atomic.AtomicReferenceFieldUpdater; 12 13 import junit.framework.Test; 14 import junit.framework.TestSuite; 15 16 public class AtomicReferenceFieldUpdaterTest extends JSR166TestCase { 17 volatile Integer x = null; 18 protected volatile Integer protectedField; 19 private volatile Integer privateField; 20 Object z; 21 Integer w; 22 volatile int i; 23 24 // android-note: Removed because the CTS runner does a bad job of 25 // retrying tests that have suite() declarations. 26 // 27 // public static void main(String[] args) { 28 // main(suite(), args); 29 // } 30 // public static Test suite() { 31 // return new TestSuite(AtomicReferenceFieldUpdaterTest.class); 32 // } 33 34 // for testing subclass access 35 static class AtomicReferenceFieldUpdaterTestSubclass extends AtomicReferenceFieldUpdaterTest { 36 public void checkPrivateAccess() { 37 try { 38 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest,Integer> a = 39 AtomicReferenceFieldUpdater.newUpdater 40 (AtomicReferenceFieldUpdaterTest.class, Integer.class, "privateField"); 41 shouldThrow(); 42 } catch (RuntimeException success) { 43 assertNotNull(success.getCause()); 44 } 45 } 46 47 public void checkCompareAndSetProtectedSub() { 48 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest,Integer> a = 49 AtomicReferenceFieldUpdater.newUpdater 50 (AtomicReferenceFieldUpdaterTest.class, Integer.class, "protectedField"); 51 this.protectedField = one; 52 assertTrue(a.compareAndSet(this, one, two)); 53 assertTrue(a.compareAndSet(this, two, m4)); 54 assertSame(m4, a.get(this)); 55 assertFalse(a.compareAndSet(this, m5, seven)); 56 assertFalse(seven == a.get(this)); 57 assertTrue(a.compareAndSet(this, m4, seven)); 58 assertSame(seven, a.get(this)); 59 } 60 } 61 62 static class UnrelatedClass { 63 public void checkPackageAccess(AtomicReferenceFieldUpdaterTest obj) { 64 obj.x = one; 65 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest,Integer> a = 66 AtomicReferenceFieldUpdater.newUpdater 67 (AtomicReferenceFieldUpdaterTest.class, Integer.class, "x"); 68 assertSame(one, a.get(obj)); 69 assertTrue(a.compareAndSet(obj, one, two)); 70 assertSame(two, a.get(obj)); 71 } 72 73 public void checkPrivateAccess(AtomicReferenceFieldUpdaterTest obj) { 74 try { 75 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest,Integer> a = 76 AtomicReferenceFieldUpdater.newUpdater 77 (AtomicReferenceFieldUpdaterTest.class, Integer.class, "privateField"); 78 throw new AssertionError("should throw"); 79 } catch (RuntimeException success) { 80 assertNotNull(success.getCause()); 81 } 82 } 83 } 84 85 static AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> updaterFor(String fieldName) { 86 return AtomicReferenceFieldUpdater.newUpdater 87 (AtomicReferenceFieldUpdaterTest.class, Integer.class, fieldName); 88 } 89 90 /** 91 * Construction with non-existent field throws RuntimeException 92 */ 93 public void testConstructor() { 94 try { 95 updaterFor("y"); 96 shouldThrow(); 97 } catch (RuntimeException success) { 98 assertNotNull(success.getCause()); 99 } 100 } 101 102 /** 103 * construction with field not of given type throws ClassCastException 104 */ 105 public void testConstructor2() { 106 try { 107 updaterFor("z"); 108 shouldThrow(); 109 } catch (ClassCastException success) {} 110 } 111 112 /** 113 * Constructor with non-volatile field throws IllegalArgumentException 114 */ 115 public void testConstructor3() { 116 try { 117 updaterFor("w"); 118 shouldThrow(); 119 } catch (IllegalArgumentException success) {} 120 } 121 122 /** 123 * Constructor with non-reference field throws ClassCastException 124 */ 125 public void testConstructor4() { 126 try { 127 updaterFor("i"); 128 shouldThrow(); 129 } catch (ClassCastException success) {} 130 } 131 132 /** 133 * construction using private field from subclass throws RuntimeException 134 */ 135 public void testPrivateFieldInSubclass() { 136 AtomicReferenceFieldUpdaterTestSubclass s = 137 new AtomicReferenceFieldUpdaterTestSubclass(); 138 s.checkPrivateAccess(); 139 } 140 141 /** 142 * construction from unrelated class; package access is allowed, 143 * private access is not 144 */ 145 public void testUnrelatedClassAccess() { 146 new UnrelatedClass().checkPackageAccess(this); 147 new UnrelatedClass().checkPrivateAccess(this); 148 } 149 150 /** 151 * get returns the last value set or assigned 152 */ 153 public void testGetSet() { 154 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> a; 155 a = updaterFor("x"); 156 x = one; 157 assertSame(one, a.get(this)); 158 a.set(this, two); 159 assertSame(two, a.get(this)); 160 a.set(this, m3); 161 assertSame(m3, a.get(this)); 162 } 163 164 /** 165 * get returns the last value lazySet by same thread 166 */ 167 public void testGetLazySet() { 168 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> a; 169 a = updaterFor("x"); 170 x = one; 171 assertSame(one, a.get(this)); 172 a.lazySet(this, two); 173 assertSame(two, a.get(this)); 174 a.lazySet(this, m3); 175 assertSame(m3, a.get(this)); 176 } 177 178 /** 179 * compareAndSet succeeds in changing value if equal to expected else fails 180 */ 181 public void testCompareAndSet() { 182 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> a; 183 a = updaterFor("x"); 184 x = one; 185 assertTrue(a.compareAndSet(this, one, two)); 186 assertTrue(a.compareAndSet(this, two, m4)); 187 assertSame(m4, a.get(this)); 188 assertFalse(a.compareAndSet(this, m5, seven)); 189 assertFalse(seven == a.get(this)); 190 assertTrue(a.compareAndSet(this, m4, seven)); 191 assertSame(seven, a.get(this)); 192 } 193 194 /** 195 * compareAndSet in one thread enables another waiting for value 196 * to succeed 197 */ 198 public void testCompareAndSetInMultipleThreads() throws Exception { 199 x = one; 200 final AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> a; 201 a = updaterFor("x"); 202 203 Thread t = new Thread(new CheckedRunnable() { 204 public void realRun() { 205 while (!a.compareAndSet(AtomicReferenceFieldUpdaterTest.this, two, three)) 206 Thread.yield(); 207 }}); 208 209 t.start(); 210 assertTrue(a.compareAndSet(this, one, two)); 211 t.join(LONG_DELAY_MS); 212 assertFalse(t.isAlive()); 213 assertSame(three, a.get(this)); 214 } 215 216 /** 217 * repeated weakCompareAndSet succeeds in changing value when equal 218 * to expected 219 */ 220 public void testWeakCompareAndSet() { 221 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> a; 222 a = updaterFor("x"); 223 x = one; 224 do {} while (!a.weakCompareAndSet(this, one, two)); 225 do {} while (!a.weakCompareAndSet(this, two, m4)); 226 assertSame(m4, a.get(this)); 227 do {} while (!a.weakCompareAndSet(this, m4, seven)); 228 assertSame(seven, a.get(this)); 229 } 230 231 /** 232 * getAndSet returns previous value and sets to given value 233 */ 234 public void testGetAndSet() { 235 AtomicReferenceFieldUpdater<AtomicReferenceFieldUpdaterTest, Integer> a; 236 a = updaterFor("x"); 237 x = one; 238 assertSame(one, a.getAndSet(this, zero)); 239 assertSame(zero, a.getAndSet(this, m10)); 240 assertSame(m10, a.getAndSet(this, 1)); 241 } 242 243 } 244