2022-12-27 20:08:17 +00:00
|
|
|
mod commands;
|
2022-12-27 20:03:06 +00:00
|
|
|
pub mod target;
|
|
|
|
|
2023-02-20 19:52:59 +00:00
|
|
|
use crate::backend::RenderTargetMode;
|
2022-12-27 20:42:55 +00:00
|
|
|
use crate::blend::ComplexBlend;
|
2022-12-27 20:03:06 +00:00
|
|
|
use crate::buffer_pool::TexturePool;
|
2023-07-26 23:25:26 +00:00
|
|
|
use crate::filters::FilterSource;
|
2022-09-07 20:14:14 +00:00
|
|
|
use crate::mesh::Mesh;
|
2023-07-26 23:25:26 +00:00
|
|
|
use crate::pixel_bender::{run_pixelbender_shader_impl, ShaderMode};
|
2023-06-02 21:16:05 +00:00
|
|
|
use crate::surface::commands::{chunk_blends, Chunk, CommandRenderer};
|
2023-02-03 15:44:21 +00:00
|
|
|
use crate::utils::{remove_srgb, supported_sample_count};
|
2023-06-27 19:47:47 +00:00
|
|
|
use crate::{ColorAdjustments, Descriptors, MaskState, Pipelines, Transforms, UniformBuffer};
|
2022-09-07 19:53:21 +00:00
|
|
|
use ruffle_render::commands::CommandList;
|
2023-07-26 23:25:26 +00:00
|
|
|
use ruffle_render::pixel_bender::{ImageInputTexture, PixelBenderShaderArgument};
|
2023-02-03 15:44:21 +00:00
|
|
|
use ruffle_render::quality::StageQuality;
|
2022-09-08 08:05:34 +00:00
|
|
|
use std::sync::Arc;
|
2022-12-27 20:03:06 +00:00
|
|
|
use target::CommandTarget;
|
2023-01-06 12:17:12 +00:00
|
|
|
use tracing::instrument;
|
2022-09-21 00:44:56 +00:00
|
|
|
|
2023-06-23 22:41:29 +00:00
|
|
|
use crate::utils::run_copy_pipeline;
|
2023-02-20 19:52:59 +00:00
|
|
|
|
2023-06-02 21:16:05 +00:00
|
|
|
pub use crate::surface::commands::LayerRef;
|
|
|
|
|
2023-07-26 23:25:26 +00:00
|
|
|
use self::commands::ChunkBlendMode;
|
|
|
|
|
2022-09-21 00:44:56 +00:00
|
|
|
#[derive(Debug)]
|
2022-12-18 20:08:26 +00:00
|
|
|
pub struct Surface {
|
2022-09-21 00:44:56 +00:00
|
|
|
size: wgpu::Extent3d,
|
2023-02-03 15:44:21 +00:00
|
|
|
quality: StageQuality,
|
2022-09-21 00:44:56 +00:00
|
|
|
sample_count: u32,
|
2022-09-20 20:26:39 +00:00
|
|
|
pipelines: Arc<Pipelines>,
|
2022-09-21 00:44:56 +00:00
|
|
|
format: wgpu::TextureFormat,
|
|
|
|
actual_surface_format: wgpu::TextureFormat,
|
2022-09-07 14:54:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Surface {
|
|
|
|
pub fn new(
|
|
|
|
descriptors: &Descriptors,
|
2023-02-03 15:44:21 +00:00
|
|
|
quality: StageQuality,
|
2022-09-07 14:54:05 +00:00
|
|
|
width: u32,
|
|
|
|
height: u32,
|
2022-09-21 00:44:56 +00:00
|
|
|
surface_format: wgpu::TextureFormat,
|
2022-09-07 14:54:05 +00:00
|
|
|
) -> Self {
|
2022-09-21 00:44:56 +00:00
|
|
|
let size = wgpu::Extent3d {
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
depth_or_array_layers: 1,
|
2022-09-07 14:54:05 +00:00
|
|
|
};
|
2022-09-21 00:44:56 +00:00
|
|
|
let frame_buffer_format = remove_srgb(surface_format);
|
2022-09-07 14:54:05 +00:00
|
|
|
|
2024-01-21 17:21:34 +00:00
|
|
|
let sample_count = supported_sample_count(
|
|
|
|
&descriptors.adapter,
|
|
|
|
quality.sample_count(),
|
|
|
|
frame_buffer_format,
|
|
|
|
);
|
2022-09-21 00:44:56 +00:00
|
|
|
let pipelines = descriptors.pipelines(sample_count, frame_buffer_format);
|
2022-09-20 20:26:39 +00:00
|
|
|
Self {
|
2022-12-18 20:08:26 +00:00
|
|
|
size,
|
2023-02-03 15:44:21 +00:00
|
|
|
quality,
|
2022-12-18 20:08:26 +00:00
|
|
|
sample_count,
|
2022-09-20 20:26:39 +00:00
|
|
|
pipelines,
|
2022-09-21 00:44:56 +00:00
|
|
|
format: frame_buffer_format,
|
|
|
|
actual_surface_format: surface_format,
|
2022-09-07 14:54:05 +00:00
|
|
|
}
|
|
|
|
}
|
2022-09-07 19:53:21 +00:00
|
|
|
|
2022-12-23 23:13:26 +00:00
|
|
|
#[allow(clippy::too_many_arguments)]
|
2023-01-06 12:17:12 +00:00
|
|
|
#[instrument(level = "debug", skip_all)]
|
2023-06-19 21:40:53 +00:00
|
|
|
pub fn draw_commands_and_copy_to<'frame, 'global: 'frame>(
|
2022-09-21 00:44:56 +00:00
|
|
|
&mut self,
|
2022-09-07 19:53:21 +00:00
|
|
|
frame_view: &wgpu::TextureView,
|
2023-02-20 19:52:59 +00:00
|
|
|
render_target_mode: RenderTargetMode,
|
2023-06-19 21:40:53 +00:00
|
|
|
descriptors: &'global Descriptors,
|
|
|
|
uniform_buffers: &'frame mut UniformBuffer<'global, Transforms>,
|
|
|
|
color_buffers: &'frame mut UniformBuffer<'global, ColorAdjustments>,
|
|
|
|
uniform_encoder: &'frame mut wgpu::CommandEncoder,
|
|
|
|
draw_encoder: &'frame mut wgpu::CommandEncoder,
|
|
|
|
meshes: &'global Vec<Mesh>,
|
2022-09-07 19:53:21 +00:00
|
|
|
commands: CommandList,
|
2023-06-02 21:16:05 +00:00
|
|
|
layer: LayerRef,
|
2022-12-23 23:13:26 +00:00
|
|
|
texture_pool: &mut TexturePool,
|
2023-06-19 21:40:53 +00:00
|
|
|
) {
|
2022-12-22 00:57:38 +00:00
|
|
|
let target = self.draw_commands(
|
2023-03-27 10:19:57 +00:00
|
|
|
render_target_mode,
|
2022-12-22 00:57:38 +00:00
|
|
|
descriptors,
|
|
|
|
meshes,
|
2022-12-23 01:47:33 +00:00
|
|
|
commands,
|
2023-06-19 21:40:53 +00:00
|
|
|
uniform_buffers,
|
|
|
|
color_buffers,
|
|
|
|
uniform_encoder,
|
|
|
|
draw_encoder,
|
2023-06-02 21:16:05 +00:00
|
|
|
layer,
|
2022-12-23 23:13:26 +00:00
|
|
|
texture_pool,
|
2022-09-21 00:44:56 +00:00
|
|
|
);
|
2022-09-28 15:47:35 +00:00
|
|
|
|
2023-03-27 10:19:27 +00:00
|
|
|
run_copy_pipeline(
|
|
|
|
descriptors,
|
|
|
|
self.format,
|
|
|
|
self.actual_surface_format,
|
|
|
|
self.size,
|
|
|
|
frame_view,
|
|
|
|
target.color_view(),
|
|
|
|
target.whole_frame_bind_group(descriptors),
|
|
|
|
target.globals(),
|
|
|
|
1,
|
2023-06-19 21:40:53 +00:00
|
|
|
draw_encoder,
|
2023-06-19 21:14:18 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-12-22 00:57:38 +00:00
|
|
|
#[allow(clippy::too_many_arguments)]
|
2023-01-06 12:17:12 +00:00
|
|
|
#[instrument(level = "debug", skip_all)]
|
2022-12-22 00:57:38 +00:00
|
|
|
pub fn draw_commands<'frame, 'global: 'frame>(
|
|
|
|
&mut self,
|
2023-02-20 19:52:59 +00:00
|
|
|
render_target_mode: RenderTargetMode,
|
2022-12-22 00:57:38 +00:00
|
|
|
descriptors: &'global Descriptors,
|
|
|
|
meshes: &'global Vec<Mesh>,
|
2022-12-23 01:47:33 +00:00
|
|
|
commands: CommandList,
|
2022-12-22 00:57:38 +00:00
|
|
|
uniform_buffers: &'frame mut UniformBuffer<'global, Transforms>,
|
2022-12-27 22:18:03 +00:00
|
|
|
color_buffers: &'frame mut UniformBuffer<'global, ColorAdjustments>,
|
2022-12-22 00:57:38 +00:00
|
|
|
uniform_encoder: &'frame mut wgpu::CommandEncoder,
|
|
|
|
draw_encoder: &'frame mut wgpu::CommandEncoder,
|
2023-03-19 07:34:33 +00:00
|
|
|
nearest_layer: LayerRef<'frame>,
|
2022-12-22 22:45:08 +00:00
|
|
|
texture_pool: &mut TexturePool,
|
2022-12-22 00:57:38 +00:00
|
|
|
) -> CommandTarget {
|
2022-12-22 22:45:08 +00:00
|
|
|
let target = CommandTarget::new(
|
2023-02-11 18:28:53 +00:00
|
|
|
descriptors,
|
2022-12-22 22:45:08 +00:00
|
|
|
texture_pool,
|
|
|
|
self.size,
|
|
|
|
self.format,
|
|
|
|
self.sample_count,
|
2023-02-20 19:52:59 +00:00
|
|
|
render_target_mode,
|
|
|
|
draw_encoder,
|
2022-12-22 22:45:08 +00:00
|
|
|
);
|
2023-02-20 19:52:59 +00:00
|
|
|
|
2022-12-27 20:42:55 +00:00
|
|
|
let mut num_masks = 0;
|
|
|
|
let mut mask_state = MaskState::NoMask;
|
|
|
|
let chunks = chunk_blends(
|
2022-12-27 23:13:52 +00:00
|
|
|
commands.commands,
|
2022-12-27 20:42:55 +00:00
|
|
|
descriptors,
|
2022-12-22 00:57:38 +00:00
|
|
|
uniform_buffers,
|
2022-12-27 22:18:03 +00:00
|
|
|
color_buffers,
|
2022-12-22 00:57:38 +00:00
|
|
|
uniform_encoder,
|
|
|
|
draw_encoder,
|
2022-12-27 20:42:55 +00:00
|
|
|
meshes,
|
2023-02-03 15:44:21 +00:00
|
|
|
self.quality,
|
2022-12-27 20:42:55 +00:00
|
|
|
target.width(),
|
|
|
|
target.height(),
|
2023-03-19 07:34:33 +00:00
|
|
|
match nearest_layer {
|
|
|
|
LayerRef::Current => LayerRef::Parent(&target),
|
|
|
|
layer => layer,
|
|
|
|
},
|
2022-12-22 22:45:08 +00:00
|
|
|
texture_pool,
|
2022-12-22 00:57:38 +00:00
|
|
|
);
|
|
|
|
|
2022-12-27 20:42:55 +00:00
|
|
|
for chunk in chunks {
|
|
|
|
match chunk {
|
2023-09-24 20:19:14 +00:00
|
|
|
Chunk::Draw(chunk, needs_stencil) => {
|
2022-12-27 20:42:55 +00:00
|
|
|
let mut render_pass =
|
|
|
|
draw_encoder.begin_render_pass(&wgpu::RenderPassDescriptor {
|
|
|
|
label: create_debug_label!(
|
|
|
|
"Chunked draw calls {}",
|
2023-09-24 20:19:14 +00:00
|
|
|
if needs_stencil {
|
|
|
|
"(with stencil)"
|
2022-12-27 20:42:55 +00:00
|
|
|
} else {
|
2023-09-24 20:19:14 +00:00
|
|
|
"(Stencilless)"
|
2022-12-27 20:42:55 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
.as_deref(),
|
2023-01-25 03:44:37 +00:00
|
|
|
color_attachments: &[target.color_attachments()],
|
2023-09-24 20:19:14 +00:00
|
|
|
depth_stencil_attachment: if needs_stencil {
|
|
|
|
target.stencil_attachment(descriptors, texture_pool)
|
2022-12-27 20:42:55 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
},
|
2023-11-28 09:24:55 +00:00
|
|
|
..Default::default()
|
2022-12-27 20:42:55 +00:00
|
|
|
});
|
|
|
|
render_pass.set_bind_group(0, target.globals().bind_group(), &[]);
|
|
|
|
let mut renderer = CommandRenderer::new(
|
|
|
|
&self.pipelines,
|
2023-02-11 18:28:53 +00:00
|
|
|
descriptors,
|
2022-12-27 20:42:55 +00:00
|
|
|
uniform_buffers,
|
2022-12-27 22:18:03 +00:00
|
|
|
color_buffers,
|
2022-12-27 20:42:55 +00:00
|
|
|
uniform_encoder,
|
|
|
|
render_pass,
|
|
|
|
num_masks,
|
|
|
|
mask_state,
|
2023-09-24 20:19:14 +00:00
|
|
|
needs_stencil,
|
2022-12-27 20:42:55 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
for command in &chunk {
|
2022-12-27 20:50:53 +00:00
|
|
|
renderer.execute(command);
|
2022-12-27 20:42:55 +00:00
|
|
|
}
|
|
|
|
|
2022-12-27 20:50:53 +00:00
|
|
|
num_masks = renderer.num_masks();
|
|
|
|
mask_state = renderer.mask_state();
|
2022-12-27 20:42:55 +00:00
|
|
|
}
|
2023-09-24 20:19:14 +00:00
|
|
|
Chunk::Blend(texture, ChunkBlendMode::Shader(shader), needs_stencil) => {
|
|
|
|
assert!(
|
|
|
|
!needs_stencil,
|
|
|
|
"Shader blend should not need stencil buffer"
|
|
|
|
);
|
2023-07-26 23:25:26 +00:00
|
|
|
let parent_blend_buffer =
|
|
|
|
target.update_blend_buffer(descriptors, texture_pool, draw_encoder);
|
|
|
|
run_pixelbender_shader_impl(
|
|
|
|
descriptors,
|
|
|
|
shader,
|
|
|
|
ShaderMode::Filter,
|
|
|
|
&[
|
|
|
|
PixelBenderShaderArgument::ImageInput {
|
|
|
|
index: 0,
|
|
|
|
channels: 0xFF,
|
|
|
|
name: "background".to_string(),
|
|
|
|
texture: Some(ImageInputTexture::TextureRef(
|
|
|
|
parent_blend_buffer.texture(),
|
|
|
|
)),
|
|
|
|
},
|
|
|
|
PixelBenderShaderArgument::ImageInput {
|
|
|
|
index: 1,
|
|
|
|
channels: 0xff,
|
|
|
|
name: "foreground".to_string(),
|
|
|
|
texture: Some(ImageInputTexture::TextureRef(texture.texture())),
|
|
|
|
},
|
|
|
|
],
|
|
|
|
parent_blend_buffer.texture(),
|
|
|
|
draw_encoder,
|
|
|
|
target.color_attachments(),
|
|
|
|
target.sample_count(),
|
|
|
|
&FilterSource::for_entire_texture(texture.texture()),
|
|
|
|
)
|
|
|
|
.expect("Failed to run PixelBender blend mode");
|
|
|
|
}
|
2023-09-24 20:19:14 +00:00
|
|
|
Chunk::Blend(texture, ChunkBlendMode::Complex(blend_mode), needs_stencil) => {
|
2022-12-27 20:42:55 +00:00
|
|
|
let parent = match blend_mode {
|
2023-07-26 23:25:26 +00:00
|
|
|
ComplexBlend::Alpha | ComplexBlend::Erase => {
|
|
|
|
match nearest_layer {
|
|
|
|
LayerRef::None => {
|
|
|
|
// An Alpha or Erase with no Layer above it should be ignored
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
LayerRef::Current => &target,
|
|
|
|
LayerRef::Parent(layer) => layer,
|
2023-03-19 07:34:33 +00:00
|
|
|
}
|
2023-07-26 23:25:26 +00:00
|
|
|
}
|
2022-12-27 20:42:55 +00:00
|
|
|
_ => &target,
|
|
|
|
};
|
|
|
|
|
|
|
|
let parent_blend_buffer =
|
2023-02-11 18:28:53 +00:00
|
|
|
parent.update_blend_buffer(descriptors, texture_pool, draw_encoder);
|
2022-12-27 20:42:55 +00:00
|
|
|
|
|
|
|
let blend_bind_group =
|
|
|
|
descriptors
|
|
|
|
.device
|
|
|
|
.create_bind_group(&wgpu::BindGroupDescriptor {
|
|
|
|
label: create_debug_label!(
|
|
|
|
"Complex blend binds {:?} {}",
|
|
|
|
blend_mode,
|
2023-09-24 20:19:14 +00:00
|
|
|
if needs_stencil {
|
|
|
|
"(with stencil)"
|
2022-12-27 20:42:55 +00:00
|
|
|
} else {
|
2023-09-24 20:19:14 +00:00
|
|
|
"(Stencilless)"
|
2022-12-27 20:42:55 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
.as_deref(),
|
|
|
|
layout: &descriptors.bind_layouts.blend,
|
|
|
|
entries: &[
|
|
|
|
wgpu::BindGroupEntry {
|
|
|
|
binding: 0,
|
|
|
|
resource: wgpu::BindingResource::TextureView(
|
|
|
|
parent_blend_buffer.view(),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
wgpu::BindGroupEntry {
|
|
|
|
binding: 1,
|
2023-02-20 19:52:59 +00:00
|
|
|
resource: wgpu::BindingResource::TextureView(
|
|
|
|
texture.view(),
|
|
|
|
),
|
2022-12-27 20:42:55 +00:00
|
|
|
},
|
|
|
|
wgpu::BindGroupEntry {
|
|
|
|
binding: 2,
|
|
|
|
resource: wgpu::BindingResource::Sampler(
|
|
|
|
descriptors.bitmap_samplers.get_sampler(false, false),
|
|
|
|
),
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
let mut render_pass =
|
|
|
|
draw_encoder.begin_render_pass(&wgpu::RenderPassDescriptor {
|
|
|
|
label: create_debug_label!(
|
|
|
|
"Complex blend {:?} {}",
|
|
|
|
blend_mode,
|
2023-09-24 20:19:14 +00:00
|
|
|
if needs_stencil {
|
|
|
|
"(with stencil)"
|
2022-12-27 20:42:55 +00:00
|
|
|
} else {
|
2023-09-24 20:19:14 +00:00
|
|
|
"(Stencilless)"
|
2022-12-27 20:42:55 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
.as_deref(),
|
2023-01-25 03:44:37 +00:00
|
|
|
color_attachments: &[target.color_attachments()],
|
2023-09-24 20:19:14 +00:00
|
|
|
depth_stencil_attachment: if needs_stencil {
|
|
|
|
target.stencil_attachment(descriptors, texture_pool)
|
2022-12-27 20:42:55 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
},
|
2023-11-28 09:24:55 +00:00
|
|
|
..Default::default()
|
2022-12-27 20:42:55 +00:00
|
|
|
});
|
|
|
|
render_pass.set_bind_group(0, target.globals().bind_group(), &[]);
|
|
|
|
|
2023-09-24 20:19:14 +00:00
|
|
|
if needs_stencil {
|
2022-12-27 20:42:55 +00:00
|
|
|
match mask_state {
|
|
|
|
MaskState::NoMask => {}
|
|
|
|
MaskState::DrawMaskStencil => {
|
|
|
|
render_pass.set_stencil_reference(num_masks - 1);
|
|
|
|
}
|
|
|
|
MaskState::DrawMaskedContent => {
|
|
|
|
render_pass.set_stencil_reference(num_masks);
|
|
|
|
}
|
|
|
|
MaskState::ClearMaskStencil => {
|
|
|
|
render_pass.set_stencil_reference(num_masks);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
render_pass.set_pipeline(
|
|
|
|
self.pipelines.complex_blends[blend_mode].pipeline_for(mask_state),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
render_pass.set_pipeline(
|
2023-09-24 20:19:14 +00:00
|
|
|
self.pipelines.complex_blends[blend_mode].stencilless_pipeline(),
|
2022-12-27 20:42:55 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-01-06 17:41:24 +00:00
|
|
|
if descriptors.limits.max_push_constant_size > 0 {
|
|
|
|
render_pass.set_push_constants(
|
|
|
|
wgpu::ShaderStages::VERTEX,
|
|
|
|
0,
|
|
|
|
bytemuck::cast_slice(&[Transforms {
|
|
|
|
world_matrix: [
|
|
|
|
[self.size.width as f32, 0.0, 0.0, 0.0],
|
|
|
|
[0.0, self.size.height as f32, 0.0, 0.0],
|
|
|
|
[0.0, 0.0, 1.0, 0.0],
|
|
|
|
[0.0, 0.0, 0.0, 1.0],
|
|
|
|
],
|
|
|
|
}]),
|
|
|
|
);
|
2023-01-07 09:19:14 +00:00
|
|
|
render_pass.set_bind_group(1, &blend_bind_group, &[]);
|
2023-01-06 17:41:24 +00:00
|
|
|
} else {
|
|
|
|
render_pass.set_bind_group(
|
|
|
|
1,
|
|
|
|
target.whole_frame_bind_group(descriptors),
|
|
|
|
&[0],
|
|
|
|
);
|
2023-01-07 09:19:14 +00:00
|
|
|
render_pass.set_bind_group(2, &blend_bind_group, &[]);
|
2023-01-06 17:41:24 +00:00
|
|
|
}
|
|
|
|
|
2023-01-30 20:45:44 +00:00
|
|
|
render_pass.set_vertex_buffer(0, descriptors.quad.vertices_pos.slice(..));
|
2022-12-27 20:42:55 +00:00
|
|
|
render_pass.set_index_buffer(
|
|
|
|
descriptors.quad.indices.slice(..),
|
|
|
|
wgpu::IndexFormat::Uint32,
|
|
|
|
);
|
|
|
|
|
|
|
|
render_pass.draw_indexed(0..6, 0, 0..1);
|
|
|
|
drop(render_pass);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-19 21:40:53 +00:00
|
|
|
// If nothing happened, ensure it's cleared so we don't operate on garbage data
|
|
|
|
target.ensure_cleared(draw_encoder);
|
|
|
|
|
2022-12-22 00:57:38 +00:00
|
|
|
target
|
|
|
|
}
|
2023-01-03 14:53:22 +00:00
|
|
|
|
2023-02-03 15:44:21 +00:00
|
|
|
pub fn quality(&self) -> StageQuality {
|
|
|
|
self.quality
|
|
|
|
}
|
|
|
|
|
2023-01-03 14:53:22 +00:00
|
|
|
pub fn sample_count(&self) -> u32 {
|
|
|
|
self.sample_count
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn size(&self) -> wgpu::Extent3d {
|
|
|
|
self.size
|
|
|
|
}
|
2023-03-25 21:01:31 +00:00
|
|
|
}
|