167ab875fSJohannes Doerfert //===- Synchronization.cpp - OpenMP Device synchronization API ---- c++ -*-===//
267ab875fSJohannes Doerfert //
367ab875fSJohannes Doerfert // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
467ab875fSJohannes Doerfert // See https://llvm.org/LICENSE.txt for license information.
567ab875fSJohannes Doerfert // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
667ab875fSJohannes Doerfert //
767ab875fSJohannes Doerfert //===----------------------------------------------------------------------===//
867ab875fSJohannes Doerfert //
967ab875fSJohannes Doerfert // Include all synchronization.
1067ab875fSJohannes Doerfert //
1167ab875fSJohannes Doerfert //===----------------------------------------------------------------------===//
1267ab875fSJohannes Doerfert 
1367ab875fSJohannes Doerfert #include "Synchronization.h"
1467ab875fSJohannes Doerfert 
1567ab875fSJohannes Doerfert #include "Debug.h"
1667ab875fSJohannes Doerfert #include "Interface.h"
1767ab875fSJohannes Doerfert #include "Mapping.h"
1867ab875fSJohannes Doerfert #include "State.h"
1967ab875fSJohannes Doerfert #include "Types.h"
2067ab875fSJohannes Doerfert #include "Utils.h"
2167ab875fSJohannes Doerfert 
22b4f8443dSJoseph Huber #pragma omp begin declare target device_type(nohost)
2367ab875fSJohannes Doerfert 
2467ab875fSJohannes Doerfert using namespace _OMP;
2567ab875fSJohannes Doerfert 
2667ab875fSJohannes Doerfert namespace impl {
2767ab875fSJohannes Doerfert 
2867ab875fSJohannes Doerfert /// Atomics
2967ab875fSJohannes Doerfert ///
3067ab875fSJohannes Doerfert ///{
3167ab875fSJohannes Doerfert /// NOTE: This function needs to be implemented by every target.
3267ab875fSJohannes Doerfert uint32_t atomicInc(uint32_t *Address, uint32_t Val, int Ordering);
3367ab875fSJohannes Doerfert 
atomicLoad(uint32_t * Address,int Ordering)347272982eSJon Chesterfield uint32_t atomicLoad(uint32_t *Address, int Ordering) {
3567ab875fSJohannes Doerfert   return __atomic_fetch_add(Address, 0U, __ATOMIC_SEQ_CST);
3667ab875fSJohannes Doerfert }
3767ab875fSJohannes Doerfert 
atomicStore(uint32_t * Address,uint32_t Val,int Ordering)387272982eSJon Chesterfield void atomicStore(uint32_t *Address, uint32_t Val, int Ordering) {
397272982eSJon Chesterfield   __atomic_store_n(Address, Val, Ordering);
407272982eSJon Chesterfield }
417272982eSJon Chesterfield 
atomicAdd(uint32_t * Address,uint32_t Val,int Ordering)4267ab875fSJohannes Doerfert uint32_t atomicAdd(uint32_t *Address, uint32_t Val, int Ordering) {
4367ab875fSJohannes Doerfert   return __atomic_fetch_add(Address, Val, Ordering);
4467ab875fSJohannes Doerfert }
atomicMax(uint32_t * Address,uint32_t Val,int Ordering)4567ab875fSJohannes Doerfert uint32_t atomicMax(uint32_t *Address, uint32_t Val, int Ordering) {
4667ab875fSJohannes Doerfert   return __atomic_fetch_max(Address, Val, Ordering);
4767ab875fSJohannes Doerfert }
4867ab875fSJohannes Doerfert 
atomicExchange(uint32_t * Address,uint32_t Val,int Ordering)4967ab875fSJohannes Doerfert uint32_t atomicExchange(uint32_t *Address, uint32_t Val, int Ordering) {
5067ab875fSJohannes Doerfert   uint32_t R;
5167ab875fSJohannes Doerfert   __atomic_exchange(Address, &Val, &R, Ordering);
5267ab875fSJohannes Doerfert   return R;
5367ab875fSJohannes Doerfert }
atomicCAS(uint32_t * Address,uint32_t Compare,uint32_t Val,int Ordering)5467ab875fSJohannes Doerfert uint32_t atomicCAS(uint32_t *Address, uint32_t Compare, uint32_t Val,
5567ab875fSJohannes Doerfert                    int Ordering) {
5667ab875fSJohannes Doerfert   (void)__atomic_compare_exchange(Address, &Compare, &Val, false, Ordering,
5767ab875fSJohannes Doerfert                                   Ordering);
5867ab875fSJohannes Doerfert   return Compare;
5967ab875fSJohannes Doerfert }
6067ab875fSJohannes Doerfert 
atomicAdd(uint64_t * Address,uint64_t Val,int Ordering)6167ab875fSJohannes Doerfert uint64_t atomicAdd(uint64_t *Address, uint64_t Val, int Ordering) {
6267ab875fSJohannes Doerfert   return __atomic_fetch_add(Address, Val, Ordering);
6367ab875fSJohannes Doerfert }
6467ab875fSJohannes Doerfert ///}
6567ab875fSJohannes Doerfert 
66b4f8443dSJoseph Huber // Forward declarations defined to be defined for AMDGCN and NVPTX.
67b4f8443dSJoseph Huber uint32_t atomicInc(uint32_t *A, uint32_t V, int Ordering);
68b4f8443dSJoseph Huber void namedBarrierInit();
69b4f8443dSJoseph Huber void namedBarrier();
70b4f8443dSJoseph Huber void fenceTeam(int Ordering);
71b4f8443dSJoseph Huber void fenceKernel(int Ordering);
72b4f8443dSJoseph Huber void fenceSystem(int Ordering);
73b4f8443dSJoseph Huber void syncWarp(__kmpc_impl_lanemask_t);
74b4f8443dSJoseph Huber void syncThreads();
syncThreadsAligned()75b4f8443dSJoseph Huber void syncThreadsAligned() { syncThreads(); }
76b4f8443dSJoseph Huber void unsetLock(omp_lock_t *);
77b4f8443dSJoseph Huber int testLock(omp_lock_t *);
78b4f8443dSJoseph Huber void initLock(omp_lock_t *);
79b4f8443dSJoseph Huber void destroyLock(omp_lock_t *);
80b4f8443dSJoseph Huber void setLock(omp_lock_t *);
81b4f8443dSJoseph Huber 
8267ab875fSJohannes Doerfert /// AMDGCN Implementation
8367ab875fSJohannes Doerfert ///
8467ab875fSJohannes Doerfert ///{
8567ab875fSJohannes Doerfert #pragma omp begin declare variant match(device = {arch(amdgcn)})
8667ab875fSJohannes Doerfert 
atomicInc(uint32_t * A,uint32_t V,int Ordering)874d50803cSJon Chesterfield uint32_t atomicInc(uint32_t *A, uint32_t V, int Ordering) {
884d50803cSJon Chesterfield   // builtin_amdgcn_atomic_inc32 should expand to this switch when
894d50803cSJon Chesterfield   // passed a runtime value, but does not do so yet. Workaround here.
904d50803cSJon Chesterfield   switch (Ordering) {
914d50803cSJon Chesterfield   default:
924d50803cSJon Chesterfield     __builtin_unreachable();
934d50803cSJon Chesterfield   case __ATOMIC_RELAXED:
944d50803cSJon Chesterfield     return __builtin_amdgcn_atomic_inc32(A, V, __ATOMIC_RELAXED, "");
954d50803cSJon Chesterfield   case __ATOMIC_ACQUIRE:
964d50803cSJon Chesterfield     return __builtin_amdgcn_atomic_inc32(A, V, __ATOMIC_ACQUIRE, "");
974d50803cSJon Chesterfield   case __ATOMIC_RELEASE:
984d50803cSJon Chesterfield     return __builtin_amdgcn_atomic_inc32(A, V, __ATOMIC_RELEASE, "");
994d50803cSJon Chesterfield   case __ATOMIC_ACQ_REL:
1004d50803cSJon Chesterfield     return __builtin_amdgcn_atomic_inc32(A, V, __ATOMIC_ACQ_REL, "");
1014d50803cSJon Chesterfield   case __ATOMIC_SEQ_CST:
1024d50803cSJon Chesterfield     return __builtin_amdgcn_atomic_inc32(A, V, __ATOMIC_SEQ_CST, "");
1034d50803cSJon Chesterfield   }
10467ab875fSJohannes Doerfert }
10567ab875fSJohannes Doerfert 
1067272982eSJon Chesterfield uint32_t SHARED(namedBarrierTracker);
10767ab875fSJohannes Doerfert 
namedBarrierInit()10867ab875fSJohannes Doerfert void namedBarrierInit() {
10967ab875fSJohannes Doerfert   // Don't have global ctors, and shared memory is not zero init
11067ab875fSJohannes Doerfert   atomic::store(&namedBarrierTracker, 0u, __ATOMIC_RELEASE);
11167ab875fSJohannes Doerfert }
11267ab875fSJohannes Doerfert 
namedBarrier()11367ab875fSJohannes Doerfert void namedBarrier() {
11467ab875fSJohannes Doerfert   uint32_t NumThreads = omp_get_num_threads();
11567ab875fSJohannes Doerfert   // assert(NumThreads % 32 == 0);
11667ab875fSJohannes Doerfert 
1177272982eSJon Chesterfield   uint32_t WarpSize = mapping::getWarpSize();
11867ab875fSJohannes Doerfert   uint32_t NumWaves = NumThreads / WarpSize;
11967ab875fSJohannes Doerfert 
12067ab875fSJohannes Doerfert   fence::team(__ATOMIC_ACQUIRE);
12167ab875fSJohannes Doerfert 
12267ab875fSJohannes Doerfert   // named barrier implementation for amdgcn.
12367ab875fSJohannes Doerfert   // Uses two 16 bit unsigned counters. One for the number of waves to have
12467ab875fSJohannes Doerfert   // reached the barrier, and one to count how many times the barrier has been
12567ab875fSJohannes Doerfert   // passed. These are packed in a single atomically accessed 32 bit integer.
12667ab875fSJohannes Doerfert   // Low bits for the number of waves, assumed zero before this call.
12767ab875fSJohannes Doerfert   // High bits to count the number of times the barrier has been passed.
12867ab875fSJohannes Doerfert 
12967ab875fSJohannes Doerfert   // precondition: NumWaves != 0;
13067ab875fSJohannes Doerfert   // invariant: NumWaves * WarpSize == NumThreads;
13167ab875fSJohannes Doerfert   // precondition: NumWaves < 0xffffu;
13267ab875fSJohannes Doerfert 
13367ab875fSJohannes Doerfert   // Increment the low 16 bits once, using the lowest active thread.
13467ab875fSJohannes Doerfert   if (mapping::isLeaderInWarp()) {
13567ab875fSJohannes Doerfert     uint32_t load = atomic::add(&namedBarrierTracker, 1,
13667ab875fSJohannes Doerfert                                 __ATOMIC_RELAXED); // commutative
13767ab875fSJohannes Doerfert 
13867ab875fSJohannes Doerfert     // Record the number of times the barrier has been passed
13967ab875fSJohannes Doerfert     uint32_t generation = load & 0xffff0000u;
14067ab875fSJohannes Doerfert 
14167ab875fSJohannes Doerfert     if ((load & 0x0000ffffu) == (NumWaves - 1)) {
14267ab875fSJohannes Doerfert       // Reached NumWaves in low bits so this is the last wave.
14367ab875fSJohannes Doerfert       // Set low bits to zero and increment high bits
14467ab875fSJohannes Doerfert       load += 0x00010000u; // wrap is safe
14567ab875fSJohannes Doerfert       load &= 0xffff0000u; // because bits zeroed second
14667ab875fSJohannes Doerfert 
14767ab875fSJohannes Doerfert       // Reset the wave counter and release the waiting waves
14867ab875fSJohannes Doerfert       atomic::store(&namedBarrierTracker, load, __ATOMIC_RELAXED);
14967ab875fSJohannes Doerfert     } else {
15067ab875fSJohannes Doerfert       // more waves still to go, spin until generation counter changes
15167ab875fSJohannes Doerfert       do {
15267ab875fSJohannes Doerfert         __builtin_amdgcn_s_sleep(0);
1537272982eSJon Chesterfield         load = atomic::load(&namedBarrierTracker, __ATOMIC_RELAXED);
15467ab875fSJohannes Doerfert       } while ((load & 0xffff0000u) == generation);
15567ab875fSJohannes Doerfert     }
15667ab875fSJohannes Doerfert   }
15767ab875fSJohannes Doerfert   fence::team(__ATOMIC_RELEASE);
15867ab875fSJohannes Doerfert }
15967ab875fSJohannes Doerfert 
1604d50803cSJon Chesterfield // sema checking of amdgcn_fence is aggressive. Intention is to patch clang
1614d50803cSJon Chesterfield // so that it is usable within a template environment and so that a runtime
1624d50803cSJon Chesterfield // value of the memory order is expanded to this switch within clang/llvm.
fenceTeam(int Ordering)1634d50803cSJon Chesterfield void fenceTeam(int Ordering) {
1644d50803cSJon Chesterfield   switch (Ordering) {
1654d50803cSJon Chesterfield   default:
1664d50803cSJon Chesterfield     __builtin_unreachable();
1674d50803cSJon Chesterfield   case __ATOMIC_ACQUIRE:
1684d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_ACQUIRE, "workgroup");
1694d50803cSJon Chesterfield   case __ATOMIC_RELEASE:
1704d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_RELEASE, "workgroup");
1714d50803cSJon Chesterfield   case __ATOMIC_ACQ_REL:
1724d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_ACQ_REL, "workgroup");
1734d50803cSJon Chesterfield   case __ATOMIC_SEQ_CST:
1744d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_SEQ_CST, "workgroup");
1754d50803cSJon Chesterfield   }
1764d50803cSJon Chesterfield }
fenceKernel(int Ordering)1774d50803cSJon Chesterfield void fenceKernel(int Ordering) {
1784d50803cSJon Chesterfield   switch (Ordering) {
1794d50803cSJon Chesterfield   default:
1804d50803cSJon Chesterfield     __builtin_unreachable();
1814d50803cSJon Chesterfield   case __ATOMIC_ACQUIRE:
1824d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_ACQUIRE, "agent");
1834d50803cSJon Chesterfield   case __ATOMIC_RELEASE:
1844d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_RELEASE, "agent");
1854d50803cSJon Chesterfield   case __ATOMIC_ACQ_REL:
1864d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_ACQ_REL, "agent");
1874d50803cSJon Chesterfield   case __ATOMIC_SEQ_CST:
1884d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_SEQ_CST, "agent");
1894d50803cSJon Chesterfield   }
1904d50803cSJon Chesterfield }
fenceSystem(int Ordering)1914d50803cSJon Chesterfield void fenceSystem(int Ordering) {
1924d50803cSJon Chesterfield   switch (Ordering) {
1934d50803cSJon Chesterfield   default:
1944d50803cSJon Chesterfield     __builtin_unreachable();
1954d50803cSJon Chesterfield   case __ATOMIC_ACQUIRE:
1964d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_ACQUIRE, "");
1974d50803cSJon Chesterfield   case __ATOMIC_RELEASE:
1984d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_RELEASE, "");
1994d50803cSJon Chesterfield   case __ATOMIC_ACQ_REL:
2004d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_ACQ_REL, "");
2014d50803cSJon Chesterfield   case __ATOMIC_SEQ_CST:
2024d50803cSJon Chesterfield     return __builtin_amdgcn_fence(__ATOMIC_SEQ_CST, "");
2034d50803cSJon Chesterfield   }
2044d50803cSJon Chesterfield }
2054d50803cSJon Chesterfield 
syncWarp(__kmpc_impl_lanemask_t)20667ab875fSJohannes Doerfert void syncWarp(__kmpc_impl_lanemask_t) {
20767ab875fSJohannes Doerfert   // AMDGCN doesn't need to sync threads in a warp
20867ab875fSJohannes Doerfert }
20967ab875fSJohannes Doerfert 
syncThreads()21067ab875fSJohannes Doerfert void syncThreads() { __builtin_amdgcn_s_barrier(); }
syncThreadsAligned()211cf37a94cSJon Chesterfield void syncThreadsAligned() { syncThreads(); }
21267ab875fSJohannes Doerfert 
2134d50803cSJon Chesterfield // TODO: Don't have wavefront lane locks. Possibly can't have them.
unsetLock(omp_lock_t *)2144d50803cSJon Chesterfield void unsetLock(omp_lock_t *) { __builtin_trap(); }
testLock(omp_lock_t *)2154d50803cSJon Chesterfield int testLock(omp_lock_t *) { __builtin_trap(); }
initLock(omp_lock_t *)2164d50803cSJon Chesterfield void initLock(omp_lock_t *) { __builtin_trap(); }
destroyLock(omp_lock_t *)2174d50803cSJon Chesterfield void destroyLock(omp_lock_t *) { __builtin_trap(); }
setLock(omp_lock_t *)2184d50803cSJon Chesterfield void setLock(omp_lock_t *) { __builtin_trap(); }
21967ab875fSJohannes Doerfert 
22067ab875fSJohannes Doerfert #pragma omp end declare variant
22167ab875fSJohannes Doerfert ///}
22267ab875fSJohannes Doerfert 
22367ab875fSJohannes Doerfert /// NVPTX Implementation
22467ab875fSJohannes Doerfert ///
22567ab875fSJohannes Doerfert ///{
22667ab875fSJohannes Doerfert #pragma omp begin declare variant match(                                       \
22767ab875fSJohannes Doerfert     device = {arch(nvptx, nvptx64)}, implementation = {extension(match_any)})
22867ab875fSJohannes Doerfert 
atomicInc(uint32_t * Address,uint32_t Val,int Ordering)22967ab875fSJohannes Doerfert uint32_t atomicInc(uint32_t *Address, uint32_t Val, int Ordering) {
23067ab875fSJohannes Doerfert   return __nvvm_atom_inc_gen_ui(Address, Val);
23167ab875fSJohannes Doerfert }
23267ab875fSJohannes Doerfert 
namedBarrierInit()23367ab875fSJohannes Doerfert void namedBarrierInit() {}
23467ab875fSJohannes Doerfert 
namedBarrier()23567ab875fSJohannes Doerfert void namedBarrier() {
23667ab875fSJohannes Doerfert   uint32_t NumThreads = omp_get_num_threads();
23767ab875fSJohannes Doerfert   ASSERT(NumThreads % 32 == 0);
23867ab875fSJohannes Doerfert 
23967ab875fSJohannes Doerfert   // The named barrier for active parallel threads of a team in an L1 parallel
24067ab875fSJohannes Doerfert   // region to synchronize with each other.
24167ab875fSJohannes Doerfert   constexpr int BarrierNo = 7;
24267ab875fSJohannes Doerfert   asm volatile("barrier.sync %0, %1;"
24367ab875fSJohannes Doerfert                :
24467ab875fSJohannes Doerfert                : "r"(BarrierNo), "r"(NumThreads)
24567ab875fSJohannes Doerfert                : "memory");
24667ab875fSJohannes Doerfert }
24767ab875fSJohannes Doerfert 
fenceTeam(int)24867ab875fSJohannes Doerfert void fenceTeam(int) { __nvvm_membar_cta(); }
24967ab875fSJohannes Doerfert 
fenceKernel(int)25067ab875fSJohannes Doerfert void fenceKernel(int) { __nvvm_membar_gl(); }
25167ab875fSJohannes Doerfert 
fenceSystem(int)25267ab875fSJohannes Doerfert void fenceSystem(int) { __nvvm_membar_sys(); }
25367ab875fSJohannes Doerfert 
syncWarp(__kmpc_impl_lanemask_t Mask)25467ab875fSJohannes Doerfert void syncWarp(__kmpc_impl_lanemask_t Mask) { __nvvm_bar_warp_sync(Mask); }
25567ab875fSJohannes Doerfert 
syncThreads()25667ab875fSJohannes Doerfert void syncThreads() {
25767ab875fSJohannes Doerfert   constexpr int BarrierNo = 8;
25867ab875fSJohannes Doerfert   asm volatile("barrier.sync %0;" : : "r"(BarrierNo) : "memory");
25967ab875fSJohannes Doerfert }
26067ab875fSJohannes Doerfert 
syncThreadsAligned()261b16aadf0SJohannes Doerfert void syncThreadsAligned() { __syncthreads(); }
262b16aadf0SJohannes Doerfert 
26367ab875fSJohannes Doerfert constexpr uint32_t OMP_SPIN = 1000;
26467ab875fSJohannes Doerfert constexpr uint32_t UNSET = 0;
26567ab875fSJohannes Doerfert constexpr uint32_t SET = 1;
26667ab875fSJohannes Doerfert 
26767ab875fSJohannes Doerfert // TODO: This seems to hide a bug in the declare variant handling. If it is
26867ab875fSJohannes Doerfert // called before it is defined
26967ab875fSJohannes Doerfert //       here the overload won't happen. Investigate lalter!
unsetLock(omp_lock_t * Lock)27067ab875fSJohannes Doerfert void unsetLock(omp_lock_t *Lock) {
27167ab875fSJohannes Doerfert   (void)atomicExchange((uint32_t *)Lock, UNSET, __ATOMIC_SEQ_CST);
27267ab875fSJohannes Doerfert }
27367ab875fSJohannes Doerfert 
testLock(omp_lock_t * Lock)27467ab875fSJohannes Doerfert int testLock(omp_lock_t *Lock) {
27567ab875fSJohannes Doerfert   return atomicAdd((uint32_t *)Lock, 0u, __ATOMIC_SEQ_CST);
27667ab875fSJohannes Doerfert }
27767ab875fSJohannes Doerfert 
initLock(omp_lock_t * Lock)27867ab875fSJohannes Doerfert void initLock(omp_lock_t *Lock) { unsetLock(Lock); }
27967ab875fSJohannes Doerfert 
destroyLock(omp_lock_t * Lock)2807272982eSJon Chesterfield void destroyLock(omp_lock_t *Lock) { unsetLock(Lock); }
28167ab875fSJohannes Doerfert 
setLock(omp_lock_t * Lock)28267ab875fSJohannes Doerfert void setLock(omp_lock_t *Lock) {
28367ab875fSJohannes Doerfert   // TODO: not sure spinning is a good idea here..
28467ab875fSJohannes Doerfert   while (atomicCAS((uint32_t *)Lock, UNSET, SET, __ATOMIC_SEQ_CST) != UNSET) {
28567ab875fSJohannes Doerfert     int32_t start = __nvvm_read_ptx_sreg_clock();
28667ab875fSJohannes Doerfert     int32_t now;
28767ab875fSJohannes Doerfert     for (;;) {
28867ab875fSJohannes Doerfert       now = __nvvm_read_ptx_sreg_clock();
28967ab875fSJohannes Doerfert       int32_t cycles = now > start ? now - start : now + (0xffffffff - start);
29067ab875fSJohannes Doerfert       if (cycles >= OMP_SPIN * mapping::getBlockId()) {
29167ab875fSJohannes Doerfert         break;
29267ab875fSJohannes Doerfert       }
29367ab875fSJohannes Doerfert     }
29467ab875fSJohannes Doerfert   } // wait for 0 to be the read value
29567ab875fSJohannes Doerfert }
29667ab875fSJohannes Doerfert 
29767ab875fSJohannes Doerfert #pragma omp end declare variant
29867ab875fSJohannes Doerfert ///}
29967ab875fSJohannes Doerfert 
30067ab875fSJohannes Doerfert } // namespace impl
30167ab875fSJohannes Doerfert 
init(bool IsSPMD)30267ab875fSJohannes Doerfert void synchronize::init(bool IsSPMD) {
30367ab875fSJohannes Doerfert   if (!IsSPMD)
30467ab875fSJohannes Doerfert     impl::namedBarrierInit();
30567ab875fSJohannes Doerfert }
30667ab875fSJohannes Doerfert 
warp(LaneMaskTy Mask)30767ab875fSJohannes Doerfert void synchronize::warp(LaneMaskTy Mask) { impl::syncWarp(Mask); }
30867ab875fSJohannes Doerfert 
threads()30967ab875fSJohannes Doerfert void synchronize::threads() { impl::syncThreads(); }
31067ab875fSJohannes Doerfert 
threadsAligned()311b16aadf0SJohannes Doerfert void synchronize::threadsAligned() { impl::syncThreadsAligned(); }
312b16aadf0SJohannes Doerfert 
team(int Ordering)31367ab875fSJohannes Doerfert void fence::team(int Ordering) { impl::fenceTeam(Ordering); }
31467ab875fSJohannes Doerfert 
kernel(int Ordering)31567ab875fSJohannes Doerfert void fence::kernel(int Ordering) { impl::fenceKernel(Ordering); }
31667ab875fSJohannes Doerfert 
system(int Ordering)31767ab875fSJohannes Doerfert void fence::system(int Ordering) { impl::fenceSystem(Ordering); }
31867ab875fSJohannes Doerfert 
load(uint32_t * Addr,int Ordering)3197272982eSJon Chesterfield uint32_t atomic::load(uint32_t *Addr, int Ordering) {
3207272982eSJon Chesterfield   return impl::atomicLoad(Addr, Ordering);
3217272982eSJon Chesterfield }
3227272982eSJon Chesterfield 
store(uint32_t * Addr,uint32_t V,int Ordering)3237272982eSJon Chesterfield void atomic::store(uint32_t *Addr, uint32_t V, int Ordering) {
3247272982eSJon Chesterfield   impl::atomicStore(Addr, V, Ordering);
32567ab875fSJohannes Doerfert }
32667ab875fSJohannes Doerfert 
inc(uint32_t * Addr,uint32_t V,int Ordering)32767ab875fSJohannes Doerfert uint32_t atomic::inc(uint32_t *Addr, uint32_t V, int Ordering) {
32867ab875fSJohannes Doerfert   return impl::atomicInc(Addr, V, Ordering);
32967ab875fSJohannes Doerfert }
33067ab875fSJohannes Doerfert 
add(uint32_t * Addr,uint32_t V,int Ordering)33167ab875fSJohannes Doerfert uint32_t atomic::add(uint32_t *Addr, uint32_t V, int Ordering) {
33267ab875fSJohannes Doerfert   return impl::atomicAdd(Addr, V, Ordering);
33367ab875fSJohannes Doerfert }
33467ab875fSJohannes Doerfert 
add(uint64_t * Addr,uint64_t V,int Ordering)33567ab875fSJohannes Doerfert uint64_t atomic::add(uint64_t *Addr, uint64_t V, int Ordering) {
33667ab875fSJohannes Doerfert   return impl::atomicAdd(Addr, V, Ordering);
33767ab875fSJohannes Doerfert }
33867ab875fSJohannes Doerfert 
33967ab875fSJohannes Doerfert extern "C" {
__kmpc_ordered(IdentTy * Loc,int32_t TId)34074f91741SJoseph Huber void __kmpc_ordered(IdentTy *Loc, int32_t TId) { FunctionTracingRAII(); }
34167ab875fSJohannes Doerfert 
__kmpc_end_ordered(IdentTy * Loc,int32_t TId)34274f91741SJoseph Huber void __kmpc_end_ordered(IdentTy *Loc, int32_t TId) { FunctionTracingRAII(); }
34367ab875fSJohannes Doerfert 
__kmpc_cancel_barrier(IdentTy * Loc,int32_t TId)34467ab875fSJohannes Doerfert int32_t __kmpc_cancel_barrier(IdentTy *Loc, int32_t TId) {
34574f91741SJoseph Huber   FunctionTracingRAII();
34667ab875fSJohannes Doerfert   __kmpc_barrier(Loc, TId);
34767ab875fSJohannes Doerfert   return 0;
34867ab875fSJohannes Doerfert }
34967ab875fSJohannes Doerfert 
__kmpc_barrier(IdentTy * Loc,int32_t TId)35067ab875fSJohannes Doerfert void __kmpc_barrier(IdentTy *Loc, int32_t TId) {
35174f91741SJoseph Huber   FunctionTracingRAII();
35267ab875fSJohannes Doerfert   if (mapping::isMainThreadInGenericMode())
35367ab875fSJohannes Doerfert     return __kmpc_flush(Loc);
35467ab875fSJohannes Doerfert 
355f52927c1SJon Chesterfield   if (mapping::isSPMDMode())
356f52927c1SJon Chesterfield     return __kmpc_barrier_simple_spmd(Loc, TId);
357f52927c1SJon Chesterfield 
35867ab875fSJohannes Doerfert   impl::namedBarrier();
35967ab875fSJohannes Doerfert }
36067ab875fSJohannes Doerfert 
__kmpc_barrier_simple_spmd(IdentTy * Loc,int32_t TId)361*fd8fd9e5SJoseph Huber __attribute__((noinline)) void __kmpc_barrier_simple_spmd(IdentTy *Loc,
362*fd8fd9e5SJoseph Huber                                                           int32_t TId) {
36374f91741SJoseph Huber   FunctionTracingRAII();
364b16aadf0SJohannes Doerfert   synchronize::threadsAligned();
36567ab875fSJohannes Doerfert }
36667ab875fSJohannes Doerfert 
__kmpc_barrier_simple_generic(IdentTy * Loc,int32_t TId)367*fd8fd9e5SJoseph Huber __attribute__((noinline)) void __kmpc_barrier_simple_generic(IdentTy *Loc,
368*fd8fd9e5SJoseph Huber                                                              int32_t TId) {
36973720c80SJohannes Doerfert   FunctionTracingRAII();
37073720c80SJohannes Doerfert   synchronize::threads();
37173720c80SJohannes Doerfert }
37273720c80SJohannes Doerfert 
__kmpc_master(IdentTy * Loc,int32_t TId)37367ab875fSJohannes Doerfert int32_t __kmpc_master(IdentTy *Loc, int32_t TId) {
37474f91741SJoseph Huber   FunctionTracingRAII();
37567ab875fSJohannes Doerfert   return omp_get_team_num() == 0;
37667ab875fSJohannes Doerfert }
37767ab875fSJohannes Doerfert 
__kmpc_end_master(IdentTy * Loc,int32_t TId)37874f91741SJoseph Huber void __kmpc_end_master(IdentTy *Loc, int32_t TId) { FunctionTracingRAII(); }
37967ab875fSJohannes Doerfert 
__kmpc_single(IdentTy * Loc,int32_t TId)38067ab875fSJohannes Doerfert int32_t __kmpc_single(IdentTy *Loc, int32_t TId) {
38174f91741SJoseph Huber   FunctionTracingRAII();
38267ab875fSJohannes Doerfert   return __kmpc_master(Loc, TId);
38367ab875fSJohannes Doerfert }
38467ab875fSJohannes Doerfert 
__kmpc_end_single(IdentTy * Loc,int32_t TId)38567ab875fSJohannes Doerfert void __kmpc_end_single(IdentTy *Loc, int32_t TId) {
38674f91741SJoseph Huber   FunctionTracingRAII();
38767ab875fSJohannes Doerfert   // The barrier is explicitly called.
38867ab875fSJohannes Doerfert }
38967ab875fSJohannes Doerfert 
__kmpc_flush(IdentTy * Loc)39074f91741SJoseph Huber void __kmpc_flush(IdentTy *Loc) {
39174f91741SJoseph Huber   FunctionTracingRAII();
39274f91741SJoseph Huber   fence::kernel(__ATOMIC_SEQ_CST);
39374f91741SJoseph Huber }
39467ab875fSJohannes Doerfert 
__kmpc_warp_active_thread_mask(void)39574f91741SJoseph Huber uint64_t __kmpc_warp_active_thread_mask(void) {
39674f91741SJoseph Huber   FunctionTracingRAII();
39774f91741SJoseph Huber   return mapping::activemask();
39874f91741SJoseph Huber }
39967ab875fSJohannes Doerfert 
__kmpc_syncwarp(uint64_t Mask)40074f91741SJoseph Huber void __kmpc_syncwarp(uint64_t Mask) {
40174f91741SJoseph Huber   FunctionTracingRAII();
40274f91741SJoseph Huber   synchronize::warp(Mask);
40374f91741SJoseph Huber }
40467ab875fSJohannes Doerfert 
__kmpc_critical(IdentTy * Loc,int32_t TId,CriticalNameTy * Name)40567ab875fSJohannes Doerfert void __kmpc_critical(IdentTy *Loc, int32_t TId, CriticalNameTy *Name) {
40674f91741SJoseph Huber   FunctionTracingRAII();
40767ab875fSJohannes Doerfert   omp_set_lock(reinterpret_cast<omp_lock_t *>(Name));
40867ab875fSJohannes Doerfert }
40967ab875fSJohannes Doerfert 
__kmpc_end_critical(IdentTy * Loc,int32_t TId,CriticalNameTy * Name)41067ab875fSJohannes Doerfert void __kmpc_end_critical(IdentTy *Loc, int32_t TId, CriticalNameTy *Name) {
41174f91741SJoseph Huber   FunctionTracingRAII();
41267ab875fSJohannes Doerfert   omp_unset_lock(reinterpret_cast<omp_lock_t *>(Name));
41367ab875fSJohannes Doerfert }
41467ab875fSJohannes Doerfert 
omp_init_lock(omp_lock_t * Lock)41567ab875fSJohannes Doerfert void omp_init_lock(omp_lock_t *Lock) { impl::initLock(Lock); }
41667ab875fSJohannes Doerfert 
omp_destroy_lock(omp_lock_t * Lock)4177272982eSJon Chesterfield void omp_destroy_lock(omp_lock_t *Lock) { impl::destroyLock(Lock); }
41867ab875fSJohannes Doerfert 
omp_set_lock(omp_lock_t * Lock)41967ab875fSJohannes Doerfert void omp_set_lock(omp_lock_t *Lock) { impl::setLock(Lock); }
42067ab875fSJohannes Doerfert 
omp_unset_lock(omp_lock_t * Lock)42167ab875fSJohannes Doerfert void omp_unset_lock(omp_lock_t *Lock) { impl::unsetLock(Lock); }
42267ab875fSJohannes Doerfert 
omp_test_lock(omp_lock_t * Lock)42367ab875fSJohannes Doerfert int omp_test_lock(omp_lock_t *Lock) { return impl::testLock(Lock); }
42467ab875fSJohannes Doerfert } // extern "C"
42567ab875fSJohannes Doerfert 
42667ab875fSJohannes Doerfert #pragma omp end declare target
427