mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			317 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			317 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Java
		
	
	
	
| /*
 | |
|  * Written by Doug Lea with assistance from members of JCP JSR-166
 | |
|  * Expert Group and released to the public domain, as explained at
 | |
|  * http://creativecommons.org/licenses/publicdomain
 | |
|  */
 | |
| 
 | |
| package java.util.concurrent.atomic;
 | |
| import sun.misc.Unsafe;
 | |
| import java.lang.reflect.*;
 | |
| 
 | |
| /**
 | |
|  * A reflection-based utility that enables atomic updates to
 | |
|  * designated <tt>volatile int</tt> fields of designated classes.
 | |
|  * This class is designed for use in atomic data structures in which
 | |
|  * several fields of the same node are independently subject to atomic
 | |
|  * updates.
 | |
|  *
 | |
|  * <p>Note that the guarantees of the {@code compareAndSet}
 | |
|  * method in this class are weaker than in other atomic classes.
 | |
|  * Because this class cannot ensure that all uses of the field
 | |
|  * are appropriate for purposes of atomic access, it can
 | |
|  * guarantee atomicity only with respect to other invocations of
 | |
|  * {@code compareAndSet} and {@code set} on the same updater.
 | |
|  *
 | |
|  * @since 1.5
 | |
|  * @author Doug Lea
 | |
|  * @param <T> The type of the object holding the updatable field
 | |
|  */
 | |
