/* * Copyright (C) 2018-2021 Intel Corporation * * SPDX-License-Identifier: MIT * */ #include "opencl/source/sampler/sampler.h" #include "shared/source/helpers/get_info.h" #include "shared/source/helpers/hw_info.h" #include "opencl/source/cl_device/cl_device.h" #include "opencl/source/context/context.h" #include "opencl/source/helpers/get_info_status_mapper.h" #include "patch_list.h" #include namespace NEO { SamplerCreateFunc samplerFactory[IGFX_MAX_CORE] = {}; Sampler::Sampler(Context *context, cl_bool normalizedCoordinates, cl_addressing_mode addressingMode, cl_filter_mode filterMode, cl_filter_mode mipFilterMode, float lodMin, float lodMax) : context(context), normalizedCoordinates(normalizedCoordinates), addressingMode(addressingMode), filterMode(filterMode), mipFilterMode(mipFilterMode), lodMin(lodMin), lodMax(lodMax) { } Sampler::Sampler(Context *context, cl_bool normalizedCoordinates, cl_addressing_mode addressingMode, cl_filter_mode filterMode) : Sampler(context, normalizedCoordinates, addressingMode, filterMode, CL_FILTER_NEAREST, 0.0f, std::numeric_limits::max()) { } Sampler *Sampler::create(Context *context, cl_bool normalizedCoordinates, cl_addressing_mode addressingMode, cl_filter_mode filterMode, cl_filter_mode mipFilterMode, float lodMin, float lodMax, cl_int &errcodeRet) { errcodeRet = CL_SUCCESS; Sampler *sampler = nullptr; DEBUG_BREAK_IF(nullptr == context); const auto device = context->getDevice(0); const auto &hwInfo = device->getHardwareInfo(); auto funcCreate = samplerFactory[hwInfo.platform.eRenderCoreFamily]; DEBUG_BREAK_IF(nullptr == funcCreate); sampler = funcCreate(context, normalizedCoordinates, addressingMode, filterMode, mipFilterMode, lodMin, lodMax); if (sampler == nullptr) { errcodeRet = CL_OUT_OF_HOST_MEMORY; } return sampler; } template struct SetOnce { SetOnce(ParameterType defaultValue, ParameterType min, ParameterType max) : value(defaultValue), min(min), max(max) { } cl_int setValue(ParameterType property) { if (alreadySet) { return CL_INVALID_VALUE; } if ((property < min) || (property > max)) { return CL_INVALID_VALUE; } this->value = property; alreadySet = true; return CL_SUCCESS; } bool alreadySet = false; ParameterType value; ParameterType min; ParameterType max; }; Sampler *Sampler::create(Context *context, const cl_sampler_properties *samplerProperties, cl_int &errcodeRet) { SetOnce normalizedCoords(CL_TRUE, CL_FALSE, CL_TRUE); SetOnce filterMode(CL_FILTER_NEAREST, CL_FILTER_NEAREST, CL_FILTER_LINEAR); SetOnce addressingMode(CL_ADDRESS_CLAMP, CL_ADDRESS_NONE, CL_ADDRESS_MIRRORED_REPEAT); SetOnce mipFilterMode(CL_FILTER_NEAREST, CL_FILTER_NEAREST, CL_FILTER_LINEAR); SetOnce lodMin(0.0f, 0.0f, std::numeric_limits::max()); SetOnce lodMax(std::numeric_limits::max(), 0.0f, std::numeric_limits::max()); errcodeRet = CL_SUCCESS; auto samplerProperty = samplerProperties; if (samplerProperty) { cl_ulong samType; while ((samType = *samplerProperty) != 0) { ++samplerProperty; auto samValue = *samplerProperty; switch (samType) { case CL_SAMPLER_NORMALIZED_COORDS: errcodeRet = normalizedCoords.setValue(static_cast(samValue)); break; case CL_SAMPLER_ADDRESSING_MODE: errcodeRet = addressingMode.setValue(static_cast(samValue)); break; case CL_SAMPLER_FILTER_MODE: errcodeRet = filterMode.setValue(static_cast(samValue)); break; case CL_SAMPLER_MIP_FILTER_MODE: errcodeRet = mipFilterMode.setValue(static_cast(samValue)); break; case CL_SAMPLER_LOD_MIN: { SamplerLodProperty lodData; lodData.data = samValue; errcodeRet = lodMin.setValue(lodData.lod); break; } case CL_SAMPLER_LOD_MAX: { SamplerLodProperty lodData; lodData.data = samValue; errcodeRet = lodMax.setValue(lodData.lod); break; } default: errcodeRet = CL_INVALID_VALUE; break; } ++samplerProperty; } } Sampler *sampler = nullptr; if (errcodeRet == CL_SUCCESS) { sampler = create(context, normalizedCoords.value, addressingMode.value, filterMode.value, mipFilterMode.value, lodMin.value, lodMax.value, errcodeRet); } if (errcodeRet == CL_SUCCESS) { sampler->storeProperties(samplerProperties); } return sampler; } unsigned int Sampler::getSnapWaValue() const { if (filterMode == CL_FILTER_NEAREST && addressingMode == CL_ADDRESS_CLAMP) { return iOpenCL::CONSTANT_REGISTER_BOOLEAN_TRUE; } else { return iOpenCL::CONSTANT_REGISTER_BOOLEAN_FALSE; } } cl_int Sampler::getInfo(cl_sampler_info paramName, size_t paramValueSize, void *paramValue, size_t *paramValueSizeRet) { cl_int retVal; size_t valueSize = GetInfo::invalidSourceSize; const void *pValue = nullptr; cl_uint refCount = 0; switch (paramName) { case CL_SAMPLER_CONTEXT: valueSize = sizeof(cl_device_id); pValue = &this->context; break; case CL_SAMPLER_NORMALIZED_COORDS: valueSize = sizeof(cl_bool); pValue = &this->normalizedCoordinates; break; case CL_SAMPLER_ADDRESSING_MODE: valueSize = sizeof(cl_addressing_mode); pValue = &this->addressingMode; break; case CL_SAMPLER_FILTER_MODE: valueSize = sizeof(cl_filter_mode); pValue = &this->filterMode; break; case CL_SAMPLER_MIP_FILTER_MODE: valueSize = sizeof(cl_filter_mode); pValue = &this->mipFilterMode; break; case CL_SAMPLER_LOD_MIN: valueSize = sizeof(float); pValue = &this->lodMin; break; case CL_SAMPLER_LOD_MAX: valueSize = sizeof(float); pValue = &this->lodMax; break; case CL_SAMPLER_REFERENCE_COUNT: refCount = static_cast(this->getReference()); valueSize = sizeof(refCount); pValue = &refCount; break; case CL_SAMPLER_PROPERTIES: valueSize = propertiesVector.size() * sizeof(cl_sampler_properties); pValue = propertiesVector.data(); break; default: break; } auto getInfoStatus = GetInfo::getInfo(paramValue, paramValueSize, pValue, valueSize); retVal = changeGetInfoStatusToCLResultType(getInfoStatus); GetInfo::setParamValueReturnSize(paramValueSizeRet, valueSize, getInfoStatus); return retVal; } bool Sampler::isTransformable() const { return addressingMode == CL_ADDRESS_CLAMP_TO_EDGE && filterMode == CL_FILTER_NEAREST && normalizedCoordinates == CL_FALSE; } void Sampler::storeProperties(const cl_sampler_properties *properties) { if (properties) { for (size_t i = 0; properties[i] != 0; i += 2) { propertiesVector.push_back(properties[i]); propertiesVector.push_back(properties[i + 1]); } propertiesVector.push_back(0); } } } // namespace NEO