blob: f6242c371759a9e8ff31a26207e9d33637d6c413 [file] [log] [blame]
jvanverth992ad362016-02-26 09:21:02 -08001/*
2* Copyright 2016 Google Inc.
3*
4* Use of this source code is governed by a BSD-style license that can be
5* found in the LICENSE file.
6*/
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/vk/GrVkSampler.h"
jvanverth992ad362016-02-26 09:21:02 -08009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "src/gpu/vk/GrVkGpu.h"
11#include "src/gpu/vk/GrVkSamplerYcbcrConversion.h"
jvanverth992ad362016-02-26 09:21:02 -080012
Brian Salomon2bbdcc42017-09-07 12:36:34 -040013static inline VkSamplerAddressMode wrap_mode_to_vk_sampler_address(
14 GrSamplerState::WrapMode wrapMode) {
15 switch (wrapMode) {
16 case GrSamplerState::WrapMode::kClamp:
17 return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
18 case GrSamplerState::WrapMode::kRepeat:
19 return VK_SAMPLER_ADDRESS_MODE_REPEAT;
20 case GrSamplerState::WrapMode::kMirrorRepeat:
21 return VK_SAMPLER_ADDRESS_MODE_MIRRORED_REPEAT;
Michael Ludwigf23a1522018-12-10 11:36:13 -050022 case GrSamplerState::WrapMode::kClampToBorder:
23 return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER;
Brian Salomon2bbdcc42017-09-07 12:36:34 -040024 }
25 SK_ABORT("Unknown wrap mode.");
jvanverth992ad362016-02-26 09:21:02 -080026}
27
Greg Daniel7e000222018-12-03 10:08:21 -050028GrVkSampler* GrVkSampler::Create(GrVkGpu* gpu, const GrSamplerState& samplerState,
29 const GrVkYcbcrConversionInfo& ycbcrInfo) {
jvanverth992ad362016-02-26 09:21:02 -080030 static VkFilter vkMinFilterModes[] = {
31 VK_FILTER_NEAREST,
32 VK_FILTER_LINEAR,
33 VK_FILTER_LINEAR
34 };
35 static VkFilter vkMagFilterModes[] = {
36 VK_FILTER_NEAREST,
37 VK_FILTER_LINEAR,
38 VK_FILTER_LINEAR
39 };
40
jvanverth992ad362016-02-26 09:21:02 -080041 VkSamplerCreateInfo createInfo;
42 memset(&createInfo, 0, sizeof(VkSamplerCreateInfo));
43 createInfo.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO;
Greg Daniel7e000222018-12-03 10:08:21 -050044 createInfo.pNext = nullptr;
jvanverth992ad362016-02-26 09:21:02 -080045 createInfo.flags = 0;
Brian Salomon2bbdcc42017-09-07 12:36:34 -040046 createInfo.magFilter = vkMagFilterModes[static_cast<int>(samplerState.filter())];
47 createInfo.minFilter = vkMinFilterModes[static_cast<int>(samplerState.filter())];
brianosman88791862016-05-23 10:15:27 -070048 createInfo.mipmapMode = VK_SAMPLER_MIPMAP_MODE_LINEAR;
Brian Salomon2bbdcc42017-09-07 12:36:34 -040049 createInfo.addressModeU = wrap_mode_to_vk_sampler_address(samplerState.wrapModeX());
50 createInfo.addressModeV = wrap_mode_to_vk_sampler_address(samplerState.wrapModeY());
jvanverth992ad362016-02-26 09:21:02 -080051 createInfo.addressModeW = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE; // Shouldn't matter
52 createInfo.mipLodBias = 0.0f;
53 createInfo.anisotropyEnable = VK_FALSE;
54 createInfo.maxAnisotropy = 1.0f;
55 createInfo.compareEnable = VK_FALSE;
56 createInfo.compareOp = VK_COMPARE_OP_NEVER;
egdaniel8f1dcaa2016-04-01 10:10:45 -070057 // Vulkan doesn't have a direct mapping of GL's nearest or linear filters for minFilter since
58 // there is always a mipmapMode. To get the same effect as GL we can set minLod = maxLod = 0.0.
59 // This works since our min and mag filters are the same (this forces us to use mag on the 0
60 // level mip). If the filters weren't the same we could set min = 0 and max = 0.25 to force
61 // the minFilter on mip level 0.
jvanverth992ad362016-02-26 09:21:02 -080062 createInfo.minLod = 0.0f;
Greg Daniel6616efb2018-11-27 16:56:29 -050063 bool useMipMaps = GrSamplerState::Filter::kMipMap == samplerState.filter();
64 createInfo.maxLod = !useMipMaps ? 0.0f : 10000.0f;
jvanverth992ad362016-02-26 09:21:02 -080065 createInfo.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK;
66 createInfo.unnormalizedCoordinates = VK_FALSE;
67
Greg Daniel7e000222018-12-03 10:08:21 -050068 VkSamplerYcbcrConversionInfo conversionInfo;
69 GrVkSamplerYcbcrConversion* ycbcrConversion = nullptr;
70 if (ycbcrInfo.isValid()) {
71 SkASSERT(gpu->vkCaps().supportsYcbcrConversion());
72
73 ycbcrConversion =
74 gpu->resourceProvider().findOrCreateCompatibleSamplerYcbcrConversion(ycbcrInfo);
75 if (!ycbcrConversion) {
76 return nullptr;
77 }
78
79 conversionInfo.sType = VK_STRUCTURE_TYPE_SAMPLER_YCBCR_CONVERSION_INFO;
80 conversionInfo.pNext = nullptr;
81 conversionInfo.conversion = ycbcrConversion->ycbcrConversion();
82
83 createInfo.pNext = &conversionInfo;
84
Sergey Ulanov2739fd22019-08-11 22:46:33 -070085 VkFormatFeatureFlags flags = ycbcrInfo.fFormatFeatures;
Greg Daniel7e000222018-12-03 10:08:21 -050086 if (!SkToBool(flags & VK_FORMAT_FEATURE_SAMPLED_IMAGE_YCBCR_CONVERSION_LINEAR_FILTER_BIT)) {
87 createInfo.magFilter = VK_FILTER_NEAREST;
88 createInfo.minFilter = VK_FILTER_NEAREST;
89 } else if (
90 !(flags &
91 VK_FORMAT_FEATURE_SAMPLED_IMAGE_YCBCR_CONVERSION_SEPARATE_RECONSTRUCTION_FILTER_BIT)) {
92 createInfo.magFilter = ycbcrInfo.fChromaFilter;
93 createInfo.minFilter = ycbcrInfo.fChromaFilter;
94 }
95
96 // Required values when using ycbcr conversion
97 createInfo.addressModeU = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
98 createInfo.addressModeV = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
99 createInfo.addressModeW = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
100 createInfo.anisotropyEnable = VK_FALSE;
101 createInfo.unnormalizedCoordinates = VK_FALSE;
102 }
103
jvanverth992ad362016-02-26 09:21:02 -0800104 VkSampler sampler;
105 GR_VK_CALL_ERRCHECK(gpu->vkInterface(), CreateSampler(gpu->device(),
106 &createInfo,
107 nullptr,
108 &sampler));
109
Greg Daniel7e000222018-12-03 10:08:21 -0500110 return new GrVkSampler(sampler, ycbcrConversion, GenerateKey(samplerState, ycbcrInfo));
jvanverth992ad362016-02-26 09:21:02 -0800111}
112
Ethan Nicholas8e265a72018-12-12 16:22:40 -0500113void GrVkSampler::freeGPUData(GrVkGpu* gpu) const {
jvanverth992ad362016-02-26 09:21:02 -0800114 SkASSERT(fSampler);
115 GR_VK_CALL(gpu->vkInterface(), DestroySampler(gpu->device(), fSampler, nullptr));
Greg Daniel7e000222018-12-03 10:08:21 -0500116 if (fYcbcrConversion) {
117 fYcbcrConversion->unref(gpu);
118 }
jvanverth992ad362016-02-26 09:21:02 -0800119}
egdaniel8b6394c2016-03-04 07:35:10 -0800120
Greg Daniel7e000222018-12-03 10:08:21 -0500121void GrVkSampler::abandonGPUData() const {
122 if (fYcbcrConversion) {
123 fYcbcrConversion->unrefAndAbandon();
124 }
125}
126
127GrVkSampler::Key GrVkSampler::GenerateKey(const GrSamplerState& samplerState,
128 const GrVkYcbcrConversionInfo& ycbcrInfo) {
jvanverth62340062016-04-26 08:01:44 -0700129 const int kTileModeXShift = 2;
130 const int kTileModeYShift = 4;
egdaniel8b6394c2016-03-04 07:35:10 -0800131
Brian Salomonf4b9bf72017-09-21 13:35:12 -0400132 SkASSERT(static_cast<int>(samplerState.filter()) <= 3);
Greg Daniel7e000222018-12-03 10:08:21 -0500133 uint8_t samplerKey = static_cast<uint16_t>(samplerState.filter());
halcanary9d524f22016-03-29 09:03:52 -0700134
Greg Daniel6616efb2018-11-27 16:56:29 -0500135 SkASSERT(static_cast<int>(samplerState.wrapModeX()) <= 3);
Greg Daniel7e000222018-12-03 10:08:21 -0500136 samplerKey |= (static_cast<uint8_t>(samplerState.wrapModeX()) << kTileModeXShift);
egdaniel8b6394c2016-03-04 07:35:10 -0800137
Greg Daniel6616efb2018-11-27 16:56:29 -0500138 SkASSERT(static_cast<int>(samplerState.wrapModeY()) <= 3);
Greg Daniel7e000222018-12-03 10:08:21 -0500139 samplerKey |= (static_cast<uint8_t>(samplerState.wrapModeY()) << kTileModeYShift);
egdaniel8b6394c2016-03-04 07:35:10 -0800140
Greg Daniel7e000222018-12-03 10:08:21 -0500141 return {samplerKey, GrVkSamplerYcbcrConversion::GenerateKey(ycbcrInfo)};
egdaniel8b6394c2016-03-04 07:35:10 -0800142}
Greg Daniel7e000222018-12-03 10:08:21 -0500143