| public abstract class  AtomicIntegerFieldUpdater<T>  {
 | |
|     /**
 | |
|      * Creates and returns an updater for objects with the given field.
 | |
|      * The Class argument is needed to check that reflective types and
 | |
|      * generic types match.
 | |
|      *
 | |
|      * @param tclass the class of the objects holding the field
 | |
|      * @param fieldName the name of the field to be updated
 | |
|      * @return the updater
 | |
|      * @throws IllegalArgumentException if the field is not a
 | |
|      * volatile integer type
 | |
|      * @throws RuntimeException with a nested reflection-based
 | |
|      * exception if the class does not hold field or is the wrong type
 | |
|      */
 | |
|     public static <U> AtomicIntegerFieldUpdater<U> newUpdater(Class<U> tclass, String fieldName) {
 | |
|         return new AtomicIntegerFieldUpdaterImpl<U>(tclass, fieldName);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Protected do-nothing constructor for use by subclasses.
 | |
|      */
 | |
|     protected AtomicIntegerFieldUpdater() {
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically sets the field of the given object managed by this updater
 | |
|      * to the given updated value if the current value <tt>==</tt> the
 | |
|      * expected value. This method is guaranteed to be atomic with respect to
 | |
|      * other calls to <tt>compareAndSet</tt> and <tt>set</tt>, but not
 | |
|      * necessarily with respect to other changes in the field.
 | |
|      *
 | |
|      * @param obj An object whose field to conditionally set
 | |
|      * @param expect the expected value
 | |
|      * @param update the new value
 | |
|      * @return true if successful
 | |
|      * @throws ClassCastException if <tt>obj</tt> is not an instance
 | |
|      * of the class possessing the field established in the constructor
 | |
|      */
 | |
|     public abstract boolean compareAndSet(T obj, int expect, int update);
 | |
| 
 | |
|     /**
 | |
|      * Atomically sets the field of the given object managed by this updater
 | |
|      * to the given updated value if the current value <tt>==</tt> the
 | |
|      * expected value. This method is guaranteed to be atomic with respect to
 | |
|      * other calls to <tt>compareAndSet</tt> and <tt>set</tt>, but not
 | |
|      * necessarily with respect to other changes in the field.
 | |
|      * May fail spuriously and does not provide ordering guarantees,
 | |
|      * so is only rarely an appropriate alternative to <tt>compareAndSet</tt>.
 | |
|      *
 | |
|      * @param obj An object whose field to conditionally set
 | |
|      * @param expect the expected value
 | |
|      * @param update the new value
 | |
|      * @return true if successful
 | |
|      * @throws ClassCastException if <tt>obj</tt> is not an instance
 | |
|      * of the class possessing the field established in the constructor
 | |
|      */
 | |
|     public abstract boolean weakCompareAndSet(T obj, int expect, int update);
 | |
| 
 | |
|     /**
 | |
|      * Sets the field of the given object managed by this updater to the
 | |
|      * given updated value. This operation is guaranteed to act as a volatile
 | |
|      * store with respect to subsequent invocations of
 | |
|      * <tt>compareAndSet</tt>.
 | |
|      *
 | |
|      * @param obj An object whose field to set
 | |
|      * @param newValue the new value
 | |
|      */
 | |
|     public abstract void set(T obj, int newValue);
 | |
| 
 | |
|     /**
 | |
|      * Eventually sets the field of the given object managed by this
 | |
|      * updater to the given updated value.
 | |
|      *
 | |
|      * @param obj An object whose field to set
 | |
|      * @param newValue the new value
 | |
|      * @since 1.6
 | |
|      */
 | |
|     public abstract void lazySet(T obj, int newValue);
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * Gets the current value held in the field of the given object managed
 | |
|      * by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get
 | |
|      * @return the current value
 | |
|      */
 | |
|     public abstract int get(T obj);
 | |
| 
 | |
|     /**
 | |
|      * Atomically sets the field of the given object managed by this updater
 | |
|      * to the given value and returns the old value.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @param newValue the new value
 | |
|      * @return the previous value
 | |
|      */
 | |
|     public int getAndSet(T obj, int newValue) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             if (compareAndSet(obj, current, newValue))
 | |
|                 return current;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically increments by one the current value of the field of the
 | |
|      * given object managed by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @return the previous value
 | |
|      */
 | |
|     public int getAndIncrement(T obj) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             int next = current + 1;
 | |
|             if (compareAndSet(obj, current, next))
 | |
|                 return current;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically decrements by one the current value of the field of the
 | |
|      * given object managed by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @return the previous value
 | |
|      */
 | |
|     public int getAndDecrement(T obj) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             int next = current - 1;
 | |
|             if (compareAndSet(obj, current, next))
 | |
|                 return current;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically adds the given value to the current value of the field of
 | |
|      * the given object managed by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @param delta the value to add
 | |
|      * @return the previous value
 | |
|      */
 | |
|     public int getAndAdd(T obj, int delta) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             int next = current + delta;
 | |
|             if (compareAndSet(obj, current, next))
 | |
|                 return current;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically increments by one the current value of the field of the
 | |
|      * given object managed by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @return the updated value
 | |
|      */
 | |
|     public int incrementAndGet(T obj) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             int next = current + 1;
 | |
|             if (compareAndSet(obj, current, next))
 | |
|                 return next;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically decrements by one the current value of the field of the
 | |
|      * given object managed by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @return the updated value
 | |
|      */
 | |
|     public int decrementAndGet(T obj) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             int next = current - 1;
 | |
|             if (compareAndSet(obj, current, next))
 | |
|                 return next;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Atomically adds the given value to the current value of the field of
 | |
|      * the given object managed by this updater.
 | |
|      *
 | |
|      * @param obj An object whose field to get and set
 | |
|      * @param delta the value to add
 | |
|      * @return the updated value
 | |
|      */
 | |
|     public int addAndGet(T obj, int delta) {
 | |
|         for (;;) {
 | |
|             int current = get(obj);
 | |
|             int next = current + delta;
 | |
|             if (compareAndSet(obj, current, next))
 | |
|                 return next;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Standard hotspot implementation using intrinsics
 | |
|      */
 | |
|     private static class AtomicIntegerFieldUpdaterImpl<T> extends AtomicIntegerFieldUpdater<T> {
 | |
|         private static final Unsafe unsafe = Unsafe.getUnsafe();
 | |
|         private final long offset;
 | |
|         private final Class<T> tclass;
 | |
|         private final Class cclass;
 | |
| 
 | |
|         AtomicIntegerFieldUpdaterImpl(Class<T> tclass, String fieldName) {
 | |
|             Field field = null;
 | |
|             Class caller = null;
 | |
|             int modifiers = 0;
 | |
|             try {
 | |
|                 field = tclass.getDeclaredField(fieldName);
 | |
|                 caller = sun.reflect.Reflection.getCallerClass(3);
 | |
|                 modifiers = field.getModifiers();
 | |
|                 sun.reflect.misc.ReflectUtil.ensureMemberAccess(
 | |
|                     caller, tclass, null, modifiers);
 | |
|                 sun.reflect.misc.ReflectUtil.checkPackageAccess(tclass);
 | |
|             } catch(Exception ex) {
 | |
|                 throw new RuntimeException(ex);
 | |
|             }
 | |
| 
 | |
|             Class fieldt = field.getType();
 | |
|             if (fieldt != int.class)
 | |
|                 throw new IllegalArgumentException("Must be integer type");
 | |
| 
 | |
|             if (!Modifier.isVolatile(modifiers))
 | |
|                 throw new IllegalArgumentException("Must be volatile type");
 | |
| 
 | |
|             this.cclass = (Modifier.isProtected(modifiers) &&
 | |
|                            caller != tclass) ? caller : null;
 | |
|             this.tclass = tclass;
 | |
|             offset = unsafe.objectFieldOffset(field);
 | |
|         }
 | |
| 
 | |
|         private void fullCheck(T obj) {
 | |
|             if (!tclass.isInstance(obj))
 | |
|                 throw new ClassCastException();
 | |
|             if (cclass != null)
 | |
|                 ensureProtectedAccess(obj);
 | |
|         }
 | |
| 
 | |
|         public boolean compareAndSet(T obj, int expect, int update) {
 | |
|             if (obj == null || obj.getClass() != tclass || cclass != null) fullCheck(obj);
 | |
|             return unsafe.compareAndSwapInt(obj, offset, expect, update);
 | |
|         }
 | |
| 
 | |
|         public boolean weakCompareAndSet(T obj, int expect, int update) {
 | |
|             if (obj == null || obj.getClass() != tclass || cclass != null) fullCheck(obj);
 | |
|             return unsafe.compareAndSwapInt(obj, offset, expect, update);
 | |
|         }
 | |
| 
 | |
|         public void set(T obj, int newValue) {
 | |
|             if (obj == null || obj.getClass() != tclass || cclass != null) fullCheck(obj);
 | |
|             unsafe.putIntVolatile(obj, offset, newValue);
 | |
|         }
 | |
| 
 | |
|         public void lazySet(T obj, int newValue) {
 | |
|             if (obj == null || obj.getClass() != tclass || cclass != null) fullCheck(obj);
 | |
|             unsafe.putOrderedInt(obj, offset, newValue);
 | |
|         }
 | |
| 
 | |
|         public final int get(T obj) {
 | |
|             if (obj == null || obj.getClass() != tclass || cclass != null) fullCheck(obj);
 | |
|             return unsafe.getIntVolatile(obj, offset);
 | |
|         }
 | |
| 
 | |
|         private void ensureProtectedAccess(T obj) {
 | |
|             if (cclass.isInstance(obj)) {
 | |
|                 return;
 | |
|             }
 | |
|             throw new RuntimeException(
 | |
|                 new IllegalAccessException("Class " +
 | |
|                     cclass.getName() +
 | |
|                     " can not access a protected member of class " +
 | |
|                     tclass.getName() +
 | |
|                     " using an instance of " +
 | |
|                     obj.getClass().getName()
 | |
|                 )
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| }
 |