344 řádky
12 KiB
C++
344 řádky
12 KiB
C++
//
|
|
// Copyright (c) 2008 Advanced Micro Devices, Inc. All rights reserved.
|
|
//
|
|
|
|
#include "cl_common.hpp"
|
|
|
|
#include "platform/context.hpp"
|
|
#include "platform/sampler.hpp"
|
|
|
|
|
|
/*! \addtogroup API
|
|
* @{
|
|
*
|
|
* \addtogroup CL_Samplers
|
|
*
|
|
* A sampler object describes how to sample an image when the image is read
|
|
* in the kernel. The built-in functions to read from an image in a kernel
|
|
* take a sampler as an argument. The sampler arguments to the image read
|
|
* function can be sampler objects created using OpenCL functions and passed
|
|
* as argument values to the kernel or can be samplers declared inside
|
|
* a kernel.
|
|
*
|
|
* @{
|
|
*/
|
|
|
|
/*! \brief Create a sampler object.
|
|
*
|
|
* \param context must be a valid OpenCL context.
|
|
*
|
|
* \param specifies a list of sampler property names and their corresponding
|
|
* values. Each sampler property name is immediately followed by the
|
|
* corresponding desired value. The list is terminated with 0. If a supported
|
|
* property and its value is not specified in sampler_properties, its default
|
|
* value will be used. sampler_properties can be NULL in which case the default
|
|
* values for supported sampler properties will be used.
|
|
*
|
|
* \param errcode_ret will return an appropriate error code. If \a errcode_ret
|
|
* is NULL, no error code is returned.
|
|
*
|
|
* \return A valid non-zero sampler object and \a errcode_ret is set to
|
|
* CL_SUCCESS if the sampler object is created successfully. It returns a NULL
|
|
* value with one of the following error values returned in \a errcode_ret:
|
|
* - CL_INVALID_CONTEXT if \a context is not a valid context.
|
|
* - CL_INVALID_VALUE if the property name in sampler_properties is not a
|
|
* supported property name, if the value specified for a supported property
|
|
* name is not valid, or if the same property name is specified more than
|
|
* once
|
|
* - CL_INVALID_OPERATION if images are not supported by any device associated
|
|
* with context
|
|
* - CL_OUT_OF_HOST_MEMORY if there is a failure to allocate resources required
|
|
* by the runtime.
|
|
*
|
|
* \version 2.0r19
|
|
*/
|
|
RUNTIME_ENTRY_RET(cl_sampler, clCreateSamplerWithProperties,
|
|
(cl_context context, const cl_sampler_properties* sampler_properties,
|
|
cl_int* errcode_ret)) {
|
|
if (!is_valid(context)) {
|
|
*not_null(errcode_ret) = CL_INVALID_CONTEXT;
|
|
LogWarning("invalid parameter \"context\"");
|
|
return (cl_sampler)0;
|
|
}
|
|
|
|
cl_bool normalizedCoords = CL_TRUE;
|
|
cl_addressing_mode addressingMode = CL_ADDRESS_CLAMP;
|
|
cl_filter_mode filterMode = CL_FILTER_NEAREST;
|
|
#ifndef CL_FILTER_NONE
|
|
#define CL_FILTER_NONE 0x1142
|
|
#endif
|
|
cl_filter_mode mipFilterMode = CL_FILTER_NONE;
|
|
float minLod = 0.f;
|
|
float maxLod = CL_MAXFLOAT;
|
|
|
|
const struct SamplerProperty {
|
|
cl_sampler_properties name;
|
|
union {
|
|
cl_sampler_properties raw;
|
|
cl_bool normalizedCoords;
|
|
cl_addressing_mode addressingMode;
|
|
cl_filter_mode filterMode;
|
|
cl_float lod;
|
|
} value;
|
|
}* p = reinterpret_cast<const SamplerProperty*>(sampler_properties);
|
|
|
|
if (p != NULL)
|
|
while (p->name != 0) {
|
|
switch (p->name) {
|
|
case CL_SAMPLER_NORMALIZED_COORDS:
|
|
normalizedCoords = p->value.normalizedCoords;
|
|
break;
|
|
case CL_SAMPLER_ADDRESSING_MODE:
|
|
addressingMode = p->value.addressingMode;
|
|
break;
|
|
case CL_SAMPLER_FILTER_MODE:
|
|
filterMode = p->value.filterMode;
|
|
break;
|
|
case CL_SAMPLER_MIP_FILTER_MODE:
|
|
mipFilterMode = p->value.filterMode;
|
|
break;
|
|
case CL_SAMPLER_LOD_MIN:
|
|
minLod = p->value.lod;
|
|
break;
|
|
case CL_SAMPLER_LOD_MAX:
|
|
maxLod = p->value.lod;
|
|
break;
|
|
default:
|
|
*not_null(errcode_ret) = CL_INVALID_VALUE;
|
|
LogWarning("invalid property name");
|
|
return (cl_sampler)0;
|
|
}
|
|
++p;
|
|
}
|
|
|
|
// Check sampler validity
|
|
// Check addressing mode
|
|
switch (addressingMode) {
|
|
case CL_ADDRESS_NONE:
|
|
case CL_ADDRESS_CLAMP_TO_EDGE:
|
|
case CL_ADDRESS_CLAMP:
|
|
break;
|
|
case CL_ADDRESS_REPEAT:
|
|
if (!normalizedCoords) {
|
|
// repeat mode cannot be used with unnormalized coordinates
|
|
*not_null(errcode_ret) = CL_INVALID_VALUE;
|
|
LogWarning("invalid combination for sampler");
|
|
return (cl_sampler)0;
|
|
}
|
|
break;
|
|
case CL_ADDRESS_MIRRORED_REPEAT:
|
|
if (!normalizedCoords) {
|
|
// repeat mode cannot be used with unnormalized coordinates
|
|
*not_null(errcode_ret) = CL_INVALID_VALUE;
|
|
LogWarning("invalid combination for sampler");
|
|
return (cl_sampler)0;
|
|
}
|
|
break;
|
|
default:
|
|
*not_null(errcode_ret) = CL_INVALID_VALUE;
|
|
LogWarning("invalid addressing mode");
|
|
return (cl_sampler)0;
|
|
}
|
|
// Check filter mode
|
|
switch (filterMode) {
|
|
case CL_FILTER_NEAREST:
|
|
case CL_FILTER_LINEAR:
|
|
break;
|
|
default:
|
|
*not_null(errcode_ret) = CL_INVALID_VALUE;
|
|
LogWarning("invalid filter mode");
|
|
return (cl_sampler)0;
|
|
}
|
|
switch (mipFilterMode) {
|
|
case CL_FILTER_NONE:
|
|
case CL_FILTER_NEAREST:
|
|
case CL_FILTER_LINEAR:
|
|
break;
|
|
default:
|
|
*not_null(errcode_ret) = CL_INVALID_VALUE;
|
|
LogWarning("invalid filter mode");
|
|
return (cl_sampler)0;
|
|
}
|
|
// Create instance of Sampler
|
|
amd::Sampler* sampler =
|
|
new amd::Sampler(*as_amd(context),
|
|
normalizedCoords == CL_TRUE, // To get rid of VS warning C4800
|
|
addressingMode, filterMode, mipFilterMode, minLod, maxLod);
|
|
if (!sampler) {
|
|
*not_null(errcode_ret) = CL_OUT_OF_HOST_MEMORY;
|
|
LogWarning("not enough host memory");
|
|
return (cl_sampler)0;
|
|
}
|
|
|
|
if (!sampler->create()) {
|
|
delete sampler;
|
|
*not_null(errcode_ret) = CL_OUT_OF_HOST_MEMORY;
|
|
LogWarning("Runtime failed sampler creation!");
|
|
return as_cl<amd::Sampler>(0);
|
|
}
|
|
|
|
*not_null(errcode_ret) = CL_SUCCESS;
|
|
return as_cl<amd::Sampler>(sampler);
|
|
}
|
|
RUNTIME_EXIT
|
|
|
|
/*! \brief Create a sampler object.
|
|
*
|
|
* \param context must be a valid OpenCL context.
|
|
*
|
|
* \param addressing_mode specifies how out of range image coordinates are
|
|
* handled when reading from an image. This can be set to CL_ADDRESS_REPEAT,
|
|
* CL_ADDRESS_CLAMP_TO_EDGE, CL_ADDRESS_CLAMP and CL_ADDRESS_NONE.
|
|
*
|
|
* \param filter_mode specifies the type of filter that must be applied when
|
|
* reading an image. This can be CL_FILTER_NEAREST or CL_FILTER_LINEAR.
|
|
*
|
|
* \param normalized_coords determines if the image coordinates specified are
|
|
* normalized (if \a normalized_coords is not zero) or not (if
|
|
* \a normalized_coords is zero).
|
|
*
|
|
* \param errcode_ret will return an appropriate error code. If \a errcode_ret
|
|
* is NULL, no error code is returned.
|
|
*
|
|
* \return A valid non-zero sampler object and \a errcode_ret is set to
|
|
* CL_SUCCESS if the sampler object is created successfully. It returns a NULL
|
|
* value with one of the following error values returned in \a errcode_ret:
|
|
* - CL_INVALID_CONTEXT if \a context is not a valid context.
|
|
* - CL_INVALID_VALUE if \a addressing_mode, \a filter_mode or
|
|
* \a normalized_coords or combination of these argument values are not
|
|
* valid.
|
|
* - CL_INVALID_OPERATION if images are not supported by any device associated
|
|
* with context
|
|
* - CL_OUT_OF_HOST_MEMORY if there is a failure to allocate resources required
|
|
* by the runtime.
|
|
*
|
|
* \version 1.0r33
|
|
*/
|
|
RUNTIME_ENTRY_RET(cl_sampler, clCreateSampler, (cl_context context, cl_bool normalized_coords,
|
|
cl_addressing_mode addressing_mode,
|
|
cl_filter_mode filter_mode, cl_int* errcode_ret)) {
|
|
const cl_sampler_properties sprops[] = {CL_SAMPLER_NORMALIZED_COORDS,
|
|
static_cast<cl_sampler_properties>(normalized_coords),
|
|
CL_SAMPLER_ADDRESSING_MODE,
|
|
static_cast<cl_sampler_properties>(addressing_mode),
|
|
CL_SAMPLER_FILTER_MODE,
|
|
static_cast<cl_sampler_properties>(filter_mode),
|
|
0};
|
|
return clCreateSamplerWithProperties(context, sprops, errcode_ret);
|
|
}
|
|
RUNTIME_EXIT
|
|
|
|
/*! \brief Increment the sampler reference count.
|
|
*
|
|
* clCreateSampler does an implicit retain.
|
|
*
|
|
* \return CL_SUCCESS if the function is executed successfully. It returns
|
|
* CL_INVALID_SAMPLER if \a sampler is not a valid sampler object.
|
|
*
|
|
* \version 1.0r33
|
|
*/
|
|
RUNTIME_ENTRY(cl_int, clRetainSampler, (cl_sampler sampler)) {
|
|
if (!is_valid(sampler)) {
|
|
return CL_INVALID_SAMPLER;
|
|
}
|
|
as_amd(sampler)->retain();
|
|
return CL_SUCCESS;
|
|
}
|
|
RUNTIME_EXIT
|
|
|
|
/*! \brief Decrement the sampler reference count.
|
|
*
|
|
* The sampler object is deleted after the reference count becomes zero and
|
|
* commands queued for execution on a command-queue(s) that use sampler have
|
|
* finished.
|
|
*
|
|
* \return CL_SUCCESS if the function is executed successfully. It returns
|
|
* CL_INVALID_SAMPLER if \a sampler is not a valid sampler object.
|
|
*
|
|
* \version 1.0r33
|
|
*/
|
|
RUNTIME_ENTRY(cl_int, clReleaseSampler, (cl_sampler sampler)) {
|
|
if (!is_valid(sampler)) {
|
|
return CL_INVALID_SAMPLER;
|
|
}
|
|
as_amd(sampler)->release();
|
|
return CL_SUCCESS;
|
|
}
|
|
RUNTIME_EXIT
|
|
|
|
/*! \brief Return information about the sampler object.
|
|
*
|
|
* \param sampler specifies the sampler being queried.
|
|
*
|
|
* \param param_name specifies the information to query.
|
|
*
|
|
* \param param_value is a pointer to memory where the appropriate result
|
|
* being queried is returned. If \a param_value is NULL, it is ignored.
|
|
*
|
|
* \param param_value_size is used to specify the size in bytes of memory
|
|
* pointed to by \a param_value. This size must be >= size of return type.
|
|
*
|
|
* \param param_value_size_ret returns the actual size in bytes of data copied
|
|
* to \a param_value. If \a param_value_size_ret is NULL, it is ignored.
|
|
*
|
|
* \return One of the following values:
|
|
* - CL_SUCCESS if the function is executed successfully
|
|
* - CL_INVALID_VALUE if \a param_name is not valid, or if size in bytes
|
|
* specified by \a param_value_size is < size of return type and
|
|
* \a param_value is not NULL
|
|
* - CL_INVALID_SAMPLER if \a sampler is a not a valid sampler object.
|
|
*
|
|
* \version 1.0r33
|
|
*/
|
|
RUNTIME_ENTRY(cl_int, clGetSamplerInfo,
|
|
(cl_sampler sampler, cl_sampler_info param_name, size_t param_value_size,
|
|
void* param_value, size_t* param_value_size_ret)) {
|
|
if (!is_valid(sampler)) {
|
|
return CL_INVALID_SAMPLER;
|
|
}
|
|
|
|
switch (param_name) {
|
|
case CL_SAMPLER_REFERENCE_COUNT: {
|
|
cl_uint count = as_amd(sampler)->referenceCount();
|
|
return amd::clGetInfo(count, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_CONTEXT: {
|
|
cl_context context = as_cl(&as_amd(sampler)->context());
|
|
return amd::clGetInfo(context, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_ADDRESSING_MODE: {
|
|
cl_addressing_mode addressing = as_amd(sampler)->addressingMode();
|
|
return amd::clGetInfo(addressing, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_FILTER_MODE: {
|
|
cl_filter_mode filter = as_amd(sampler)->filterMode();
|
|
return amd::clGetInfo(filter, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_NORMALIZED_COORDS: {
|
|
cl_bool normalized = as_amd(sampler)->normalizedCoords();
|
|
return amd::clGetInfo(normalized, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_MIP_FILTER_MODE: {
|
|
cl_filter_mode mipFilter = as_amd(sampler)->mipFilter();
|
|
return amd::clGetInfo(mipFilter, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_LOD_MIN: {
|
|
cl_float minLod = as_amd(sampler)->minLod();
|
|
return amd::clGetInfo(minLod, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
case CL_SAMPLER_LOD_MAX: {
|
|
cl_float maxLod = as_amd(sampler)->maxLod();
|
|
return amd::clGetInfo(maxLod, param_value_size, param_value, param_value_size_ret);
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return CL_INVALID_VALUE;
|
|
}
|
|
RUNTIME_EXIT
|
|
|
|
/*! @}
|
|
* @}
|
|
*/
|