// // Copyright 2012 Francisco Jerez // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR // OTHER DEALINGS IN THE SOFTWARE. // #include "api/util.hpp" #include "core/program.hpp" using namespace clover; CLOVER_API cl_program clCreateProgramWithSource(cl_context d_ctx, cl_uint count, const char **strings, const size_t *lengths, cl_int *r_errcode) try { auto &ctx = obj(d_ctx); std::string source; if (!count || !strings || any_of(is_zero(), range(strings, count))) throw error(CL_INVALID_VALUE); // Concatenate all the provided fragments together for (unsigned i = 0; i < count; ++i) source += (lengths && lengths[i] ? std::string(strings[i], strings[i] + lengths[i]) : std::string(strings[i])); // ...and create a program object for them. ret_error(r_errcode, CL_SUCCESS); return new program(ctx, source); } catch (error &e) { ret_error(r_errcode, e); return NULL; } CLOVER_API cl_program clCreateProgramWithBinary(cl_context d_ctx, cl_uint n, const cl_device_id *d_devs, const size_t *lengths, const unsigned char **binaries, cl_int *r_status, cl_int *r_errcode) try { auto &ctx = obj(d_ctx); auto devs = objs(d_devs, n); if (!lengths || !binaries) throw error(CL_INVALID_VALUE); if (any_of([&](const device &dev) { return !count(dev, ctx.devs()); }, devs)) throw error(CL_INVALID_DEVICE); // Deserialize the provided binaries, auto result = map( [](const unsigned char *p, size_t l) -> std::pair { if (!p || !l) return { CL_INVALID_VALUE, {} }; try { compat::istream::buffer_t bin(p, l); compat::istream s(bin); return { CL_SUCCESS, module::deserialize(s) }; } catch (compat::istream::error &e) { return { CL_INVALID_BINARY, {} }; } }, range(binaries, n), range(lengths, n)); // update the status array, if (r_status) copy(map(keys(), result), r_status); if (any_of(key_equals(CL_INVALID_VALUE), result)) throw error(CL_INVALID_VALUE); if (any_of(key_equals(CL_INVALID_BINARY), result)) throw error(CL_INVALID_BINARY); // initialize a program object with them. ret_error(r_errcode, CL_SUCCESS); return new program(ctx, devs, map(values(), result)); } catch (error &e) { ret_error(r_errcode, e); return NULL; } CLOVER_API cl_int clRetainProgram(cl_program d_prog) try { obj(d_prog).retain(); return CL_SUCCESS; } catch (error &e) { return e.get(); } CLOVER_API cl_int clReleaseProgram(cl_program d_prog) try { if (obj(d_prog).release()) delete pobj(d_prog); return CL_SUCCESS; } catch (error &e) { return e.get(); } CLOVER_API cl_int clBuildProgram(cl_program d_prog, cl_uint num_devs, const cl_device_id *d_devs, const char *p_opts, void (*pfn_notify)(cl_program, void *), void *user_data) try { auto &prog = obj(d_prog); auto devs = (d_devs ? objs(d_devs, num_devs) : ref_vector(prog.ctx.devs())); auto opts = (p_opts ? p_opts : ""); if (bool(num_devs) != bool(d_devs) || (!pfn_notify && user_data)) throw error(CL_INVALID_VALUE); if (any_of([&](const device &dev) { return !count(dev, prog.ctx.devs()); }, devs)) throw error(CL_INVALID_DEVICE); prog.build(devs, opts); return CL_SUCCESS; } catch (error &e) { return e.get(); } CLOVER_API cl_int clUnloadCompiler() { return CL_SUCCESS; } CLOVER_API cl_int clGetProgramInfo(cl_program d_prog, cl_program_info param, size_t size, void *r_buf, size_t *r_size) try { property_buffer buf { r_buf, size, r_size }; auto &prog = obj(d_prog); switch (param) { case CL_PROGRAM_REFERENCE_COUNT: buf.as_scalar() = prog.ref_count(); break; case CL_PROGRAM_CONTEXT: buf.as_scalar() = desc(prog.ctx); break; case CL_PROGRAM_NUM_DEVICES: buf.as_scalar() = prog.devices().size() ? prog.devices().size() : prog.ctx.devs().size(); break; case CL_PROGRAM_DEVICES: buf.as_vector() = prog.devices().size() ? descs(prog.devices()) : descs(prog.ctx.devs()); break; case CL_PROGRAM_SOURCE: buf.as_string() = prog.source(); break; case CL_PROGRAM_BINARY_SIZES: buf.as_vector() = map([&](const device &dev) { compat::ostream::buffer_t bin; compat::ostream s(bin); prog.binary(dev).serialize(s); return bin.size(); }, prog.devices()); break; case CL_PROGRAM_BINARIES: buf.as_matrix() = map([&](const device &dev) { compat::ostream::buffer_t bin; compat::ostream s(bin); prog.binary(dev).serialize(s); return bin; }, prog.devices()); break; default: throw error(CL_INVALID_VALUE); } return CL_SUCCESS; } catch (error &e) { return e.get(); } CLOVER_API cl_int clGetProgramBuildInfo(cl_program d_prog, cl_device_id d_dev, cl_program_build_info param, size_t size, void *r_buf, size_t *r_size) try { property_buffer buf { r_buf, size, r_size }; auto &prog = obj(d_prog); auto &dev = obj(d_dev); if (!count(dev, prog.ctx.devs())) return CL_INVALID_DEVICE; switch (param) { case CL_PROGRAM_BUILD_STATUS: buf.as_scalar() = prog.build_status(dev); break; case CL_PROGRAM_BUILD_OPTIONS: buf.as_string() = prog.build_opts(dev); break; case CL_PROGRAM_BUILD_LOG: buf.as_string() = prog.build_log(dev); break; default: throw error(CL_INVALID_VALUE); } return CL_SUCCESS; } catch (error &e) { return e.get(); }