1 /*
2  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
3  *
4  * This code is free software; you can redistribute it and/or modify it
5  * under the terms of the GNU General Public License version 2 only, as
6  * published by the Free Software Foundation.  Oracle designates this
7  * particular file as subject to the "Classpath" exception as provided
8  * by Oracle in the LICENSE file that accompanied this code.
9  *
10  * This code is distributed in the hope that it will be useful, but WITHOUT
11  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
13  * version 2 for more details (a copy is included in the LICENSE file that
14  * accompanied this code).
15  *
16  * You should have received a copy of the GNU General Public License version
17  * 2 along with this work; if not, write to the Free Software Foundation,
18  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
19  *
20  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
21  * or visit www.oracle.com if you need additional information or have any
22  * questions.
23  */
24 
25 /*
26  * This file is available under and governed by the GNU General Public
27  * License version 2 only, as published by the Free Software Foundation.
28  * However, the following notice accompanied the original version of this
29  * file:
30  *
31  * Written by Doug Lea with assistance from members of JCP JSR-166
32  * Expert Group and released to the public domain, as explained at
33  * http://creativecommons.org/publicdomain/zero/1.0/
34  */
35 
36 package java.util.concurrent.atomic;
37 
38 import java.lang.invoke.MethodHandles;
39 import java.lang.invoke.VarHandle;
40 import java.util.function.LongBinaryOperator;
41 import java.util.function.LongUnaryOperator;
42 
43 /**
44  * A {@code long} array in which elements may be updated atomically.
45  * See the {@link VarHandle} specification for descriptions of the
46  * properties of atomic accesses.
47  * @since 1.5
48  * @author Doug Lea
49  */
50 public class AtomicLongArray implements java.io.Serializable {
51     private static final long serialVersionUID = -2308431214976778248L;
52     private static final VarHandle AA
53         = MethodHandles.arrayElementVarHandle(long[].class);
54     private final long[] array;
55 
56     /**
57      * Creates a new AtomicLongArray of the given length, with all
58      * elements initially zero.
59      *
60      * @param length the length of the array
61      */
AtomicLongArray(int length)62     public AtomicLongArray(int length) {
63         array = new long[length];
64     }
65 
66     /**
67      * Creates a new AtomicLongArray with the same length as, and
68      * all elements copied from, the given array.
69      *
70      * @param array the array to copy elements from
71      * @throws NullPointerException if array is null
72      */
AtomicLongArray(long[] array)73     public AtomicLongArray(long[] array) {
74         // Visibility guaranteed by final field guarantees
75         this.array = array.clone();
76     }
77 
78     /**
79      * Returns the length of the array.
80      *
81      * @return the length of the array
82      */
length()83     public final int length() {
84         return array.length;
85     }
86 
87     /**
88      * Returns the current value of the element at index {@code i},
89      * with memory effects as specified by {@link VarHandle#getVolatile}.
90      *
91      * @param i the index
92      * @return the current value
93      */
get(int i)94     public final long get(int i) {
95         return (long)AA.getVolatile(array, i);
96     }
97 
98     /**
99      * Sets the element at index {@code i} to {@code newValue},
100      * with memory effects as specified by {@link VarHandle#setVolatile}.
101      *
102      * @param i the index
103      * @param newValue the new value
104      */
set(int i, long newValue)105     public final void set(int i, long newValue) {
106         AA.setVolatile(array, i, newValue);
107     }
108 
109     /**
110      * Sets the element at index {@code i} to {@code newValue},
111      * with memory effects as specified by {@link VarHandle#setRelease}.
112      *
113      * @param i the index
114      * @param newValue the new value
115      * @since 1.6
116      */
lazySet(int i, long newValue)117     public final void lazySet(int i, long newValue) {
118         AA.setRelease(array, i, newValue);
119     }
120 
121     /**
122      * Atomically sets the element at index {@code i} to {@code
123      * newValue} and returns the old value,
124      * with memory effects as specified by {@link VarHandle#getAndSet}.
125      *
126      * @param i the index
127      * @param newValue the new value
128      * @return the previous value
129      */
getAndSet(int i, long newValue)130     public final long getAndSet(int i, long newValue) {
131         return (long)AA.getAndSet(array, i, newValue);
132     }
133 
134     /**
135      * Atomically sets the element at index {@code i} to {@code newValue}
136      * if the element's current value {@code == expectedValue},
137      * with memory effects as specified by {@link VarHandle#compareAndSet}.
138      *
139      * @param i the index
140      * @param expectedValue the expected value
141      * @param newValue the new value
142      * @return {@code true} if successful. False return indicates that
143      * the actual value was not equal to the expected value.
144      */
compareAndSet(int i, long expectedValue, long newValue)145     public final boolean compareAndSet(int i, long expectedValue, long newValue) {
146         return AA.compareAndSet(array, i, expectedValue, newValue);
147     }
148 
149     /**
150      * Possibly atomically sets the element at index {@code i} to
151      * {@code newValue} if the element's current value {@code == expectedValue},
152      * with memory effects as specified by {@link VarHandle#weakCompareAndSetPlain}.
153      *
154      * @deprecated This method has plain memory effects but the method
155      * name implies volatile memory effects (see methods such as
156      * {@link #compareAndExchange} and {@link #compareAndSet}).  To avoid
157      * confusion over plain or volatile memory effects it is recommended that
158      * the method {@link #weakCompareAndSetPlain} be used instead.
159      *
160      * @param i the index
161      * @param expectedValue the expected value
162      * @param newValue the new value
163      * @return {@code true} if successful
164      * @see #weakCompareAndSetPlain
165      */
166     @Deprecated(since="9")
weakCompareAndSet(int i, long expectedValue, long newValue)167     public final boolean weakCompareAndSet(int i, long expectedValue, long newValue) {
168         return AA.weakCompareAndSetPlain(array, i, expectedValue, newValue);
169     }
170 
171     /**
172      * Possibly atomically sets the element at index {@code i} to
173      * {@code newValue} if the element's current value {@code == expectedValue},
174      * with memory effects as specified by {@link VarHandle#weakCompareAndSetPlain}.
175      *
176      * @param i the index
177      * @param expectedValue the expected value
178      * @param newValue the new value
179      * @return {@code true} if successful
180      * @since 9
181      */
weakCompareAndSetPlain(int i, long expectedValue, long newValue)182     public final boolean weakCompareAndSetPlain(int i, long expectedValue, long newValue) {
183         return AA.weakCompareAndSetPlain(array, i, expectedValue, newValue);
184     }
185 
186     /**
187      * Atomically increments the value of the element at index {@code i},
188      * with memory effects as specified by {@link VarHandle#getAndAdd}.
189      *
190      * <p>Equivalent to {@code getAndAdd(i, 1)}.
191      *
192      * @param i the index
193      * @return the previous value
194      */
getAndIncrement(int i)195     public final long getAndIncrement(int i) {
196         return (long)AA.getAndAdd(array, i, 1L);
197     }
198 
199     /**
200      * Atomically decrements the value of the element at index {@code i},
201      * with memory effects as specified by {@link VarHandle#getAndAdd}.
202      *
203      * <p>Equivalent to {@code getAndAdd(i, -1)}.
204      *
205      * @param i the index
206      * @return the previous value
207      */
getAndDecrement(int i)208     public final long getAndDecrement(int i) {
209         return (long)AA.getAndAdd(array, i, -1L);
210     }
211 
212     /**
213      * Atomically adds the given value to the element at index {@code i},
214      * with memory effects as specified by {@link VarHandle#getAndAdd}.
215      *
216      * @param i the index
217      * @param delta the value to add
218      * @return the previous value
219      */
getAndAdd(int i, long delta)220     public final long getAndAdd(int i, long delta) {
221         return (long)AA.getAndAdd(array, i, delta);
222     }
223 
224     /**
225      * Atomically increments the value of the element at index {@code i},
226      * with memory effects as specified by {@link VarHandle#getAndAdd}.
227      *
228      * <p>Equivalent to {@code addAndGet(i, 1)}.
229      *
230      * @param i the index
231      * @return the updated value
232      */
incrementAndGet(int i)233     public final long incrementAndGet(int i) {
234         return (long)AA.getAndAdd(array, i, 1L) + 1L;
235     }
236 
237     /**
238      * Atomically decrements the value of the element at index {@code i},
239      * with memory effects as specified by {@link VarHandle#getAndAdd}.
240      *
241      * <p>Equivalent to {@code addAndGet(i, -1)}.
242      *
243      * @param i the index
244      * @return the updated value
245      */
decrementAndGet(int i)246     public final long decrementAndGet(int i) {
247         return (long)AA.getAndAdd(array, i, -1L) - 1L;
248     }
249 
250     /**
251      * Atomically adds the given value to the element at index {@code i},
252      * with memory effects as specified by {@link VarHandle#getAndAdd}.
253      *
254      * @param i the index
255      * @param delta the value to add
256      * @return the updated value
257      */
addAndGet(int i, long delta)258     public long addAndGet(int i, long delta) {
259         return (long)AA.getAndAdd(array, i, delta) + delta;
260     }
261 
262     /**
263      * Atomically updates (with memory effects as specified by {@link
264      * VarHandle#compareAndSet}) the element at index {@code i} with
265      * the results of applying the given function, returning the
266      * previous value. The function should be side-effect-free, since
267      * it may be re-applied when attempted updates fail due to
268      * contention among threads.
269      *
270      * @param i the index
271      * @param updateFunction a side-effect-free function
272      * @return the previous value
273      * @since 1.8
274      */
getAndUpdate(int i, LongUnaryOperator updateFunction)275     public final long getAndUpdate(int i, LongUnaryOperator updateFunction) {
276         long prev = get(i), next = 0L;
277         for (boolean haveNext = false;;) {
278             if (!haveNext)
279                 next = updateFunction.applyAsLong(prev);
280             if (weakCompareAndSetVolatile(i, prev, next))
281                 return prev;
282             haveNext = (prev == (prev = get(i)));
283         }
284     }
285 
286     /**
287      * Atomically updates (with memory effects as specified by {@link
288      * VarHandle#compareAndSet}) the element at index {@code i} with
289      * the results of applying the given function, returning the
290      * updated value. The function should be side-effect-free, since it
291      * may be re-applied when attempted updates fail due to contention
292      * among threads.
293      *
294      * @param i the index
295      * @param updateFunction a side-effect-free function
296      * @return the updated value
297      * @since 1.8
298      */
updateAndGet(int i, LongUnaryOperator updateFunction)299     public final long updateAndGet(int i, LongUnaryOperator updateFunction) {
300         long prev = get(i), next = 0L;
301         for (boolean haveNext = false;;) {
302             if (!haveNext)
303                 next = updateFunction.applyAsLong(prev);
304             if (weakCompareAndSetVolatile(i, prev, next))
305                 return next;
306             haveNext = (prev == (prev = get(i)));
307         }
308     }
309 
310     /**
311      * Atomically updates (with memory effects as specified by {@link
312      * VarHandle#compareAndSet}) the element at index {@code i} with
313      * the results of applying the given function to the current and
314      * given values, returning the previous value. The function should
315      * be side-effect-free, since it may be re-applied when attempted
316      * updates fail due to contention among threads.  The function is
317      * applied with the current value of the element at index {@code i}
318      * as its first argument, and the given update as the second
319      * argument.
320      *
321      * @param i the index
322      * @param x the update value
323      * @param accumulatorFunction a side-effect-free function of two arguments
324      * @return the previous value
325      * @since 1.8
326      */
getAndAccumulate(int i, long x, LongBinaryOperator accumulatorFunction)327     public final long getAndAccumulate(int i, long x,
328                                       LongBinaryOperator accumulatorFunction) {
329         long prev = get(i), next = 0L;
330         for (boolean haveNext = false;;) {
331             if (!haveNext)
332                 next = accumulatorFunction.applyAsLong(prev, x);
333             if (weakCompareAndSetVolatile(i, prev, next))
334                 return prev;
335             haveNext = (prev == (prev = get(i)));
336         }
337     }
338 
339     /**
340      * Atomically updates (with memory effects as specified by {@link
341      * VarHandle#compareAndSet}) the element at index {@code i} with
342      * the results of applying the given function to the current and
343      * given values, returning the updated value. The function should
344      * be side-effect-free, since it may be re-applied when attempted
345      * updates fail due to contention among threads.  The function is
346      * applied with the current value of the element at index {@code i}
347      * as its first argument, and the given update as the second
348      * argument.
349      *
350      * @param i the index
351      * @param x the update value
352      * @param accumulatorFunction a side-effect-free function of two arguments
353      * @return the updated value
354      * @since 1.8
355      */
accumulateAndGet(int i, long x, LongBinaryOperator accumulatorFunction)356     public final long accumulateAndGet(int i, long x,
357                                       LongBinaryOperator accumulatorFunction) {
358         long prev = get(i), next = 0L;
359         for (boolean haveNext = false;;) {
360             if (!haveNext)
361                 next = accumulatorFunction.applyAsLong(prev, x);
362             if (weakCompareAndSetVolatile(i, prev, next))
363                 return next;
364             haveNext = (prev == (prev = get(i)));
365         }
366     }
367 
368     /**
369      * Returns the String representation of the current values of array.
370      * @return the String representation of the current values of array
371      */
toString()372     public String toString() {
373         int iMax = array.length - 1;
374         if (iMax == -1)
375             return "[]";
376 
377         StringBuilder b = new StringBuilder();
378         b.append('[');
379         for (int i = 0; ; i++) {
380             b.append(get(i));
381             if (i == iMax)
382                 return b.append(']').toString();
383             b.append(',').append(' ');
384         }
385     }
386 
387     // jdk9
388 
389     /**
390      * Returns the current value of the element at index {@code i},
391      * with memory semantics of reading as if the variable was declared
392      * non-{@code volatile}.
393      *
394      * @param i the index
395      * @return the value
396      * @since 9
397      */
getPlain(int i)398     public final long getPlain(int i) {
399         return (long)AA.get(array, i);
400     }
401 
402     /**
403      * Sets the element at index {@code i} to {@code newValue},
404      * with memory semantics of setting as if the variable was
405      * declared non-{@code volatile} and non-{@code final}.
406      *
407      * @param i the index
408      * @param newValue the new value
409      * @since 9
410      */
setPlain(int i, long newValue)411     public final void setPlain(int i, long newValue) {
412         AA.set(array, i, newValue);
413     }
414 
415     /**
416      * Returns the current value of the element at index {@code i},
417      * with memory effects as specified by {@link VarHandle#getOpaque}.
418      *
419      * @param i the index
420      * @return the value
421      * @since 9
422      */
getOpaque(int i)423     public final long getOpaque(int i) {
424         return (long)AA.getOpaque(array, i);
425     }
426 
427     /**
428      * Sets the element at index {@code i} to {@code newValue},
429      * with memory effects as specified by {@link VarHandle#setOpaque}.
430      *
431      * @param i the index
432      * @param newValue the new value
433      * @since 9
434      */
setOpaque(int i, long newValue)435     public final void setOpaque(int i, long newValue) {
436         AA.setOpaque(array, i, newValue);
437     }
438 
439     /**
440      * Returns the current value of the element at index {@code i},
441      * with memory effects as specified by {@link VarHandle#getAcquire}.
442      *
443      * @param i the index
444      * @return the value
445      * @since 9
446      */
getAcquire(int i)447     public final long getAcquire(int i) {
448         return (long)AA.getAcquire(array, i);
449     }
450 
451     /**
452      * Sets the element at index {@code i} to {@code newValue},
453      * with memory effects as specified by {@link VarHandle#setRelease}.
454      *
455      * @param i the index
456      * @param newValue the new value
457      * @since 9
458      */
setRelease(int i, long newValue)459     public final void setRelease(int i, long newValue) {
460         AA.setRelease(array, i, newValue);
461     }
462 
463     /**
464      * Atomically sets the element at index {@code i} to {@code newValue}
465      * if the element's current value, referred to as the <em>witness
466      * value</em>, {@code == expectedValue},
467      * with memory effects as specified by
468      * {@link VarHandle#compareAndExchange}.
469      *
470      * @param i the index
471      * @param expectedValue the expected value
472      * @param newValue the new value
473      * @return the witness value, which will be the same as the
474      * expected value if successful
475      * @since 9
476      */
compareAndExchange(int i, long expectedValue, long newValue)477     public final long compareAndExchange(int i, long expectedValue, long newValue) {
478         return (long)AA.compareAndExchange(array, i, expectedValue, newValue);
479     }
480 
481     /**
482      * Atomically sets the element at index {@code i} to {@code newValue}
483      * if the element's current value, referred to as the <em>witness
484      * value</em>, {@code == expectedValue},
485      * with memory effects as specified by
486      * {@link VarHandle#compareAndExchangeAcquire}.
487      *
488      * @param i the index
489      * @param expectedValue the expected value
490      * @param newValue the new value
491      * @return the witness value, which will be the same as the
492      * expected value if successful
493      * @since 9
494      */
compareAndExchangeAcquire(int i, long expectedValue, long newValue)495     public final long compareAndExchangeAcquire(int i, long expectedValue, long newValue) {
496         return (long)AA.compareAndExchangeAcquire(array, i, expectedValue, newValue);
497     }
498 
499     /**
500      * Atomically sets the element at index {@code i} to {@code newValue}
501      * if the element's current value, referred to as the <em>witness
502      * value</em>, {@code == expectedValue},
503      * with memory effects as specified by
504      * {@link VarHandle#compareAndExchangeRelease}.
505      *
506      * @param i the index
507      * @param expectedValue the expected value
508      * @param newValue the new value
509      * @return the witness value, which will be the same as the
510      * expected value if successful
511      * @since 9
512      */
compareAndExchangeRelease(int i, long expectedValue, long newValue)513     public final long compareAndExchangeRelease(int i, long expectedValue, long newValue) {
514         return (long)AA.compareAndExchangeRelease(array, i, expectedValue, newValue);
515     }
516 
517     /**
518      * Possibly atomically sets the element at index {@code i} to
519      * {@code newValue} if the element's current value {@code == expectedValue},
520      * with memory effects as specified by
521      * {@link VarHandle#weakCompareAndSet}.
522      *
523      * @param i the index
524      * @param expectedValue the expected value
525      * @param newValue the new value
526      * @return {@code true} if successful
527      * @since 9
528      */
weakCompareAndSetVolatile(int i, long expectedValue, long newValue)529     public final boolean weakCompareAndSetVolatile(int i, long expectedValue, long newValue) {
530         return AA.weakCompareAndSet(array, i, expectedValue, newValue);
531     }
532 
533     /**
534      * Possibly atomically sets the element at index {@code i} to
535      * {@code newValue} if the element's current value {@code == expectedValue},
536      * with memory effects as specified by
537      * {@link VarHandle#weakCompareAndSetAcquire}.
538      *
539      * @param i the index
540      * @param expectedValue the expected value
541      * @param newValue the new value
542      * @return {@code true} if successful
543      * @since 9
544      */
weakCompareAndSetAcquire(int i, long expectedValue, long newValue)545     public final boolean weakCompareAndSetAcquire(int i, long expectedValue, long newValue) {
546         return AA.weakCompareAndSetAcquire(array, i, expectedValue, newValue);
547     }
548 
549     /**
550      * Possibly atomically sets the element at index {@code i} to
551      * {@code newValue} if the element's current value {@code == expectedValue},
552      * with memory effects as specified by
553      * {@link VarHandle#weakCompareAndSetRelease}.
554      *
555      * @param i the index
556      * @param expectedValue the expected value
557      * @param newValue the new value
558      * @return {@code true} if successful
559      * @since 9
560      */
weakCompareAndSetRelease(int i, long expectedValue, long newValue)561     public final boolean weakCompareAndSetRelease(int i, long expectedValue, long newValue) {
562         return AA.weakCompareAndSetRelease(array, i, expectedValue, newValue);
563     }
564 
565 }
566