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/context.hpp"
25 #include "core/platform.hpp"
26 
27 using namespace clover;
28 
29 CLOVER_API cl_context
clCreateContext(const cl_context_properties * d_props,cl_uint num_devs,const cl_device_id * d_devs,void (CL_CALLBACK * pfn_notify)(const char *,const void *,size_t,void *),void * user_data,cl_int * r_errcode)30 clCreateContext(const cl_context_properties *d_props, cl_uint num_devs,
31                 const cl_device_id *d_devs,
32                 void (CL_CALLBACK *pfn_notify)(const char *, const void *,
33                                                size_t, void *),
34                 void *user_data, cl_int *r_errcode) try {
35    auto props = obj<property_list_tag>(d_props);
36    auto devs = objs(d_devs, num_devs);
37 
38    if (!pfn_notify && user_data)
39       throw error(CL_INVALID_VALUE);
40 
41    for (auto &prop : props) {
42       if (prop.first == CL_CONTEXT_PLATFORM)
43          obj(prop.second.as<cl_platform_id>());
44       else
45          throw error(CL_INVALID_PROPERTY);
46    }
47 
48    const auto notify = (!pfn_notify ? context::notify_action() :
49                         [=](const char *s) {
50                            pfn_notify(s, NULL, 0, user_data);
51                         });
52 
53    ret_error(r_errcode, CL_SUCCESS);
54    return desc(new context(props, devs, notify));
55 
56 } catch (error &e) {
57    ret_error(r_errcode, e);
58    return NULL;
59 }
60 
61 CLOVER_API cl_context
clCreateContextFromType(const cl_context_properties * d_props,cl_device_type type,void (CL_CALLBACK * pfn_notify)(const char *,const void *,size_t,void *),void * user_data,cl_int * r_errcode)62 clCreateContextFromType(const cl_context_properties *d_props,
63                         cl_device_type type,
64                         void (CL_CALLBACK *pfn_notify)(
65                            const char *, const void *, size_t, void *),
66                         void *user_data, cl_int *r_errcode) try {
67    cl_platform_id d_platform;
68    cl_uint num_platforms;
69    cl_int ret;
70    std::vector<cl_device_id> devs;
71    cl_uint num_devices;
72 
73    ret = clGetPlatformIDs(1, &d_platform, &num_platforms);
74    if (ret || !num_platforms)
75       throw error(CL_INVALID_PLATFORM);
76 
77    ret = clGetDeviceIDs(d_platform, type, 0, NULL, &num_devices);
78    if (ret)
79       throw error(CL_DEVICE_NOT_FOUND);
80    devs.resize(num_devices);
81    ret = clGetDeviceIDs(d_platform, type, num_devices, devs.data(), 0);
82    if (ret)
83       throw error(CL_DEVICE_NOT_FOUND);
84 
85    return clCreateContext(d_props, num_devices, devs.data(), pfn_notify,
86                           user_data, r_errcode);
87 
88 } catch (error &e) {
89    ret_error(r_errcode, e);
90    return NULL;
91 }
92 
93 CLOVER_API cl_int
clRetainContext(cl_context d_ctx)94 clRetainContext(cl_context d_ctx) try {
95    obj(d_ctx).retain();
96    return CL_SUCCESS;
97 
98 } catch (error &e) {
99    return e.get();
100 }
101 
102 CLOVER_API cl_int
clReleaseContext(cl_context d_ctx)103 clReleaseContext(cl_context d_ctx) try {
104    if (obj(d_ctx).release())
105       delete pobj(d_ctx);
106 
107    return CL_SUCCESS;
108 
109 } catch (error &e) {
110    return e.get();
111 }
112 
113 CLOVER_API cl_int
clGetContextInfo(cl_context d_ctx,cl_context_info param,size_t size,void * r_buf,size_t * r_size)114 clGetContextInfo(cl_context d_ctx, cl_context_info param,
115                  size_t size, void *r_buf, size_t *r_size) try {
116    property_buffer buf { r_buf, size, r_size };
117    auto &ctx = obj(d_ctx);
118 
119    switch (param) {
120    case CL_CONTEXT_REFERENCE_COUNT:
121       buf.as_scalar<cl_uint>() = ctx.ref_count();
122       break;
123 
124    case CL_CONTEXT_NUM_DEVICES:
125       buf.as_scalar<cl_uint>() = ctx.devices().size();
126       break;
127 
128    case CL_CONTEXT_DEVICES:
129       buf.as_vector<cl_device_id>() = descs(ctx.devices());
130       break;
131 
132    case CL_CONTEXT_PROPERTIES:
133       buf.as_vector<cl_context_properties>() = desc(ctx.properties());
134       break;
135 
136    default:
137       throw error(CL_INVALID_VALUE);
138    }
139 
140    return CL_SUCCESS;
141 
142 } catch (error &e) {
143    return e.get();
144 }
145 
146 CLOVER_API cl_int
clSetContextDestructorCallback(cl_context d_ctx,void (CL_CALLBACK * pfn_notify)(cl_context,void *),void * user_data)147 clSetContextDestructorCallback(cl_context d_ctx,
148                                void (CL_CALLBACK *pfn_notify)(cl_context, void *),
149                                void *user_data) try {
150    CLOVER_NOT_SUPPORTED_UNTIL("3.0");
151    auto &ctx = obj(d_ctx);
152 
153    if (!pfn_notify)
154       return CL_INVALID_VALUE;
155 
156    ctx.destroy_notify([=]{ pfn_notify(d_ctx, user_data); });
157 
158    return CL_SUCCESS;
159 
160 } catch (error &e) {
161    return e.get();
162 }
163