From 4a4345600b0a43f7281c73158e92f4a3044ae1ee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fran=C3=A7ois?= Date: Thu, 14 Jul 2022 20:37:49 +0000 Subject: [PATCH] update wgpu to 0.13 (#5168) # Objective - Update wgpu to 0.13 - ~~Wait, is wgpu 0.13 released? No, but I had most of the changes already ready since playing with webgpu~~ well it has been released now - Also update parking_lot to 0.12 and naga to 0.9 ## Solution - Update syntax for wgsl shaders https://github.com/gfx-rs/wgpu/blob/master/CHANGELOG.md#wgsl-syntax - Add a few options, remove some references: https://github.com/gfx-rs/wgpu/blob/master/CHANGELOG.md#other-breaking-changes - fragment inputs should now exactly match vertex outputs for locations, so I added exports for those to be able to reuse them https://github.com/gfx-rs/wgpu/pull/2704 --- .github/bors.toml | 2 +- .github/workflows/validation-jobs.yml | 4 +- assets/shaders/animate_shader.wgsl | 22 +- assets/shaders/array_texture.wgsl | 22 +- assets/shaders/custom_material.wgsl | 14 +- .../custom_material_chromatic_aberration.wgsl | 12 +- .../custom_material_screenspace_texture.wgsl | 11 +- assets/shaders/custom_vertex_attribute.wgsl | 20 +- assets/shaders/game_of_life.wgsl | 30 +- assets/shaders/instancing.wgsl | 22 +- assets/shaders/shader_defs.wgsl | 10 +- crates/bevy_asset/Cargo.toml | 2 +- crates/bevy_audio/Cargo.toml | 2 +- .../src/core_2d/main_pass_2d_node.rs | 8 +- .../src/core_3d/main_pass_3d_node.rs | 16 +- crates/bevy_pbr/src/material.rs | 8 +- crates/bevy_pbr/src/render/depth.wgsl | 16 +- crates/bevy_pbr/src/render/mesh.rs | 12 +- crates/bevy_pbr/src/render/mesh.wgsl | 48 +-- crates/bevy_pbr/src/render/mesh_bindings.wgsl | 4 +- crates/bevy_pbr/src/render/mesh_types.wgsl | 8 +- .../src/render/mesh_vertex_output.wgsl | 13 + .../src/render/mesh_view_bindings.wgsl | 28 +- .../bevy_pbr/src/render/mesh_view_types.wgsl | 68 ++-- crates/bevy_pbr/src/render/pbr.wgsl | 20 +- crates/bevy_pbr/src/render/pbr_bindings.wgsl | 22 +- crates/bevy_pbr/src/render/pbr_functions.wgsl | 16 +- crates/bevy_pbr/src/render/pbr_lighting.wgsl | 4 +- crates/bevy_pbr/src/render/pbr_types.wgsl | 14 +- crates/bevy_pbr/src/render/skinning.wgsl | 22 +- crates/bevy_pbr/src/render/wireframe.wgsl | 18 +- crates/bevy_reflect/Cargo.toml | 2 +- crates/bevy_render/Cargo.toml | 6 +- .../src/camera/camera_driver_node.rs | 4 +- .../src/render_resource/bind_group.rs | 14 +- .../src/render_resource/pipeline.rs | 2 +- .../src/render_resource/pipeline_cache.rs | 4 +- .../bevy_render/src/render_resource/shader.rs | 228 ++++++------ crates/bevy_render/src/renderer/mod.rs | 3 + .../bevy_render/src/renderer/render_device.rs | 19 +- crates/bevy_render/src/texture/basis.rs | 13 +- crates/bevy_render/src/texture/image.rs | 31 +- crates/bevy_render/src/texture/ktx2.rs | 337 +++++++----------- crates/bevy_render/src/view/window.rs | 2 + .../src/mesh2d/color_material.wgsl | 30 +- crates/bevy_sprite/src/mesh2d/mesh.rs | 12 +- crates/bevy_sprite/src/mesh2d/mesh2d.wgsl | 41 +-- .../src/mesh2d/mesh2d_bindings.wgsl | 2 +- .../bevy_sprite/src/mesh2d/mesh2d_types.wgsl | 6 +- .../src/mesh2d/mesh2d_vertex_output.wgsl | 11 + .../src/mesh2d/mesh2d_view_bindings.wgsl | 2 +- .../src/mesh2d/mesh2d_view_types.wgsl | 14 +- crates/bevy_sprite/src/render/mod.rs | 4 +- crates/bevy_sprite/src/render/sprite.wgsl | 28 +- crates/bevy_ui/src/render/pipeline.rs | 4 +- crates/bevy_ui/src/render/render_pass.rs | 4 +- crates/bevy_ui/src/render/ui.wgsl | 28 +- crates/bevy_window/src/window.rs | 20 +- deny.toml | 2 + examples/2d/mesh2d_manual.rs | 22 +- .../shader/compute_shader_game_of_life.rs | 6 +- examples/stress_tests/bevymark.rs | 2 +- examples/stress_tests/many_cubes.rs | 2 +- examples/stress_tests/many_foxes.rs | 2 +- examples/stress_tests/many_lights.rs | 2 +- examples/stress_tests/many_sprites.rs | 2 +- examples/ui/text_debug.rs | 2 +- examples/window/low_power.rs | 2 +- examples/window/multiple_windows.rs | 2 +- examples/window/window_settings.rs | 2 +- 70 files changed, 664 insertions(+), 773 deletions(-) create mode 100644 crates/bevy_pbr/src/render/mesh_vertex_output.wgsl create mode 100644 crates/bevy_sprite/src/mesh2d/mesh2d_vertex_output.wgsl diff --git a/.github/bors.toml b/.github/bors.toml index 4aa3080f4c7fa5..3390d8f2d668e3 100644 --- a/.github/bors.toml +++ b/.github/bors.toml @@ -16,7 +16,7 @@ status = [ "miri", "check-compiles", "build-and-install-on-iOS", - "run-examples-on-windows", + "run-examples-on-windows-dx12", ] use_squash_merge = true diff --git a/.github/workflows/validation-jobs.yml b/.github/workflows/validation-jobs.yml index 1c5e8d553ecd65..474c7f4a703fee 100644 --- a/.github/workflows/validation-jobs.yml +++ b/.github/workflows/validation-jobs.yml @@ -61,7 +61,7 @@ jobs: - name: Build APK run: cargo apk build --example android_example - run-examples-on-windows: + run-examples-on-windows-dx12: runs-on: windows-latest timeout-minutes: 30 steps: @@ -91,7 +91,7 @@ jobs: for example in .github/example-run/*.ron; do example_name=`basename $example .ron` echo "running $example_name - "`date` - time CI_TESTING_CONFIG=$example cargo run --example $example_name --features "bevy_ci_testing" + time WGPU_BACKEND=dx12 CI_TESTING_CONFIG=$example cargo run --example $example_name --features "bevy_ci_testing" sleep 10 done diff --git a/assets/shaders/animate_shader.wgsl b/assets/shaders/animate_shader.wgsl index d71b30bfafe7cc..947c4a5420aff8 100644 --- a/assets/shaders/animate_shader.wgsl +++ b/assets/shaders/animate_shader.wgsl @@ -1,24 +1,24 @@ #import bevy_pbr::mesh_types #import bevy_pbr::mesh_view_bindings -[[group(1), binding(0)]] +@group(1) @binding(0) var mesh: Mesh; // NOTE: Bindings must come before functions that use them! #import bevy_pbr::mesh_functions struct Vertex { - [[location(0)]] position: vec3; - [[location(1)]] normal: vec3; - [[location(2)]] uv: vec2; + @location(0) position: vec3, + @location(1) normal: vec3, + @location(2) uv: vec2, }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; - [[location(0)]] uv: vec2; + @builtin(position) clip_position: vec4, + @location(0) uv: vec2, }; -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { var out: VertexOutput; out.clip_position = mesh_position_local_to_clip(mesh.model, vec4(vertex.position, 1.0)); @@ -28,9 +28,9 @@ fn vertex(vertex: Vertex) -> VertexOutput { struct Time { - time_since_startup: f32; + time_since_startup: f32, }; -[[group(2), binding(0)]] +@group(2) @binding(0) var time: Time; @@ -54,8 +54,8 @@ fn oklab_to_linear_srgb(c: vec3) -> vec3 { ); } -[[stage(fragment)]] -fn fragment(in: VertexOutput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: VertexOutput) -> @location(0) vec4 { let speed = 2.0; let t_1 = sin(time.time_since_startup * speed) * 0.5 + 0.5; let t_2 = cos(time.time_since_startup * speed); diff --git a/assets/shaders/array_texture.wgsl b/assets/shaders/array_texture.wgsl index a3b0c0fb588529..91f4564e7fe27e 100644 --- a/assets/shaders/array_texture.wgsl +++ b/assets/shaders/array_texture.wgsl @@ -8,27 +8,19 @@ #import bevy_pbr::shadows #import bevy_pbr::pbr_functions -[[group(1), binding(0)]] +@group(1) @binding(0) var my_array_texture: texture_2d_array; -[[group(1), binding(1)]] +@group(1) @binding(1) var my_array_texture_sampler: sampler; struct FragmentInput { - [[builtin(front_facing)]] is_front: bool; - [[builtin(position)]] frag_coord: vec4; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; - [[location(2)]] uv: vec2; -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif -#ifdef VERTEX_COLORS - [[location(4)]] color: vec4; -#endif + @builtin(front_facing) is_front: bool, + @builtin(position) frag_coord: vec4, + #import bevy_pbr::mesh_vertex_output }; -[[stage(fragment)]] -fn fragment(in: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: FragmentInput) -> @location(0) vec4 { let layer = i32(in.world_position.x) & 0x3; // Prepare a 'processed' StandardMaterial by sampling all textures to resolve diff --git a/assets/shaders/custom_material.wgsl b/assets/shaders/custom_material.wgsl index 33d706f9d885d2..95b1b7d26a196a 100644 --- a/assets/shaders/custom_material.wgsl +++ b/assets/shaders/custom_material.wgsl @@ -1,15 +1,17 @@ struct CustomMaterial { - color: vec4; + color: vec4, }; -[[group(1), binding(0)]] +@group(1) @binding(0) var material: CustomMaterial; -[[group(1), binding(1)]] +@group(1) @binding(1) var base_color_texture: texture_2d; -[[group(1), binding(2)]] +@group(1) @binding(2) var base_color_sampler: sampler; -[[stage(fragment)]] -fn fragment([[location(2)]] uv: vec2) -> [[location(0)]] vec4 { +@fragment +fn fragment( + #import bevy_pbr::mesh_vertex_output +) -> @location(0) vec4 { return material.color * textureSample(base_color_texture, base_color_sampler, uv); } diff --git a/assets/shaders/custom_material_chromatic_aberration.wgsl b/assets/shaders/custom_material_chromatic_aberration.wgsl index 811cfb8810abc1..e8ccdcfb625133 100644 --- a/assets/shaders/custom_material_chromatic_aberration.wgsl +++ b/assets/shaders/custom_material_chromatic_aberration.wgsl @@ -1,14 +1,16 @@ #import bevy_pbr::mesh_view_bindings -[[group(1), binding(0)]] +@group(1) @binding(0) var texture: texture_2d; -[[group(1), binding(1)]] +@group(1) @binding(1) var our_sampler: sampler; - -[[stage(fragment)]] -fn fragment([[builtin(position)]] position: vec4) -> [[location(0)]] vec4 { +@fragment +fn fragment( + @builtin(position) position: vec4, + #import bevy_sprite::mesh2d_vertex_output +) -> @location(0) vec4 { // Get screen position with coordinates from 0 to 1 let uv = position.xy / vec2(view.width, view.height); let offset_strength = 0.02; diff --git a/assets/shaders/custom_material_screenspace_texture.wgsl b/assets/shaders/custom_material_screenspace_texture.wgsl index 2228b79083af9f..aad35920c093ec 100644 --- a/assets/shaders/custom_material_screenspace_texture.wgsl +++ b/assets/shaders/custom_material_screenspace_texture.wgsl @@ -1,12 +1,15 @@ #import bevy_pbr::mesh_view_bindings -[[group(1), binding(0)]] +@group(1) @binding(0) var texture: texture_2d; -[[group(1), binding(1)]] +@group(1) @binding(1) var texture_sampler: sampler; -[[stage(fragment)]] -fn fragment([[builtin(position)]] position: vec4) -> [[location(0)]] vec4 { +@fragment +fn fragment( + @builtin(position) position: vec4, + #import bevy_pbr::mesh_vertex_output +) -> @location(0) vec4 { let uv = position.xy / vec2(view.width, view.height); let color = textureSample(texture, texture_sampler, uv); return color; diff --git a/assets/shaders/custom_vertex_attribute.wgsl b/assets/shaders/custom_vertex_attribute.wgsl index 2def33a0986025..dd7cfc150308df 100644 --- a/assets/shaders/custom_vertex_attribute.wgsl +++ b/assets/shaders/custom_vertex_attribute.wgsl @@ -2,25 +2,25 @@ #import bevy_pbr::mesh_bindings struct CustomMaterial { - color: vec4; + color: vec4, }; -[[group(1), binding(0)]] +@group(1) @binding(0) var material: CustomMaterial; // NOTE: Bindings must come before functions that use them! #import bevy_pbr::mesh_functions struct Vertex { - [[location(0)]] position: vec3; - [[location(1)]] blend_color: vec4; + @location(0) position: vec3, + @location(1) blend_color: vec4, }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; - [[location(0)]] blend_color: vec4; + @builtin(position) clip_position: vec4, + @location(0) blend_color: vec4, }; -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { var out: VertexOutput; out.clip_position = mesh_position_local_to_clip(mesh.model, vec4(vertex.position, 1.0)); @@ -29,10 +29,10 @@ fn vertex(vertex: Vertex) -> VertexOutput { } struct FragmentInput { - [[location(0)]] blend_color: vec4; + @location(0) blend_color: vec4, }; -[[stage(fragment)]] -fn fragment(input: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(input: FragmentInput) -> @location(0) vec4 { return material.color * input.blend_color; } diff --git a/assets/shaders/game_of_life.wgsl b/assets/shaders/game_of_life.wgsl index 6acb095fe78bcd..d2a63c975cde42 100644 --- a/assets/shaders/game_of_life.wgsl +++ b/assets/shaders/game_of_life.wgsl @@ -1,4 +1,4 @@ -[[group(0), binding(0)]] +@group(0) @binding(0) var texture: texture_storage_2d; fn hash(value: u32) -> u32 { @@ -15,8 +15,8 @@ fn randomFloat(value: u32) -> f32 { return f32(hash(value)) / 4294967295.0; } -[[stage(compute), workgroup_size(8, 8, 1)]] -fn init([[builtin(global_invocation_id)]] invocation_id: vec3, [[builtin(num_workgroups)]] num_workgroups: vec3) { +@compute @workgroup_size(8, 8, 1) +fn init(@builtin(global_invocation_id) invocation_id: vec3, @builtin(num_workgroups) num_workgroups: vec3) { let location = vec2(i32(invocation_id.x), i32(invocation_id.y)); let location_f32 = vec2(f32(invocation_id.x), f32(invocation_id.y)); @@ -28,24 +28,24 @@ fn init([[builtin(global_invocation_id)]] invocation_id: vec3, [[builtin(nu } -fn get(location: vec2, offset_x: i32, offset_y: i32) -> i32 { +fn is_alive(location: vec2, offset_x: i32, offset_y: i32) -> i32 { let value: vec4 = textureLoad(texture, location + vec2(offset_x, offset_y)); return i32(value.x); } fn count_alive(location: vec2) -> i32 { - return get(location, -1, -1) + - get(location, -1, 0) + - get(location, -1, 1) + - get(location, 0, -1) + - get(location, 0, 1) + - get(location, 1, -1) + - get(location, 1, 0) + - get(location, 1, 1); + return is_alive(location, -1, -1) + + is_alive(location, -1, 0) + + is_alive(location, -1, 1) + + is_alive(location, 0, -1) + + is_alive(location, 0, 1) + + is_alive(location, 1, -1) + + is_alive(location, 1, 0) + + is_alive(location, 1, 1); } -[[stage(compute), workgroup_size(8, 8, 1)]] -fn update([[builtin(global_invocation_id)]] invocation_id: vec3) { +@compute @workgroup_size(8, 8, 1) +fn update(@builtin(global_invocation_id) invocation_id: vec3) { let location = vec2(i32(invocation_id.x), i32(invocation_id.y)); let n_alive = count_alive(location); @@ -55,7 +55,7 @@ fn update([[builtin(global_invocation_id)]] invocation_id: vec3) { if (n_alive == 3) { alive = true; } else if (n_alive == 2) { - let currently_alive = get(location, 0, 0); + let currently_alive = is_alive(location, 0, 0); alive = bool(currently_alive); } else { alive = false; diff --git a/assets/shaders/instancing.wgsl b/assets/shaders/instancing.wgsl index c83a853b078426..7cb00b039a84ad 100644 --- a/assets/shaders/instancing.wgsl +++ b/assets/shaders/instancing.wgsl @@ -1,27 +1,27 @@ #import bevy_pbr::mesh_types #import bevy_pbr::mesh_view_bindings -[[group(1), binding(0)]] +@group(1) @binding(0) var mesh: Mesh; // NOTE: Bindings must come before functions that use them! #import bevy_pbr::mesh_functions struct Vertex { - [[location(0)]] position: vec3; - [[location(1)]] normal: vec3; - [[location(2)]] uv: vec2; + @location(0) position: vec3, + @location(1) normal: vec3, + @location(2) uv: vec2, - [[location(3)]] i_pos_scale: vec4; - [[location(4)]] i_color: vec4; + @location(3) i_pos_scale: vec4, + @location(4) i_color: vec4, }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; - [[location(0)]] color: vec4; + @builtin(position) clip_position: vec4, + @location(0) color: vec4, }; -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { let position = vertex.position * vertex.i_pos_scale.w + vertex.i_pos_scale.xyz; var out: VertexOutput; @@ -30,7 +30,7 @@ fn vertex(vertex: Vertex) -> VertexOutput { return out; } -[[stage(fragment)]] -fn fragment(in: VertexOutput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: VertexOutput) -> @location(0) vec4 { return in.color; } diff --git a/assets/shaders/shader_defs.wgsl b/assets/shaders/shader_defs.wgsl index 5518a73b0247b2..0efa91231a622b 100644 --- a/assets/shaders/shader_defs.wgsl +++ b/assets/shaders/shader_defs.wgsl @@ -1,12 +1,14 @@ struct CustomMaterial { - color: vec4; + color: vec4, }; -[[group(1), binding(0)]] +@group(1) @binding(0) var material: CustomMaterial; -[[stage(fragment)]] -fn fragment() -> [[location(0)]] vec4 { +@fragment +fn fragment( + #import bevy_pbr::mesh_vertex_output +) -> @location(0) vec4 { #ifdef IS_RED return vec4(1.0, 0.0, 0.0, 1.0); #else diff --git a/crates/bevy_asset/Cargo.toml b/crates/bevy_asset/Cargo.toml index 96c0bd222e12b3..e16455537f78a3 100644 --- a/crates/bevy_asset/Cargo.toml +++ b/crates/bevy_asset/Cargo.toml @@ -31,7 +31,7 @@ thiserror = "1.0" downcast-rs = "1.2.0" fastrand = "1.7.0" notify = { version = "=5.0.0-pre.11", optional = true } -parking_lot = "0.11.0" +parking_lot = "0.12.1" [target.'cfg(target_arch = "wasm32")'.dependencies] wasm-bindgen = { version = "0.2" } diff --git a/crates/bevy_audio/Cargo.toml b/crates/bevy_audio/Cargo.toml index 9b04cc4bf826db..b8d1148c8fc6ac 100644 --- a/crates/bevy_audio/Cargo.toml +++ b/crates/bevy_audio/Cargo.toml @@ -19,7 +19,7 @@ bevy_utils = { path = "../bevy_utils", version = "0.8.0-dev" } # other anyhow = "1.0.4" rodio = { version = "0.15", default-features = false } -parking_lot = "0.11.0" +parking_lot = "0.12.1" [target.'cfg(target_arch = "wasm32")'.dependencies] rodio = { version = "0.15", default-features = false, features = ["wasm-bindgen"] } diff --git a/crates/bevy_core_pipeline/src/core_2d/main_pass_2d_node.rs b/crates/bevy_core_pipeline/src/core_2d/main_pass_2d_node.rs index 626b022d46eded..f4006822f3436b 100644 --- a/crates/bevy_core_pipeline/src/core_2d/main_pass_2d_node.rs +++ b/crates/bevy_core_pipeline/src/core_2d/main_pass_2d_node.rs @@ -64,7 +64,7 @@ impl Node for MainPass2dNode { let _main_pass_2d = info_span!("main_pass_2d").entered(); let pass_descriptor = RenderPassDescriptor { label: Some("main_pass_2d"), - color_attachments: &[target.get_color_attachment(Operations { + color_attachments: &[Some(target.get_color_attachment(Operations { load: match camera_2d.clear_color { ClearColorConfig::Default => { LoadOp::Clear(world.resource::().0.into()) @@ -73,7 +73,7 @@ impl Node for MainPass2dNode { ClearColorConfig::None => LoadOp::Load, }, store: true, - })], + }))], depth_stencil_attachment: None, }; @@ -102,10 +102,10 @@ impl Node for MainPass2dNode { let _reset_viewport_pass_2d = info_span!("reset_viewport_pass_2d").entered(); let pass_descriptor = RenderPassDescriptor { label: Some("reset_viewport_pass_2d"), - color_attachments: &[target.get_color_attachment(Operations { + color_attachments: &[Some(target.get_color_attachment(Operations { load: LoadOp::Load, store: true, - })], + }))], depth_stencil_attachment: None, }; diff --git a/crates/bevy_core_pipeline/src/core_3d/main_pass_3d_node.rs b/crates/bevy_core_pipeline/src/core_3d/main_pass_3d_node.rs index ec9aa7a6ff8847..cf3a3d38fa6032 100644 --- a/crates/bevy_core_pipeline/src/core_3d/main_pass_3d_node.rs +++ b/crates/bevy_core_pipeline/src/core_3d/main_pass_3d_node.rs @@ -73,7 +73,7 @@ impl Node for MainPass3dNode { label: Some("main_opaque_pass_3d"), // NOTE: The opaque pass loads the color // buffer as well as writing to it. - color_attachments: &[target.get_color_attachment(Operations { + color_attachments: &[Some(target.get_color_attachment(Operations { load: match camera_3d.clear_color { ClearColorConfig::Default => { LoadOp::Clear(world.resource::().0.into()) @@ -82,7 +82,7 @@ impl Node for MainPass3dNode { ClearColorConfig::None => LoadOp::Load, }, store: true, - })], + }))], depth_stencil_attachment: Some(RenderPassDepthStencilAttachment { view: &depth.view, // NOTE: The opaque main pass loads the depth buffer and possibly overwrites it @@ -119,10 +119,10 @@ impl Node for MainPass3dNode { let pass_descriptor = RenderPassDescriptor { label: Some("main_alpha_mask_pass_3d"), // NOTE: The alpha_mask pass loads the color buffer as well as overwriting it where appropriate. - color_attachments: &[target.get_color_attachment(Operations { + color_attachments: &[Some(target.get_color_attachment(Operations { load: LoadOp::Load, store: true, - })], + }))], depth_stencil_attachment: Some(RenderPassDepthStencilAttachment { view: &depth.view, // NOTE: The alpha mask pass loads the depth buffer and possibly overwrites it @@ -158,10 +158,10 @@ impl Node for MainPass3dNode { let pass_descriptor = RenderPassDescriptor { label: Some("main_transparent_pass_3d"), // NOTE: The transparent pass loads the color buffer as well as overwriting it where appropriate. - color_attachments: &[target.get_color_attachment(Operations { + color_attachments: &[Some(target.get_color_attachment(Operations { load: LoadOp::Load, store: true, - })], + }))], depth_stencil_attachment: Some(RenderPassDepthStencilAttachment { view: &depth.view, // NOTE: For the transparent pass we load the depth buffer. There should be no @@ -202,10 +202,10 @@ impl Node for MainPass3dNode { let _reset_viewport_pass_3d = info_span!("reset_viewport_pass_3d").entered(); let pass_descriptor = RenderPassDescriptor { label: Some("reset_viewport_pass_3d"), - color_attachments: &[target.get_color_attachment(Operations { + color_attachments: &[Some(target.get_color_attachment(Operations { load: LoadOp::Load, store: true, - })], + }))], depth_stencil_attachment: None, }; diff --git a/crates/bevy_pbr/src/material.rs b/crates/bevy_pbr/src/material.rs index 5e840632c19066..0dcbe50de7a737 100644 --- a/crates/bevy_pbr/src/material.rs +++ b/crates/bevy_pbr/src/material.rs @@ -97,14 +97,14 @@ use std::marker::PhantomData; /// /// ```wgsl /// struct CustomMaterial { -/// color: vec4; +/// color: vec4, /// }; /// -/// [[group(1), binding(0)]] +/// @group(1) @binding(0) /// var material: CustomMaterial; -/// [[group(1), binding(1)]] +/// @group(1) @binding(1) /// var color_texture: texture_2d; -/// [[group(1), binding(2)]] +/// @group(1) @binding(2) /// var color_sampler: sampler; /// ``` pub trait Material: AsBindGroup + Send + Sync + Clone + TypeUuid + Sized + 'static { diff --git a/crates/bevy_pbr/src/render/depth.wgsl b/crates/bevy_pbr/src/render/depth.wgsl index 23c797d9dfa694..5a238cb1e5f212 100644 --- a/crates/bevy_pbr/src/render/depth.wgsl +++ b/crates/bevy_pbr/src/render/depth.wgsl @@ -1,14 +1,14 @@ #import bevy_pbr::mesh_view_types #import bevy_pbr::mesh_types -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; -[[group(1), binding(0)]] +@group(1) @binding(0) var mesh: Mesh; #ifdef SKINNED -[[group(1), binding(1)]] +@group(1) @binding(1) var joint_matrices: SkinnedMesh; #import bevy_pbr::skinning #endif @@ -17,18 +17,18 @@ var joint_matrices: SkinnedMesh; #import bevy_pbr::mesh_functions struct Vertex { - [[location(0)]] position: vec3; + @location(0) position: vec3, #ifdef SKINNED - [[location(4)]] joint_indices: vec4; - [[location(5)]] joint_weights: vec4; + @location(4) joint_indices: vec4, + @location(5) joint_weights: vec4, #endif }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; + @builtin(position) clip_position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { #ifdef SKINNED let model = skin_model(vertex.joint_indices, vertex.joint_weights); diff --git a/crates/bevy_pbr/src/render/mesh.rs b/crates/bevy_pbr/src/render/mesh.rs index 991735bd43397c..3bf3fb2a0a5011 100644 --- a/crates/bevy_pbr/src/render/mesh.rs +++ b/crates/bevy_pbr/src/render/mesh.rs @@ -37,6 +37,8 @@ const MAX_JOINTS: usize = 256; const JOINT_SIZE: usize = std::mem::size_of::(); pub(crate) const JOINT_BUFFER_SIZE: usize = MAX_JOINTS * JOINT_SIZE; +pub const MESH_VERTEX_OUTPUT: HandleUntyped = + HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 2645551199423808407); pub const MESH_VIEW_TYPES_HANDLE: HandleUntyped = HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 8140454348013264787); pub const MESH_VIEW_BINDINGS_HANDLE: HandleUntyped = @@ -54,6 +56,12 @@ pub const SKINNING_HANDLE: HandleUntyped = impl Plugin for MeshRenderPlugin { fn build(&self, app: &mut bevy_app::App) { + load_internal_asset!( + app, + MESH_VERTEX_OUTPUT, + "mesh_vertex_output.wgsl", + Shader::from_wgsl + ); load_internal_asset!( app, MESH_VIEW_TYPES_HANDLE, @@ -601,11 +609,11 @@ impl SpecializedMeshPipeline for MeshPipeline { shader: MESH_SHADER_HANDLE.typed::(), shader_defs, entry_point: "fragment".into(), - targets: vec![ColorTargetState { + targets: vec![Some(ColorTargetState { format: TextureFormat::bevy_default(), blend, write_mask: ColorWrites::ALL, - }], + })], }), layout: Some(bind_group_layout), primitive: PrimitiveState { diff --git a/crates/bevy_pbr/src/render/mesh.wgsl b/crates/bevy_pbr/src/render/mesh.wgsl index 4d6de7c5bfdc43..32cbe31c8d1610 100644 --- a/crates/bevy_pbr/src/render/mesh.wgsl +++ b/crates/bevy_pbr/src/render/mesh.wgsl @@ -5,39 +5,29 @@ #import bevy_pbr::mesh_functions struct Vertex { - [[location(0)]] position: vec3; - [[location(1)]] normal: vec3; + @location(0) position: vec3, + @location(1) normal: vec3, #ifdef VERTEX_UVS - [[location(2)]] uv: vec2; + @location(2) uv: vec2, #endif #ifdef VERTEX_TANGENTS - [[location(3)]] tangent: vec4; + @location(3) tangent: vec4, #endif #ifdef VERTEX_COLORS - [[location(4)]] color: vec4; + @location(4) color: vec4, #endif #ifdef SKINNED - [[location(5)]] joint_indices: vec4; - [[location(6)]] joint_weights: vec4; + @location(5) joint_indices: vec4, + @location(6) joint_weights: vec4, #endif }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; -#ifdef VERTEX_UVS - [[location(2)]] uv: vec2; -#endif -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif -#ifdef VERTEX_COLORS - [[location(4)]] color: vec4; -#endif + @builtin(position) clip_position: vec4, + #import bevy_pbr::mesh_vertex_output }; -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { var out: VertexOutput; #ifdef SKINNED @@ -63,22 +53,12 @@ fn vertex(vertex: Vertex) -> VertexOutput { } struct FragmentInput { - [[builtin(front_facing)]] is_front: bool; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; -#ifdef VERTEX_UVS - [[location(2)]] uv: vec2; -#endif -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif -#ifdef VERTEX_COLORS - [[location(4)]] color: vec4; -#endif + @builtin(front_facing) is_front: bool, + #import bevy_pbr::mesh_vertex_output }; -[[stage(fragment)]] -fn fragment(in: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: FragmentInput) -> @location(0) vec4 { #ifdef VERTEX_COLORS return in.color; #else diff --git a/crates/bevy_pbr/src/render/mesh_bindings.wgsl b/crates/bevy_pbr/src/render/mesh_bindings.wgsl index 8f0c69a7818328..d0781e149b4203 100644 --- a/crates/bevy_pbr/src/render/mesh_bindings.wgsl +++ b/crates/bevy_pbr/src/render/mesh_bindings.wgsl @@ -2,10 +2,10 @@ #import bevy_pbr::mesh_types -[[group(2), binding(0)]] +@group(2) @binding(0) var mesh: Mesh; #ifdef SKINNED -[[group(2), binding(1)]] +@group(2) @binding(1) var joint_matrices: SkinnedMesh; #import bevy_pbr::skinning #endif diff --git a/crates/bevy_pbr/src/render/mesh_types.wgsl b/crates/bevy_pbr/src/render/mesh_types.wgsl index c04720d551e42e..e0bfd2c2899cdb 100644 --- a/crates/bevy_pbr/src/render/mesh_types.wgsl +++ b/crates/bevy_pbr/src/render/mesh_types.wgsl @@ -1,15 +1,15 @@ #define_import_path bevy_pbr::mesh_types struct Mesh { - model: mat4x4; - inverse_transpose_model: mat4x4; + model: mat4x4, + inverse_transpose_model: mat4x4, // 'flags' is a bit field indicating various options. u32 is 32 bits so we have up to 32 options. - flags: u32; + flags: u32, }; #ifdef SKINNED struct SkinnedMesh { - data: array, 256u>; + data: array, 256u>, }; #endif diff --git a/crates/bevy_pbr/src/render/mesh_vertex_output.wgsl b/crates/bevy_pbr/src/render/mesh_vertex_output.wgsl new file mode 100644 index 00000000000000..ba0c6b2237c0ae --- /dev/null +++ b/crates/bevy_pbr/src/render/mesh_vertex_output.wgsl @@ -0,0 +1,13 @@ +#define_import_path bevy_pbr::mesh_vertex_output + +@location(0) world_position: vec4, +@location(1) world_normal: vec3, +#ifdef VERTEX_UVS +@location(2) uv: vec2, +#endif +#ifdef VERTEX_TANGENTS +@location(3) world_tangent: vec4, +#endif +#ifdef VERTEX_COLORS +@location(4) color: vec4, +#endif diff --git a/crates/bevy_pbr/src/render/mesh_view_bindings.wgsl b/crates/bevy_pbr/src/render/mesh_view_bindings.wgsl index ec6f5dbb4398c4..04780be3f0c961 100644 --- a/crates/bevy_pbr/src/render/mesh_view_bindings.wgsl +++ b/crates/bevy_pbr/src/render/mesh_view_bindings.wgsl @@ -2,41 +2,41 @@ #import bevy_pbr::mesh_view_types -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; -[[group(0), binding(1)]] +@group(0) @binding(1) var lights: Lights; #ifdef NO_ARRAY_TEXTURES_SUPPORT -[[group(0), binding(2)]] +@group(0) @binding(2) var point_shadow_textures: texture_depth_cube; #else -[[group(0), binding(2)]] +@group(0) @binding(2) var point_shadow_textures: texture_depth_cube_array; #endif -[[group(0), binding(3)]] +@group(0) @binding(3) var point_shadow_textures_sampler: sampler_comparison; #ifdef NO_ARRAY_TEXTURES_SUPPORT -[[group(0), binding(4)]] +@group(0) @binding(4) var directional_shadow_textures: texture_depth_2d; #else -[[group(0), binding(4)]] +@group(0) @binding(4) var directional_shadow_textures: texture_depth_2d_array; #endif -[[group(0), binding(5)]] +@group(0) @binding(5) var directional_shadow_textures_sampler: sampler_comparison; #ifdef NO_STORAGE_BUFFERS_SUPPORT -[[group(0), binding(6)]] +@group(0) @binding(6) var point_lights: PointLights; -[[group(0), binding(7)]] +@group(0) @binding(7) var cluster_light_index_lists: ClusterLightIndexLists; -[[group(0), binding(8)]] +@group(0) @binding(8) var cluster_offsets_and_counts: ClusterOffsetsAndCounts; #else -[[group(0), binding(6)]] +@group(0) @binding(6) var point_lights: PointLights; -[[group(0), binding(7)]] +@group(0) @binding(7) var cluster_light_index_lists: ClusterLightIndexLists; -[[group(0), binding(8)]] +@group(0) @binding(8) var cluster_offsets_and_counts: ClusterOffsetsAndCounts; #endif diff --git a/crates/bevy_pbr/src/render/mesh_view_types.wgsl b/crates/bevy_pbr/src/render/mesh_view_types.wgsl index 98b447209cf0eb..ba6da2bb664f9c 100644 --- a/crates/bevy_pbr/src/render/mesh_view_types.wgsl +++ b/crates/bevy_pbr/src/render/mesh_view_types.wgsl @@ -1,51 +1,51 @@ #define_import_path bevy_pbr::mesh_view_types struct View { - view_proj: mat4x4; - inverse_view_proj: mat4x4; - view: mat4x4; - inverse_view: mat4x4; - projection: mat4x4; - inverse_projection: mat4x4; - world_position: vec3; - width: f32; - height: f32; + view_proj: mat4x4, + inverse_view_proj: mat4x4, + view: mat4x4, + inverse_view: mat4x4, + projection: mat4x4, + inverse_projection: mat4x4, + world_position: vec3, + width: f32, + height: f32, }; struct PointLight { // For point lights: the lower-right 2x2 values of the projection matrix [2][2] [2][3] [3][2] [3][3] // For spot lights: the direction (x,z), spot_scale and spot_offset - light_custom_data: vec4; - color_inverse_square_range: vec4; - position_radius: vec4; + light_custom_data: vec4, + color_inverse_square_range: vec4, + position_radius: vec4, // 'flags' is a bit field indicating various options. u32 is 32 bits so we have up to 32 options. - flags: u32; - shadow_depth_bias: f32; - shadow_normal_bias: f32; - spot_light_tan_angle: f32; + flags: u32, + shadow_depth_bias: f32, + shadow_normal_bias: f32, + spot_light_tan_angle: f32, }; let POINT_LIGHT_FLAGS_SHADOWS_ENABLED_BIT: u32 = 1u; let POINT_LIGHT_FLAGS_SPOT_LIGHT_Y_NEGATIVE: u32 = 2u; struct DirectionalLight { - view_projection: mat4x4; - color: vec4; - direction_to_light: vec3; + view_projection: mat4x4, + color: vec4, + direction_to_light: vec3, // 'flags' is a bit field indicating various options. u32 is 32 bits so we have up to 32 options. - flags: u32; - shadow_depth_bias: f32; - shadow_normal_bias: f32; + flags: u32, + shadow_depth_bias: f32, + shadow_normal_bias: f32, }; let DIRECTIONAL_LIGHT_FLAGS_SHADOWS_ENABLED_BIT: u32 = 1u; struct Lights { // NOTE: this array size must be kept in sync with the constants defined bevy_pbr2/src/render/light.rs - directional_lights: array; - ambient_color: vec4; + directional_lights: array, + ambient_color: vec4, // x/y/z dimensions and n_clusters in w - cluster_dimensions: vec4; + cluster_dimensions: vec4, // xy are vec2(cluster_dimensions.xy) / vec2(view.width, view.height) // // For perspective projections: @@ -56,32 +56,32 @@ struct Lights { // NOTE: near and far are +ve but -z is infront of the camera // z is -near // w is cluster_dimensions.z / (-far - -near) - cluster_factors: vec4; - n_directional_lights: u32; - spot_light_shadowmap_offset: i32; + cluster_factors: vec4, + n_directional_lights: u32, + spot_light_shadowmap_offset: i32, }; #ifdef NO_STORAGE_BUFFERS_SUPPORT struct PointLights { - data: array; + data: array, }; struct ClusterLightIndexLists { // each u32 contains 4 u8 indices into the PointLights array - data: array, 1024u>; + data: array, 1024u>, }; struct ClusterOffsetsAndCounts { // each u32 contains a 24-bit index into ClusterLightIndexLists in the high 24 bits // and an 8-bit count of the number of lights in the low 8 bits - data: array, 1024u>; + data: array, 1024u>, }; #else struct PointLights { - data: array; + data: array, }; struct ClusterLightIndexLists { - data: array; + data: array, }; struct ClusterOffsetsAndCounts { - data: array>; + data: array>, }; #endif diff --git a/crates/bevy_pbr/src/render/pbr.wgsl b/crates/bevy_pbr/src/render/pbr.wgsl index f632974a464a8c..b70b5b333943cf 100644 --- a/crates/bevy_pbr/src/render/pbr.wgsl +++ b/crates/bevy_pbr/src/render/pbr.wgsl @@ -9,23 +9,13 @@ #import bevy_pbr::pbr_functions struct FragmentInput { - [[builtin(front_facing)]] is_front: bool; - [[builtin(position)]] frag_coord: vec4; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; -#ifdef VERTEX_UVS - [[location(2)]] uv: vec2; -#endif -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif -#ifdef VERTEX_COLORS - [[location(4)]] color: vec4; -#endif + @builtin(front_facing) is_front: bool, + @builtin(position) frag_coord: vec4, + #import bevy_pbr::mesh_vertex_output }; -[[stage(fragment)]] -fn fragment(in: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: FragmentInput) -> @location(0) vec4 { var output_color: vec4 = material.base_color; #ifdef VERTEX_COLORS output_color = output_color * in.color; diff --git a/crates/bevy_pbr/src/render/pbr_bindings.wgsl b/crates/bevy_pbr/src/render/pbr_bindings.wgsl index 4c9205461ce115..f4e4d34f3b4c21 100644 --- a/crates/bevy_pbr/src/render/pbr_bindings.wgsl +++ b/crates/bevy_pbr/src/render/pbr_bindings.wgsl @@ -2,25 +2,25 @@ #import bevy_pbr::pbr_types -[[group(1), binding(0)]] +@group(1) @binding(0) var material: StandardMaterial; -[[group(1), binding(1)]] +@group(1) @binding(1) var base_color_texture: texture_2d; -[[group(1), binding(2)]] +@group(1) @binding(2) var base_color_sampler: sampler; -[[group(1), binding(3)]] +@group(1) @binding(3) var emissive_texture: texture_2d; -[[group(1), binding(4)]] +@group(1) @binding(4) var emissive_sampler: sampler; -[[group(1), binding(5)]] +@group(1) @binding(5) var metallic_roughness_texture: texture_2d; -[[group(1), binding(6)]] +@group(1) @binding(6) var metallic_roughness_sampler: sampler; -[[group(1), binding(7)]] +@group(1) @binding(7) var occlusion_texture: texture_2d; -[[group(1), binding(8)]] +@group(1) @binding(8) var occlusion_sampler: sampler; -[[group(1), binding(9)]] +@group(1) @binding(9) var normal_map_texture: texture_2d; -[[group(1), binding(10)]] +@group(1) @binding(10) var normal_map_sampler: sampler; diff --git a/crates/bevy_pbr/src/render/pbr_functions.wgsl b/crates/bevy_pbr/src/render/pbr_functions.wgsl index 35752353daae01..cd922bf74412ca 100644 --- a/crates/bevy_pbr/src/render/pbr_functions.wgsl +++ b/crates/bevy_pbr/src/render/pbr_functions.wgsl @@ -87,19 +87,19 @@ fn calculate_view( } struct PbrInput { - material: StandardMaterial; - occlusion: f32; - frag_coord: vec4; - world_position: vec4; + material: StandardMaterial, + occlusion: f32, + frag_coord: vec4, + world_position: vec4, // Normalized world normal used for shadow mapping as normal-mapping is not used for shadow // mapping - world_normal: vec3; + world_normal: vec3, // Normalized normal-mapped world normal used for lighting - N: vec3; + N: vec3, // Normalized view vector in world space, pointing from the fragment world position toward the // view world position - V: vec3; - is_orthographic: bool; + V: vec3, + is_orthographic: bool, }; // Creates a PbrInput with default values diff --git a/crates/bevy_pbr/src/render/pbr_lighting.wgsl b/crates/bevy_pbr/src/render/pbr_lighting.wgsl index c373177c415497..ff9451ef302b4a 100644 --- a/crates/bevy_pbr/src/render/pbr_lighting.wgsl +++ b/crates/bevy_pbr/src/render/pbr_lighting.wgsl @@ -244,7 +244,7 @@ fn spot_light( R: vec3, F0: vec3, diffuseColor: vec3 ) -> vec3 { // reuse the point light calculations - let point = point_light(world_position, light, roughness, NdotV, N, V, R, F0, diffuseColor); + let point_light = point_light(world_position, light, roughness, NdotV, N, V, R, F0, diffuseColor); // reconstruct spot dir from x/z and y-direction flag var spot_dir = vec3(light.light_custom_data.x, 0.0, light.light_custom_data.y); @@ -261,7 +261,7 @@ fn spot_light( let attenuation = saturate(cd * light.light_custom_data.z + light.light_custom_data.w); let spot_attenuation = attenuation * attenuation; - return point * spot_attenuation; + return point_light * spot_attenuation; } fn directional_light(light: DirectionalLight, roughness: f32, NdotV: f32, normal: vec3, view: vec3, R: vec3, F0: vec3, diffuseColor: vec3) -> vec3 { diff --git a/crates/bevy_pbr/src/render/pbr_types.wgsl b/crates/bevy_pbr/src/render/pbr_types.wgsl index 77a511510b6a1f..879cc64e2d7446 100644 --- a/crates/bevy_pbr/src/render/pbr_types.wgsl +++ b/crates/bevy_pbr/src/render/pbr_types.wgsl @@ -1,14 +1,14 @@ #define_import_path bevy_pbr::pbr_types struct StandardMaterial { - base_color: vec4; - emissive: vec4; - perceptual_roughness: f32; - metallic: f32; - reflectance: f32; + base_color: vec4, + emissive: vec4, + perceptual_roughness: f32, + metallic: f32, + reflectance: f32, // 'flags' is a bit field indicating various options. u32 is 32 bits so we have up to 32 options. - flags: u32; - alpha_cutoff: f32; + flags: u32, + alpha_cutoff: f32, }; let STANDARD_MATERIAL_FLAGS_BASE_COLOR_TEXTURE_BIT: u32 = 1u; diff --git a/crates/bevy_pbr/src/render/skinning.wgsl b/crates/bevy_pbr/src/render/skinning.wgsl index 3596ba4c762315..9fe80c904163dc 100644 --- a/crates/bevy_pbr/src/render/skinning.wgsl +++ b/crates/bevy_pbr/src/render/skinning.wgsl @@ -4,28 +4,14 @@ #define_import_path bevy_pbr::skinning -/// HACK: This works around naga not supporting matrix addition in SPIR-V -// translations. See https://github.com/gfx-rs/naga/issues/1527 -fn add_matrix( - a: mat4x4, - b: mat4x4, -) -> mat4x4 { - return mat4x4( - a[0] + b[0], - a[1] + b[1], - a[2] + b[2], - a[3] + b[3], - ); -} - fn skin_model( indexes: vec4, weights: vec4, ) -> mat4x4 { - var matrix = weights.x * joint_matrices.data[indexes.x]; - matrix = add_matrix(matrix, weights.y * joint_matrices.data[indexes.y]); - matrix = add_matrix(matrix, weights.z * joint_matrices.data[indexes.z]); - return add_matrix(matrix, weights.w * joint_matrices.data[indexes.w]); + return weights.x * joint_matrices.data[indexes.x] + + weights.y * joint_matrices.data[indexes.y] + + weights.z * joint_matrices.data[indexes.z] + + weights.w * joint_matrices.data[indexes.w]; } fn inverse_transpose_3x3(in: mat3x3) -> mat3x3 { diff --git a/crates/bevy_pbr/src/render/wireframe.wgsl b/crates/bevy_pbr/src/render/wireframe.wgsl index b3dc23c86fa277..651a10d491b063 100644 --- a/crates/bevy_pbr/src/render/wireframe.wgsl +++ b/crates/bevy_pbr/src/render/wireframe.wgsl @@ -1,11 +1,11 @@ #import bevy_pbr::mesh_types #import bevy_pbr::mesh_view_bindings -[[group(1), binding(0)]] +@group(1) @binding(0) var mesh: Mesh; #ifdef SKINNED -[[group(1), binding(1)]] +@group(1) @binding(1) var joint_matrices: SkinnedMesh; #import bevy_pbr::skinning #endif @@ -14,18 +14,18 @@ var joint_matrices: SkinnedMesh; #import bevy_pbr::mesh_functions struct Vertex { - [[location(0)]] position: vec3; + @location(0) position: vec3, #ifdef SKINNED - [[location(4)]] joint_indexes: vec4; - [[location(5)]] joint_weights: vec4; + @location(4) joint_indexes: vec4, + @location(5) joint_weights: vec4, #endif }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; + @builtin(position) clip_position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { #ifdef SKINNED let model = skin_model(vertex.joint_indexes, vertex.joint_weights); @@ -38,7 +38,7 @@ fn vertex(vertex: Vertex) -> VertexOutput { return out; } -[[stage(fragment)]] -fn fragment() -> [[location(0)]] vec4 { +@fragment +fn fragment() -> @location(0) vec4 { return vec4(1.0, 1.0, 1.0, 1.0); } diff --git a/crates/bevy_reflect/Cargo.toml b/crates/bevy_reflect/Cargo.toml index bc2af4179d9450..e632d4a25db3bb 100644 --- a/crates/bevy_reflect/Cargo.toml +++ b/crates/bevy_reflect/Cargo.toml @@ -20,7 +20,7 @@ bevy_utils = { path = "../bevy_utils", version = "0.8.0-dev" } # other erased-serde = "0.3" downcast-rs = "1.2" -parking_lot = "0.11.0" +parking_lot = "0.12.1" thiserror = "1.0" once_cell = "1.11" serde = "1" diff --git a/crates/bevy_render/Cargo.toml b/crates/bevy_render/Cargo.toml index 000c845aca0c2d..a773d01affac59 100644 --- a/crates/bevy_render/Cargo.toml +++ b/crates/bevy_render/Cargo.toml @@ -48,9 +48,9 @@ bevy_utils = { path = "../bevy_utils", version = "0.8.0-dev" } image = { version = "0.24", default-features = false } # misc -wgpu = { version = "0.12.0", features = ["spirv"] } +wgpu = { version = "0.13.1", features = ["spirv"] } codespan-reporting = "0.11.0" -naga = { version = "0.8.0", features = ["glsl-in", "spv-in", "spv-out", "wgsl-in", "wgsl-out"] } +naga = { version = "0.9.0", features = ["glsl-in", "spv-in", "spv-out", "wgsl-in", "wgsl-out"] } serde = { version = "1", features = ["derive"] } bitflags = "1.2.1" smallvec = { version = "1.6", features = ["union", "const_generics"] } @@ -62,7 +62,7 @@ futures-lite = "1.4.0" anyhow = "1.0" hex = "0.4.2" hexasphere = "7.2" -parking_lot = "0.11.0" +parking_lot = "0.12.1" regex = "1.5" copyless = "0.1.5" ddsfile = { version = "0.5.0", optional = true } diff --git a/crates/bevy_render/src/camera/camera_driver_node.rs b/crates/bevy_render/src/camera/camera_driver_node.rs index fab057dc2eb7bd..9cf23c106f3f34 100644 --- a/crates/bevy_render/src/camera/camera_driver_node.rs +++ b/crates/bevy_render/src/camera/camera_driver_node.rs @@ -88,14 +88,14 @@ impl Node for CameraDriverNode { let _span = bevy_utils::tracing::info_span!("no_camera_clear_pass").entered(); let pass_descriptor = RenderPassDescriptor { label: Some("no_camera_clear_pass"), - color_attachments: &[RenderPassColorAttachment { + color_attachments: &[Some(RenderPassColorAttachment { view: swap_chain_texture, resolve_target: None, ops: Operations { load: LoadOp::Clear(wgpu::Color::BLACK), store: true, }, - }], + })], depth_stencil_attachment: None, }; diff --git a/crates/bevy_render/src/render_resource/bind_group.rs b/crates/bevy_render/src/render_resource/bind_group.rs index 4356a085d71a13..8362a5c899ad57 100644 --- a/crates/bevy_render/src/render_resource/bind_group.rs +++ b/crates/bevy_render/src/render_resource/bind_group.rs @@ -91,14 +91,14 @@ impl Deref for BindGroup { /// /// ```wgsl /// struct CoolMaterial { -/// color: vec4; +/// color: vec4, /// }; /// -/// [[group(1), binding(0)]] +/// @group(1) @binding(0) /// var material: CoolMaterial; -/// [[group(1), binding(1)]] +/// @group(1) @binding(1) /// var color_texture: texture_2d; -/// [[group(1), binding(2)]] +/// @group(1) @binding(2) /// var color_sampler: sampler; /// ``` /// Note that the "group" index is determined by the usage context. It is not defined in [`AsBindGroup`]. For example, in Bevy material bind groups @@ -165,11 +165,11 @@ impl Deref for BindGroup { /// In WGSL shaders, the binding would look like this: /// ```wgsl /// struct CoolMaterial { -/// color: vec4; -/// roughness: f32; +/// color: vec4, +/// roughness: f32, /// }; /// -/// [[group(1), binding(0)]] +/// @group(1) @binding(0) /// var material: CoolMaterial; /// ``` /// diff --git a/crates/bevy_render/src/render_resource/pipeline.rs b/crates/bevy_render/src/render_resource/pipeline.rs index c5bee4a56b5be9..7cf6ded22c2d46 100644 --- a/crates/bevy_render/src/render_resource/pipeline.rs +++ b/crates/bevy_render/src/render_resource/pipeline.rs @@ -166,7 +166,7 @@ pub struct FragmentState { /// function with this name in the shader. pub entry_point: Cow<'static, str>, /// The color state of the render targets. - pub targets: Vec, + pub targets: Vec>, } /// Describes a compute pipeline. diff --git a/crates/bevy_render/src/render_resource/pipeline_cache.rs b/crates/bevy_render/src/render_resource/pipeline_cache.rs index e93d2f3008af53..341e3b21646ef9 100644 --- a/crates/bevy_render/src/render_resource/pipeline_cache.rs +++ b/crates/bevy_render/src/render_resource/pipeline_cache.rs @@ -155,7 +155,7 @@ impl ShaderCache { render_device .wgpu_device() .push_error_scope(wgpu::ErrorFilter::Validation); - let shader_module = render_device.create_shader_module(&module_descriptor); + let shader_module = render_device.create_shader_module(module_descriptor); let error = render_device.wgpu_device().pop_error_scope(); // `now_or_never` will return Some if the future is ready and None otherwise. @@ -410,7 +410,7 @@ impl PipelineCache { Some(( fragment_module, fragment.entry_point.deref(), - &fragment.targets, + fragment.targets.as_slice(), )) } else { None diff --git a/crates/bevy_render/src/render_resource/shader.rs b/crates/bevy_render/src/render_resource/shader.rs index c952e600779fae..5913d44dd3aab1 100644 --- a/crates/bevy_render/src/render_resource/shader.rs +++ b/crates/bevy_render/src/render_resource/shader.rs @@ -557,26 +557,26 @@ mod tests { #[rustfmt::skip] const WGSL: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; #ifdef TEXTURE -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; #endif struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -587,29 +587,29 @@ fn vertex( const WGSL_ELSE: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; #ifdef TEXTURE -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; #else -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d_array; #endif struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -620,28 +620,28 @@ fn vertex( const WGSL_NESTED_IFDEF: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; # ifdef TEXTURE # ifdef ATTRIBUTE -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; # endif # endif struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -652,31 +652,31 @@ fn vertex( const WGSL_NESTED_IFDEF_ELSE: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; # ifdef TEXTURE # ifdef ATTRIBUTE -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; #else -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d_array; # endif # endif struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -690,24 +690,24 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -732,22 +732,22 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -772,24 +772,24 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d_array; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -937,22 +937,22 @@ void bar() { } #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -977,24 +977,24 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d_array; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -1019,22 +1019,22 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -1059,22 +1059,22 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -1099,22 +1099,22 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -1139,24 +1139,24 @@ fn vertex( #[rustfmt::skip] const EXPECTED: &str = r" struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; struct VertexOutput { - [[location(0)]] uv: vec2; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2 + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2 ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; diff --git a/crates/bevy_render/src/renderer/mod.rs b/crates/bevy_render/src/renderer/mod.rs index 115893d76d6731..bc95217dabcaf3 100644 --- a/crates/bevy_render/src/renderer/mod.rs +++ b/crates/bevy_render/src/renderer/mod.rs @@ -226,6 +226,9 @@ pub async fn initialize_renderer( max_compute_workgroups_per_dimension: limits .max_compute_workgroups_per_dimension .min(constrained_limits.max_compute_workgroups_per_dimension), + max_buffer_size: limits + .max_buffer_size + .min(constrained_limits.max_buffer_size), }; } diff --git a/crates/bevy_render/src/renderer/render_device.rs b/crates/bevy_render/src/renderer/render_device.rs index 4430e7001273ff..e15ed803f777f6 100644 --- a/crates/bevy_render/src/renderer/render_device.rs +++ b/crates/bevy_render/src/renderer/render_device.rs @@ -2,9 +2,8 @@ use crate::render_resource::{ BindGroup, BindGroupLayout, Buffer, ComputePipeline, RawRenderPipelineDescriptor, RenderPipeline, Sampler, Texture, }; -use futures_lite::future; use std::sync::Arc; -use wgpu::{util::DeviceExt, BufferBindingType}; +use wgpu::{util::DeviceExt, BufferAsyncError, BufferBindingType}; use super::RenderQueue; @@ -39,7 +38,7 @@ impl RenderDevice { /// Creates a [`ShaderModule`](wgpu::ShaderModule) from either SPIR-V or WGSL source code. #[inline] - pub fn create_shader_module(&self, desc: &wgpu::ShaderModuleDescriptor) -> wgpu::ShaderModule { + pub fn create_shader_module(&self, desc: wgpu::ShaderModuleDescriptor) -> wgpu::ShaderModule { self.device.create_shader_module(desc) } @@ -170,13 +169,13 @@ impl RenderDevice { &self.device } - pub fn map_buffer(&self, buffer: &wgpu::BufferSlice, map_mode: wgpu::MapMode) { - let data = buffer.map_async(map_mode); - self.poll(wgpu::Maintain::Wait); - assert!( - future::block_on(data).is_ok(), - "Failed to map buffer to host." - ); + pub fn map_buffer( + &self, + buffer: &wgpu::BufferSlice, + map_mode: wgpu::MapMode, + callback: impl FnOnce(Result<(), BufferAsyncError>) + Send + 'static, + ) { + buffer.map_async(map_mode, callback); } pub fn align_copy_bytes_per_row(row_bytes: usize) -> usize { diff --git a/crates/bevy_render/src/texture/basis.rs b/crates/bevy_render/src/texture/basis.rs index 256bbf8d79ce3d..45240c229016b3 100644 --- a/crates/bevy_render/src/texture/basis.rs +++ b/crates/bevy_render/src/texture/basis.rs @@ -1,7 +1,7 @@ use basis_universal::{ BasisTextureType, DecodeFlags, TranscodeParameters, Transcoder, TranscoderTextureFormat, }; -use wgpu::{Extent3d, TextureDimension, TextureFormat}; +use wgpu::{AstcBlock, AstcChannel, Extent3d, TextureDimension, TextureFormat}; use super::{CompressedImageFormats, Image, TextureError}; @@ -139,10 +139,13 @@ pub fn get_transcoded_formats( if supported_compressed_formats.contains(CompressedImageFormats::ASTC_LDR) { ( TranscoderTextureFormat::ASTC_4x4_RGBA, - if is_srgb { - TextureFormat::Astc4x4RgbaUnormSrgb - } else { - TextureFormat::Astc4x4RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B4x4, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, }, ) } else if supported_compressed_formats.contains(CompressedImageFormats::BC) { diff --git a/crates/bevy_render/src/texture/image.rs b/crates/bevy_render/src/texture/image.rs index 577fc33780121d..ceab85cb994989 100644 --- a/crates/bevy_render/src/texture/image.rs +++ b/crates/bevy_render/src/texture/image.rs @@ -742,36 +742,7 @@ impl CompressedImageFormats { | TextureFormat::EacR11Snorm | TextureFormat::EacRg11Unorm | TextureFormat::EacRg11Snorm => self.contains(CompressedImageFormats::ETC2), - TextureFormat::Astc4x4RgbaUnorm - | TextureFormat::Astc4x4RgbaUnormSrgb - | TextureFormat::Astc5x4RgbaUnorm - | TextureFormat::Astc5x4RgbaUnormSrgb - | TextureFormat::Astc5x5RgbaUnorm - | TextureFormat::Astc5x5RgbaUnormSrgb - | TextureFormat::Astc6x5RgbaUnorm - | TextureFormat::Astc6x5RgbaUnormSrgb - | TextureFormat::Astc6x6RgbaUnorm - | TextureFormat::Astc6x6RgbaUnormSrgb - | TextureFormat::Astc8x5RgbaUnorm - | TextureFormat::Astc8x5RgbaUnormSrgb - | TextureFormat::Astc8x6RgbaUnorm - | TextureFormat::Astc8x6RgbaUnormSrgb - | TextureFormat::Astc10x5RgbaUnorm - | TextureFormat::Astc10x5RgbaUnormSrgb - | TextureFormat::Astc10x6RgbaUnorm - | TextureFormat::Astc10x6RgbaUnormSrgb - | TextureFormat::Astc8x8RgbaUnorm - | TextureFormat::Astc8x8RgbaUnormSrgb - | TextureFormat::Astc10x8RgbaUnorm - | TextureFormat::Astc10x8RgbaUnormSrgb - | TextureFormat::Astc10x10RgbaUnorm - | TextureFormat::Astc10x10RgbaUnormSrgb - | TextureFormat::Astc12x10RgbaUnorm - | TextureFormat::Astc12x10RgbaUnormSrgb - | TextureFormat::Astc12x12RgbaUnorm - | TextureFormat::Astc12x12RgbaUnormSrgb => { - self.contains(CompressedImageFormats::ASTC_LDR) - } + TextureFormat::Astc { .. } => self.contains(CompressedImageFormats::ASTC_LDR), _ => true, } } diff --git a/crates/bevy_render/src/texture/ktx2.rs b/crates/bevy_render/src/texture/ktx2.rs index 7a4fb62682ba28..b26c6c1bcebe2b 100644 --- a/crates/bevy_render/src/texture/ktx2.rs +++ b/crates/bevy_render/src/texture/ktx2.rs @@ -11,7 +11,7 @@ use ktx2::{ BasicDataFormatDescriptor, ChannelTypeQualifiers, ColorModel, DataFormatDescriptorHeader, Header, SampleInformation, }; -use wgpu::{Extent3d, TextureDimension, TextureFormat}; +use wgpu::{AstcBlock, AstcChannel, Extent3d, TextureDimension, TextureFormat}; use super::{CompressedImageFormats, DataFormat, Image, TextureError, TranscodeFormat}; @@ -239,10 +239,13 @@ pub fn get_transcoded_formats( if supported_compressed_formats.contains(CompressedImageFormats::ASTC_LDR) { ( TranscoderBlockFormat::ASTC_4x4, - if is_srgb { - TextureFormat::Astc4x4RgbaUnormSrgb - } else { - TextureFormat::Astc4x4RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B4x4, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, }, ) } else if supported_compressed_formats.contains(CompressedImageFormats::BC) { @@ -1016,159 +1019,35 @@ pub fn ktx2_dfd_to_texture_format( ))); } }, - Some(ColorModel::ASTC) => match data_format_descriptor.texel_block_dimensions[0] { - 4 => match data_format_descriptor.texel_block_dimensions[1] { - 4 => { - if is_srgb { - TextureFormat::Astc4x4RgbaUnormSrgb - } else { - TextureFormat::Astc4x4RgbaUnorm - } - } - d => { - return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC y-dimension: {}", - d - ))) - } - }, - 5 => match data_format_descriptor.texel_block_dimensions[1] { - 4 => { - if is_srgb { - TextureFormat::Astc5x4RgbaUnormSrgb - } else { - TextureFormat::Astc5x4RgbaUnorm - } - } - 5 => { - if is_srgb { - TextureFormat::Astc5x5RgbaUnormSrgb - } else { - TextureFormat::Astc5x5RgbaUnorm - } - } - d => { - return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC y-dimension: {}", - d - ))) - } - }, - 6 => match data_format_descriptor.texel_block_dimensions[1] { - 5 => { - if is_srgb { - TextureFormat::Astc6x5RgbaUnormSrgb - } else { - TextureFormat::Astc6x5RgbaUnorm - } - } - 6 => { - if is_srgb { - TextureFormat::Astc6x6RgbaUnormSrgb - } else { - TextureFormat::Astc6x6RgbaUnorm - } - } + Some(ColorModel::ASTC) => TextureFormat::Astc { + block: match ( + data_format_descriptor.texel_block_dimensions[0], + data_format_descriptor.texel_block_dimensions[1], + ) { + (4, 4) => AstcBlock::B4x4, + (5, 4) => AstcBlock::B5x4, + (5, 5) => AstcBlock::B5x5, + (6, 5) => AstcBlock::B6x5, + (8, 5) => AstcBlock::B8x5, + (8, 8) => AstcBlock::B8x8, + (10, 5) => AstcBlock::B10x5, + (10, 6) => AstcBlock::B10x6, + (10, 8) => AstcBlock::B10x8, + (10, 10) => AstcBlock::B10x10, + (12, 10) => AstcBlock::B12x10, + (12, 12) => AstcBlock::B12x12, d => { return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC y-dimension: {}", - d + "Invalid ASTC dimension: {} x {}", + d.0, d.1 ))) } }, - 8 => match data_format_descriptor.texel_block_dimensions[1] { - 5 => { - if is_srgb { - TextureFormat::Astc8x5RgbaUnormSrgb - } else { - TextureFormat::Astc8x5RgbaUnorm - } - } - 6 => { - if is_srgb { - TextureFormat::Astc8x6RgbaUnormSrgb - } else { - TextureFormat::Astc8x6RgbaUnorm - } - } - 8 => { - if is_srgb { - TextureFormat::Astc8x8RgbaUnormSrgb - } else { - TextureFormat::Astc8x8RgbaUnorm - } - } - d => { - return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC y-dimension: {}", - d - ))) - } - }, - 10 => match data_format_descriptor.texel_block_dimensions[1] { - 5 => { - if is_srgb { - TextureFormat::Astc10x5RgbaUnormSrgb - } else { - TextureFormat::Astc10x5RgbaUnorm - } - } - 6 => { - if is_srgb { - TextureFormat::Astc10x6RgbaUnormSrgb - } else { - TextureFormat::Astc10x6RgbaUnorm - } - } - 8 => { - if is_srgb { - TextureFormat::Astc10x8RgbaUnormSrgb - } else { - TextureFormat::Astc10x8RgbaUnorm - } - } - 10 => { - if is_srgb { - TextureFormat::Astc10x10RgbaUnormSrgb - } else { - TextureFormat::Astc10x10RgbaUnorm - } - } - d => { - return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC y-dimension: {}", - d - ))) - } - }, - 12 => match data_format_descriptor.texel_block_dimensions[1] { - 10 => { - if is_srgb { - TextureFormat::Astc12x10RgbaUnormSrgb - } else { - TextureFormat::Astc12x10RgbaUnorm - } - } - 12 => { - if is_srgb { - TextureFormat::Astc12x12RgbaUnormSrgb - } else { - TextureFormat::Astc12x12RgbaUnorm - } - } - d => { - return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC y-dimension: {}", - d - ))) - } + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm }, - d => { - return Err(TextureError::UnsupportedTextureFormat(format!( - "Invalid ASTC x-dimension: {}", - d - ))) - } }, Some(ColorModel::ETC1S) => { return Err(TextureError::FormatRequiresTranscodingError( @@ -1362,101 +1241,143 @@ pub fn ktx2_format_to_texture_format( ktx2::Format::EAC_R11G11_UNORM_BLOCK => TextureFormat::EacRg11Unorm, ktx2::Format::EAC_R11G11_SNORM_BLOCK => TextureFormat::EacRg11Snorm, ktx2::Format::ASTC_4x4_UNORM_BLOCK | ktx2::Format::ASTC_4x4_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc4x4RgbaUnormSrgb - } else { - TextureFormat::Astc4x4RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B4x4, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_5x4_UNORM_BLOCK | ktx2::Format::ASTC_5x4_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc5x4RgbaUnormSrgb - } else { - TextureFormat::Astc5x4RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B5x4, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_5x5_UNORM_BLOCK | ktx2::Format::ASTC_5x5_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc5x5RgbaUnormSrgb - } else { - TextureFormat::Astc5x5RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B5x5, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_6x5_UNORM_BLOCK | ktx2::Format::ASTC_6x5_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc6x5RgbaUnormSrgb - } else { - TextureFormat::Astc6x5RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B6x5, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_6x6_UNORM_BLOCK | ktx2::Format::ASTC_6x6_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc6x6RgbaUnormSrgb - } else { - TextureFormat::Astc6x6RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B6x6, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_8x5_UNORM_BLOCK | ktx2::Format::ASTC_8x5_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc8x5RgbaUnormSrgb - } else { - TextureFormat::Astc8x5RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B8x5, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_8x6_UNORM_BLOCK | ktx2::Format::ASTC_8x6_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc8x6RgbaUnormSrgb - } else { - TextureFormat::Astc8x6RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B8x6, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_8x8_UNORM_BLOCK | ktx2::Format::ASTC_8x8_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc8x8RgbaUnormSrgb - } else { - TextureFormat::Astc8x8RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B8x8, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_10x5_UNORM_BLOCK | ktx2::Format::ASTC_10x5_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc10x5RgbaUnormSrgb - } else { - TextureFormat::Astc10x5RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B10x5, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_10x6_UNORM_BLOCK | ktx2::Format::ASTC_10x6_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc10x6RgbaUnormSrgb - } else { - TextureFormat::Astc10x6RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B10x6, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_10x8_UNORM_BLOCK | ktx2::Format::ASTC_10x8_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc10x8RgbaUnormSrgb - } else { - TextureFormat::Astc10x8RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B10x8, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_10x10_UNORM_BLOCK | ktx2::Format::ASTC_10x10_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc10x10RgbaUnormSrgb - } else { - TextureFormat::Astc10x10RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B10x10, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_12x10_UNORM_BLOCK | ktx2::Format::ASTC_12x10_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc12x10RgbaUnormSrgb - } else { - TextureFormat::Astc12x10RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B12x10, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } ktx2::Format::ASTC_12x12_UNORM_BLOCK | ktx2::Format::ASTC_12x12_SRGB_BLOCK => { - if is_srgb { - TextureFormat::Astc12x12RgbaUnormSrgb - } else { - TextureFormat::Astc12x12RgbaUnorm + TextureFormat::Astc { + block: AstcBlock::B12x12, + channel: if is_srgb { + AstcChannel::UnormSrgb + } else { + AstcChannel::Unorm + }, } } _ => { diff --git a/crates/bevy_render/src/view/window.rs b/crates/bevy_render/src/view/window.rs index 0104b85e86bd55..4699cef2020068 100644 --- a/crates/bevy_render/src/view/window.rs +++ b/crates/bevy_render/src/view/window.rs @@ -169,6 +169,8 @@ pub fn prepare_windows( PresentMode::Fifo => wgpu::PresentMode::Fifo, PresentMode::Mailbox => wgpu::PresentMode::Mailbox, PresentMode::Immediate => wgpu::PresentMode::Immediate, + PresentMode::AutoVsync => wgpu::PresentMode::AutoVsync, + PresentMode::AutoNoVsync => wgpu::PresentMode::AutoNoVsync, }, }; diff --git a/crates/bevy_sprite/src/mesh2d/color_material.wgsl b/crates/bevy_sprite/src/mesh2d/color_material.wgsl index 24b55e448177d9..3b6a2b52171761 100644 --- a/crates/bevy_sprite/src/mesh2d/color_material.wgsl +++ b/crates/bevy_sprite/src/mesh2d/color_material.wgsl @@ -2,41 +2,33 @@ #import bevy_sprite::mesh2d_view_bindings struct ColorMaterial { - color: vec4; + color: vec4, // 'flags' is a bit field indicating various options. u32 is 32 bits so we have up to 32 options. - flags: u32; + flags: u32, }; let COLOR_MATERIAL_FLAGS_TEXTURE_BIT: u32 = 1u; -[[group(1), binding(0)]] +@group(1) @binding(0) var material: ColorMaterial; -[[group(1), binding(1)]] +@group(1) @binding(1) var texture: texture_2d; -[[group(1), binding(2)]] +@group(1) @binding(2) var texture_sampler: sampler; -[[group(2), binding(0)]] +@group(2) @binding(0) var mesh: Mesh2d; struct FragmentInput { - [[builtin(front_facing)]] is_front: bool; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; - [[location(2)]] uv: vec2; -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif -#ifdef VERTEX_COLORS - [[location(4)]] colors: vec4; -#endif + @builtin(front_facing) is_front: bool, + #import bevy_sprite::mesh2d_vertex_output }; -[[stage(fragment)]] -fn fragment(in: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: FragmentInput) -> @location(0) vec4 { var output_color: vec4 = material.color; if ((material.flags & COLOR_MATERIAL_FLAGS_TEXTURE_BIT) != 0u) { #ifdef VERTEX_COLORS - output_color = output_color * textureSample(texture, texture_sampler, in.uv) * in.colors; + output_color = output_color * textureSample(texture, texture_sampler, in.uv) * in.color; #else output_color = output_color * textureSample(texture, texture_sampler, in.uv); #endif diff --git a/crates/bevy_sprite/src/mesh2d/mesh.rs b/crates/bevy_sprite/src/mesh2d/mesh.rs index 9196026932cc08..0c858e17eaca11 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh.rs +++ b/crates/bevy_sprite/src/mesh2d/mesh.rs @@ -37,6 +37,8 @@ impl From> for Mesh2dHandle { #[derive(Default)] pub struct Mesh2dRenderPlugin; +pub const MESH2D_VERTEX_OUTPUT: HandleUntyped = + HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 7646632476603252194); pub const MESH2D_VIEW_TYPES_HANDLE: HandleUntyped = HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 12677582416765805110); pub const MESH2D_VIEW_BINDINGS_HANDLE: HandleUntyped = @@ -52,6 +54,12 @@ pub const MESH2D_SHADER_HANDLE: HandleUntyped = impl Plugin for Mesh2dRenderPlugin { fn build(&self, app: &mut bevy_app::App) { + load_internal_asset!( + app, + MESH2D_VERTEX_OUTPUT, + "mesh2d_vertex_output.wgsl", + Shader::from_wgsl + ); load_internal_asset!( app, MESH2D_VIEW_TYPES_HANDLE, @@ -344,11 +352,11 @@ impl SpecializedMeshPipeline for Mesh2dPipeline { shader: MESH2D_SHADER_HANDLE.typed::(), shader_defs, entry_point: "fragment".into(), - targets: vec![ColorTargetState { + targets: vec![Some(ColorTargetState { format: TextureFormat::bevy_default(), blend: Some(BlendState::ALPHA_BLENDING), write_mask: ColorWrites::ALL, - }], + })], }), layout: Some(vec![self.view_layout.clone(), self.mesh_layout.clone()]), primitive: PrimitiveState { diff --git a/crates/bevy_sprite/src/mesh2d/mesh2d.wgsl b/crates/bevy_sprite/src/mesh2d/mesh2d.wgsl index c1d29aef334192..00c3a9ab74416c 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh2d.wgsl +++ b/crates/bevy_sprite/src/mesh2d/mesh2d.wgsl @@ -5,31 +5,23 @@ #import bevy_sprite::mesh2d_functions struct Vertex { - [[location(0)]] position: vec3; - [[location(1)]] normal: vec3; - [[location(2)]] uv: vec2; + @location(0) position: vec3, + @location(1) normal: vec3, + @location(2) uv: vec2, #ifdef VERTEX_TANGENTS - [[location(3)]] tangent: vec4; + @location(3) tangent: vec4, #endif #ifdef VERTEX_COLORS - [[location(4)]] colors: vec4; + @location(4) color: vec4, #endif }; struct VertexOutput { - [[builtin(position)]] clip_position: vec4; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; - [[location(2)]] uv: vec2; -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif -#ifdef VERTEX_COLORS - [[location(4)]] colors: vec4; -#endif -}; + @builtin(position) clip_position: vec4, + #import bevy_sprite::mesh2d_vertex_output +} -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { var out: VertexOutput; out.uv = vertex.uv; @@ -40,22 +32,17 @@ fn vertex(vertex: Vertex) -> VertexOutput { out.world_tangent = mesh2d_tangent_local_to_world(vertex.tangent); #endif #ifdef VERTEX_COLORS - out.colors = vertex.colors; + out.color = vertex.color; #endif return out; } struct FragmentInput { - [[builtin(front_facing)]] is_front: bool; - [[location(0)]] world_position: vec4; - [[location(1)]] world_normal: vec3; - [[location(2)]] uv: vec2; -#ifdef VERTEX_TANGENTS - [[location(3)]] world_tangent: vec4; -#endif + @builtin(front_facing) is_front: bool, + #import bevy_sprite::mesh2d_vertex_output }; -[[stage(fragment)]] -fn fragment(in: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: FragmentInput) -> @location(0) vec4 { return vec4(1.0, 0.0, 1.0, 1.0); } diff --git a/crates/bevy_sprite/src/mesh2d/mesh2d_bindings.wgsl b/crates/bevy_sprite/src/mesh2d/mesh2d_bindings.wgsl index 1bb2ec5959b8ee..f26a0442c95db5 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh2d_bindings.wgsl +++ b/crates/bevy_sprite/src/mesh2d/mesh2d_bindings.wgsl @@ -2,5 +2,5 @@ #import bevy_sprite::mesh2d_types -[[group(2), binding(0)]] +@group(2) @binding(0) var mesh: Mesh2d; diff --git a/crates/bevy_sprite/src/mesh2d/mesh2d_types.wgsl b/crates/bevy_sprite/src/mesh2d/mesh2d_types.wgsl index 83f83366eb25ca..1de0218112a47a 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh2d_types.wgsl +++ b/crates/bevy_sprite/src/mesh2d/mesh2d_types.wgsl @@ -1,8 +1,8 @@ #define_import_path bevy_sprite::mesh2d_types struct Mesh2d { - model: mat4x4; - inverse_transpose_model: mat4x4; + model: mat4x4, + inverse_transpose_model: mat4x4, // 'flags' is a bit field indicating various options. u32 is 32 bits so we have up to 32 options. - flags: u32; + flags: u32, }; diff --git a/crates/bevy_sprite/src/mesh2d/mesh2d_vertex_output.wgsl b/crates/bevy_sprite/src/mesh2d/mesh2d_vertex_output.wgsl new file mode 100644 index 00000000000000..cd0c2e8f42f446 --- /dev/null +++ b/crates/bevy_sprite/src/mesh2d/mesh2d_vertex_output.wgsl @@ -0,0 +1,11 @@ +#define_import_path bevy_sprite::mesh2d_vertex_output + +@location(0) world_position: vec4, +@location(1) world_normal: vec3, +@location(2) uv: vec2, +#ifdef VERTEX_TANGENTS +@location(3) world_tangent: vec4, +#endif +#ifdef VERTEX_COLORS +@location(4) color: vec4, +#endif diff --git a/crates/bevy_sprite/src/mesh2d/mesh2d_view_bindings.wgsl b/crates/bevy_sprite/src/mesh2d/mesh2d_view_bindings.wgsl index fedf12bff00722..cc138d28facb7b 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh2d_view_bindings.wgsl +++ b/crates/bevy_sprite/src/mesh2d/mesh2d_view_bindings.wgsl @@ -2,5 +2,5 @@ #import bevy_sprite::mesh2d_view_types -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; diff --git a/crates/bevy_sprite/src/mesh2d/mesh2d_view_types.wgsl b/crates/bevy_sprite/src/mesh2d/mesh2d_view_types.wgsl index 2e54a898e56211..8f90ebdfdc1e1d 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh2d_view_types.wgsl +++ b/crates/bevy_sprite/src/mesh2d/mesh2d_view_types.wgsl @@ -1,11 +1,11 @@ #define_import_path bevy_sprite::mesh2d_view_types struct View { - view_proj: mat4x4; - view: mat4x4; - inverse_view: mat4x4; - projection: mat4x4; - world_position: vec3; - width: f32; - height: f32; + view_proj: mat4x4, + view: mat4x4, + inverse_view: mat4x4, + projection: mat4x4, + world_position: vec3, + width: f32, + height: f32, }; diff --git a/crates/bevy_sprite/src/render/mod.rs b/crates/bevy_sprite/src/render/mod.rs index ead7f71895f8fb..6d739dd0295609 100644 --- a/crates/bevy_sprite/src/render/mod.rs +++ b/crates/bevy_sprite/src/render/mod.rs @@ -143,11 +143,11 @@ impl SpecializedRenderPipeline for SpritePipeline { shader: SPRITE_SHADER_HANDLE.typed::(), shader_defs, entry_point: "fragment".into(), - targets: vec![ColorTargetState { + targets: vec![Some(ColorTargetState { format: TextureFormat::bevy_default(), blend: Some(BlendState::ALPHA_BLENDING), write_mask: ColorWrites::ALL, - }], + })], }), layout: Some(vec![self.view_layout.clone(), self.material_layout.clone()]), primitive: PrimitiveState { diff --git a/crates/bevy_sprite/src/render/sprite.wgsl b/crates/bevy_sprite/src/render/sprite.wgsl index e04559dd004129..78494b5a72ee54 100644 --- a/crates/bevy_sprite/src/render/sprite.wgsl +++ b/crates/bevy_sprite/src/render/sprite.wgsl @@ -1,24 +1,24 @@ struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; + @location(0) uv: vec2, #ifdef COLORED - [[location(1)]] color: vec4; + @location(1) color: vec4, #endif - [[builtin(position)]] position: vec4; + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2, + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2, #ifdef COLORED - [[location(2)]] vertex_color: vec4, + @location(2) vertex_color: vec4, #endif ) -> VertexOutput { var out: VertexOutput; @@ -30,13 +30,13 @@ fn vertex( return out; } -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; -[[group(1), binding(1)]] +@group(1) @binding(1) var sprite_sampler: sampler; -[[stage(fragment)]] -fn fragment(in: VertexOutput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: VertexOutput) -> @location(0) vec4 { var color = textureSample(sprite_texture, sprite_sampler, in.uv); #ifdef COLORED color = in.color * color; diff --git a/crates/bevy_ui/src/render/pipeline.rs b/crates/bevy_ui/src/render/pipeline.rs index 3fcfcda7338f02..4404429afffe78 100644 --- a/crates/bevy_ui/src/render/pipeline.rs +++ b/crates/bevy_ui/src/render/pipeline.rs @@ -86,11 +86,11 @@ impl SpecializedRenderPipeline for UiPipeline { shader: super::UI_SHADER_HANDLE.typed::(), shader_defs, entry_point: "fragment".into(), - targets: vec![ColorTargetState { + targets: vec![Some(ColorTargetState { format: TextureFormat::bevy_default(), blend: Some(BlendState::ALPHA_BLENDING), write_mask: ColorWrites::ALL, - }], + })], }), layout: Some(vec![self.view_layout.clone(), self.image_layout.clone()]), primitive: PrimitiveState { diff --git a/crates/bevy_ui/src/render/render_pass.rs b/crates/bevy_ui/src/render/render_pass.rs index 2a129694f71a31..66445c92a17c3b 100644 --- a/crates/bevy_ui/src/render/render_pass.rs +++ b/crates/bevy_ui/src/render/render_pass.rs @@ -81,14 +81,14 @@ impl Node for UiPassNode { }; let pass_descriptor = RenderPassDescriptor { label: Some("ui_pass"), - color_attachments: &[RenderPassColorAttachment { + color_attachments: &[Some(RenderPassColorAttachment { view: &target.view, resolve_target: None, ops: Operations { load: LoadOp::Load, store: true, }, - }], + })], depth_stencil_attachment: None, }; diff --git a/crates/bevy_ui/src/render/ui.wgsl b/crates/bevy_ui/src/render/ui.wgsl index 4a7c4bff3e379c..70edf23f24c147 100644 --- a/crates/bevy_ui/src/render/ui.wgsl +++ b/crates/bevy_ui/src/render/ui.wgsl @@ -1,21 +1,21 @@ struct View { - view_proj: mat4x4; - world_position: vec3; + view_proj: mat4x4, + world_position: vec3, }; -[[group(0), binding(0)]] +@group(0) @binding(0) var view: View; struct VertexOutput { - [[location(0)]] uv: vec2; - [[location(1)]] color: vec4; - [[builtin(position)]] position: vec4; + @location(0) uv: vec2, + @location(1) color: vec4, + @builtin(position) position: vec4, }; -[[stage(vertex)]] +@vertex fn vertex( - [[location(0)]] vertex_position: vec3, - [[location(1)]] vertex_uv: vec2, - [[location(2)]] vertex_color: vec4, + @location(0) vertex_position: vec3, + @location(1) vertex_uv: vec2, + @location(2) vertex_color: vec4, ) -> VertexOutput { var out: VertexOutput; out.uv = vertex_uv; @@ -24,13 +24,13 @@ fn vertex( return out; } -[[group(1), binding(0)]] +@group(1) @binding(0) var sprite_texture: texture_2d; -[[group(1), binding(1)]] +@group(1) @binding(1) var sprite_sampler: sampler; -[[stage(fragment)]] -fn fragment(in: VertexOutput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: VertexOutput) -> @location(0) vec4 { var color = textureSample(sprite_texture, sprite_sampler, in.uv); color = in.color * color; return color; diff --git a/crates/bevy_window/src/window.rs b/crates/bevy_window/src/window.rs index c67949e4033dfe..561feb7598c37a 100644 --- a/crates/bevy_window/src/window.rs +++ b/crates/bevy_window/src/window.rs @@ -23,20 +23,34 @@ pub struct WindowId(Uuid); #[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)] #[doc(alias = "vsync")] pub enum PresentMode { + /// Chooses FifoRelaxed -> Fifo based on availability. + /// + /// Because of the fallback behavior, it is supported everywhere. + AutoVsync = 0, + /// Chooses Immediate -> Mailbox -> Fifo (on web) based on availability. + /// + /// Because of the fallback behavior, it is supported everywhere. + AutoNoVsync = 1, /// The presentation engine does **not** wait for a vertical blanking period and /// the request is presented immediately. This is a low-latency presentation mode, /// but visible tearing may be observed. Will fallback to `Fifo` if unavailable on the /// selected platform and backend. Not optimal for mobile. - Immediate = 0, + /// + /// Selecting this variant will panic if not supported, it is preferred to use + /// [`PresentMode::AutoNoVsync`]. + Immediate = 2, /// The presentation engine waits for the next vertical blanking period to update /// the current image, but frames may be submitted without delay. This is a low-latency /// presentation mode and visible tearing will **not** be observed. Will fallback to `Fifo` /// if unavailable on the selected platform and backend. Not optimal for mobile. - Mailbox = 1, + /// + /// Selecting this variant will panic if not supported, it is preferred to use + /// [`PresentMode::AutoNoVsync`]. + Mailbox = 3, /// The presentation engine waits for the next vertical blanking period to update /// the current image. The framerate will be capped at the display refresh rate, /// corresponding to the `VSync`. Tearing cannot be observed. Optimal for mobile. - Fifo = 2, // NOTE: The explicit ordinal values mirror wgpu and the vulkan spec. + Fifo = 4, // NOTE: The explicit ordinal values mirror wgpu. } impl WindowId { diff --git a/deny.toml b/deny.toml index bbabea5a99605d..685fd68f93ab5c 100644 --- a/deny.toml +++ b/deny.toml @@ -40,6 +40,8 @@ skip = [ { name = "ndk", version = "0.5" }, # from winit v0.26.1 { name = "ndk-glue", version = "0.5" }, # from winit v0.26.1 { name = "ndk-sys", version = "0.2" }, # from winit v0.26.1 + { name = "parking_lot", version = "0.11" }, # from rodio v0.15.0 + { name = "parking_lot_core", version = "0.8" }, # from rodio v0.15.0 { name = "stdweb", version = "0.1" }, # from rodio v0.15.0 { name = "nix", version = "0.23.1" }, # from alsa v0.6.0 ] diff --git a/examples/2d/mesh2d_manual.rs b/examples/2d/mesh2d_manual.rs index 187317797eeb4a..e1828d92e6fb08 100644 --- a/examples/2d/mesh2d_manual.rs +++ b/examples/2d/mesh2d_manual.rs @@ -160,11 +160,11 @@ impl SpecializedRenderPipeline for ColoredMesh2dPipeline { shader: COLORED_MESH2D_SHADER_HANDLE.typed::(), shader_defs: Vec::new(), entry_point: "fragment".into(), - targets: vec![ColorTargetState { + targets: vec![Some(ColorTargetState { format: TextureFormat::bevy_default(), blend: Some(BlendState::ALPHA_BLENDING), write_mask: ColorWrites::ALL, - }], + })], }), // Use the two standard uniforms for 2d meshes layout: Some(vec![ @@ -212,7 +212,7 @@ const COLORED_MESH2D_SHADER: &str = r" #import bevy_sprite::mesh2d_types #import bevy_sprite::mesh2d_view_bindings -[[group(1), binding(0)]] +@group(1) @binding(0) var mesh: Mesh2d; // NOTE: Bindings must come before functions that use them! @@ -220,19 +220,19 @@ var mesh: Mesh2d; // The structure of the vertex buffer is as specified in `specialize()` struct Vertex { - [[location(0)]] position: vec3; - [[location(1)]] color: u32; + @location(0) position: vec3, + @location(1) color: u32, }; struct VertexOutput { // The vertex shader must set the on-screen position of the vertex - [[builtin(position)]] clip_position: vec4; + @builtin(position) clip_position: vec4, // We pass the vertex color to the fragment shader in location 0 - [[location(0)]] color: vec4; + @location(0) color: vec4, }; /// Entry point for the vertex shader -[[stage(vertex)]] +@vertex fn vertex(vertex: Vertex) -> VertexOutput { var out: VertexOutput; // Project the world position of the mesh into screen position @@ -245,12 +245,12 @@ fn vertex(vertex: Vertex) -> VertexOutput { // The input of the fragment shader must correspond to the output of the vertex shader for all `location`s struct FragmentInput { // The color is interpolated between vertices by default - [[location(0)]] color: vec4; + @location(0) color: vec4, }; /// Entry point for the fragment shader -[[stage(fragment)]] -fn fragment(in: FragmentInput) -> [[location(0)]] vec4 { +@fragment +fn fragment(in: FragmentInput) -> @location(0) vec4 { return in.color; } "; diff --git a/examples/shader/compute_shader_game_of_life.rs b/examples/shader/compute_shader_game_of_life.rs index 450e6e80b2a4a6..9b4df3e0a1c545 100644 --- a/examples/shader/compute_shader_game_of_life.rs +++ b/examples/shader/compute_shader_game_of_life.rs @@ -25,7 +25,7 @@ fn main() { .insert_resource(ClearColor(Color::BLACK)) .insert_resource(WindowDescriptor { // uncomment for unthrottled FPS - // present_mode: bevy::window::PresentMode::Immediate, + // present_mode: bevy::window::PresentMode::AutoNoVsync, ..default() }) .add_plugins(DefaultPlugins) @@ -227,14 +227,14 @@ impl render_graph::Node for GameOfLifeNode { .get_compute_pipeline(pipeline.init_pipeline) .unwrap(); pass.set_pipeline(init_pipeline); - pass.dispatch(SIZE.0 / WORKGROUP_SIZE, SIZE.1 / WORKGROUP_SIZE, 1); + pass.dispatch_workgroups(SIZE.0 / WORKGROUP_SIZE, SIZE.1 / WORKGROUP_SIZE, 1); } GameOfLifeState::Update => { let update_pipeline = pipeline_cache .get_compute_pipeline(pipeline.update_pipeline) .unwrap(); pass.set_pipeline(update_pipeline); - pass.dispatch(SIZE.0 / WORKGROUP_SIZE, SIZE.1 / WORKGROUP_SIZE, 1); + pass.dispatch_workgroups(SIZE.0 / WORKGROUP_SIZE, SIZE.1 / WORKGROUP_SIZE, 1); } } diff --git a/examples/stress_tests/bevymark.rs b/examples/stress_tests/bevymark.rs index 6fb85a5d8e76d4..bf3955428b1d9e 100644 --- a/examples/stress_tests/bevymark.rs +++ b/examples/stress_tests/bevymark.rs @@ -32,7 +32,7 @@ fn main() { title: "BevyMark".to_string(), width: 800., height: 600., - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, resizable: true, ..default() }) diff --git a/examples/stress_tests/many_cubes.rs b/examples/stress_tests/many_cubes.rs index f1f6d9ba40e1a4..f4b119b61d4f6d 100644 --- a/examples/stress_tests/many_cubes.rs +++ b/examples/stress_tests/many_cubes.rs @@ -20,7 +20,7 @@ use bevy::{ fn main() { App::new() .insert_resource(WindowDescriptor { - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, ..default() }) .add_plugins(DefaultPlugins) diff --git a/examples/stress_tests/many_foxes.rs b/examples/stress_tests/many_foxes.rs index 4251354a5fc9bb..a279c59582f388 100644 --- a/examples/stress_tests/many_foxes.rs +++ b/examples/stress_tests/many_foxes.rs @@ -17,7 +17,7 @@ fn main() { App::new() .insert_resource(WindowDescriptor { title: "🦊🦊🦊 Many Foxes! 🦊🦊🦊".to_string(), - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, ..default() }) .add_plugins(DefaultPlugins) diff --git a/examples/stress_tests/many_lights.rs b/examples/stress_tests/many_lights.rs index 0236980d56d4c5..c65776bd9a367a 100644 --- a/examples/stress_tests/many_lights.rs +++ b/examples/stress_tests/many_lights.rs @@ -17,7 +17,7 @@ fn main() { width: 1024.0, height: 768.0, title: "many_lights".to_string(), - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, ..default() }) .add_plugins(DefaultPlugins) diff --git a/examples/stress_tests/many_sprites.rs b/examples/stress_tests/many_sprites.rs index 25858eeb7f4426..324c5fae1d5f62 100644 --- a/examples/stress_tests/many_sprites.rs +++ b/examples/stress_tests/many_sprites.rs @@ -26,7 +26,7 @@ struct ColorTint(bool); fn main() { App::new() .insert_resource(WindowDescriptor { - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, ..default() }) .insert_resource(ColorTint( diff --git a/examples/ui/text_debug.rs b/examples/ui/text_debug.rs index af64d403954805..685eb4c639e784 100644 --- a/examples/ui/text_debug.rs +++ b/examples/ui/text_debug.rs @@ -9,7 +9,7 @@ use bevy::{ fn main() { App::new() .insert_resource(WindowDescriptor { - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, ..default() }) .add_plugins(DefaultPlugins) diff --git a/examples/window/low_power.rs b/examples/window/low_power.rs index 18161684d6f78c..5667e310b39df5 100644 --- a/examples/window/low_power.rs +++ b/examples/window/low_power.rs @@ -27,7 +27,7 @@ fn main() { }) // Turn off vsync to maximize CPU/GPU usage .insert_resource(WindowDescriptor { - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, ..default() }) .insert_resource(ExampleMode::Game) diff --git a/examples/window/multiple_windows.rs b/examples/window/multiple_windows.rs index 7d9e4a643de697..82e8a5e99b928b 100644 --- a/examples/window/multiple_windows.rs +++ b/examples/window/multiple_windows.rs @@ -43,7 +43,7 @@ fn setup( descriptor: WindowDescriptor { width: 800., height: 600., - present_mode: PresentMode::Immediate, + present_mode: PresentMode::AutoNoVsync, title: "Second window".to_string(), ..default() }, diff --git a/examples/window/window_settings.rs b/examples/window/window_settings.rs index ef11098faa9048..5bda8151b72da4 100644 --- a/examples/window/window_settings.rs +++ b/examples/window/window_settings.rs @@ -9,7 +9,7 @@ fn main() { title: "I am a window!".to_string(), width: 500., height: 300., - present_mode: PresentMode::Fifo, + present_mode: PresentMode::AutoVsync, ..default() }) .add_plugins(DefaultPlugins)