Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support DXR in wgpu-hal & naga. #6777

Open
wants to merge 51 commits into
base: trunk
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 44 commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
4be11ec
implement device calls
Vecvec Sep 9, 2024
647ddd0
implement todo and fmt
Vecvec Sep 9, 2024
0d0b2ab
properly implement todo and fix other clippy errors
Vecvec Sep 9, 2024
f91a482
don't use buffers in getting build sizes
Vecvec Sep 9, 2024
951d437
fmt
Vecvec Sep 9, 2024
f017080
implement commands
Vecvec Sep 9, 2024
23cc3a1
fix errors
Vecvec Sep 9, 2024
6a2f362
implement hlsl out
Vecvec Sep 11, 2024
7e2fc34
fix clippy warnings
Vecvec Sep 11, 2024
2350e04
fix CI errors
Vecvec Sep 11, 2024
4bdf0ed
fix CI errors #2
Vecvec Sep 11, 2024
1750afb
fix CI errors #3
Vecvec Sep 11, 2024
45ec097
fix CI errors #4
Vecvec Sep 11, 2024
275d082
fix CI errors #5
Vecvec Sep 11, 2024
3cc9801
check for sm 6.5
Vecvec Sep 12, 2024
dd99a08
implement rest of direct x ray queries
Vecvec Sep 13, 2024
499fa9b
fix issues with ray-traced triangle
Vecvec Sep 13, 2024
4f762e0
clippy & fmt
Vecvec Sep 13, 2024
f92221d
Merge remote-tracking branch 'refs/remotes/gfx-rs/trunk' into dx-ray-…
Vecvec Sep 21, 2024
0877129
Merge remote-tracking branch 'refs/remotes/gfx-rs/trunk' into dx-ray-…
Vecvec Dec 17, 2024
7576bbe
fix merge
Vecvec Dec 17, 2024
44f73e6
fix build
Vecvec Dec 17, 2024
02c8699
remove unused import
Vecvec Dec 17, 2024
19caada
don't use `Direct3D12::D3D12_RESOURCE_FLAG_RAYTRACING_ACCELERATION_ST…
Vecvec Dec 17, 2024
2493f04
support candidate intersections
Vecvec Dec 17, 2024
3810051
write out values of constants, not their wgsl names.
Vecvec Dec 17, 2024
d6a4fee
format
Vecvec Dec 17, 2024
ef1ffd7
Merge remote-tracking branch 'refs/remotes/gfx-rs/trunk' into dx-ray-…
Vecvec Dec 17, 2024
8a88223
fix merge
Vecvec Dec 17, 2024
f74a6da
remove println!
Vecvec Dec 17, 2024
01e31e4
Remove unnecessary feature. This was preventing DXR from being used.
Vecvec Dec 18, 2024
9a57f86
Remove println!()
Vecvec Dec 18, 2024
1506561
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 18, 2024
a66ebcf
Changelog
Vecvec Dec 18, 2024
f93a97d
Correctly use candidate instead of committed int the candidate inters…
Vecvec Dec 18, 2024
def77b5
swap the instance calls because have different names
Vecvec Dec 18, 2024
93f700b
fmt
Vecvec Dec 18, 2024
31715f4
regen snapshots
Vecvec Dec 18, 2024
f46ec3c
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 18, 2024
0de5a51
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 18, 2024
3b89172
Merge branch 'trunk' into dx-ray-tracing
cwfitzgerald Dec 18, 2024
72676fc
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 18, 2024
45ed3f3
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 20, 2024
0aeed65
remove layout variable (all paths lead to the same value). Fix incorr…
Vecvec Dec 22, 2024
9c54f4f
begin addressing comments
Vecvec Dec 24, 2024
0665c93
address comments
Vecvec Dec 24, 2024
31c6990
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 24, 2024
c5fdde4
use rgba8
Vecvec Dec 26, 2024
d86619d
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 26, 2024
06fec21
clarify that DXR means DirectX Ray-tracing
Vecvec Dec 30, 2024
0093945
Merge branch 'trunk' into dx-ray-tracing
Vecvec Dec 30, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -149,6 +149,7 @@ By @ErichDonGubler in [#6456](https://github.com/gfx-rs/wgpu/pull/6456), [#6148]
- Return submission index in `map_async` and `on_submitted_work_done` to track down completion of async callbacks. By @eliemichel in [#6360](https://github.com/gfx-rs/wgpu/pull/6360).
- Move raytracing alignments into HAL instead of in core. By @Vecvec in [#6563](https://github.com/gfx-rs/wgpu/pull/6563).
- Allow for statically linking DXC rather than including separate `.dll` files. By @DouglasDwyer in [#6574](https://github.com/gfx-rs/wgpu/pull/6574).
- Support DXR in wgpu-hal. By @Vecvec in [#6777](https://github.com/gfx-rs/wgpu/pull/6777)
Vecvec marked this conversation as resolved.
Show resolved Hide resolved

### Changes

Expand Down
1 change: 0 additions & 1 deletion examples/src/ray_cube_compute/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -137,7 +137,6 @@ struct Example {
impl crate::framework::Example for Example {
fn required_features() -> wgpu::Features {
wgpu::Features::TEXTURE_BINDING_ARRAY
| wgpu::Features::STORAGE_RESOURCE_BINDING_ARRAY
| wgpu::Features::VERTEX_WRITABLE_STORAGE
| wgpu::Features::EXPERIMENTAL_RAY_QUERY
| wgpu::Features::EXPERIMENTAL_RAY_TRACING_ACCELERATION_STRUCTURE
Expand Down
29 changes: 21 additions & 8 deletions naga/src/back/hlsl/help.rs
Original file line number Diff line number Diff line change
Expand Up @@ -841,6 +841,9 @@ impl<W: Write> super::Writer<'_, W> {
&crate::PredeclaredType::AtomicCompareExchangeWeakResult { .. } => {}
}
}
if module.special_types.ray_desc.is_some() {
self.write_ray_desc_from_ray_desc_constructor_function(module)?;
}

Ok(())
}
Expand All @@ -852,16 +855,26 @@ impl<W: Write> super::Writer<'_, W> {
expressions: &crate::Arena<crate::Expression>,
) -> BackendResult {
for (handle, _) in expressions.iter() {
if let crate::Expression::Compose { ty, .. } = expressions[handle] {
match module.types[ty].inner {
crate::TypeInner::Struct { .. } | crate::TypeInner::Array { .. } => {
let constructor = WrappedConstructor { ty };
if self.wrapped.constructors.insert(constructor) {
self.write_wrapped_constructor_function(module, constructor)?;
match expressions[handle] {
crate::Expression::Compose { ty, .. } => {
match module.types[ty].inner {
crate::TypeInner::Struct { .. } | crate::TypeInner::Array { .. } => {
let constructor = WrappedConstructor { ty };
if self.wrapped.constructors.insert(constructor) {
self.write_wrapped_constructor_function(module, constructor)?;
}
}
_ => {}
};
}
crate::Expression::RayQueryGetIntersection { committed, .. } => {
if committed {
self.write_committed_intersection_function(module)?;
} else {
self.write_candidate_intersection_function(module)?;
}
_ => {}
};
}
_ => {}
}
}
Ok(())
Expand Down
1 change: 1 addition & 0 deletions naga/src/back/hlsl/keywords.rs
Original file line number Diff line number Diff line change
Expand Up @@ -814,6 +814,7 @@ pub const RESERVED: &[&str] = &[
"TextureBuffer",
"ConstantBuffer",
"RayQuery",
"RayDesc",
// Naga utilities
super::writer::MODF_FUNCTION,
super::writer::FREXP_FUNCTION,
Expand Down
2 changes: 2 additions & 0 deletions naga/src/back/hlsl/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,7 @@ accessing individual columns by dynamic index.
mod conv;
mod help;
mod keywords;
mod ray;
mod storage;
mod writer;

Expand Down Expand Up @@ -331,6 +332,7 @@ pub struct Writer<'a, W> {
/// Set of expressions that have associated temporary variables
named_expressions: crate::NamedExpressions,
wrapped: Wrapped,
written_committed_intersection: bool,
continue_ctx: back::continue_forward::ContinueCtx,

/// A reference to some part of a global variable, lowered to a series of
Expand Down
163 changes: 163 additions & 0 deletions naga/src/back/hlsl/ray.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,163 @@
use crate::back::hlsl::BackendResult;
use crate::{RayQueryIntersection, TypeInner};
use std::fmt::Write;

impl<W: Write> super::Writer<'_, W> {
// constructs hlsl RayDesc from wgsl RayDesc
pub(super) fn write_ray_desc_from_ray_desc_constructor_function(
&mut self,
module: &crate::Module,
) -> BackendResult {
write!(self.out, "RayDesc RayDescFromRayDesc_(")?;
self.write_type(module, module.special_types.ray_desc.unwrap())?;
writeln!(self.out, " arg0) {{")?;
writeln!(self.out, " RayDesc ret = (RayDesc)0;")?;
writeln!(self.out, " ret.Origin = arg0.origin;")?;
writeln!(self.out, " ret.TMin = arg0.tmin;")?;
writeln!(self.out, " ret.Direction = arg0.dir;")?;
writeln!(self.out, " ret.TMax = arg0.tmax;")?;
writeln!(self.out, " return ret;")?;
writeln!(self.out, "}}")?;
writeln!(self.out)?;
Ok(())
}
pub(super) fn write_committed_intersection_function(
&mut self,
module: &crate::Module,
) -> BackendResult {
self.write_type(module, module.special_types.ray_intersection.unwrap())?;
write!(self.out, " GetCommittedIntersection(")?;
self.write_value_type(module, &TypeInner::RayQuery)?;
writeln!(self.out, " rq) {{")?;
write!(self.out, " ")?;
self.write_type(module, module.special_types.ray_intersection.unwrap())?;
write!(self.out, " ret = (")?;
self.write_type(module, module.special_types.ray_intersection.unwrap())?;
writeln!(self.out, ")0;")?;
writeln!(self.out, " ret.kind = rq.CommittedStatus();")?;
writeln!(
self.out,
" if( rq.CommittedStatus() == COMMITTED_NOTHING) {{}} else {{"
)?;
writeln!(self.out, " ret.t = rq.CommittedRayT();")?;
writeln!(
self.out,
" ret.instance_custom_index = rq.CommittedInstanceID();"
)?;
writeln!(
self.out,
" ret.instance_id = rq.CommittedInstanceIndex();"
)?;
writeln!(
self.out,
" ret.sbt_record_offset = rq.CommittedInstanceContributionToHitGroupIndex();"
)?;
writeln!(
self.out,
" ret.geometry_index = rq.CommittedGeometryIndex();"
)?;
writeln!(
self.out,
" ret.primitive_index = rq.CommittedPrimitiveIndex();"
)?;
writeln!(
self.out,
" if( rq.CommittedStatus() == COMMITTED_TRIANGLE_HIT ) {{"
)?;
writeln!(
self.out,
" ret.barycentrics = rq.CommittedTriangleBarycentrics();"
)?;
writeln!(
self.out,
" ret.front_face = rq.CommittedTriangleFrontFace();"
)?;
writeln!(self.out, " }}")?;
writeln!(
self.out,
" ret.object_to_world = rq.CommittedObjectToWorld4x3();"
)?;
writeln!(
self.out,
" ret.world_to_object = rq.CommittedWorldToObject4x3();"
)?;
writeln!(self.out, " }}")?;
writeln!(self.out, " return ret;")?;
writeln!(self.out, "}}")?;
writeln!(self.out)?;
Ok(())
}
pub(super) fn write_candidate_intersection_function(
&mut self,
module: &crate::Module,
) -> BackendResult {
self.write_type(module, module.special_types.ray_intersection.unwrap())?;
write!(self.out, " GetCandidateIntersection(")?;
self.write_value_type(module, &TypeInner::RayQuery)?;
writeln!(self.out, " rq) {{")?;
write!(self.out, " ")?;
self.write_type(module, module.special_types.ray_intersection.unwrap())?;
write!(self.out, " ret = (")?;
self.write_type(module, module.special_types.ray_intersection.unwrap())?;
writeln!(self.out, ")0;")?;
writeln!(self.out, " CANDIDATE_TYPE kind = rq.CandidateType();")?;
writeln!(
self.out,
" if (kind == CANDIDATE_NON_OPAQUE_TRIANGLE) {{"
)?;
writeln!(
self.out,
" ret.kind = {};",
RayQueryIntersection::Triangle as u32
)?;
writeln!(self.out, " ret.t = rq.CandidateTriangleRayT();")?;
writeln!(
self.out,
" ret.barycentrics = rq.CandidateTriangleBarycentrics();"
)?;
writeln!(
self.out,
" ret.front_face = rq.CandidateTriangleFrontFace();"
)?;
writeln!(self.out, " }} else {{")?;
writeln!(
self.out,
" ret.kind = {};",
RayQueryIntersection::Aabb as u32
)?;
writeln!(self.out, " }}")?;

writeln!(
self.out,
" ret.instance_custom_index = rq.CandidateInstanceID();"
)?;
writeln!(
self.out,
" ret.instance_id = rq.CandidateInstanceIndex();"
)?;
writeln!(
self.out,
" ret.sbt_record_offset = rq.CandidateInstanceContributionToHitGroupIndex();"
)?;
writeln!(
self.out,
" ret.geometry_index = rq.CandidateGeometryIndex();"
)?;
writeln!(
self.out,
" ret.primitive_index = rq.CandidatePrimitiveIndex();"
)?;
writeln!(
self.out,
" ret.object_to_world = rq.CandidateObjectToWorld4x3();"
)?;
writeln!(
self.out,
" ret.world_to_object = rq.CandidateWorldToObject4x3();"
)?;
writeln!(self.out, " return ret;")?;
writeln!(self.out, "}}")?;
writeln!(self.out)?;
Ok(())
}
}
77 changes: 65 additions & 12 deletions naga/src/back/hlsl/writer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ use super::{
use crate::{
back::{self, Baked},
proc::{self, index, ExpressionKindTracker, NameKey},
valid, Handle, Module, Scalar, ScalarKind, ShaderStage, TypeInner,
valid, Handle, Module, RayQueryFunction, Scalar, ScalarKind, ShaderStage, TypeInner,
};
use std::{fmt, mem};

Expand Down Expand Up @@ -104,6 +104,7 @@ impl<'a, W: fmt::Write> super::Writer<'a, W> {
entry_point_io: Vec::new(),
named_expressions: crate::NamedExpressions::default(),
wrapped: super::Wrapped::default(),
written_committed_intersection: false,
continue_ctx: back::continue_forward::ContinueCtx::default(),
temp_access_chain: Vec::new(),
need_bake_expressions: Default::default(),
Expand All @@ -123,6 +124,7 @@ impl<'a, W: fmt::Write> super::Writer<'a, W> {
self.entry_point_io.clear();
self.named_expressions.clear();
self.wrapped.clear();
self.written_committed_intersection = false;
self.continue_ctx.clear();
self.need_bake_expressions.clear();
}
Expand Down Expand Up @@ -1218,6 +1220,13 @@ impl<'a, W: fmt::Write> super::Writer<'a, W> {
TypeInner::Array { base, size, .. } | TypeInner::BindingArray { base, size } => {
self.write_array_size(module, base, size)?;
}
TypeInner::AccelerationStructure => {
write!(self.out, "RaytracingAccelerationStructure")?;
}
TypeInner::RayQuery => {
// these are constant flags, there are dynamic flags also but constant flags are not supported by naga
write!(self.out, "RayQuery<RAY_FLAG_NONE>")?;
}
_ => return Err(Error::Unimplemented(format!("write_value_type {inner:?}"))),
}

Expand Down Expand Up @@ -1375,15 +1384,20 @@ impl<'a, W: fmt::Write> super::Writer<'a, W> {
self.write_array_size(module, base, size)?;
}

write!(self.out, " = ")?;
// Write the local initializer if needed
if let Some(init) = local.init {
self.write_expr(module, init, func_ctx)?;
} else {
// Zero initialize local variables
self.write_default_init(module, local.ty)?;
match module.types[local.ty].inner {
// from https://microsoft.github.io/DirectX-Specs/d3d/Raytracing.html#tracerayinline-example-1 it seems that ray queries shouldn't be zeroed
TypeInner::RayQuery => {}
_ => {
write!(self.out, " = ")?;
// Write the local initializer if needed
if let Some(init) = local.init {
self.write_expr(module, init, func_ctx)?;
} else {
// Zero initialize local variables
self.write_default_init(module, local.ty)?;
}
}
}

// Finish the local with `;` and add a newline (only for readability)
writeln!(self.out, ";")?
}
Expand Down Expand Up @@ -2224,7 +2238,37 @@ impl<'a, W: fmt::Write> super::Writer<'a, W> {
} => {
self.write_switch(module, func_ctx, level, selector, cases)?;
}
Statement::RayQuery { .. } => unreachable!(),
Statement::RayQuery { query, ref fun } => match *fun {
RayQueryFunction::Initialize {
acceleration_structure,
descriptor,
} => {
write!(self.out, "{level}")?;
self.write_expr(module, query, func_ctx)?;
write!(self.out, ".TraceRayInline(")?;
self.write_expr(module, acceleration_structure, func_ctx)?;
write!(self.out, ", ")?;
self.write_expr(module, descriptor, func_ctx)?;
write!(self.out, ".flags, ")?;
self.write_expr(module, descriptor, func_ctx)?;
write!(self.out, ".cull_mask, ")?;
write!(self.out, "RayDescFromRayDesc_(")?;
self.write_expr(module, descriptor, func_ctx)?;
writeln!(self.out, "));")?;
}
RayQueryFunction::Proceed { result } => {
write!(self.out, "{level}")?;
let name = Baked(result).to_string();
write!(self.out, "const bool {name} = ")?;
self.named_expressions.insert(result, name);
self.write_expr(module, query, func_ctx)?;
writeln!(self.out, ".Proceed();")?;
}
RayQueryFunction::Terminate => {
self.write_expr(module, query, func_ctx)?;
writeln!(self.out, ".Abort();")?;
}
},
Statement::SubgroupBallot { result, predicate } => {
write!(self.out, "{level}")?;
let name = Baked(result).to_string();
Expand Down Expand Up @@ -3581,8 +3625,17 @@ impl<'a, W: fmt::Write> super::Writer<'a, W> {
self.write_expr(module, reject, func_ctx)?;
write!(self.out, ")")?
}
// Not supported yet
Expression::RayQueryGetIntersection { .. } => unreachable!(),
Expression::RayQueryGetIntersection { query, committed } => {
if committed {
write!(self.out, "GetCommittedIntersection(")?;
self.write_expr(module, query, func_ctx)?;
write!(self.out, ")")?;
} else {
write!(self.out, "GetCandidateIntersection(")?;
self.write_expr(module, query, func_ctx)?;
write!(self.out, ")")?;
}
}
// Nothing to do here, since call expression already cached
Expression::CallResult(_)
| Expression::AtomicResult { .. }
Expand Down
7 changes: 7 additions & 0 deletions naga/tests/in/ray-query.param.ron
Original file line number Diff line number Diff line change
Expand Up @@ -11,4 +11,11 @@
per_entry_point_map: {},
inline_samplers: [],
),
hlsl: (
shader_model: V6_5,
binding_map: {},
fake_missing_bindings: true,
special_constants_binding: None,
zero_initialize_workgroup_memory: true,
)
)
Loading
Loading