1
0
Fork 0
forked from wry/wry

vulkan: use specialization constants

This commit is contained in:
Julian Orth 2025-01-30 13:54:37 +01:00
parent 218f8d7ba2
commit cca0af3d0d
7 changed files with 58 additions and 39 deletions

View file

@ -1,7 +1,7 @@
use { use {
crate::open, crate::open,
anyhow::{bail, Context}, anyhow::{anyhow, bail, Context},
shaderc::CompileOptions, shaderc::{CompileOptions, ResolvedInclude},
std::{io::Write, path::Path}, std::{io::Write, path::Path},
}; };
@ -12,21 +12,17 @@ pub fn main() -> anyhow::Result<()> {
compile_simple("fill.frag")?; compile_simple("fill.frag")?;
compile_simple("fill.vert")?; compile_simple("fill.vert")?;
compile_simple("tex.vert")?; compile_simple("tex.vert")?;
compile_tex_frag("tex.frag.spv", false, false)?; compile_tex_frag("tex.frag.spv", false)?;
compile_tex_frag("tex.frag.mult+opaque.spv", false, true)?; compile_tex_frag("tex.frag.mult.spv", true)?;
compile_tex_frag("tex.frag.mult+alpha.spv", true, true)?;
Ok(()) Ok(())
} }
fn compile_tex_frag(out: &str, alpha: bool, alpha_multiplier: bool) -> anyhow::Result<()> { fn compile_tex_frag(out: &str, alpha_multiplier: bool) -> anyhow::Result<()> {
let mut opts = CompileOptions::new().unwrap(); let mut opts = CompileOptions::new().unwrap();
if alpha {
opts.add_macro_definition("ALPHA", None);
}
if alpha_multiplier { if alpha_multiplier {
opts.add_macro_definition("ALPHA_MULTIPLIER", None); opts.add_macro_definition("ALPHA_MULTIPLIER", None);
} }
compile_shader("tex.frag", out, Some(&opts)).with_context(|| out.to_string())?; compile_shader("tex.frag", out, Some(opts)).with_context(|| out.to_string())?;
Ok(()) Ok(())
} }
@ -34,7 +30,15 @@ fn compile_simple(name: &str) -> anyhow::Result<()> {
compile_shader(name, &format!("{name}.spv"), None).with_context(|| name.to_string()) compile_shader(name, &format!("{name}.spv"), None).with_context(|| name.to_string())
} }
fn compile_shader(name: &str, out: &str, options: Option<&CompileOptions>) -> anyhow::Result<()> { fn compile_shader(name: &str, out: &str, options: Option<CompileOptions>) -> anyhow::Result<()> {
let read = |path: &str| std::fs::read_to_string(format!("{}/{}", ROOT, path));
let mut options = options.unwrap_or_else(|| CompileOptions::new().unwrap());
options.set_include_callback(|name, _, _, _| {
Ok(ResolvedInclude {
resolved_name: name.to_string(),
content: read(name).map_err(|e| anyhow!(e).to_string())?,
})
});
let stage = match Path::new(name) let stage = match Path::new(name)
.extension() .extension()
.and_then(|e| e.to_str()) .and_then(|e| e.to_str())
@ -44,10 +48,10 @@ fn compile_shader(name: &str, out: &str, options: Option<&CompileOptions>) -> an
"vert" => shaderc::ShaderKind::Vertex, "vert" => shaderc::ShaderKind::Vertex,
n => bail!("Unknown shader stage {}", n), n => bail!("Unknown shader stage {}", n),
}; };
let src = std::fs::read_to_string(format!("{}/{}", ROOT, name))?; let src = read(name)?;
let compiler = shaderc::Compiler::new().unwrap(); let compiler = shaderc::Compiler::new().unwrap();
let binary = compiler let binary = compiler
.compile_into_spirv(&src, stage, name, "main", options) .compile_into_spirv(&src, stage, name, "main", Some(&options))
.unwrap(); .unwrap();
let mut file = open(out)?; let mut file = open(out)?;
file.write_all(binary.as_binary_u8())?; file.write_all(binary.as_binary_u8())?;

View file

