aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/ports/SkBarriers_arm.h
blob: 386294e9b1fcbde4b9583dc1471fdcc3771dbb6f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/*
 * Copyright 2014 Google Inc.
 *
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */

#ifndef SkBarriers_arm_DEFINED
#define SkBarriers_arm_DEFINED

static inline void sk_compiler_barrier() { asm volatile("" : : : "memory"); }

template <typename T>
T sk_acquire_load(T* ptr) {
    T val = *ptr;
    __sync_synchronize();  // Issue a full barrier, which is an overkill acquire barrier.
    return val;
}

template <typename T>
T sk_consume_load(T* ptr) {
    T val = *ptr;
    // Unlike acquire, consume loads (data-dependent loads) are guaranteed not to reorder on ARM.
    // No memory barrier is needed, so we just use a compiler barrier.
    // C.f. http://preshing.com/20140709/the-purpose-of-memory_order_consume-in-cpp11/
    sk_compiler_barrier();
    return val;
}

template <typename T>
void sk_release_store(T* ptr, T val) {
    __sync_synchronize();  // Issue a full barrier, which is an overkill release barrier.
    *ptr = val;
}

#endif//SkBarriers_x86_DEFINED