summaryrefslogtreecommitdiff
path: root/src/gallium/state_trackers/clover/api/program.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/gallium/state_trackers/clover/api/program.cpp')
-rw-r--r--src/gallium/state_trackers/clover/api/program.cpp241
1 files changed, 241 insertions, 0 deletions
diff --git a/src/gallium/state_trackers/clover/api/program.cpp b/src/gallium/state_trackers/clover/api/program.cpp
new file mode 100644
index 00000000000..e874c51ad7d
--- /dev/null
+++ b/src/gallium/state_trackers/clover/api/program.cpp
@@ -0,0 +1,241 @@
+//
+// 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 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;
+
+PUBLIC cl_program
+clCreateProgramWithSource(cl_context ctx, cl_uint count,
+ const char **strings, const size_t *lengths,
+ cl_int *errcode_ret) try {
+ std::string source;
+
+ if (!ctx)
+ throw error(CL_INVALID_CONTEXT);
+
+ if (!count || !strings ||
+ any_of(is_zero<const char *>(), strings, 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(errcode_ret, CL_SUCCESS);
+ return new program(*ctx, source);
+
+} catch (error &e) {
+ ret_error(errcode_ret, e);
+ return NULL;
+}
+
+PUBLIC cl_program
+clCreateProgramWithBinary(cl_context ctx, cl_uint count,
+ const cl_device_id *devs, const size_t *lengths,
+ const unsigned char **binaries, cl_int *status_ret,
+ cl_int *errcode_ret) try {
+ if (!ctx)
+ throw error(CL_INVALID_CONTEXT);
+
+ if (!count || !devs || !lengths || !binaries)
+ throw error(CL_INVALID_VALUE);
+
+ if (any_of([&](const cl_device_id dev) {
+ return !ctx->has_device(dev);
+ }, devs, devs + count))
+ throw error(CL_INVALID_DEVICE);
+
+ // Deserialize the provided binaries,
+ auto modules = map(
+ [](const unsigned char *p, size_t l) -> std::pair<cl_int, module> {
+ 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, {} };
+ }
+ },
+ binaries, binaries + count, lengths);
+
+ // update the status array,
+ if (status_ret)
+ std::transform(modules.begin(), modules.end(), status_ret,
+ keys<cl_int, module>);
+
+ if (any_of(key_equals<cl_int, module>(CL_INVALID_VALUE),
+ modules.begin(), modules.end()))
+ throw error(CL_INVALID_VALUE);
+
+ if (any_of(key_equals<cl_int, module>(CL_INVALID_BINARY),
+ modules.begin(), modules.end()))
+ throw error(CL_INVALID_BINARY);
+
+ // initialize a program object with them.
+ ret_error(errcode_ret, CL_SUCCESS);
+ return new program(*ctx, { devs, devs + count },
+ map(values<cl_int, module>,
+ modules.begin(), modules.end()));
+
+} catch (error &e) {
+ ret_error(errcode_ret, e);
+ return NULL;
+}
+
+PUBLIC cl_int
+clRetainProgram(cl_program prog) {
+ if (!prog)
+ return CL_INVALID_PROGRAM;
+
+ prog->retain();
+ return CL_SUCCESS;
+}
+
+PUBLIC cl_int
+clReleaseProgram(cl_program prog) {
+ if (!prog)
+ return CL_INVALID_PROGRAM;
+
+ if (prog->release())
+ delete prog;
+
+ return CL_SUCCESS;
+}
+
+PUBLIC cl_int
+clBuildProgram(cl_program prog, cl_uint count, const cl_device_id *devs,
+ const char *opts, void (*pfn_notify)(cl_program, void *),
+ void *user_data) try {
+ if (!prog)
+ throw error(CL_INVALID_PROGRAM);
+
+ if (bool(count) != bool(devs) ||
+ (!pfn_notify && user_data))
+ throw error(CL_INVALID_VALUE);
+
+ if (any_of([&](const cl_device_id dev) {
+ return !prog->ctx.has_device(dev);
+ }, devs, devs + count))
+ throw error(CL_INVALID_DEVICE);
+
+ prog->build({ devs, devs + count });
+ return CL_SUCCESS;
+
+} catch (error &e) {
+ return e.get();
+}
+
+PUBLIC cl_int
+clUnloadCompiler() {
+ return CL_SUCCESS;
+}
+
+PUBLIC cl_int
+clGetProgramInfo(cl_program prog, cl_program_info param,
+ size_t size, void *buf, size_t *size_ret) {
+ if (!prog)
+ return CL_INVALID_PROGRAM;
+
+ switch (param) {
+ case CL_PROGRAM_REFERENCE_COUNT:
+ return scalar_property<cl_uint>(buf, size, size_ret,
+ prog->ref_count());
+
+ case CL_PROGRAM_CONTEXT:
+ return scalar_property<cl_context>(buf, size, size_ret,
+ &prog->ctx);
+
+ case CL_PROGRAM_NUM_DEVICES:
+ return scalar_property<cl_uint>(buf, size, size_ret,
+ prog->binaries().size());
+
+ case CL_PROGRAM_DEVICES:
+ return vector_property<cl_device_id>(
+ buf, size, size_ret,
+ map(keys<device *, module>,
+ prog->binaries().begin(), prog->binaries().end()));
+
+ case CL_PROGRAM_SOURCE:
+ return string_property(buf, size, size_ret, prog->source());
+
+ case CL_PROGRAM_BINARY_SIZES:
+ return vector_property<size_t>(
+ buf, size, size_ret,
+ map([](const std::pair<device *, module> &ent) {
+ compat::ostream::buffer_t bin;
+ compat::ostream s(bin);
+ ent.second.serialize(s);
+ return bin.size();
+ },
+ prog->binaries().begin(), prog->binaries().end()));
+
+ case CL_PROGRAM_BINARIES:
+ return matrix_property<unsigned char>(
+ buf, size, size_ret,
+ map([](const std::pair<device *, module> &ent) {
+ compat::ostream::buffer_t bin;
+ compat::ostream s(bin);
+ ent.second.serialize(s);
+ return bin;
+ },
+ prog->binaries().begin(), prog->binaries().end()));
+
+ default:
+ return CL_INVALID_VALUE;
+ }
+}
+
+PUBLIC cl_int
+clGetProgramBuildInfo(cl_program prog, cl_device_id dev,
+ cl_program_build_info param,
+ size_t size, void *buf, size_t *size_ret) {
+ if (!prog)
+ return CL_INVALID_PROGRAM;
+
+ if (!prog->ctx.has_device(dev))
+ return CL_INVALID_DEVICE;
+
+ switch (param) {
+ case CL_PROGRAM_BUILD_STATUS:
+ return scalar_property<cl_build_status>(buf, size, size_ret,
+ prog->build_status(dev));
+
+ case CL_PROGRAM_BUILD_OPTIONS:
+ return string_property(buf, size, size_ret, prog->build_opts(dev));
+
+ case CL_PROGRAM_BUILD_LOG:
+ return string_property(buf, size, size_ret, prog->build_log(dev));
+
+ default:
+ return CL_INVALID_VALUE;
+ }
+}