1 //
2 // Copyright 2012 Francisco Jerez
3 //
4 // Permission is hereby granted, free of charge, to any person obtaining a
5 // copy of this software and associated documentation files (the "Software"),
6 // to deal in the Software without restriction, including without limitation
7 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 // and/or sell copies of the Software, and to permit persons to whom the
9 // Software is furnished to do so, subject to the following conditions:
10 //
11 // The above copyright notice and this permission notice shall be included in
12 // all copies or substantial portions of the Software.
13 //
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17 // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 // OTHER DEALINGS IN THE SOFTWARE.
21 //
22 
23 #include "core/event.hpp"
24 #include "pipe/p_screen.h"
25 
26 using namespace clover;
27 
event(clover::context & ctx,const ref_vector<event> & deps,action action_ok,action action_fail)28 event::event(clover::context &ctx, const ref_vector<event> &deps,
29              action action_ok, action action_fail) :
30    context(ctx), _wait_count(1), _status(0),
31    action_ok(action_ok), action_fail(action_fail) {
32    for (auto &ev : deps)
33       ev.chain(*this);
34 }
35 
~event()36 event::~event() {
37 }
38 
39 std::vector<intrusive_ref<event>>
trigger_self()40 event::trigger_self() {
41    std::lock_guard<std::mutex> lock(mutex);
42    std::vector<intrusive_ref<event>> evs;
43 
44    if (_wait_count && !--_wait_count)
45       std::swap(_chain, evs);
46 
47    cv.notify_all();
48    return evs;
49 }
50 
51 void
trigger()52 event::trigger() try {
53    if (wait_count() == 1)
54       action_ok(*this);
55 
56    for (event &ev : trigger_self())
57       ev.trigger();
58 } catch (error &e) {
59    abort(e.get());
60 }
61 
62 std::vector<intrusive_ref<event>>
abort_self(cl_int status)63 event::abort_self(cl_int status) {
64    std::lock_guard<std::mutex> lock(mutex);
65    std::vector<intrusive_ref<event>> evs;
66 
67    _status = status;
68    _wait_count = 0;
69    std::swap(_chain, evs);
70 
71    cv.notify_all();
72    return evs;
73 }
74 
75 void
abort(cl_int status)76 event::abort(cl_int status) {
77    action_fail(*this);
78 
79    for (event &ev : abort_self(status))
80       ev.abort(status);
81 }
82 
83 unsigned
wait_count() const84 event::wait_count() const {
85    std::lock_guard<std::mutex> lock(mutex);
86    return _wait_count;
87 }
88 
89 bool
signalled() const90 event::signalled() const {
91    return !wait_count();
92 }
93 
94 cl_int
status() const95 event::status() const {
96    std::lock_guard<std::mutex> lock(mutex);
97    return _status;
98 }
99 
100 void
chain(event & ev)101 event::chain(event &ev) {
102    std::unique_lock<std::mutex> lock(mutex, std::defer_lock);
103    std::unique_lock<std::mutex> lock_ev(ev.mutex, std::defer_lock);
104    std::lock(lock, lock_ev);
105 
106    if (_wait_count) {
107       ev._wait_count++;
108       _chain.push_back(ev);
109    }
110    ev.deps.push_back(*this);
111 }
112 
113 void
wait_signalled() const114 event::wait_signalled() const {
115    std::unique_lock<std::mutex> lock(mutex);
116    cv.wait(lock, [=]{ return !_wait_count; });
117 }
118 
119 void
wait() const120 event::wait() const {
121    std::vector<intrusive_ref<event>> evs;
122    std::swap(deps, evs);
123 
124    for (event &ev : evs)
125       ev.wait();
126 
127    wait_signalled();
128 }
129 
hard_event(command_queue & q,cl_command_type command,const ref_vector<event> & deps,action action)130 hard_event::hard_event(command_queue &q, cl_command_type command,
131                        const ref_vector<event> &deps, action action) :
132    event(q.context(), deps, profile(q, action), [](event &ev){}),
133    _queue(q), _command(command), _fence(NULL) {
134    if (q.profiling_enabled())
135       _time_queued = timestamp::current(q);
136 
137    q.sequence(*this);
138    trigger();
139 }
140 
~hard_event()141 hard_event::~hard_event() {
142    pipe_screen *screen = queue()->device().pipe;
143    screen->fence_reference(screen, &_fence, NULL);
144 }
145 
146 cl_int
status() const147 hard_event::status() const {
148    pipe_screen *screen = queue()->device().pipe;
149 
150    if (event::status() < 0)
151       return event::status();
152 
153    else if (!_fence)
154       return CL_QUEUED;
155 
156    else if (!screen->fence_finish(screen, NULL, _fence, 0))
157       return CL_SUBMITTED;
158 
159    else
160       return CL_COMPLETE;
161 }
162 
163 command_queue *
queue() const164 hard_event::queue() const {
165    return &_queue();
166 }
167 
168 cl_command_type
command() const169 hard_event::command() const {
170    return _command;
171 }
172 
173 void
wait() const174 hard_event::wait() const {
175    pipe_screen *screen = queue()->device().pipe;
176 
177    event::wait();
178 
179    if (status() == CL_QUEUED)
180       queue()->flush();
181 
182    if (!_fence ||
183        !screen->fence_finish(screen, NULL, _fence, PIPE_TIMEOUT_INFINITE))
184       throw error(CL_EXEC_STATUS_ERROR_FOR_EVENTS_IN_WAIT_LIST);
185 }
186 
187 const lazy<cl_ulong> &
time_queued() const188 hard_event::time_queued() const {
189    return _time_queued;
190 }
191 
192 const lazy<cl_ulong> &
time_submit() const193 hard_event::time_submit() const {
194    return _time_submit;
195 }
196 
197 const lazy<cl_ulong> &
time_start() const198 hard_event::time_start() const {
199    return _time_start;
200 }
201 
202 const lazy<cl_ulong> &
time_end() const203 hard_event::time_end() const {
204    return _time_end;
205 }
206 
207 void
fence(pipe_fence_handle * fence)208 hard_event::fence(pipe_fence_handle *fence) {
209    assert(fence);
210    pipe_screen *screen = queue()->device().pipe;
211    screen->fence_reference(screen, &_fence, fence);
212    deps.clear();
213 }
214 
215 event::action
profile(command_queue & q,const action & action) const216 hard_event::profile(command_queue &q, const action &action) const {
217    if (q.profiling_enabled()) {
218       return [&q, action] (event &ev) {
219          auto &hev = static_cast<hard_event &>(ev);
220 
221          hev._time_submit = timestamp::current(q);
222          hev._time_start = timestamp::query(q);
223 
224          action(ev);
225 
226          hev._time_end = timestamp::query(q);
227       };
228 
229    } else {
230       return action;
231    }
232 }
233 
soft_event(clover::context & ctx,const ref_vector<event> & deps,bool _trigger,action action)234 soft_event::soft_event(clover::context &ctx, const ref_vector<event> &deps,
235                        bool _trigger, action action) :
236    event(ctx, deps, action, action) {
237    if (_trigger)
238       trigger();
239 }
240 
241 cl_int
status() const242 soft_event::status() const {
243    if (event::status() < 0)
244       return event::status();
245 
246    else if (!signalled() ||
247             any_of([](const event &ev) {
248                   return ev.status() != CL_COMPLETE;
249                }, deps))
250       return CL_SUBMITTED;
251 
252    else
253       return CL_COMPLETE;
254 }
255 
256 command_queue *
queue() const257 soft_event::queue() const {
258    return NULL;
259 }
260 
261 cl_command_type
command() const262 soft_event::command() const {
263    return CL_COMMAND_USER;
264 }
265 
266 void
wait() const267 soft_event::wait() const {
268    event::wait();
269 
270    if (status() != CL_COMPLETE)
271       throw error(CL_EXEC_STATUS_ERROR_FOR_EVENTS_IN_WAIT_LIST);
272 }
273