| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2021 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup gpu | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2022-01-18 14:27:29 +11:00
										 |  |  |  * Descriptor type used to define shader structure, resources and interfaces. | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_map.hh"
 | 
					
						
							|  |  |  | #include "BLI_set.hh"
 | 
					
						
							|  |  |  | #include "BLI_string_ref.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "GPU_shader.h"
 | 
					
						
							|  |  |  | #include "GPU_texture.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "gpu_shader_create_info.hh"
 | 
					
						
							|  |  |  | #include "gpu_shader_create_info_private.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef GPU_SHADER_INTERFACE_INFO
 | 
					
						
							|  |  |  | #undef GPU_SHADER_CREATE_INFO
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace blender::gpu::shader { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using CreateInfoDictionnary = Map<StringRef, ShaderCreateInfo *>; | 
					
						
							|  |  |  | using InterfaceDictionnary = Map<StringRef, StageInterfaceInfo *>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static CreateInfoDictionnary *g_create_infos = nullptr; | 
					
						
							|  |  |  | static InterfaceDictionnary *g_interfaces = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ShaderCreateInfo::finalize() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (finalized_) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   finalized_ = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (auto &info_name : additional_infos_) { | 
					
						
							|  |  |  |     const ShaderCreateInfo &info = *reinterpret_cast<const ShaderCreateInfo *>( | 
					
						
							|  |  |  |         gpu_shader_create_info_get(info_name.c_str())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Recursive. */ | 
					
						
							|  |  |  |     const_cast<ShaderCreateInfo &>(info).finalize(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     interface_names_size_ += info.interface_names_size_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vertex_inputs_.extend(info.vertex_inputs_); | 
					
						
							|  |  |  |     fragment_outputs_.extend(info.fragment_outputs_); | 
					
						
							|  |  |  |     vertex_out_interfaces_.extend(info.vertex_out_interfaces_); | 
					
						
							|  |  |  |     geometry_out_interfaces_.extend(info.geometry_out_interfaces_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     push_constants_.extend(info.push_constants_); | 
					
						
							|  |  |  |     defines_.extend(info.defines_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     batch_resources_.extend(info.batch_resources_); | 
					
						
							|  |  |  |     pass_resources_.extend(info.pass_resources_); | 
					
						
							|  |  |  |     typedef_sources_.extend(info.typedef_sources_); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-24 13:23:50 +01:00
										 |  |  |     validate(info); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-25 14:46:25 +01:00
										 |  |  |     if (info.compute_layout_.local_size_x != -1) { | 
					
						
							|  |  |  |       compute_layout_.local_size_x = info.compute_layout_.local_size_x; | 
					
						
							|  |  |  |       compute_layout_.local_size_y = info.compute_layout_.local_size_y; | 
					
						
							|  |  |  |       compute_layout_.local_size_z = info.compute_layout_.local_size_z; | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-25 14:46:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  |     if (!info.vertex_source_.is_empty()) { | 
					
						
							|  |  |  |       BLI_assert(vertex_source_.is_empty()); | 
					
						
							|  |  |  |       vertex_source_ = info.vertex_source_; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!info.geometry_source_.is_empty()) { | 
					
						
							|  |  |  |       BLI_assert(geometry_source_.is_empty()); | 
					
						
							|  |  |  |       geometry_source_ = info.geometry_source_; | 
					
						
							| 
									
										
										
										
											2022-01-19 11:48:39 +01:00
										 |  |  |       geometry_layout_ = info.geometry_layout_; | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (!info.fragment_source_.is_empty()) { | 
					
						
							|  |  |  |       BLI_assert(fragment_source_.is_empty()); | 
					
						
							|  |  |  |       fragment_source_ = info.fragment_source_; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!info.compute_source_.is_empty()) { | 
					
						
							|  |  |  |       BLI_assert(compute_source_.is_empty()); | 
					
						
							|  |  |  |       compute_source_ = info.compute_source_; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do_static_compilation_ = do_static_compilation_ || info.do_static_compilation_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-24 13:23:50 +01:00
										 |  |  | void ShaderCreateInfo::validate(const ShaderCreateInfo &other_info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-01-26 16:06:22 +11:00
										 |  |  |     /* Check same bind-points usage in OGL. */ | 
					
						
							| 
									
										
										
										
											2022-01-24 13:23:50 +01:00
										 |  |  |     Set<int> images, samplers, ubos, ssbos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto register_resource = [&](const Resource &res) -> bool { | 
					
						
							|  |  |  |       switch (res.bind_type) { | 
					
						
							|  |  |  |         case Resource::BindType::UNIFORM_BUFFER: | 
					
						
							|  |  |  |           return images.add(res.slot); | 
					
						
							|  |  |  |         case Resource::BindType::STORAGE_BUFFER: | 
					
						
							|  |  |  |           return samplers.add(res.slot); | 
					
						
							|  |  |  |         case Resource::BindType::SAMPLER: | 
					
						
							|  |  |  |           return ubos.add(res.slot); | 
					
						
							|  |  |  |         case Resource::BindType::IMAGE: | 
					
						
							|  |  |  |           return ssbos.add(res.slot); | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |           return false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto print_error_msg = [&](const Resource &res) { | 
					
						
							|  |  |  |       std::cerr << name_ << ": Validation failed : Overlapping "; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       switch (res.bind_type) { | 
					
						
							|  |  |  |         case Resource::BindType::UNIFORM_BUFFER: | 
					
						
							|  |  |  |           std::cerr << "Uniform Buffer " << res.uniformbuf.name; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         case Resource::BindType::STORAGE_BUFFER: | 
					
						
							|  |  |  |           std::cerr << "Storage Buffer " << res.storagebuf.name; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         case Resource::BindType::SAMPLER: | 
					
						
							|  |  |  |           std::cerr << "Sampler " << res.sampler.name; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         case Resource::BindType::IMAGE: | 
					
						
							|  |  |  |           std::cerr << "Image " << res.image.name; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |           std::cerr << "Unknown Type"; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       std::cerr << " (" << res.slot << ") while merging " << other_info.name_ << std::endl; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto &res : batch_resources_) { | 
					
						
							|  |  |  |       if (register_resource(res) == false) { | 
					
						
							|  |  |  |         print_error_msg(res); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto &res : pass_resources_) { | 
					
						
							|  |  |  |       if (register_resource(res) == false) { | 
					
						
							|  |  |  |         print_error_msg(res); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     /* TODO(fclem) Push constant validation. */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  | }  // namespace blender::gpu::shader
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace blender::gpu::shader; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_shader_create_info_init() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   g_create_infos = new CreateInfoDictionnary(); | 
					
						
							|  |  |  |   g_interfaces = new InterfaceDictionnary(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GPU_SHADER_INTERFACE_INFO(_interface, _inst_name) \
 | 
					
						
							|  |  |  |   auto *ptr_##_interface = new StageInterfaceInfo(#_interface, _inst_name); \ | 
					
						
							|  |  |  |   auto &_interface = *ptr_##_interface; \ | 
					
						
							|  |  |  |   g_interfaces->add_new(#_interface, ptr_##_interface); \ | 
					
						
							|  |  |  |   _interface | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GPU_SHADER_CREATE_INFO(_info) \
 | 
					
						
							|  |  |  |   auto *ptr_##_info = new ShaderCreateInfo(#_info); \ | 
					
						
							|  |  |  |   auto &_info = *ptr_##_info; \ | 
					
						
							|  |  |  |   g_create_infos->add_new(#_info, ptr_##_info); \ | 
					
						
							|  |  |  |   _info | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Declare, register and construct the infos. */ | 
					
						
							|  |  |  | #include "gpu_shader_create_info_list.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Baked shader data appended to create infos. */ | 
					
						
							| 
									
										
										
										
											2022-01-24 14:29:19 +11:00
										 |  |  | /* TODO(jbakker): should call a function with a callback. so we could switch implementations.
 | 
					
						
							|  |  |  |  * We cannot compile bf_gpu twice. */ | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  | #ifdef GPU_RUNTIME
 | 
					
						
							|  |  |  | #  include "gpu_shader_baked.hh"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* TEST */ | 
					
						
							|  |  |  |   // gpu_shader_create_info_compile_all();
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_shader_create_info_exit() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for (auto *value : g_create_infos->values()) { | 
					
						
							|  |  |  |     delete value; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   delete g_create_infos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (auto *value : g_interfaces->values()) { | 
					
						
							|  |  |  |     delete value; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   delete g_interfaces; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool gpu_shader_create_info_compile_all() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for (ShaderCreateInfo *info : g_create_infos->values()) { | 
					
						
							|  |  |  |     if (info->do_static_compilation_) { | 
					
						
							|  |  |  |       // printf("Compiling %s: ... \n", info->name_.c_str());
 | 
					
						
							|  |  |  |       GPUShader *shader = GPU_shader_create_from_info( | 
					
						
							|  |  |  |           reinterpret_cast<const GPUShaderCreateInfo *>(info)); | 
					
						
							|  |  |  |       if (shader == nullptr) { | 
					
						
							|  |  |  |         printf("Compilation %s Failed\n", info->name_.c_str()); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       GPU_shader_free(shader); | 
					
						
							|  |  |  |       // printf("Success\n");
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 14:27:29 +11:00
										 |  |  | /* Runtime create infos are not registered in the dictionary and cannot be searched. */ | 
					
						
							| 
									
										
										
										
											2022-01-17 14:45:22 +01:00
										 |  |  | const GPUShaderCreateInfo *gpu_shader_create_info_get(const char *info_name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ShaderCreateInfo *info = g_create_infos->lookup(info_name); | 
					
						
							|  |  |  |   return reinterpret_cast<const GPUShaderCreateInfo *>(info); | 
					
						
							|  |  |  | } |