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 "api/util.hpp"
24 #include "core/event.hpp"
25 
26 using namespace clover;
27 
28 CLOVER_API cl_event
clCreateUserEvent(cl_context d_ctx,cl_int * r_errcode)29 clCreateUserEvent(cl_context d_ctx, cl_int *r_errcode) try {
30    auto &ctx = obj(d_ctx);
31 
32    ret_error(r_errcode, CL_SUCCESS);
33    return desc(new soft_event(ctx, {}, false));
34 
35 } catch (error &e) {
36    ret_error(r_errcode, e);
37    return NULL;
38 }
39 
40 CLOVER_API cl_int
clSetUserEventStatus(cl_event d_ev,cl_int status)41 clSetUserEventStatus(cl_event d_ev, cl_int status) try {
42    auto &sev = obj<soft_event>(d_ev);
43 
44    if (status > 0)
45       return CL_INVALID_VALUE;
46 
47    if (sev.status() <= 0)
48       return CL_INVALID_OPERATION;
49 
50    if (status)
51       sev.abort(status);
52    else
53       sev.trigger();
54 
55    return CL_SUCCESS;
56 
57 } catch (error &e) {
58    return e.get();
59 }
60 
61 CLOVER_API cl_int
clWaitForEvents(cl_uint num_evs,const cl_event * d_evs)62 clWaitForEvents(cl_uint num_evs, const cl_event *d_evs) try {
63    auto evs = objs(d_evs, num_evs);
64 
65    for (auto &ev : evs) {
66       if (ev.context() != evs.front().context())
67          throw error(CL_INVALID_CONTEXT);
68 
69       if (ev.status() < 0)
70          throw error(CL_EXEC_STATUS_ERROR_FOR_EVENTS_IN_WAIT_LIST);
71    }
72 
73    // Create a temporary soft event that depends on all the events in
74    // the wait list
75    auto sev = create<soft_event>(evs.front().context(), evs, true);
76 
77    // ...and wait on it.
78    sev().wait();
79 
80    return CL_SUCCESS;
81 
82 } catch (error &e) {
83    return e.get();
84 }
85 
86 CLOVER_API cl_int
clGetEventInfo(cl_event d_ev,cl_event_info param,size_t size,void * r_buf,size_t * r_size)87 clGetEventInfo(cl_event d_ev, cl_event_info param,
88                size_t size, void *r_buf, size_t *r_size) try {
89    property_buffer buf { r_buf, size, r_size };
90    auto &ev = obj(d_ev);
91 
92    switch (param) {
93    case CL_EVENT_COMMAND_QUEUE:
94       buf.as_scalar<cl_command_queue>() = desc(ev.queue());
95       break;
96 
97    case CL_EVENT_CONTEXT:
98       buf.as_scalar<cl_context>() = desc(ev.context());
99       break;
100 
101    case CL_EVENT_COMMAND_TYPE:
102       buf.as_scalar<cl_command_type>() = ev.command();
103       break;
104 
105    case CL_EVENT_COMMAND_EXECUTION_STATUS:
106       buf.as_scalar<cl_int>() = ev.status();
107       break;
108 
109    case CL_EVENT_REFERENCE_COUNT:
110       buf.as_scalar<cl_uint>() = ev.ref_count();
111       break;
112 
113    default:
114       throw error(CL_INVALID_VALUE);
115    }
116 
117    return CL_SUCCESS;
118 
119 } catch (error &e) {
120    return e.get();
121 }
122 
123 CLOVER_API cl_int
clSetEventCallback(cl_event d_ev,cl_int type,void (CL_CALLBACK * pfn_notify)(cl_event,cl_int,void *),void * user_data)124 clSetEventCallback(cl_event d_ev, cl_int type,
125                    void (CL_CALLBACK *pfn_notify)(cl_event, cl_int, void *),
126                    void *user_data) try {
127    auto &ev = obj(d_ev);
128 
129    if (!pfn_notify ||
130        (type != CL_COMPLETE && type != CL_SUBMITTED && type != CL_RUNNING))
131       throw error(CL_INVALID_VALUE);
132 
133    // Create a temporary soft event that depends on ev, with
134    // pfn_notify as completion action.
135    create<soft_event>(ev.context(), ref_vector<event> { ev }, true,
136                       [=, &ev](event &) {
137                          ev.wait();
138                          pfn_notify(desc(ev), ev.status(), user_data);
139                       });
140 
141    return CL_SUCCESS;
142 
143 } catch (error &e) {
144    return e.get();
145 }
146 
147 CLOVER_API cl_int
clRetainEvent(cl_event d_ev)148 clRetainEvent(cl_event d_ev) try {
149    obj(d_ev).retain();
150    return CL_SUCCESS;
151 
152 } catch (error &e) {
153    return e.get();
154 }
155 
156 CLOVER_API cl_int
clReleaseEvent(cl_event d_ev)157 clReleaseEvent(cl_event d_ev) try {
158    if (obj(d_ev).release())
159       delete pobj(d_ev);
160 
161    return CL_SUCCESS;
162 
163 } catch (error &e) {
164    return e.get();
165 }
166 
167 CLOVER_API cl_int
clEnqueueMarker(cl_command_queue d_q,cl_event * rd_ev)168 clEnqueueMarker(cl_command_queue d_q, cl_event *rd_ev) try {
169    auto &q = obj(d_q);
170 
171    if (!rd_ev)
172       throw error(CL_INVALID_VALUE);
173 
174    *rd_ev = desc(new hard_event(q, CL_COMMAND_MARKER, {}));
175 
176    return CL_SUCCESS;
177 
178 } catch (error &e) {
179    return e.get();
180 }
181 
182 CLOVER_API cl_int
clEnqueueMarkerWithWaitList(cl_command_queue d_q,cl_uint num_deps,const cl_event * d_deps,cl_event * rd_ev)183 clEnqueueMarkerWithWaitList(cl_command_queue d_q, cl_uint num_deps,
184                             const cl_event *d_deps, cl_event *rd_ev) try {
185    auto &q = obj(d_q);
186    auto deps = objs<wait_list_tag>(d_deps, num_deps);
187 
188    for (auto &ev : deps) {
189       if (ev.context() != q.context())
190          throw error(CL_INVALID_CONTEXT);
191    }
192 
193    // Create a hard event that depends on the events in the wait list:
194    // previous commands in the same queue are implicitly serialized
195    // with respect to it -- hard events always are.
196    auto hev = create<hard_event>(q, CL_COMMAND_MARKER, deps);
197 
198    ret_object(rd_ev, hev);
199    return CL_SUCCESS;
200 
201 } catch (error &e) {
202    return e.get();
203 }
204 
205 CLOVER_API cl_int
clEnqueueBarrier(cl_command_queue d_q)206 clEnqueueBarrier(cl_command_queue d_q) try {
207    obj(d_q);
208 
209    // No need to do anything, q preserves data ordering strictly.
210 
211    return CL_SUCCESS;
212 
213 } catch (error &e) {
214    return e.get();
215 }
216 
217 CLOVER_API cl_int
clEnqueueBarrierWithWaitList(cl_command_queue d_q,cl_uint num_deps,const cl_event * d_deps,cl_event * rd_ev)218 clEnqueueBarrierWithWaitList(cl_command_queue d_q, cl_uint num_deps,
219                              const cl_event *d_deps, cl_event *rd_ev) try {
220    auto &q = obj(d_q);
221    auto deps = objs<wait_list_tag>(d_deps, num_deps);
222 
223    for (auto &ev : deps) {
224       if (ev.context() != q.context())
225          throw error(CL_INVALID_CONTEXT);
226    }
227 
228    // Create a hard event that depends on the events in the wait list:
229    // subsequent commands in the same queue will be implicitly
230    // serialized with respect to it -- hard events always are.
231    auto hev = create<hard_event>(q, CL_COMMAND_BARRIER, deps);
232 
233    ret_object(rd_ev, hev);
234    return CL_SUCCESS;
235 
236 } catch (error &e) {
237    return e.get();
238 }
239 
240 CLOVER_API cl_int
clEnqueueWaitForEvents(cl_command_queue d_q,cl_uint num_evs,const cl_event * d_evs)241 clEnqueueWaitForEvents(cl_command_queue d_q, cl_uint num_evs,
242                        const cl_event *d_evs) try {
243    // The wait list is mandatory for clEnqueueWaitForEvents().
244    objs(d_evs, num_evs);
245 
246    return clEnqueueBarrierWithWaitList(d_q, num_evs, d_evs, NULL);
247 
248 } catch (error &e) {
249    return e.get();
250 }
251 
252 CLOVER_API cl_int
clGetEventProfilingInfo(cl_event d_ev,cl_profiling_info param,size_t size,void * r_buf,size_t * r_size)253 clGetEventProfilingInfo(cl_event d_ev, cl_profiling_info param,
254                         size_t size, void *r_buf, size_t *r_size) try {
255    property_buffer buf { r_buf, size, r_size };
256    hard_event &hev = dynamic_cast<hard_event &>(obj(d_ev));
257 
258    if (hev.status() != CL_COMPLETE)
259       throw error(CL_PROFILING_INFO_NOT_AVAILABLE);
260 
261    switch (param) {
262    case CL_PROFILING_COMMAND_QUEUED:
263       buf.as_scalar<cl_ulong>() = hev.time_queued();
264       break;
265 
266    case CL_PROFILING_COMMAND_SUBMIT:
267       buf.as_scalar<cl_ulong>() = hev.time_submit();
268       break;
269 
270    case CL_PROFILING_COMMAND_START:
271       buf.as_scalar<cl_ulong>() = hev.time_start();
272       break;
273 
274    case CL_PROFILING_COMMAND_END:
275    case CL_PROFILING_COMMAND_COMPLETE:
276       buf.as_scalar<cl_ulong>() = hev.time_end();
277       break;
278 
279    default:
280       throw error(CL_INVALID_VALUE);
281    }
282 
283    return CL_SUCCESS;
284 
285 } catch (std::bad_cast &) {
286    return CL_PROFILING_INFO_NOT_AVAILABLE;
287 
288 } catch (lazy<cl_ulong>::undefined_error &) {
289    return CL_PROFILING_INFO_NOT_AVAILABLE;
290 
291 } catch (error &e) {
292    return e.get();
293 }
294 
295 CLOVER_API cl_int
clFinish(cl_command_queue d_q)296 clFinish(cl_command_queue d_q) try {
297    auto &q = obj(d_q);
298 
299    // Create a temporary hard event -- it implicitly depends on all
300    // the previously queued hard events.
301    auto hev = create<hard_event>(q, 0, ref_vector<event> {});
302 
303    // And wait on it.
304    hev().wait();
305 
306    return CL_SUCCESS;
307 
308 } catch (error &e) {
309    return e.get();
310 }
311