| // Copyright 2018 The Dawn & Tint Authors |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are met: |
| // |
| // 1. Redistributions of source code must retain the above copyright notice, this |
| // list of conditions and the following disclaimer. |
| // |
| // 2. Redistributions in binary form must reproduce the above copyright notice, |
| // this list of conditions and the following disclaimer in the documentation |
| // and/or other materials provided with the distribution. |
| // |
| // 3. Neither the name of the copyright holder nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE |
| // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
| // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
| // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, |
| // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #include "dawn/native/vulkan/SamplerVk.h" |
| |
| #include <algorithm> |
| |
| #include "dawn/native/vulkan/DeviceVk.h" |
| #include "dawn/native/vulkan/FencedDeleter.h" |
| #include "dawn/native/vulkan/UtilsVulkan.h" |
| #include "dawn/native/vulkan/VulkanError.h" |
| |
| namespace dawn::native::vulkan { |
| |
| namespace { |
| VkSamplerAddressMode VulkanSamplerAddressMode(wgpu::AddressMode mode) { |
| switch (mode) { |
| case wgpu::AddressMode::Repeat: |
| return VK_SAMPLER_ADDRESS_MODE_REPEAT; |
| case wgpu::AddressMode::MirrorRepeat: |
| return VK_SAMPLER_ADDRESS_MODE_MIRRORED_REPEAT; |
| case wgpu::AddressMode::ClampToEdge: |
| return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE; |
| case wgpu::AddressMode::Undefined: |
| break; |
| } |
| DAWN_UNREACHABLE(); |
| } |
| |
| VkFilter VulkanSamplerFilter(wgpu::FilterMode filter) { |
| switch (filter) { |
| case wgpu::FilterMode::Linear: |
| return VK_FILTER_LINEAR; |
| case wgpu::FilterMode::Nearest: |
| return VK_FILTER_NEAREST; |
| case wgpu::FilterMode::Undefined: |
| break; |
| } |
| DAWN_UNREACHABLE(); |
| } |
| |
| VkSamplerMipmapMode VulkanMipMapMode(wgpu::MipmapFilterMode filter) { |
| switch (filter) { |
| case wgpu::MipmapFilterMode::Linear: |
| return VK_SAMPLER_MIPMAP_MODE_LINEAR; |
| case wgpu::MipmapFilterMode::Nearest: |
| return VK_SAMPLER_MIPMAP_MODE_NEAREST; |
| case wgpu::MipmapFilterMode::Undefined: |
| break; |
| } |
| DAWN_UNREACHABLE(); |
| } |
| } // anonymous namespace |
| |
| // static |
| ResultOrError<Ref<Sampler>> Sampler::Create(Device* device, const SamplerDescriptor* descriptor) { |
| Ref<Sampler> sampler = AcquireRef(new Sampler(device, descriptor)); |
| DAWN_TRY(sampler->Initialize(descriptor)); |
| return sampler; |
| } |
| |
| MaybeError Sampler::Initialize(const SamplerDescriptor* descriptor) { |
| VkSamplerCreateInfo createInfo = {}; |
| createInfo.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO; |
| createInfo.pNext = nullptr; |
| createInfo.flags = 0; |
| createInfo.magFilter = VulkanSamplerFilter(descriptor->magFilter); |
| createInfo.minFilter = VulkanSamplerFilter(descriptor->minFilter); |
| createInfo.mipmapMode = VulkanMipMapMode(descriptor->mipmapFilter); |
| createInfo.addressModeU = VulkanSamplerAddressMode(descriptor->addressModeU); |
| createInfo.addressModeV = VulkanSamplerAddressMode(descriptor->addressModeV); |
| createInfo.addressModeW = VulkanSamplerAddressMode(descriptor->addressModeW); |
| createInfo.mipLodBias = 0.0f; |
| if (descriptor->compare != wgpu::CompareFunction::Undefined) { |
| createInfo.compareOp = ToVulkanCompareOp(descriptor->compare); |
| createInfo.compareEnable = VK_TRUE; |
| } else { |
| // Still set the compareOp so it's not garbage. |
| createInfo.compareOp = VK_COMPARE_OP_NEVER; |
| createInfo.compareEnable = VK_FALSE; |
| } |
| createInfo.minLod = descriptor->lodMinClamp; |
| createInfo.maxLod = descriptor->lodMaxClamp; |
| createInfo.unnormalizedCoordinates = VK_FALSE; |
| |
| Device* device = ToBackend(GetDevice()); |
| uint16_t maxAnisotropy = GetMaxAnisotropy(); |
| if (device->GetDeviceInfo().features.samplerAnisotropy == VK_TRUE && maxAnisotropy > 1) { |
| createInfo.anisotropyEnable = VK_TRUE; |
| // https://www.khronos.org/registry/vulkan/specs/1.2-extensions/man/html/VkSamplerCreateInfo.html |
| createInfo.maxAnisotropy = |
| std::min(static_cast<float>(maxAnisotropy), |
| device->GetDeviceInfo().properties.limits.maxSamplerAnisotropy); |
| } else { |
| createInfo.anisotropyEnable = VK_FALSE; |
| createInfo.maxAnisotropy = 1; |
| } |
| |
| DAWN_TRY(CheckVkSuccess( |
| device->fn.CreateSampler(device->GetVkDevice(), &createInfo, nullptr, &*mHandle), |
| "CreateSampler")); |
| |
| SetLabelImpl(); |
| |
| return {}; |
| } |
| |
| Sampler::~Sampler() = default; |
| |
| void Sampler::DestroyImpl() { |
| SamplerBase::DestroyImpl(); |
| if (mHandle != VK_NULL_HANDLE) { |
| ToBackend(GetDevice())->GetFencedDeleter()->DeleteWhenUnused(mHandle); |
| mHandle = VK_NULL_HANDLE; |
| } |
| } |
| |
| VkSampler Sampler::GetHandle() const { |
| return mHandle; |
| } |
| |
| void Sampler::SetLabelImpl() { |
| SetDebugName(ToBackend(GetDevice()), mHandle, "Dawn_Sampler", GetLabel()); |
| } |
| |
| } // namespace dawn::native::vulkan |