annotate libgomp/config/nvptx/bar.c @ 158:494b0b89df80 default tip

...
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Mon, 25 May 2020 18:13:55 +0900
parents 1830386684a0
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1 /* Copyright (C) 2015-2020 Free Software Foundation, Inc.
111
kono
parents:
diff changeset
2 Contributed by Alexander Monakov <amonakov@ispras.ru>
kono
parents:
diff changeset
3
kono
parents:
diff changeset
4 This file is part of the GNU Offloading and Multi Processing Library
kono
parents:
diff changeset
5 (libgomp).
kono
parents:
diff changeset
6
kono
parents:
diff changeset
7 Libgomp is free software; you can redistribute it and/or modify it
kono
parents:
diff changeset
8 under the terms of the GNU General Public License as published by
kono
parents:
diff changeset
9 the Free Software Foundation; either version 3, or (at your option)
kono
parents:
diff changeset
10 any later version.
kono
parents:
diff changeset
11
kono
parents:
diff changeset
12 Libgomp is distributed in the hope that it will be useful, but WITHOUT ANY
kono
parents:
diff changeset
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
kono
parents:
diff changeset
14 FOR A PARTICULAR PURPOSE. See the GNU General Public License for
kono
parents:
diff changeset
15 more details.
kono
parents:
diff changeset
16
kono
parents:
diff changeset
17 Under Section 7 of GPL version 3, you are granted additional
kono
parents:
diff changeset
18 permissions described in the GCC Runtime Library Exception, version
kono
parents:
diff changeset
19 3.1, as published by the Free Software Foundation.
kono
parents:
diff changeset
20
kono
parents:
diff changeset
21 You should have received a copy of the GNU General Public License and
kono
parents:
diff changeset
22 a copy of the GCC Runtime Library Exception along with this program;
kono
parents:
diff changeset
23 see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
kono
parents:
diff changeset
24 <http://www.gnu.org/licenses/>. */
kono
parents:
diff changeset
25
kono
parents:
diff changeset
26 /* This is an NVPTX specific implementation of a barrier synchronization
kono
parents:
diff changeset
27 mechanism for libgomp. This type is private to the library. This
kono
parents:
diff changeset
28 implementation uses atomic instructions and bar.sync instruction. */
kono
parents:
diff changeset
29
kono
parents:
diff changeset
30 #include <limits.h>
kono
parents:
diff changeset
31 #include "libgomp.h"
kono
parents:
diff changeset
32
kono
parents:
diff changeset
33
kono
parents:
diff changeset
34 void
kono
parents:
diff changeset
35 gomp_barrier_wait_end (gomp_barrier_t *bar, gomp_barrier_state_t state)
kono
parents:
diff changeset
36 {
kono
parents:
diff changeset
37 if (__builtin_expect (state & BAR_WAS_LAST, 0))
kono
parents:
diff changeset
38 {
kono
parents:
diff changeset
39 /* Next time we'll be awaiting TOTAL threads again. */
kono
parents:
diff changeset
40 bar->awaited = bar->total;
kono
parents:
diff changeset
41 __atomic_store_n (&bar->generation, bar->generation + BAR_INCR,
kono
parents:
diff changeset
42 MEMMODEL_RELEASE);
kono
parents:
diff changeset
43 }
kono
parents:
diff changeset
44 asm ("bar.sync 1, %0;" : : "r" (32 * bar->total));
kono
parents:
diff changeset
45 }
kono
parents:
diff changeset
46
kono
parents:
diff changeset
47 void
kono
parents:
diff changeset
48 gomp_barrier_wait (gomp_barrier_t *bar)
kono
parents:
diff changeset
49 {
kono
parents:
diff changeset
50 gomp_barrier_wait_end (bar, gomp_barrier_wait_start (bar));
kono
parents:
diff changeset
51 }
kono
parents:
diff changeset
52
kono
parents:
diff changeset
53 /* Like gomp_barrier_wait, except that if the encountering thread
kono
parents:
diff changeset
54 is not the last one to hit the barrier, it returns immediately.
kono
parents:
diff changeset
55 The intended usage is that a thread which intends to gomp_barrier_destroy
kono
parents:
diff changeset
56 this barrier calls gomp_barrier_wait, while all other threads
kono
parents:
diff changeset
57 call gomp_barrier_wait_last. When gomp_barrier_wait returns,
kono
parents:
diff changeset
58 the barrier can be safely destroyed. */
kono
parents:
diff changeset
59
kono
parents:
diff changeset
60 void
kono
parents:
diff changeset
61 gomp_barrier_wait_last (gomp_barrier_t *bar)
kono
parents:
diff changeset
62 {
kono
parents:
diff changeset
63 /* Deferring to gomp_barrier_wait does not use the optimization opportunity
kono
parents:
diff changeset
64 allowed by the interface contract for all-but-last participants. The
kono
parents:
diff changeset
65 original implementation in config/linux/bar.c handles this better. */
kono
parents:
diff changeset
66 gomp_barrier_wait (bar);
kono
parents:
diff changeset
67 }
kono
parents:
diff changeset
68
kono
parents:
diff changeset
69 void
kono
parents:
diff changeset
70 gomp_team_barrier_wake (gomp_barrier_t *bar, int count)
kono
parents:
diff changeset
71 {
kono
parents:
diff changeset
72 asm ("bar.sync 1, %0;" : : "r" (32 * bar->total));
kono
parents:
diff changeset
73 }
kono
parents:
diff changeset
74
kono
parents:
diff changeset
75 void
kono
parents:
diff changeset
76 gomp_team_barrier_wait_end (gomp_barrier_t *bar, gomp_barrier_state_t state)
kono
parents:
diff changeset
77 {
kono
parents:
diff changeset
78 unsigned int generation, gen;
kono
parents:
diff changeset
79
kono
parents:
diff changeset
80 if (__builtin_expect (state & BAR_WAS_LAST, 0))
kono
parents:
diff changeset
81 {
kono
parents:
diff changeset
82 /* Next time we'll be awaiting TOTAL threads again. */
kono
parents:
diff changeset
83 struct gomp_thread *thr = gomp_thread ();
kono
parents:
diff changeset
84 struct gomp_team *team = thr->ts.team;
kono
parents:
diff changeset
85
kono
parents:
diff changeset
86 bar->awaited = bar->total;
kono
parents:
diff changeset
87 team->work_share_cancelled = 0;
kono
parents:
diff changeset
88 if (__builtin_expect (team->task_count, 0))
kono
parents:
diff changeset
89 {
kono
parents:
diff changeset
90 gomp_barrier_handle_tasks (state);
kono
parents:
diff changeset
91 state &= ~BAR_WAS_LAST;
kono
parents:
diff changeset
92 }
kono
parents:
diff changeset
93 else
kono
parents:
diff changeset
94 {
kono
parents:
diff changeset
95 state &= ~BAR_CANCELLED;
kono
parents:
diff changeset
96 state += BAR_INCR - BAR_WAS_LAST;
kono
parents:
diff changeset
97 __atomic_store_n (&bar->generation, state, MEMMODEL_RELEASE);
kono
parents:
diff changeset
98 asm ("bar.sync 1, %0;" : : "r" (32 * bar->total));
kono
parents:
diff changeset
99 return;
kono
parents:
diff changeset
100 }
kono
parents:
diff changeset
101 }
kono
parents:
diff changeset
102
kono
parents:
diff changeset
103 generation = state;
kono
parents:
diff changeset
104 state &= ~BAR_CANCELLED;
kono
parents:
diff changeset
105 do
kono
parents:
diff changeset
106 {
kono
parents:
diff changeset
107 asm ("bar.sync 1, %0;" : : "r" (32 * bar->total));
kono
parents:
diff changeset
108 gen = __atomic_load_n (&bar->generation, MEMMODEL_ACQUIRE);
kono
parents:
diff changeset
109 if (__builtin_expect (gen & BAR_TASK_PENDING, 0))
kono
parents:
diff changeset
110 {
kono
parents:
diff changeset
111 gomp_barrier_handle_tasks (state);
kono
parents:
diff changeset
112 gen = __atomic_load_n (&bar->generation, MEMMODEL_ACQUIRE);
kono
parents:
diff changeset
113 }
kono
parents:
diff changeset
114 generation |= gen & BAR_WAITING_FOR_TASK;
kono
parents:
diff changeset
115 }
kono
parents:
diff changeset
116 while (gen != state + BAR_INCR);
kono
parents:
diff changeset
117 }
kono
parents:
diff changeset
118
kono
parents:
diff changeset
119 void
kono
parents:
diff changeset
120 gomp_team_barrier_wait (gomp_barrier_t *bar)
kono
parents:
diff changeset
121 {
kono
parents:
diff changeset
122 gomp_team_barrier_wait_end (bar, gomp_barrier_wait_start (bar));
kono
parents:
diff changeset
123 }
kono
parents:
diff changeset
124
kono
parents:
diff changeset
125 void
kono
parents:
diff changeset
126 gomp_team_barrier_wait_final (gomp_barrier_t *bar)
kono
parents:
diff changeset
127 {
kono
parents:
diff changeset
128 gomp_barrier_state_t state = gomp_barrier_wait_final_start (bar);
kono
parents:
diff changeset
129 if (__builtin_expect (state & BAR_WAS_LAST, 0))
kono
parents:
diff changeset
130 bar->awaited_final = bar->total;
kono
parents:
diff changeset
131 gomp_team_barrier_wait_end (bar, state);
kono
parents:
diff changeset
132 }
kono
parents:
diff changeset
133
kono
parents:
diff changeset
134 bool
kono
parents:
diff changeset
135 gomp_team_barrier_wait_cancel_end (gomp_barrier_t *bar,
kono
parents:
diff changeset
136 gomp_barrier_state_t state)
kono
parents:
diff changeset
137 {
kono
parents:
diff changeset
138 unsigned int generation, gen;
kono
parents:
diff changeset
139
kono
parents:
diff changeset
140 if (__builtin_expect (state & BAR_WAS_LAST, 0))
kono
parents:
diff changeset
141 {
kono
parents:
diff changeset
142 /* Next time we'll be awaiting TOTAL threads again. */
kono
parents:
diff changeset
143 /* BAR_CANCELLED should never be set in state here, because
kono
parents:
diff changeset
144 cancellation means that at least one of the threads has been
kono
parents:
diff changeset
145 cancelled, thus on a cancellable barrier we should never see
kono
parents:
diff changeset
146 all threads to arrive. */
kono
parents:
diff changeset
147 struct gomp_thread *thr = gomp_thread ();
kono
parents:
diff changeset
148 struct gomp_team *team = thr->ts.team;
kono
parents:
diff changeset
149
kono
parents:
diff changeset
150 bar->awaited = bar->total;
kono
parents:
diff changeset
151 team->work_share_cancelled = 0;
kono
parents:
diff changeset
152 if (__builtin_expect (team->task_count, 0))
kono
parents:
diff changeset
153 {
kono
parents:
diff changeset
154 gomp_barrier_handle_tasks (state);
kono
parents:
diff changeset
155 state &= ~BAR_WAS_LAST;
kono
parents:
diff changeset
156 }
kono
parents:
diff changeset
157 else
kono
parents:
diff changeset
158 {
kono
parents:
diff changeset
159 state += BAR_INCR - BAR_WAS_LAST;
kono
parents:
diff changeset
160 __atomic_store_n (&bar->generation, state, MEMMODEL_RELEASE);
kono
parents:
diff changeset
161 asm ("bar.sync 1, %0;" : : "r" (32 * bar->total));
kono
parents:
diff changeset
162 return false;
kono
parents:
diff changeset
163 }
kono
parents:
diff changeset
164 }
kono
parents:
diff changeset
165
kono
parents:
diff changeset
166 if (__builtin_expect (state & BAR_CANCELLED, 0))
kono
parents:
diff changeset
167 return true;
kono
parents:
diff changeset
168
kono
parents:
diff changeset
169 generation = state;
kono
parents:
diff changeset
170 do
kono
parents:
diff changeset
171 {
kono
parents:
diff changeset
172 asm ("bar.sync 1, %0;" : : "r" (32 * bar->total));
kono
parents:
diff changeset
173 gen = __atomic_load_n (&bar->generation, MEMMODEL_ACQUIRE);
kono
parents:
diff changeset
174 if (__builtin_expect (gen & BAR_CANCELLED, 0))
kono
parents:
diff changeset
175 return true;
kono
parents:
diff changeset
176 if (__builtin_expect (gen & BAR_TASK_PENDING, 0))
kono
parents:
diff changeset
177 {
kono
parents:
diff changeset
178 gomp_barrier_handle_tasks (state);
kono
parents:
diff changeset
179 gen = __atomic_load_n (&bar->generation, MEMMODEL_ACQUIRE);
kono
parents:
diff changeset
180 }
kono
parents:
diff changeset
181 generation |= gen & BAR_WAITING_FOR_TASK;
kono
parents:
diff changeset
182 }
kono
parents:
diff changeset
183 while (gen != state + BAR_INCR);
kono
parents:
diff changeset
184
kono
parents:
diff changeset
185 return false;
kono
parents:
diff changeset
186 }
kono
parents:
diff changeset
187
kono
parents:
diff changeset
188 bool
kono
parents:
diff changeset
189 gomp_team_barrier_wait_cancel (gomp_barrier_t *bar)
kono
parents:
diff changeset
190 {
kono
parents:
diff changeset
191 return gomp_team_barrier_wait_cancel_end (bar, gomp_barrier_wait_start (bar));
kono
parents:
diff changeset
192 }
kono
parents:
diff changeset
193
kono
parents:
diff changeset
194 void
kono
parents:
diff changeset
195 gomp_team_barrier_cancel (struct gomp_team *team)
kono
parents:
diff changeset
196 {
kono
parents:
diff changeset
197 gomp_mutex_lock (&team->task_lock);
kono
parents:
diff changeset
198 if (team->barrier.generation & BAR_CANCELLED)
kono
parents:
diff changeset
199 {
kono
parents:
diff changeset
200 gomp_mutex_unlock (&team->task_lock);
kono
parents:
diff changeset
201 return;
kono
parents:
diff changeset
202 }
kono
parents:
diff changeset
203 team->barrier.generation |= BAR_CANCELLED;
kono
parents:
diff changeset
204 gomp_mutex_unlock (&team->task_lock);
kono
parents:
diff changeset
205 gomp_team_barrier_wake (&team->barrier, INT_MAX);
kono
parents:
diff changeset
206 }