blob: 2104cb527676564bd57d1544a9e1c060e9d88edb [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
8#include "GrVkSampler.h"
9
jvanverth992ad362016-02-26 09:21:02 -080010#include "GrVkGpu.h"
11
Brian Salomon2bbdcc42017-09-07 12:36:34 -040012static inline VkSamplerAddressMode wrap_mode_to_vk_sampler_address(
13 GrSamplerState::WrapMode wrapMode) {
14 switch (wrapMode) {
15 case GrSamplerState::WrapMode::kClamp:
16 return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
17 case GrSamplerState::WrapMode::kRepeat:
18 return VK_SAMPLER_ADDRESS_MODE_REPEAT;
19 case GrSamplerState::WrapMode::kMirrorRepeat:
20 return VK_SAMPLER_ADDRESS_MODE_MIRRORED_REPEAT;
21 }
22 SK_ABORT("Unknown wrap mode.");
23 return VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
jvanverth992ad362016-02-26 09:21:02 -080024}
25
Greg Daniel6616efb2018-11-27 16:56:29 -050026GrVkSampler* GrVkSampler::Create(const GrVkGpu* gpu, const GrSamplerState& samplerState) {
jvanverth992ad362016-02-26 09:21:02 -080027 static VkFilter vkMinFilterModes[] = {
28 VK_FILTER_NEAREST,
29 VK_FILTER_LINEAR,
30 VK_FILTER_LINEAR
31 };
32 static VkFilter vkMagFilterModes[] = {
33 VK_FILTER_NEAREST,
34 VK_FILTER_LINEAR,
35 VK_FILTER_LINEAR
36 };
37
jvanverth992ad362016-02-26 09:21:02 -080038 VkSamplerCreateInfo createInfo;
39 memset(&createInfo, 0, sizeof(VkSamplerCreateInfo));
40 createInfo.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO;
41 createInfo.pNext = 0;
42 createInfo.flags = 0;
Brian Salomon2bbdcc42017-09-07 12:36:34 -040043 createInfo.magFilter = vkMagFilterModes[static_cast<int>(samplerState.filter())];
44 createInfo.minFilter = vkMinFilterModes[static_cast<int>(samplerState.filter())];
brianosman88791862016-05-23 10:15:27 -070045 createInfo.mipmapMode = VK_SAMPLER_MIPMAP_MODE_LINEAR;
Brian Salomon2bbdcc42017-09-07 12:36:34 -040046 createInfo.addressModeU = wrap_mode_to_vk_sampler_address(samplerState.wrapModeX());
47 createInfo.addressModeV = wrap_mode_to_vk_sampler_address(samplerState.wrapModeY());
jvanverth992ad362016-02-26 09:21:02 -080048 createInfo.addressModeW = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE; // Shouldn't matter
49 createInfo.mipLodBias = 0.0f;
50 createInfo.anisotropyEnable = VK_FALSE;
51 createInfo.maxAnisotropy = 1.0f;
52 createInfo.compareEnable = VK_FALSE;
53 createInfo.compareOp = VK_COMPARE_OP_NEVER;
egdaniel8f1dcaa2016-04-01 10:10:45 -070054 // Vulkan doesn't have a direct mapping of GL's nearest or linear filters for minFilter since
55 // there is always a mipmapMode. To get the same effect as GL we can set minLod = maxLod = 0.0.
56 // This works since our min and mag filters are the same (this forces us to use mag on the 0
57 // level mip). If the filters weren't the same we could set min = 0 and max = 0.25 to force
58 // the minFilter on mip level 0.
jvanverth992ad362016-02-26 09:21:02 -080059 createInfo.minLod = 0.0f;
Greg Daniel6616efb2018-11-27 16:56:29 -050060 bool useMipMaps = GrSamplerState::Filter::kMipMap == samplerState.filter();
61 createInfo.maxLod = !useMipMaps ? 0.0f : 10000.0f;
jvanverth992ad362016-02-26 09:21:02 -080062 createInfo.borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK;
63 createInfo.unnormalizedCoordinates = VK_FALSE;
64
65 VkSampler sampler;
66 GR_VK_CALL_ERRCHECK(gpu->vkInterface(), CreateSampler(gpu->device(),
67 &createInfo,
68 nullptr,
69 &sampler));
70
Greg Daniel6616efb2018-11-27 16:56:29 -050071 return new GrVkSampler(sampler, GenerateKey(samplerState));
jvanverth992ad362016-02-26 09:21:02 -080072}
73
74void GrVkSampler::freeGPUData(const GrVkGpu* gpu) const {
75 SkASSERT(fSampler);
76 GR_VK_CALL(gpu->vkInterface(), DestroySampler(gpu->device(), fSampler, nullptr));
77}
egdaniel8b6394c2016-03-04 07:35:10 -080078
Greg Daniel6616efb2018-11-27 16:56:29 -050079uint8_t GrVkSampler::GenerateKey(const GrSamplerState& samplerState) {
jvanverth62340062016-04-26 08:01:44 -070080 const int kTileModeXShift = 2;
81 const int kTileModeYShift = 4;
egdaniel8b6394c2016-03-04 07:35:10 -080082
Brian Salomonf4b9bf72017-09-21 13:35:12 -040083 SkASSERT(static_cast<int>(samplerState.filter()) <= 3);
Greg Daniel6616efb2018-11-27 16:56:29 -050084 uint8_t key = static_cast<uint8_t>(samplerState.filter());
halcanary9d524f22016-03-29 09:03:52 -070085
Greg Daniel6616efb2018-11-27 16:56:29 -050086 SkASSERT(static_cast<int>(samplerState.wrapModeX()) <= 3);
87 key |= (static_cast<uint8_t>(samplerState.wrapModeX()) << kTileModeXShift);
egdaniel8b6394c2016-03-04 07:35:10 -080088
Greg Daniel6616efb2018-11-27 16:56:29 -050089 SkASSERT(static_cast<int>(samplerState.wrapModeY()) <= 3);
90 key |= (static_cast<uint8_t>(samplerState.wrapModeY()) << kTileModeYShift);
egdaniel8b6394c2016-03-04 07:35:10 -080091
92 return key;
93}