@ -18,6 +18,7 @@ use {
PipelineRenderingCreateInfo, PipelineShaderStageCreateInfo, PipelineRenderingCreateInfo, PipelineShaderStageCreateInfo,
PipelineVertexInputStateCreateInfo, PipelineViewportStateCreateInfo, PolygonMode, PipelineVertexInputStateCreateInfo, PipelineViewportStateCreateInfo, PolygonMode,
PrimitiveTopology, PushConstantRange, SampleCountFlags, ShaderStageFlags, PrimitiveTopology, PushConstantRange, SampleCountFlags, ShaderStageFlags,
SpecializationInfo, SpecializationMapEntry,
}, },
}, },
std::{rc::Rc, slice}, std::{rc::Rc, slice},
@ -36,7 +37,8 @@ pub(super) struct PipelineCreateInfo {
pub(super) format: vk::Format, pub(super) format: vk::Format,
pub(super) vert: Rc<VulkanShader>, pub(super) vert: Rc<VulkanShader>,
pub(super) frag: Rc<VulkanShader>, pub(super) frag: Rc<VulkanShader>,
pub(super) alpha: bool, pub(super) blend: bool,
pub(super) src_has_alpha: bool,
pub(super) frag_descriptor_set_layout: Option<Rc<VulkanDescriptorSetLayout>>, pub(super) frag_descriptor_set_layout: Option<Rc<VulkanDescriptorSetLayout>>,
} }
@ -89,6 +91,15 @@ impl VulkanDevice {
}; };
let destroy_layout = let destroy_layout =
OnDrop(|| unsafe { self.device.destroy_pipeline_layout(pipeline_layout, None) }); OnDrop(|| unsafe { self.device.destroy_pipeline_layout(pipeline_layout, None) });
let mut frag_spec_data = ArrayVec::<_, 4>::new();
frag_spec_data.extend((info.src_has_alpha as u32).to_ne_bytes());
let frag_spec_entries = [SpecializationMapEntry::default()
.constant_id(0)
.size(4)
.offset(0)];
let frag_spec = SpecializationInfo::default()
.map_entries(&frag_spec_entries)
.data(&frag_spec_data);
let pipeline = { let pipeline = {
let stages = [ let stages = [
PipelineShaderStageCreateInfo::default() PipelineShaderStageCreateInfo::default()
@ -98,6 +109,7 @@ impl VulkanDevice {
PipelineShaderStageCreateInfo::default() PipelineShaderStageCreateInfo::default()
.stage(ShaderStageFlags::FRAGMENT) .stage(ShaderStageFlags::FRAGMENT)
.module(info.frag.module) .module(info.frag.module)
.specialization_info(&frag_spec)
.name(c"main"), .name(c"main"),
]; ];
let input_assembly_state = PipelineInputAssemblyStateCreateInfo::default() let input_assembly_state = PipelineInputAssemblyStateCreateInfo::default()
@ -113,7 +125,7 @@ impl VulkanDevice {
.rasterization_samples(SampleCountFlags::TYPE_1); .rasterization_samples(SampleCountFlags::TYPE_1);
let mut blending = PipelineColorBlendAttachmentState::default() let mut blending = PipelineColorBlendAttachmentState::default()
.color_write_mask(ColorComponentFlags::RGBA); .color_write_mask(ColorComponentFlags::RGBA);
if info.alpha { if info.blend {
blending = blending blending = blending
.blend_enable(true) .blend_enable(true)
.src_color_blend_factor(BlendFactor::ONE) .src_color_blend_factor(BlendFactor::ONE)

View file

@ -18,8 +18,8 @@ use {
semaphore::VulkanSemaphore, semaphore::VulkanSemaphore,
shaders::{ shaders::{
FillFragPushConstants, FillVertPushConstants, TexFragPushConstants, FillFragPushConstants, FillVertPushConstants, TexFragPushConstants,
TexVertPushConstants, VulkanShader, FILL_FRAG, FILL_VERT, TEX_FRAG, TexVertPushConstants, VulkanShader, FILL_FRAG, FILL_VERT, TEX_FRAG, TEX_FRAG_MULT,
TEX_FRAG_MULT_ALPHA, TEX_FRAG_MULT_OPAQUE, TEX_VERT, TEX_VERT,
}, },
VulkanError, VulkanError,
}, },
@ -74,8 +74,7 @@ pub struct VulkanRenderer {
pub(super) fill_frag_shader: Rc<VulkanShader>, pub(super) fill_frag_shader: Rc<VulkanShader>,
pub(super) tex_vert_shader: Rc<VulkanShader>, pub(super) tex_vert_shader: Rc<VulkanShader>,
pub(super) tex_frag_shader: Rc<VulkanShader>, pub(super) tex_frag_shader: Rc<VulkanShader>,
pub(super) tex_frag_mult_opaque_shader: Rc<VulkanShader>, pub(super) tex_frag_mult_shader: Rc<VulkanShader>,
pub(super) tex_frag_mult_alpha_shader: Rc<VulkanShader>,
pub(super) tex_descriptor_set_layout: Rc<VulkanDescriptorSetLayout>, pub(super) tex_descriptor_set_layout: Rc<VulkanDescriptorSetLayout>,
pub(super) defunct: Cell<bool>, pub(super) defunct: Cell<bool>,
pub(super) pending_cpu_jobs: CopyHashMap<u64, PendingJob>, pub(super) pending_cpu_jobs: CopyHashMap<u64, PendingJob>,
@ -160,8 +159,7 @@ impl VulkanDevice {
let tex_descriptor_set_layout = self.create_descriptor_set_layout(&sampler)?; let tex_descriptor_set_layout = self.create_descriptor_set_layout(&sampler)?;
let tex_vert_shader = self.create_shader(TEX_VERT)?; let tex_vert_shader = self.create_shader(TEX_VERT)?;
let tex_frag_shader = self.create_shader(TEX_FRAG)?; let tex_frag_shader = self.create_shader(TEX_FRAG)?;
let tex_frag_mult_opaque_shader = self.create_shader(TEX_FRAG_MULT_OPAQUE)?; let tex_frag_mult_shader = self.create_shader(TEX_FRAG_MULT)?;
let tex_frag_mult_alpha_shader = self.create_shader(TEX_FRAG_MULT_ALPHA)?;
let gfx_command_buffers = self.create_command_pool(self.graphics_queue_idx)?; let gfx_command_buffers = self.create_command_pool(self.graphics_queue_idx)?;
let transfer_command_buffers = self let transfer_command_buffers = self
.distinct_transfer_queue_family_idx .distinct_transfer_queue_family_idx
@ -219,8 +217,7 @@ impl VulkanDevice {
fill_frag_shader, fill_frag_shader,
tex_vert_shader, tex_vert_shader,
tex_frag_shader, tex_frag_shader,
tex_frag_mult_opaque_shader, tex_frag_mult_shader,
tex_frag_mult_alpha_shader,
tex_descriptor_set_layout, tex_descriptor_set_layout,
defunct: Cell::new(false), defunct: Cell::new(false),
pending_cpu_jobs: Default::default(), pending_cpu_jobs: Default::default(),
@ -246,34 +243,37 @@ impl VulkanRenderer {
format, format,
vert: self.fill_vert_shader.clone(), vert: self.fill_vert_shader.clone(),
frag: self.fill_frag_shader.clone(), frag: self.fill_frag_shader.clone(),
alpha: true, blend: true,
src_has_alpha: true,
frag_descriptor_set_layout: None, frag_descriptor_set_layout: None,
}, },
)?; )?;
let create_tex_pipeline = |alpha| { let create_tex_pipeline = |src_has_alpha| {
self.device self.device
.create_pipeline::<TexVertPushConstants, ()>(PipelineCreateInfo { .create_pipeline::<TexVertPushConstants, ()>(PipelineCreateInfo {
format, format,
vert: self.tex_vert_shader.clone(), vert: self.tex_vert_shader.clone(),
frag: self.tex_frag_shader.clone(), frag: self.tex_frag_shader.clone(),
alpha, blend: src_has_alpha,
src_has_alpha,
frag_descriptor_set_layout: Some(self.tex_descriptor_set_layout.clone()), frag_descriptor_set_layout: Some(self.tex_descriptor_set_layout.clone()),
}) })
}; };
let create_tex_mult_pipeline = |frag: &Rc<VulkanShader>| { let create_tex_mult_pipeline = |src_has_alpha: bool| {
self.device self.device
.create_pipeline::<TexVertPushConstants, TexFragPushConstants>(PipelineCreateInfo { .create_pipeline::<TexVertPushConstants, TexFragPushConstants>(PipelineCreateInfo {
format, format,
vert: self.tex_vert_shader.clone(), vert: self.tex_vert_shader.clone(),
frag: frag.clone(), frag: self.tex_frag_mult_shader.clone(),
alpha: true, blend: true,
src_has_alpha,
frag_descriptor_set_layout: Some(self.tex_descriptor_set_layout.clone()), frag_descriptor_set_layout: Some(self.tex_descriptor_set_layout.clone()),
}) })
}; };
let tex_opaque = create_tex_pipeline(false)?; let tex_opaque = create_tex_pipeline(false)?;
let tex_alpha = create_tex_pipeline(true)?; let tex_alpha = create_tex_pipeline(true)?;
let tex_mult_opaque = create_tex_mult_pipeline(&self.tex_frag_mult_opaque_shader)?; let tex_mult_opaque = create_tex_mult_pipeline(false)?;
let tex_mult_alpha = create_tex_mult_pipeline(&self.tex_frag_mult_alpha_shader)?; let tex_mult_alpha = create_tex_mult_pipeline(true)?;
let pipelines = Rc::new(VulkanFormatPipelines { let pipelines = Rc::new(VulkanFormatPipelines {
fill, fill,
tex: static_map! { tex: static_map! {

View file

@ -9,10 +9,7 @@ pub const FILL_VERT: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/fill.vert
pub const FILL_FRAG: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/fill.frag.spv")); pub const FILL_FRAG: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/fill.frag.spv"));
pub const TEX_VERT: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/tex.vert.spv")); pub const TEX_VERT: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/tex.vert.spv"));
pub const TEX_FRAG: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/tex.frag.spv")); pub const TEX_FRAG: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/tex.frag.spv"));
pub const TEX_FRAG_MULT_OPAQUE: &[u8] = pub const TEX_FRAG_MULT: &[u8] = include_bytes!(concat!(env!("OUT_DIR"), "/tex.frag.mult.spv"));
include_bytes!(concat!(env!("OUT_DIR"), "/tex.frag.mult+opaque.spv"));
pub const TEX_FRAG_MULT_ALPHA: &[u8] =
include_bytes!(concat!(env!("OUT_DIR"), "/tex.frag.mult+alpha.spv"));
pub struct VulkanShader { pub struct VulkanShader {
pub(super) device: Rc<VulkanDevice>, pub(super) device: Rc<VulkanDevice>,

View file

@ -1,5 +1,7 @@
#version 450 #version 450
#include "frag_spec_const.glsl"
layout(push_constant, std430) uniform Data { layout(push_constant, std430) uniform Data {
layout(offset = 32) vec4 color; layout(offset = 32) vec4 color;
} data; } data;

View file

@ -0,0 +1 @@
layout(constant_id = 0) const bool src_has_alpha = false;

View file

@ -1,5 +1,7 @@
#version 450 #version 450
#include "frag_spec_const.glsl"
#ifdef ALPHA_MULTIPLIER #ifdef ALPHA_MULTIPLIER
layout(push_constant, std430) uniform Data { layout(push_constant, std430) uniform Data {
layout(offset = 64) float mul; layout(offset = 64) float mul;
@ -11,10 +13,11 @@ layout(location = 0) out vec4 out_color;
void main() { void main() {
#ifdef ALPHA_MULTIPLIER #ifdef ALPHA_MULTIPLIER
#ifdef ALPHA if (src_has_alpha) {
out_color = textureLod(tex, tex_pos, 0) * data.mul; out_color = textureLod(tex, tex_pos, 0) * data.mul;
#endif // !ALPHA } else {
out_color = vec4(textureLod(tex, tex_pos, 0).rgb * data.mul, data.mul); out_color = vec4(textureLod(tex, tex_pos, 0).rgb * data.mul, data.mul);
}
#else // !ALPHA_MULTIPLIER #else // !ALPHA_MULTIPLIER
out_color = textureLod(tex, tex_pos, 0); out_color = textureLod(tex, tex_pos, 0);
#endif #endif