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