blob: dd84e70ea2ac7dd8d443a89fd169e4f7f5269b43 [file] [log] [blame]
Ben Clayton96fbe082019-04-16 19:28:11 -04001// Copyright 2019 The SwiftShader Authors. All Rights Reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
Ben Clayton96fbe082019-04-16 19:28:11 -040015#include "SpirvShader.hpp"
16
Ben Claytonbc1c067be2019-12-17 20:37:37 +000017#include "SamplerCore.hpp" // TODO: Figure out what's needed.
18#include "Device/Config.hpp"
Ben Clayton25e06e02020-02-07 11:19:08 +000019#include "System/Debug.hpp"
Ben Clayton96fbe082019-04-16 19:28:11 -040020#include "System/Math.hpp"
Alexis Hetu6448bd62019-06-11 15:58:59 -040021#include "Vulkan/VkDescriptorSetLayout.hpp"
22#include "Vulkan/VkDevice.hpp"
Ben Clayton96fbe082019-04-16 19:28:11 -040023#include "Vulkan/VkImageView.hpp"
24#include "Vulkan/VkSampler.hpp"
Ben Clayton96fbe082019-04-16 19:28:11 -040025
26#include <spirv/unified1/spirv.hpp>
Ben Clayton96fbe082019-04-16 19:28:11 -040027
Alexis Hetu710fcd52019-05-24 17:20:21 -040028#include <climits>
Ben Clayton96fbe082019-04-16 19:28:11 -040029#include <mutex>
30
Ben Clayton96fbe082019-04-16 19:28:11 -040031namespace sw {
32
Chris Forbes45f9a932019-05-08 13:30:38 -070033SpirvShader::ImageSampler *SpirvShader::getImageSampler(uint32_t inst, vk::SampledImageDescriptor const *imageDescriptor, const vk::Sampler *sampler)
Nicolas Capens125dba02019-04-24 02:03:22 -040034{
Nicolas Capensfa7d1362019-05-01 20:26:14 -040035 ImageInstruction instruction(inst);
Nicolas Capensdbd02752019-08-20 10:59:58 -040036 const auto samplerId = sampler ? sampler->id : 0;
37 ASSERT(imageDescriptor->imageViewId != 0 && (samplerId != 0 || instruction.samplerMethod == Fetch));
Nicolas Capensfa7d1362019-05-01 20:26:14 -040038
Ben Claytonbc1c067be2019-12-17 20:37:37 +000039 vk::Device::SamplingRoutineCache::Key key = { inst, imageDescriptor->imageViewId, samplerId };
Ben Clayton96fbe082019-04-16 19:28:11 -040040
Alexis Hetu6448bd62019-06-11 15:58:59 -040041 ASSERT(imageDescriptor->device);
Ben Clayton96fbe082019-04-16 19:28:11 -040042
Nicolas Capensd53eeca2020-03-19 13:31:41 -040043 if(auto routine = imageDescriptor->device->querySnapshotCache(key))
Alexis Hetu35755502019-07-22 13:51:49 -040044 {
Ben Claytonbc1c067be2019-12-17 20:37:37 +000045 return (ImageSampler *)(routine->getEntry());
Alexis Hetu35755502019-07-22 13:51:49 -040046 }
47
Ben Claytonbc1c067be2019-12-17 20:37:37 +000048 vk::Device::SamplingRoutineCache *cache = imageDescriptor->device->getSamplingRoutineCache();
Alexis Hetu6448bd62019-06-11 15:58:59 -040049
Nicolas Capens1c294772020-03-28 23:04:55 -040050 auto createSamplingRoutine = [&](const vk::Device::SamplingRoutineCache::Key &key) {
51 auto type = imageDescriptor->type;
Ben Clayton96fbe082019-04-16 19:28:11 -040052
Nicolas Capens1c294772020-03-28 23:04:55 -040053 Sampler samplerState = {};
54 samplerState.textureType = type;
55 samplerState.textureFormat = imageDescriptor->format;
Chris Forbes45f9a932019-05-08 13:30:38 -070056
Nicolas Capens1c294772020-03-28 23:04:55 -040057 samplerState.addressingModeU = convertAddressingMode(0, sampler, type);
58 samplerState.addressingModeV = convertAddressingMode(1, sampler, type);
59 samplerState.addressingModeW = convertAddressingMode(2, sampler, type);
60 samplerState.addressingModeY = convertAddressingMode(3, sampler, type);
Nicolas Capens51e9e562019-05-16 14:01:16 -040061
Nicolas Capensdbd02752019-08-20 10:59:58 -040062 samplerState.mipmapFilter = convertMipmapMode(sampler);
Nicolas Capens1c294772020-03-28 23:04:55 -040063 samplerState.swizzle = imageDescriptor->swizzle;
64 samplerState.gatherComponent = instruction.gatherComponent;
65 samplerState.highPrecisionFiltering = false;
66 samplerState.largeTexture = (imageDescriptor->extent.width > SHRT_MAX) ||
67 (imageDescriptor->extent.height > SHRT_MAX) ||
68 (imageDescriptor->extent.depth > SHRT_MAX);
Nicolas Capensdbd02752019-08-20 10:59:58 -040069
Nicolas Capens1c294772020-03-28 23:04:55 -040070 if(sampler)
71 {
72 samplerState.textureFilter = (instruction.samplerMethod == Gather) ? FILTER_GATHER : convertFilterMode(sampler);
73 samplerState.border = sampler->borderColor;
Nicolas Capensdbd02752019-08-20 10:59:58 -040074
Nicolas Capens1c294772020-03-28 23:04:55 -040075 samplerState.mipmapFilter = convertMipmapMode(sampler);
Nicolas Capens7f469172020-03-17 17:29:11 -040076
Nicolas Capens1c294772020-03-28 23:04:55 -040077 samplerState.compareEnable = (sampler->compareEnable != VK_FALSE);
78 samplerState.compareOp = sampler->compareOp;
79 samplerState.unnormalizedCoordinates = (sampler->unnormalizedCoordinates != VK_FALSE);
Nicolas Capense2535df2019-05-06 10:37:50 -040080
Nicolas Capens1c294772020-03-28 23:04:55 -040081 samplerState.ycbcrModel = sampler->ycbcrModel;
82 samplerState.studioSwing = sampler->studioSwing;
83 samplerState.swappedChroma = sampler->swappedChroma;
Ben Clayton96fbe082019-04-16 19:28:11 -040084
Nicolas Capens1c294772020-03-28 23:04:55 -040085 samplerState.mipLodBias = sampler->mipLodBias;
86 samplerState.maxAnisotropy = sampler->maxAnisotropy;
87 samplerState.minLod = sampler->minLod;
88 samplerState.maxLod = sampler->maxLod;
89 }
90
91 return emitSamplerRoutine(instruction, samplerState);
92 };
93
94 auto routine = cache->getOrCreate(key, createSamplingRoutine);
95
Ben Claytonbc1c067be2019-12-17 20:37:37 +000096 return (ImageSampler *)(routine->getEntry());
Nicolas Capens97da7822019-04-30 17:33:26 -040097}
Nicolas Capens125dba02019-04-24 02:03:22 -040098
Ben Clayton6897e9b2019-07-16 17:27:27 +010099std::shared_ptr<rr::Routine> SpirvShader::emitSamplerRoutine(ImageInstruction instruction, const Sampler &samplerState)
Nicolas Capens97da7822019-04-30 17:33:26 -0400100{
101 // TODO(b/129523279): Hold a separate mutex lock for the sampler being built.
Nicolas Capens6c11cf22020-03-19 15:21:13 -0400102 rr::Function<Void(Pointer<Byte>, Pointer<SIMD::Float>, Pointer<SIMD::Float>, Pointer<Byte>)> function;
Nicolas Capensa47a5162019-04-24 02:41:27 -0400103 {
Nicolas Capens97da7822019-04-30 17:33:26 -0400104 Pointer<Byte> texture = function.Arg<0>();
Nicolas Capens6c11cf22020-03-19 15:21:13 -0400105 Pointer<SIMD::Float> in = function.Arg<1>();
106 Pointer<SIMD::Float> out = function.Arg<2>();
107 Pointer<Byte> constants = function.Arg<3>();
Nicolas Capensa195abb2019-04-25 17:15:56 -0400108
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000109 SIMD::Float uvw[4] = { 0, 0, 0, 0 };
Ben Clayton1d6301d2019-06-27 10:35:02 +0100110 SIMD::Float q = 0;
111 SIMD::Float lodOrBias = 0; // Explicit level-of-detail, or bias added to the implicit level-of-detail (depending on samplerMethod).
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000112 Vector4f dsx = { 0, 0, 0, 0 };
113 Vector4f dsy = { 0, 0, 0, 0 };
114 Vector4f offset = { 0, 0, 0, 0 };
Alexis Hetu8a6dcf72019-11-26 17:24:42 -0500115 SIMD::Int sampleId = 0;
Nicolas Capens97da7822019-04-30 17:33:26 -0400116 SamplerFunction samplerFunction = instruction.getSamplerFunction();
117
118 uint32_t i = 0;
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000119 for(; i < instruction.coordinates; i++)
Nicolas Capens420d9da2019-04-26 17:44:42 -0400120 {
Nicolas Capens97da7822019-04-30 17:33:26 -0400121 uvw[i] = in[i];
Nicolas Capens420d9da2019-04-26 17:44:42 -0400122 }
123
Nicolas Capens81bc9d92019-12-16 15:05:57 -0500124 if(instruction.isDref())
Chris Forbesc71c17f2019-05-04 10:01:04 -0700125 {
126 q = in[i];
127 i++;
128 }
129
Nicolas Capens8ac0bd62019-06-06 13:03:56 -0400130 // TODO(b/134669567): Currently 1D textures are treated as 2D by setting the second coordinate to 0.
Nicolas Capens97da7822019-04-30 17:33:26 -0400131 // Implement optimized 1D sampling.
Chris Forbes3aec8a32019-07-18 11:48:22 -0700132 if(samplerState.textureType == VK_IMAGE_VIEW_TYPE_1D)
Nicolas Capens420d9da2019-04-26 17:44:42 -0400133 {
Nicolas Capens97da7822019-04-30 17:33:26 -0400134 uvw[1] = SIMD::Float(0);
Nicolas Capens022bd572019-04-29 23:45:25 -0400135 }
Chris Forbes3aec8a32019-07-18 11:48:22 -0700136 else if(samplerState.textureType == VK_IMAGE_VIEW_TYPE_1D_ARRAY)
Nicolas Capense2535df2019-05-06 10:37:50 -0400137 {
138 uvw[1] = SIMD::Float(0);
139 uvw[2] = in[1]; // Move 1D layer coordinate to 2D layer coordinate index.
140 }
Nicolas Capens97da7822019-04-30 17:33:26 -0400141
Chris Forbesf7d78f72019-05-17 16:20:01 -0700142 if(instruction.samplerMethod == Lod || instruction.samplerMethod == Bias || instruction.samplerMethod == Fetch)
Nicolas Capens97da7822019-04-30 17:33:26 -0400143 {
144 lodOrBias = in[i];
145 i++;
146 }
147 else if(instruction.samplerMethod == Grad)
148 {
Nicolas Capens2a25ed82019-06-14 11:26:14 -0400149 for(uint32_t j = 0; j < instruction.grad; j++, i++)
Nicolas Capens97da7822019-04-30 17:33:26 -0400150 {
151 dsx[j] = in[i];
152 }
153
Nicolas Capens2a25ed82019-06-14 11:26:14 -0400154 for(uint32_t j = 0; j < instruction.grad; j++, i++)
Nicolas Capens97da7822019-04-30 17:33:26 -0400155 {
156 dsy[j] = in[i];
157 }
158 }
159
Nicolas Capens2a25ed82019-06-14 11:26:14 -0400160 for(uint32_t j = 0; j < instruction.offset; j++, i++)
Nicolas Capens97da7822019-04-30 17:33:26 -0400161 {
Nicolas Capens2a25ed82019-06-14 11:26:14 -0400162 offset[j] = in[i];
Nicolas Capens97da7822019-04-30 17:33:26 -0400163 }
164
Alexis Hetu8a6dcf72019-11-26 17:24:42 -0500165 if(instruction.sample)
166 {
167 sampleId = As<SIMD::Int>(in[i]);
168 }
Nicolas Capens2a25ed82019-06-14 11:26:14 -0400169
Nicolas Capens977a0a42019-05-16 16:44:05 -0400170 SamplerCore s(constants, samplerState);
Nicolas Capens97da7822019-04-30 17:33:26 -0400171
Nicolas Capens8ac0bd62019-06-06 13:03:56 -0400172 // For explicit-lod instructions the LOD can be different per SIMD lane. SamplerCore currently assumes
173 // a single LOD per four elements, so we sample the image again for each LOD separately.
174 if(samplerFunction.method == Lod || samplerFunction.method == Grad) // TODO(b/133868964): Also handle divergent Bias and Fetch with Lod.
175 {
176 auto lod = Pointer<Float>(&lodOrBias);
177
178 For(Int i = 0, i < SIMD::Width, i++)
179 {
180 SIMD::Float dPdx;
181 SIMD::Float dPdy;
182
183 dPdx.x = Pointer<Float>(&dsx.x)[i];
184 dPdx.y = Pointer<Float>(&dsx.y)[i];
185 dPdx.z = Pointer<Float>(&dsx.z)[i];
186
187 dPdy.x = Pointer<Float>(&dsy.x)[i];
188 dPdy.y = Pointer<Float>(&dsy.y)[i];
189 dPdy.z = Pointer<Float>(&dsy.z)[i];
190
191 // 1D textures are treated as 2D texture with second coordinate 0, so we also need to zero out the second grad component. TODO(b/134669567)
Chris Forbes3aec8a32019-07-18 11:48:22 -0700192 if(samplerState.textureType == VK_IMAGE_VIEW_TYPE_1D || samplerState.textureType == VK_IMAGE_VIEW_TYPE_1D_ARRAY)
Nicolas Capens8ac0bd62019-06-06 13:03:56 -0400193 {
194 dPdx.y = Float(0.0f);
195 dPdy.y = Float(0.0f);
196 }
197
Nicolas Capens6c11cf22020-03-19 15:21:13 -0400198 Vector4f sample = s.sampleTexture(texture, uvw, q, lod[i], dPdx, dPdy, offset, sampleId, samplerFunction);
Nicolas Capens8ac0bd62019-06-06 13:03:56 -0400199
200 Pointer<Float> rgba = out;
201 rgba[0 * SIMD::Width + i] = Pointer<Float>(&sample.x)[i];
202 rgba[1 * SIMD::Width + i] = Pointer<Float>(&sample.y)[i];
203 rgba[2 * SIMD::Width + i] = Pointer<Float>(&sample.z)[i];
204 rgba[3 * SIMD::Width + i] = Pointer<Float>(&sample.w)[i];
205 }
206 }
207 else
208 {
Nicolas Capens6c11cf22020-03-19 15:21:13 -0400209 Vector4f sample = s.sampleTexture(texture, uvw, q, lodOrBias.x, (dsx.x), (dsy.x), offset, sampleId, samplerFunction);
Nicolas Capens8ac0bd62019-06-06 13:03:56 -0400210
211 Pointer<SIMD::Float> rgba = out;
212 rgba[0] = sample.x;
213 rgba[1] = sample.y;
214 rgba[2] = sample.z;
215 rgba[3] = sample.w;
216 }
Nicolas Capens022bd572019-04-29 23:45:25 -0400217 }
218
Alexis Hetu6448bd62019-06-11 15:58:59 -0400219 return function("sampler");
Nicolas Capens9e735102019-04-18 15:03:06 -0400220}
221
222sw::FilterType SpirvShader::convertFilterMode(const vk::Sampler *sampler)
223{
Nicolas Capens8fff8c32020-01-22 03:07:14 -0500224 if(sampler->anisotropyEnable != VK_FALSE)
Alexis Hetu21be09d2019-12-17 16:53:53 -0500225 {
226 return FILTER_ANISOTROPIC;
227 }
228
Nicolas Capens9e735102019-04-18 15:03:06 -0400229 switch(sampler->magFilter)
230 {
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000231 case VK_FILTER_NEAREST:
232 switch(sampler->minFilter)
233 {
234 case VK_FILTER_NEAREST: return FILTER_POINT;
235 case VK_FILTER_LINEAR: return FILTER_MIN_LINEAR_MAG_POINT;
236 default:
Nicolas Capens865f8892020-01-21 14:27:10 -0500237 UNSUPPORTED("minFilter %d", sampler->minFilter);
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000238 return FILTER_POINT;
239 }
240 break;
241 case VK_FILTER_LINEAR:
242 switch(sampler->minFilter)
243 {
244 case VK_FILTER_NEAREST: return FILTER_MIN_POINT_MAG_LINEAR;
245 case VK_FILTER_LINEAR: return FILTER_LINEAR;
246 default:
Nicolas Capens865f8892020-01-21 14:27:10 -0500247 UNSUPPORTED("minFilter %d", sampler->minFilter);
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000248 return FILTER_POINT;
249 }
250 break;
Nicolas Capens9e735102019-04-18 15:03:06 -0400251 default:
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000252 break;
Nicolas Capens9e735102019-04-18 15:03:06 -0400253 }
Ben Clayton8115f1e2019-06-11 16:13:56 +0100254
Nicolas Capens865f8892020-01-21 14:27:10 -0500255 UNSUPPORTED("magFilter %d", sampler->magFilter);
Ben Clayton8115f1e2019-06-11 16:13:56 +0100256 return FILTER_POINT;
Nicolas Capens9e735102019-04-18 15:03:06 -0400257}
258
259sw::MipmapType SpirvShader::convertMipmapMode(const vk::Sampler *sampler)
260{
Nicolas Capensdbd02752019-08-20 10:59:58 -0400261 if(!sampler)
262 {
263 return MIPMAP_POINT; // Samplerless operations (OpImageFetch) can take an integer Lod operand.
264 }
265
Nicolas Capensf948cd12020-03-23 13:00:43 -0400266 if(sampler->ycbcrModel != VK_SAMPLER_YCBCR_MODEL_CONVERSION_RGB_IDENTITY)
Nicolas Capens51e9e562019-05-16 14:01:16 -0400267 {
Nicolas Capens77b43d62020-03-12 00:20:00 -0400268 // TODO(b/151263485): Check image view level count instead.
269 return MIPMAP_NONE;
Nicolas Capens51e9e562019-05-16 14:01:16 -0400270 }
271
Nicolas Capens9e735102019-04-18 15:03:06 -0400272 switch(sampler->mipmapMode)
273 {
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000274 case VK_SAMPLER_MIPMAP_MODE_NEAREST: return MIPMAP_POINT;
275 case VK_SAMPLER_MIPMAP_MODE_LINEAR: return MIPMAP_LINEAR;
276 default:
Nicolas Capens865f8892020-01-21 14:27:10 -0500277 UNSUPPORTED("mipmapMode %d", sampler->mipmapMode);
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000278 return MIPMAP_POINT;
Nicolas Capens9e735102019-04-18 15:03:06 -0400279 }
280}
281
Nicolas Capensdbd02752019-08-20 10:59:58 -0400282sw::AddressingMode SpirvShader::convertAddressingMode(int coordinateIndex, const vk::Sampler *sampler, VkImageViewType imageViewType)
Nicolas Capens9e735102019-04-18 15:03:06 -0400283{
Nicolas Capens6a12e092019-04-29 17:26:51 -0400284 switch(imageViewType)
285 {
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000286 case VK_IMAGE_VIEW_TYPE_CUBE_ARRAY:
287 if(coordinateIndex == 3)
288 {
289 return ADDRESSING_LAYER;
290 }
291 // Fall through to CUBE case:
292 case VK_IMAGE_VIEW_TYPE_CUBE:
293 if(coordinateIndex <= 1) // Cube faces themselves are addressed as 2D images.
294 {
295 // Vulkan 1.1 spec:
296 // "Cube images ignore the wrap modes specified in the sampler. Instead, if VK_FILTER_NEAREST is used within a mip level then
297 // VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE is used, and if VK_FILTER_LINEAR is used within a mip level then sampling at the edges
298 // is performed as described earlier in the Cube map edge handling section."
299 // This corresponds with our 'SEAMLESS' addressing mode.
300 return ADDRESSING_SEAMLESS;
301 }
302 else if(coordinateIndex == 2)
303 {
304 // The cube face is an index into array layers.
305 return ADDRESSING_CUBEFACE;
306 }
307 else
308 {
309 return ADDRESSING_UNUSED;
310 }
311 break;
Nicolas Capens8da52532019-05-07 14:22:31 -0400312
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000313 case VK_IMAGE_VIEW_TYPE_1D: // Treated as 2D texture with second coordinate 0. TODO(b/134669567)
314 if(coordinateIndex == 1)
315 {
316 return ADDRESSING_WRAP;
317 }
318 else if(coordinateIndex >= 2)
319 {
320 return ADDRESSING_UNUSED;
321 }
322 break;
323
324 case VK_IMAGE_VIEW_TYPE_3D:
325 if(coordinateIndex >= 3)
326 {
327 return ADDRESSING_UNUSED;
328 }
329 break;
330
331 case VK_IMAGE_VIEW_TYPE_1D_ARRAY: // Treated as 2D texture with second coordinate 0. TODO(b/134669567)
332 if(coordinateIndex == 1)
333 {
334 return ADDRESSING_WRAP;
335 }
336 // Fall through to 2D_ARRAY case:
337 case VK_IMAGE_VIEW_TYPE_2D_ARRAY:
338 if(coordinateIndex == 2)
339 {
340 return ADDRESSING_LAYER;
341 }
342 else if(coordinateIndex >= 3)
343 {
344 return ADDRESSING_UNUSED;
345 }
346 // Fall through to 2D case:
347 case VK_IMAGE_VIEW_TYPE_2D:
348 if(coordinateIndex >= 2)
349 {
350 return ADDRESSING_UNUSED;
351 }
352 break;
353
354 default:
Nicolas Capens865f8892020-01-21 14:27:10 -0500355 UNSUPPORTED("imageViewType %d", imageViewType);
Nicolas Capens8da52532019-05-07 14:22:31 -0400356 return ADDRESSING_WRAP;
Nicolas Capensa195abb2019-04-25 17:15:56 -0400357 }
358
Nicolas Capensdbd02752019-08-20 10:59:58 -0400359 if(!sampler)
360 {
361 // OpImageFetch does not take a sampler descriptor, but still needs a valid,
362 // arbitrary addressing mode that prevents out-of-bounds accesses:
363 // "The value returned by a read of an invalid texel is undefined, unless that
364 // read operation is from a buffer resource and the robustBufferAccess feature
365 // is enabled. In that case, an invalid texel is replaced as described by the
366 // robustBufferAccess feature." - Vulkan 1.1
367
368 return ADDRESSING_WRAP;
369 }
370
371 VkSamplerAddressMode addressMode = VK_SAMPLER_ADDRESS_MODE_REPEAT;
372 switch(coordinateIndex)
373 {
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000374 case 0: addressMode = sampler->addressModeU; break;
375 case 1: addressMode = sampler->addressModeV; break;
376 case 2: addressMode = sampler->addressModeW; break;
377 default: UNSUPPORTED("coordinateIndex: %d", coordinateIndex);
Nicolas Capensdbd02752019-08-20 10:59:58 -0400378 }
379
Nicolas Capens9e735102019-04-18 15:03:06 -0400380 switch(addressMode)
381 {
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000382 case VK_SAMPLER_ADDRESS_MODE_REPEAT: return ADDRESSING_WRAP;
383 case VK_SAMPLER_ADDRESS_MODE_MIRRORED_REPEAT: return ADDRESSING_MIRROR;
384 case VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE: return ADDRESSING_CLAMP;
385 case VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER: return ADDRESSING_BORDER;
386 case VK_SAMPLER_ADDRESS_MODE_MIRROR_CLAMP_TO_EDGE: return ADDRESSING_MIRRORONCE;
387 default:
Nicolas Capens865f8892020-01-21 14:27:10 -0500388 UNSUPPORTED("addressMode %d", addressMode);
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000389 return ADDRESSING_WRAP;
Nicolas Capens9e735102019-04-18 15:03:06 -0400390 }
Ben Clayton96fbe082019-04-16 19:28:11 -0400391}
392
Ben Claytonbc1c067be2019-12-17 20:37:37 +0000393} // namespace sw