mirror of
https://gitlab.freedesktop.org/mesa/mesa.git
synced 2025-12-20 22:30:12 +01:00
242 lines
7.3 KiB
C++
242 lines
7.3 KiB
C++
|
|
//
|
||
|
|
// 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;
|
||
|
|
}
|
||
|
|
}
|