Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2015 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 "GrVkUtil.h" |
| 9 | |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 10 | #include "vk/GrVkGpu.h" |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 11 | #include "SkSLCompiler.h" |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 12 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 13 | bool GrPixelConfigToVkFormat(GrPixelConfig config, VkFormat* format) { |
| 14 | VkFormat dontCare; |
| 15 | if (!format) { |
| 16 | format = &dontCare; |
| 17 | } |
| 18 | |
| 19 | switch (config) { |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 20 | case kUnknown_GrPixelConfig: |
| 21 | return false; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 22 | case kRGBA_8888_GrPixelConfig: |
| 23 | *format = VK_FORMAT_R8G8B8A8_UNORM; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 24 | return true; |
Brian Salomon | 5fba7ad | 2018-03-22 10:01:16 -0400 | [diff] [blame] | 25 | case kRGB_888_GrPixelConfig: |
Greg Daniel | 475eb70 | 2018-09-28 14:16:50 -0400 | [diff] [blame] | 26 | *format = VK_FORMAT_R8G8B8_UNORM; |
| 27 | return true; |
Jim Van Verth | 69e5785 | 2018-12-05 13:38:59 -0500 | [diff] [blame] | 28 | case kRG_88_GrPixelConfig: |
| 29 | *format = VK_FORMAT_R8G8_UNORM; |
| 30 | return true; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 31 | case kBGRA_8888_GrPixelConfig: |
| 32 | *format = VK_FORMAT_B8G8R8A8_UNORM; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 33 | return true; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 34 | case kSRGBA_8888_GrPixelConfig: |
| 35 | *format = VK_FORMAT_R8G8B8A8_SRGB; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 36 | return true; |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 37 | case kSBGRA_8888_GrPixelConfig: |
| 38 | *format = VK_FORMAT_B8G8R8A8_SRGB; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 39 | return true; |
Brian Osman | 10fc6fd | 2018-03-02 11:01:10 -0500 | [diff] [blame] | 40 | case kRGBA_1010102_GrPixelConfig: |
| 41 | *format = VK_FORMAT_A2B10G10R10_UNORM_PACK32; |
| 42 | return true; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 43 | case kRGB_565_GrPixelConfig: |
| 44 | *format = VK_FORMAT_R5G6B5_UNORM_PACK16; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 45 | return true; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 46 | case kRGBA_4444_GrPixelConfig: |
egdaniel | 3fe0327 | 2016-08-15 10:59:17 -0700 | [diff] [blame] | 47 | // R4G4B4A4 is not required to be supported so we actually |
| 48 | // store the data is if it was B4G4R4A4 and swizzle in shaders |
| 49 | *format = VK_FORMAT_B4G4R4A4_UNORM_PACK16; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 50 | return true; |
Greg Daniel | ef59d87 | 2017-11-17 16:47:21 -0500 | [diff] [blame] | 51 | case kAlpha_8_GrPixelConfig: // fall through |
| 52 | case kAlpha_8_as_Red_GrPixelConfig: |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 53 | *format = VK_FORMAT_R8_UNORM; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 54 | return true; |
Greg Daniel | ef59d87 | 2017-11-17 16:47:21 -0500 | [diff] [blame] | 55 | case kAlpha_8_as_Alpha_GrPixelConfig: |
| 56 | return false; |
Brian Osman | 986563b | 2017-01-10 14:20:02 -0500 | [diff] [blame] | 57 | case kGray_8_GrPixelConfig: |
Greg Daniel | 7af060a | 2017-12-05 16:27:11 -0500 | [diff] [blame] | 58 | case kGray_8_as_Red_GrPixelConfig: |
Brian Osman | 986563b | 2017-01-10 14:20:02 -0500 | [diff] [blame] | 59 | *format = VK_FORMAT_R8_UNORM; |
| 60 | return true; |
Greg Daniel | 7af060a | 2017-12-05 16:27:11 -0500 | [diff] [blame] | 61 | case kGray_8_as_Lum_GrPixelConfig: |
| 62 | return false; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 63 | case kRGBA_float_GrPixelConfig: |
| 64 | *format = VK_FORMAT_R32G32B32A32_SFLOAT; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 65 | return true; |
csmartdalton | 6aa0e11 | 2017-02-08 16:14:11 -0500 | [diff] [blame] | 66 | case kRG_float_GrPixelConfig: |
| 67 | *format = VK_FORMAT_R32G32_SFLOAT; |
| 68 | return true; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 69 | case kRGBA_half_GrPixelConfig: |
| 70 | *format = VK_FORMAT_R16G16B16A16_SFLOAT; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 71 | return true; |
Jim Van Verth | 1676cb9 | 2019-01-15 13:24:45 -0500 | [diff] [blame] | 72 | case kRGB_ETC1_GrPixelConfig: |
| 73 | // converting to ETC2 which is a superset of ETC1 |
| 74 | *format = VK_FORMAT_ETC2_R8G8B8_UNORM_BLOCK; |
| 75 | return true; |
Greg Daniel | ef59d87 | 2017-11-17 16:47:21 -0500 | [diff] [blame] | 76 | case kAlpha_half_GrPixelConfig: // fall through |
| 77 | case kAlpha_half_as_Red_GrPixelConfig: |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 78 | *format = VK_FORMAT_R16_SFLOAT; |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 79 | return true; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 80 | } |
Ben Wagner | b4aab9a | 2017-08-16 10:53:04 -0400 | [diff] [blame] | 81 | SK_ABORT("Unexpected config"); |
Brian Salomon | bf7b620 | 2016-11-11 16:08:03 -0500 | [diff] [blame] | 82 | return false; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 83 | } |
| 84 | |
Greg Daniel | 7e00022 | 2018-12-03 10:08:21 -0500 | [diff] [blame] | 85 | #ifdef SK_DEBUG |
Greg Daniel | 81b8059 | 2017-12-13 10:20:04 -0500 | [diff] [blame] | 86 | bool GrVkFormatPixelConfigPairIsValid(VkFormat format, GrPixelConfig config) { |
| 87 | switch (format) { |
| 88 | case VK_FORMAT_R8G8B8A8_UNORM: |
| 89 | return kRGBA_8888_GrPixelConfig == config; |
| 90 | case VK_FORMAT_B8G8R8A8_UNORM: |
| 91 | return kBGRA_8888_GrPixelConfig == config; |
| 92 | case VK_FORMAT_R8G8B8A8_SRGB: |
| 93 | return kSRGBA_8888_GrPixelConfig == config; |
| 94 | case VK_FORMAT_B8G8R8A8_SRGB: |
| 95 | return kSBGRA_8888_GrPixelConfig == config; |
Greg Daniel | 475eb70 | 2018-09-28 14:16:50 -0400 | [diff] [blame] | 96 | case VK_FORMAT_R8G8B8_UNORM: |
| 97 | return kRGB_888_GrPixelConfig == config; |
Jim Van Verth | 69e5785 | 2018-12-05 13:38:59 -0500 | [diff] [blame] | 98 | case VK_FORMAT_R8G8_UNORM: |
| 99 | return kRG_88_GrPixelConfig == config; |
Brian Osman | 10fc6fd | 2018-03-02 11:01:10 -0500 | [diff] [blame] | 100 | case VK_FORMAT_A2B10G10R10_UNORM_PACK32: |
| 101 | return kRGBA_1010102_GrPixelConfig == config; |
Greg Daniel | 81b8059 | 2017-12-13 10:20:04 -0500 | [diff] [blame] | 102 | case VK_FORMAT_R5G6B5_UNORM_PACK16: |
| 103 | return kRGB_565_GrPixelConfig == config; |
| 104 | case VK_FORMAT_B4G4R4A4_UNORM_PACK16: |
| 105 | // R4G4B4A4 is not required to be supported so we actually |
| 106 | // store RGBA_4444 data as B4G4R4A4. |
| 107 | return kRGBA_4444_GrPixelConfig == config; |
| 108 | case VK_FORMAT_R8_UNORM: |
| 109 | return kAlpha_8_GrPixelConfig == config || |
| 110 | kAlpha_8_as_Red_GrPixelConfig == config || |
| 111 | kGray_8_GrPixelConfig == config || |
| 112 | kGray_8_as_Red_GrPixelConfig == config; |
Jim Van Verth | 1676cb9 | 2019-01-15 13:24:45 -0500 | [diff] [blame] | 113 | case VK_FORMAT_ETC2_R8G8B8_UNORM_BLOCK: |
| 114 | return kRGB_ETC1_GrPixelConfig == config; |
Greg Daniel | 81b8059 | 2017-12-13 10:20:04 -0500 | [diff] [blame] | 115 | case VK_FORMAT_R32G32B32A32_SFLOAT: |
| 116 | return kRGBA_float_GrPixelConfig == config; |
| 117 | case VK_FORMAT_R32G32_SFLOAT: |
| 118 | return kRG_float_GrPixelConfig == config; |
| 119 | case VK_FORMAT_R16G16B16A16_SFLOAT: |
| 120 | return kRGBA_half_GrPixelConfig == config; |
| 121 | case VK_FORMAT_R16_SFLOAT: |
| 122 | return kAlpha_half_GrPixelConfig == config || |
| 123 | kAlpha_half_as_Red_GrPixelConfig == config; |
| 124 | default: |
| 125 | return false; |
| 126 | } |
| 127 | } |
Greg Daniel | 7e00022 | 2018-12-03 10:08:21 -0500 | [diff] [blame] | 128 | #endif |
Greg Daniel | 81b8059 | 2017-12-13 10:20:04 -0500 | [diff] [blame] | 129 | |
| 130 | bool GrVkFormatIsSupported(VkFormat format) { |
| 131 | switch (format) { |
| 132 | case VK_FORMAT_R8G8B8A8_UNORM: |
| 133 | case VK_FORMAT_B8G8R8A8_UNORM: |
| 134 | case VK_FORMAT_R8G8B8A8_SRGB: |
| 135 | case VK_FORMAT_B8G8R8A8_SRGB: |
| 136 | case VK_FORMAT_R8G8B8A8_SINT: |
Greg Daniel | 475eb70 | 2018-09-28 14:16:50 -0400 | [diff] [blame] | 137 | case VK_FORMAT_R8G8B8_UNORM: |
Jim Van Verth | 69e5785 | 2018-12-05 13:38:59 -0500 | [diff] [blame] | 138 | case VK_FORMAT_R8G8_UNORM: |
Brian Osman | 10fc6fd | 2018-03-02 11:01:10 -0500 | [diff] [blame] | 139 | case VK_FORMAT_A2B10G10R10_UNORM_PACK32: |
Greg Daniel | 81b8059 | 2017-12-13 10:20:04 -0500 | [diff] [blame] | 140 | case VK_FORMAT_R5G6B5_UNORM_PACK16: |
| 141 | case VK_FORMAT_B4G4R4A4_UNORM_PACK16: |
| 142 | case VK_FORMAT_R8_UNORM: |
Jim Van Verth | ee06b33 | 2019-01-18 10:36:32 -0500 | [diff] [blame] | 143 | case VK_FORMAT_ETC2_R8G8B8_UNORM_BLOCK: |
Greg Daniel | 81b8059 | 2017-12-13 10:20:04 -0500 | [diff] [blame] | 144 | case VK_FORMAT_R32G32B32A32_SFLOAT: |
| 145 | case VK_FORMAT_R32G32_SFLOAT: |
| 146 | case VK_FORMAT_R16G16B16A16_SFLOAT: |
| 147 | case VK_FORMAT_R16_SFLOAT: |
| 148 | return true; |
| 149 | default: |
| 150 | return false; |
| 151 | } |
| 152 | } |
| 153 | |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 154 | bool GrSampleCountToVkSampleCount(uint32_t samples, VkSampleCountFlagBits* vkSamples) { |
Brian Salomon | bdecacf | 2018-02-02 20:32:49 -0500 | [diff] [blame] | 155 | SkASSERT(samples >= 1); |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 156 | switch (samples) { |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 157 | case 1: |
| 158 | *vkSamples = VK_SAMPLE_COUNT_1_BIT; |
| 159 | return true; |
| 160 | case 2: |
| 161 | *vkSamples = VK_SAMPLE_COUNT_2_BIT; |
| 162 | return true; |
| 163 | case 4: |
egdaniel | bf63e61 | 2016-08-17 06:26:16 -0700 | [diff] [blame] | 164 | *vkSamples = VK_SAMPLE_COUNT_4_BIT; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 165 | return true; |
| 166 | case 8: |
egdaniel | bf63e61 | 2016-08-17 06:26:16 -0700 | [diff] [blame] | 167 | *vkSamples = VK_SAMPLE_COUNT_8_BIT; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 168 | return true; |
| 169 | case 16: |
egdaniel | bf63e61 | 2016-08-17 06:26:16 -0700 | [diff] [blame] | 170 | *vkSamples = VK_SAMPLE_COUNT_16_BIT; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 171 | return true; |
| 172 | case 32: |
egdaniel | bf63e61 | 2016-08-17 06:26:16 -0700 | [diff] [blame] | 173 | *vkSamples = VK_SAMPLE_COUNT_32_BIT; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 174 | return true; |
| 175 | case 64: |
egdaniel | bf63e61 | 2016-08-17 06:26:16 -0700 | [diff] [blame] | 176 | *vkSamples = VK_SAMPLE_COUNT_64_BIT; |
Greg Daniel | 164a9f0 | 2016-02-22 09:56:40 -0500 | [diff] [blame] | 177 | return true; |
| 178 | default: |
| 179 | return false; |
| 180 | } |
| 181 | } |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 182 | |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 183 | SkSL::Program::Kind vk_shader_stage_to_skiasl_kind(VkShaderStageFlagBits stage) { |
| 184 | if (VK_SHADER_STAGE_VERTEX_BIT == stage) { |
| 185 | return SkSL::Program::kVertex_Kind; |
| 186 | } |
Chris Dalton | 33607c6 | 2017-07-07 11:00:48 -0600 | [diff] [blame] | 187 | if (VK_SHADER_STAGE_GEOMETRY_BIT == stage) { |
| 188 | return SkSL::Program::kGeometry_Kind; |
| 189 | } |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 190 | SkASSERT(VK_SHADER_STAGE_FRAGMENT_BIT == stage); |
| 191 | return SkSL::Program::kFragment_Kind; |
| 192 | } |
Ethan Nicholas | 941e7e2 | 2016-12-12 15:33:30 -0500 | [diff] [blame] | 193 | |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 194 | bool GrCompileVkShaderModule(const GrVkGpu* gpu, |
| 195 | const char* shaderString, |
| 196 | VkShaderStageFlagBits stage, |
| 197 | VkShaderModule* shaderModule, |
Ethan Nicholas | 941e7e2 | 2016-12-12 15:33:30 -0500 | [diff] [blame] | 198 | VkPipelineShaderStageCreateInfo* stageInfo, |
| 199 | const SkSL::Program::Settings& settings, |
Ethan Nicholas | 92e01cf | 2018-12-19 13:12:10 -0500 | [diff] [blame] | 200 | SkSL::String* outSPIRV, |
Ethan Nicholas | 941e7e2 | 2016-12-12 15:33:30 -0500 | [diff] [blame] | 201 | SkSL::Program::Inputs* outInputs) { |
| 202 | std::unique_ptr<SkSL::Program> program = gpu->shaderCompiler()->convertProgram( |
| 203 | vk_shader_stage_to_skiasl_kind(stage), |
Brian Osman | 93ba0a4 | 2017-08-14 14:48:10 -0400 | [diff] [blame] | 204 | SkSL::String(shaderString), |
Ethan Nicholas | 941e7e2 | 2016-12-12 15:33:30 -0500 | [diff] [blame] | 205 | settings); |
| 206 | if (!program) { |
Ethan Nicholas | e1f5502 | 2019-02-05 17:17:40 -0500 | [diff] [blame] | 207 | printf("%s\n", shaderString); |
Ethan Nicholas | 941e7e2 | 2016-12-12 15:33:30 -0500 | [diff] [blame] | 208 | SkDebugf("SkSL error:\n%s\n", gpu->shaderCompiler()->errorText().c_str()); |
| 209 | SkASSERT(false); |
| 210 | } |
| 211 | *outInputs = program->fInputs; |
Ethan Nicholas | 92e01cf | 2018-12-19 13:12:10 -0500 | [diff] [blame] | 212 | if (!gpu->shaderCompiler()->toSPIRV(*program, outSPIRV)) { |
Ethan Nicholas | 941e7e2 | 2016-12-12 15:33:30 -0500 | [diff] [blame] | 213 | SkDebugf("%s\n", gpu->shaderCompiler()->errorText().c_str()); |
| 214 | return false; |
| 215 | } |
| 216 | |
Ethan Nicholas | 92e01cf | 2018-12-19 13:12:10 -0500 | [diff] [blame] | 217 | return GrInstallVkShaderModule(gpu, *outSPIRV, stage, shaderModule, stageInfo); |
| 218 | } |
| 219 | |
| 220 | bool GrInstallVkShaderModule(const GrVkGpu* gpu, |
| 221 | const SkSL::String& spirv, |
| 222 | VkShaderStageFlagBits stage, |
| 223 | VkShaderModule* shaderModule, |
| 224 | VkPipelineShaderStageCreateInfo* stageInfo) { |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 225 | VkShaderModuleCreateInfo moduleCreateInfo; |
| 226 | memset(&moduleCreateInfo, 0, sizeof(VkShaderModuleCreateInfo)); |
| 227 | moduleCreateInfo.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO; |
| 228 | moduleCreateInfo.pNext = nullptr; |
| 229 | moduleCreateInfo.flags = 0; |
Ethan Nicholas | 92e01cf | 2018-12-19 13:12:10 -0500 | [diff] [blame] | 230 | moduleCreateInfo.codeSize = spirv.size(); |
| 231 | moduleCreateInfo.pCode = (const uint32_t*)spirv.c_str(); |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 232 | |
| 233 | VkResult err = GR_VK_CALL(gpu->vkInterface(), CreateShaderModule(gpu->device(), |
| 234 | &moduleCreateInfo, |
| 235 | nullptr, |
| 236 | shaderModule)); |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 237 | if (err) { |
| 238 | return false; |
| 239 | } |
| 240 | |
| 241 | memset(stageInfo, 0, sizeof(VkPipelineShaderStageCreateInfo)); |
| 242 | stageInfo->sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO; |
| 243 | stageInfo->pNext = nullptr; |
| 244 | stageInfo->flags = 0; |
Ethan Nicholas | 92e01cf | 2018-12-19 13:12:10 -0500 | [diff] [blame] | 245 | stageInfo->stage = stage; |
egdaniel | 88987d8 | 2016-09-19 10:17:34 -0700 | [diff] [blame] | 246 | stageInfo->module = *shaderModule; |
| 247 | stageInfo->pName = "main"; |
| 248 | stageInfo->pSpecializationInfo = nullptr; |
| 249 | |
| 250 | return true; |
| 251 | } |