19 const std::shared_ptr<VKBuffer>& buffer)
21 auto info_it = buffer_info.find(buffer);
22 if (info_it == buffer_info.end()) {
23 if (buffer->has_vertex_layout()) {
24 auto vertex_layout = buffer->get_vertex_layout();
25 if (vertex_layout.has_value()) {
26 buffer_info[buffer] = { .semantic_layout = vertex_layout.value(), .use_reflection =
false };
27 info_it = buffer_info.find(buffer);
30 if (info_it == buffer_info.end()) {
34 return &info_it->second.semantic_layout;
74 window->register_rendering_buffer(buffer);
132 const std::shared_ptr<Core::VKImage>& texture,
137 "Cannot bind null texture to binding {}", binding);
143 sampler = loom.get_default_sampler();
151 foundry.update_descriptor_image(
154 texture->get_image_view(),
156 vk::ImageLayout::eShaderReadOnlyOptimal);
160 "Bound texture to binding {}", binding);
164 const std::string& descriptor_name,
165 const std::shared_ptr<Core::VKImage>& texture,
171 "No binding configured for descriptor '{}'", descriptor_name);
175 bind_texture(binding_it->second.binding, texture, sampler);
182 "Vertex shader not loaded");
188 "Fragment shader not loaded");
194 "Target window not set");
222 if (buffer->has_vertex_layout()) {
223 auto vertex_layout = buffer->get_vertex_layout();
224 if (vertex_layout.has_value()) {
226 .semantic_layout = vertex_layout.value(),
227 .use_reflection =
false
236 "initialize_pipeline: layout not yet available, deferring");
243 const auto& staging = buffer->get_pipeline_context().push_constant_staging;
244 if (!staging.empty()) {
250 auto& descriptor_bindings = buffer->get_pipeline_context().descriptor_buffer_bindings;
253 for (
const auto& binding : descriptor_bindings) {
254 unified_bindings[{ binding.set, binding.binding }] = binding;
258 auto key = std::make_pair(binding.set, binding.binding);
259 if (unified_bindings.find(key) == unified_bindings.end()) {
262 .binding = binding.binding,
263 .type = binding.type,
270 std::map<uint32_t, std::vector<Portal::Graphics::DescriptorBindingInfo>> bindings_by_set;
271 for (
const auto& [key, binding] : unified_bindings) {
272 bindings_by_set[binding.set].push_back(binding);
275 for (
const auto& [set_index, set_bindings] : bindings_by_set) {
279 vk::Format swapchain_format =
static_cast<vk::Format
>(
283 ? vk::Format::eD32Sfloat
284 : vk::Format::eUndefined;
286 m_pipeline_id = flow.create_pipeline(pipeline_config, { swapchain_format }, depth_format);
290 "Failed to create render pipeline");
295 buffer->set_needs_depth_attachment(
true);
308 "Cannot allocate descriptor sets without pipeline");
320 "Failed to allocate descriptor sets for pipeline");
326 [binding](
const auto& pair) {
327 return pair.second.binding == binding;
332 "No config for binding {}", binding);
335 uint32_t set_index = config_it->second.set;
339 "Descriptor set index {} out of range", binding);
344 foundry.update_descriptor_image(
346 config_it->second.binding,
347 tex_binding.texture->get_image_view(),
349 vk::ImageLayout::eShaderReadOnlyOptimal);
353 "Allocated {} descriptor sets and updated {} texture bindings",
366 "RenderProcessor: Set vertex range [offset={}, count={}]",
367 first_vertex, vertex_count);
371 const std::shared_ptr<VKBuffer>& buffer,
375 .semantic_layout = layout,
376 .use_reflection =
false
385 "Target window not set");
394 if (buffer->has_vertex_layout()) {
395 auto vertex_layout = buffer->get_vertex_layout();
396 if (vertex_layout.has_value()) {
398 .semantic_layout = vertex_layout.value(),
399 .use_reflection =
false
416 "VKBuffer has no vertex layout set. Use buffer->set_vertex_layout()");
425 vk::Format color_format =
static_cast<vk::Format
>(
428 auto cmd_id = foundry.begin_secondary_commands(color_format);
429 auto cmd = foundry.get_command_buffer(cmd_id);
431 uint32_t width = 0, height = 0;
434 if (width > 0 && height > 0) {
435 auto cmd = foundry.get_command_buffer(cmd_id);
437 vk::Viewport viewport {
439 static_cast<float>(height),
440 static_cast<float>(width),
441 -
static_cast<float>(height),
445 cmd.setViewport(0, 1, &viewport);
447 vk::Rect2D scissor { { 0, 0 }, { width, height } };
448 cmd.setScissor(0, 1, &scissor);
453 auto& descriptor_bindings = buffer->get_pipeline_context().descriptor_buffer_bindings;
454 if (!descriptor_bindings.empty()) {
455 for (
const auto& binding : descriptor_bindings) {
458 "Descriptor set index {} out of range", binding.set);
462 foundry.update_descriptor_buffer(
466 binding.buffer_info.buffer,
467 binding.buffer_info.offset,
468 binding.buffer_info.range);
481 auto& staging = buffer->get_pipeline_context().push_constant_staging;
483 if (!staging.empty()) {
493 const auto& staging = buffer->get_pipeline_context();
494 if (!staging.push_constant_staging.empty()) {
498 staging.push_constant_staging.data(),
499 staging.push_constant_staging.size());
510 flow.bind_vertex_buffers(cmd_id, { buffer });
512 uint32_t draw_count = 0;
516 auto current_layout = buffer->get_vertex_layout();
517 if (!current_layout.has_value() || current_layout->vertex_count == 0) {
519 "Vertex layout has zero vertices, skipping draw");
522 draw_count = current_layout->vertex_count;
527 foundry.end_commands(cmd_id);
533 "Recorded secondary command buffer {} for window '{}'",
541 auto vk_buffer = std::dynamic_pointer_cast<VKBuffer>(buffer);
542 if (vk_buffer && vk_buffer->has_vertex_layout()) {
543 auto vertex_layout = vk_buffer->get_vertex_layout();
544 if (vertex_layout.has_value()) {
546 "RenderProcessor: Auto-injecting vertex layout "
547 "({} vertices, {} attributes)",
548 vertex_layout->vertex_count,
549 vertex_layout->attributes.size());
552 m_buffer_info[vk_buffer] = { .semantic_layout = vertex_layout.value(), .use_reflection =
false };
557 vk_buffer->set_needs_depth_attachment(
true);
604 "RenderProcessor cleanup complete");
#define MF_INFO(comp, ctx,...)
#define MF_ERROR(comp, ctx,...)
#define MF_RT_ERROR(comp, ctx,...)
#define MF_RT_TRACE(comp, ctx,...)
#define MF_DEBUG(comp, ctx,...)
#define MF_RT_DEBUG(comp, ctx,...)
void bind_texture(uint32_t binding, const std::shared_ptr< Core::VKImage > &texture, vk::Sampler sampler=nullptr)
Bind a texture to a descriptor binding point.
void set_blend_attachment(const Portal::Graphics::BlendAttachmentConfig &config)
Set blend mode for color attachment.
Portal::Graphics::ShaderID m_fragment_shader_id
std::unordered_map< std::shared_ptr< VKBuffer >, VertexInfo > m_buffer_info
void set_buffer_vertex_layout(const std::shared_ptr< VKBuffer > &buffer, const Kakshya::VertexLayout &layout)
Override the vertex layout used when building the pipeline for buffer.
std::optional< Portal::Graphics::BlendAttachmentConfig > m_blend_attachment
Registry::Service::DisplayService * m_display_service
Portal::Graphics::ShaderID m_geometry_shader_id
void set_tess_eval_shader(const std::string &tess_eval_path)
bool on_before_execute(Portal::Graphics::CommandBufferID cmd_id, const std::shared_ptr< VKBuffer > &buffer) override
Called before each process callback.
void enable_alpha_blending()
Enable standard alpha blending (src_alpha, one_minus_src_alpha)
std::function< Kinesis::ViewTransform()> m_view_transform_source
void set_tess_control_shader(const std::string &tess_control_path)
Portal::Graphics::RenderPipelineID m_pipeline_id
void enable_depth_test(Portal::Graphics::CompareOp compare_op=Portal::Graphics::CompareOp::LESS)
Enable depth testing for this processor's pipeline.
Portal::Graphics::ShaderID m_tess_control_shader_id
void on_attach(const std::shared_ptr< Buffer > &buffer) override
Called when this processor is attached to a buffer.
void initialize_descriptors(const std::shared_ptr< VKBuffer > &buffer) override
std::optional< Kinesis::ViewTransform > m_view_transform
Portal::Graphics::DepthStencilConfig m_depth_stencil
void initialize_pipeline(const std::shared_ptr< VKBuffer > &buffer) override
std::unordered_map< uint32_t, TextureBinding > m_texture_bindings
void set_geometry_shader(const std::string &geometry_path)
void set_vertex_range(uint32_t first_vertex, uint32_t vertex_count)
Set vertex range for drawing subset of buffer.
Portal::Graphics::PrimitiveTopology m_primitive_topology
void set_view_transform(const Kinesis::ViewTransform &vt)
Set static view transform (evaluated once)
RenderProcessor(const ShaderConfig &config)
void set_fragment_shader(const std::string &fragment_path)
Portal::Graphics::PolygonMode m_polygon_mode
void execute_shader(const std::shared_ptr< VKBuffer > &buffer) override
void set_target_window(const std::shared_ptr< Core::Window > &window, const std::shared_ptr< VKBuffer > &buffer)
Portal::Graphics::CullMode m_cull_mode
const Kakshya::VertexLayout * get_or_cache_vertex_layout(std::unordered_map< std::shared_ptr< VKBuffer >, VertexInfo > &buffer_info, const std::shared_ptr< VKBuffer > &buffer)
std::shared_ptr< Core::Window > m_target_window
Portal::Graphics::ShaderID m_tess_eval_shader_id
void set_view_transform_source(std::function< Kinesis::ViewTransform()> fn)
Set dynamic view transform source (evaluated every frame)
Portal::Graphics::ShaderID m_shader_id
std::vector< uint8_t > m_push_constant_data
virtual void on_descriptors_created()
Called after descriptor sets are created.
bool m_needs_descriptor_rebuild
virtual void update_descriptors(const std::shared_ptr< VKBuffer > &buffer)
void set_push_constant_size()
Set push constant size from type.
virtual void on_pipeline_created(Portal::Graphics::ComputePipelineID pipeline_id)
Called after pipeline is created.
virtual void on_before_descriptors_create()
Called before descriptor sets are created.
bool m_needs_pipeline_rebuild
void on_attach(const std::shared_ptr< Buffer > &buffer) override
Called when this processor is attached to a buffer.
std::vector< Portal::Graphics::DescriptorSetID > m_descriptor_set_ids
Abstract base class for shader-based buffer processing.
void register_window_for_rendering(const std::shared_ptr< Core::Window > &window)
Register a window for dynamic rendering.
ShaderID load_shader(const std::string &content, std::optional< ShaderStage > stage=std::nullopt, const std::string &entry_point="main")
Universal shader loader - auto-detects source type.
Interface * get_service()
Query for a backend service.
static BackendRegistry & instance()
Get the global registry instance.
@ BufferProcessing
Buffer processing (Buffers::BufferManager, processing chains)
@ Buffers
Buffers, Managers, processors and processing chains.
constexpr RenderPipelineID INVALID_RENDER_PIPELINE
MAYAFLUX_API TextureLoom & get_texture_manager()
Get the global texture manager instance.
constexpr ShaderID INVALID_SHADER
MAYAFLUX_API RenderFlow & get_render_flow()
Get the global render flow instance.
MAYAFLUX_API ShaderFoundry & get_shader_foundry()
Get the global shader compiler instance.
CompareOp
Depth/stencil comparison operation.
std::string shader_path
Path to shader file.
std::unordered_map< std::string, ShaderBinding > bindings
size_t push_constant_size
Complete description of vertex data layout in a buffer.
static BlendAttachmentConfig alpha_blend()
Create standard alpha blending configuration.
Per-attachment blend configuration.
CompareOp depth_compare_op
std::vector< std::vector< DescriptorBindingInfo > > descriptor_sets
ShaderID tess_eval_shader
Optional.
std::optional< Kakshya::VertexLayout > semantic_vertex_layout
ShaderID tess_control_shader
Optional.
bool use_vertex_shader_reflection
std::vector< BlendAttachmentConfig > blend_attachments
DepthStencilConfig depth_stencil
size_t push_constant_size
PrimitiveTopology topology
ShaderID geometry_shader
Optional.
RasterizationConfig rasterization
Complete render pipeline configuration.
std::function< int(const std::shared_ptr< void > &)> get_swapchain_format
Get actual swapchain format for a window.
std::function< void(const std::shared_ptr< void > &, uint32_t &, uint32_t &)> get_swapchain_extent
Get swapchain extent for a window.
Backend display and presentation service interface.