1use alloc::format;
4use alloc::string::{String, ToString};
5
6pub trait ToWgsl: Sized {
19 fn to_wgsl(self) -> &'static str;
21}
22
23pub trait TryToWgsl: Sized {
37 fn try_to_wgsl(self) -> Option<&'static str>;
42
43 const DESCRIPTION: &'static str;
45
46 fn to_wgsl_for_diagnostics(self) -> String
57 where
58 Self: core::fmt::Debug + Copy,
59 {
60 match self.try_to_wgsl() {
61 Some(static_string) => static_string.to_string(),
62 None => format!("{{non-WGSL {} {self:?}}}", Self::DESCRIPTION),
63 }
64 }
65}
66
67impl TryToWgsl for crate::MathFunction {
68 const DESCRIPTION: &'static str = "math function";
69
70 fn try_to_wgsl(self) -> Option<&'static str> {
71 use crate::MathFunction as Mf;
72
73 Some(match self {
74 Mf::Abs => "abs",
75 Mf::Min => "min",
76 Mf::Max => "max",
77 Mf::Clamp => "clamp",
78 Mf::Saturate => "saturate",
79 Mf::Cos => "cos",
80 Mf::Cosh => "cosh",
81 Mf::Sin => "sin",
82 Mf::Sinh => "sinh",
83 Mf::Tan => "tan",
84 Mf::Tanh => "tanh",
85 Mf::Acos => "acos",
86 Mf::Asin => "asin",
87 Mf::Atan => "atan",
88 Mf::Atan2 => "atan2",
89 Mf::Asinh => "asinh",
90 Mf::Acosh => "acosh",
91 Mf::Atanh => "atanh",
92 Mf::Radians => "radians",
93 Mf::Degrees => "degrees",
94 Mf::Ceil => "ceil",
95 Mf::Floor => "floor",
96 Mf::Round => "round",
97 Mf::Fract => "fract",
98 Mf::Trunc => "trunc",
99 Mf::Modf => "modf",
100 Mf::Frexp => "frexp",
101 Mf::Ldexp => "ldexp",
102 Mf::Exp => "exp",
103 Mf::Exp2 => "exp2",
104 Mf::Log => "log",
105 Mf::Log2 => "log2",
106 Mf::Pow => "pow",
107 Mf::Dot => "dot",
108 Mf::Dot4I8Packed => "dot4I8Packed",
109 Mf::Dot4U8Packed => "dot4U8Packed",
110 Mf::Cross => "cross",
111 Mf::Distance => "distance",
112 Mf::Length => "length",
113 Mf::Normalize => "normalize",
114 Mf::FaceForward => "faceForward",
115 Mf::Reflect => "reflect",
116 Mf::Refract => "refract",
117 Mf::Sign => "sign",
118 Mf::Fma => "fma",
119 Mf::Mix => "mix",
120 Mf::Step => "step",
121 Mf::SmoothStep => "smoothstep",
122 Mf::Sqrt => "sqrt",
123 Mf::InverseSqrt => "inverseSqrt",
124 Mf::Transpose => "transpose",
125 Mf::Determinant => "determinant",
126 Mf::QuantizeToF16 => "quantizeToF16",
127 Mf::CountTrailingZeros => "countTrailingZeros",
128 Mf::CountLeadingZeros => "countLeadingZeros",
129 Mf::CountOneBits => "countOneBits",
130 Mf::ReverseBits => "reverseBits",
131 Mf::ExtractBits => "extractBits",
132 Mf::InsertBits => "insertBits",
133 Mf::FirstTrailingBit => "firstTrailingBit",
134 Mf::FirstLeadingBit => "firstLeadingBit",
135 Mf::Pack4x8snorm => "pack4x8snorm",
136 Mf::Pack4x8unorm => "pack4x8unorm",
137 Mf::Pack2x16snorm => "pack2x16snorm",
138 Mf::Pack2x16unorm => "pack2x16unorm",
139 Mf::Pack2x16float => "pack2x16float",
140 Mf::Pack4xI8 => "pack4xI8",
141 Mf::Pack4xU8 => "pack4xU8",
142 Mf::Pack4xI8Clamp => "pack4xI8Clamp",
143 Mf::Pack4xU8Clamp => "pack4xU8Clamp",
144 Mf::Unpack4x8snorm => "unpack4x8snorm",
145 Mf::Unpack4x8unorm => "unpack4x8unorm",
146 Mf::Unpack2x16snorm => "unpack2x16snorm",
147 Mf::Unpack2x16unorm => "unpack2x16unorm",
148 Mf::Unpack2x16float => "unpack2x16float",
149 Mf::Unpack4xI8 => "unpack4xI8",
150 Mf::Unpack4xU8 => "unpack4xU8",
151
152 Mf::Inverse | Mf::Outer => return None,
154 })
155 }
156}
157
158impl TryToWgsl for crate::BuiltIn {
159 const DESCRIPTION: &'static str = "builtin value";
160
161 fn try_to_wgsl(self) -> Option<&'static str> {
162 use crate::BuiltIn as Bi;
163 Some(match self {
164 Bi::Position { .. } => "position",
165 Bi::ViewIndex => "view_index",
166 Bi::InstanceIndex => "instance_index",
167 Bi::VertexIndex => "vertex_index",
168 Bi::ClipDistance => "clip_distances",
169 Bi::FragDepth => "frag_depth",
170 Bi::FrontFacing => "front_facing",
171 Bi::PrimitiveIndex => "primitive_index",
172 Bi::DrawIndex => "draw_index",
173 Bi::Barycentric { perspective: true } => "barycentric",
174 Bi::Barycentric { perspective: false } => "barycentric_no_perspective",
175 Bi::SampleIndex => "sample_index",
176 Bi::SampleMask => "sample_mask",
177 Bi::GlobalInvocationId => "global_invocation_id",
178 Bi::LocalInvocationId => "local_invocation_id",
179 Bi::LocalInvocationIndex => "local_invocation_index",
180 Bi::WorkGroupId => "workgroup_id",
181 Bi::NumWorkGroups => "num_workgroups",
182 Bi::NumSubgroups => "num_subgroups",
183 Bi::SubgroupId => "subgroup_id",
184 Bi::SubgroupSize => "subgroup_size",
185 Bi::SubgroupInvocationId => "subgroup_invocation_id",
186
187 Bi::MeshTaskSize => "mesh_task_size",
189 Bi::TriangleIndices => "triangle_indices",
190 Bi::LineIndices => "line_indices",
191 Bi::PointIndex => "point_index",
192 Bi::Vertices => "vertices",
193 Bi::Primitives => "primitives",
194 Bi::VertexCount => "vertex_count",
195 Bi::PrimitiveCount => "primitive_count",
196 Bi::CullPrimitive => "cull_primitive",
197
198 Bi::RayInvocationId => "ray_invocation_id",
199 Bi::NumRayInvocations => "num_ray_invocations",
200 Bi::InstanceCustomData => "instance_custom_data",
201 Bi::GeometryIndex => "geometry_index",
202 Bi::WorldRayOrigin => "world_ray_origin",
203 Bi::WorldRayDirection => "world_ray_direction",
204 Bi::ObjectRayOrigin => "object_ray_origin",
205 Bi::ObjectRayDirection => "object_ray_direction",
206 Bi::RayTmin => "ray_t_min",
207 Bi::RayTCurrentMax => "ray_t_current_max",
208 Bi::ObjectToWorld => "object_to_world",
209 Bi::WorldToObject => "world_to_object",
210 Bi::HitKind => "hit_kind",
211
212 Bi::BaseInstance
213 | Bi::BaseVertex
214 | Bi::CullDistance
215 | Bi::PointSize
216 | Bi::PointCoord
217 | Bi::WorkGroupSize => return None,
218 })
219 }
220}
221
222impl ToWgsl for crate::Interpolation {
223 fn to_wgsl(self) -> &'static str {
224 match self {
225 crate::Interpolation::Perspective => "perspective",
226 crate::Interpolation::Linear => "linear",
227 crate::Interpolation::Flat => "flat",
228 crate::Interpolation::PerVertex => "per_vertex",
229 }
230 }
231}
232
233impl ToWgsl for crate::Sampling {
234 fn to_wgsl(self) -> &'static str {
235 match self {
236 crate::Sampling::Center => "center",
237 crate::Sampling::Centroid => "centroid",
238 crate::Sampling::Sample => "sample",
239 crate::Sampling::First => "first",
240 crate::Sampling::Either => "either",
241 }
242 }
243}
244
245impl ToWgsl for crate::StorageFormat {
246 fn to_wgsl(self) -> &'static str {
247 use crate::StorageFormat as Sf;
248
249 match self {
250 Sf::R8Unorm => "r8unorm",
251 Sf::R8Snorm => "r8snorm",
252 Sf::R8Uint => "r8uint",
253 Sf::R8Sint => "r8sint",
254 Sf::R16Uint => "r16uint",
255 Sf::R16Sint => "r16sint",
256 Sf::R16Float => "r16float",
257 Sf::Rg8Unorm => "rg8unorm",
258 Sf::Rg8Snorm => "rg8snorm",
259 Sf::Rg8Uint => "rg8uint",
260 Sf::Rg8Sint => "rg8sint",
261 Sf::R32Uint => "r32uint",
262 Sf::R32Sint => "r32sint",
263 Sf::R32Float => "r32float",
264 Sf::Rg16Uint => "rg16uint",
265 Sf::Rg16Sint => "rg16sint",
266 Sf::Rg16Float => "rg16float",
267 Sf::Rgba8Unorm => "rgba8unorm",
268 Sf::Rgba8Snorm => "rgba8snorm",
269 Sf::Rgba8Uint => "rgba8uint",
270 Sf::Rgba8Sint => "rgba8sint",
271 Sf::Bgra8Unorm => "bgra8unorm",
272 Sf::Rgb10a2Uint => "rgb10a2uint",
273 Sf::Rgb10a2Unorm => "rgb10a2unorm",
274 Sf::Rg11b10Ufloat => "rg11b10ufloat",
275 Sf::R64Uint => "r64uint",
276 Sf::Rg32Uint => "rg32uint",
277 Sf::Rg32Sint => "rg32sint",
278 Sf::Rg32Float => "rg32float",
279 Sf::Rgba16Uint => "rgba16uint",
280 Sf::Rgba16Sint => "rgba16sint",
281 Sf::Rgba16Float => "rgba16float",
282 Sf::Rgba32Uint => "rgba32uint",
283 Sf::Rgba32Sint => "rgba32sint",
284 Sf::Rgba32Float => "rgba32float",
285 Sf::R16Unorm => "r16unorm",
286 Sf::R16Snorm => "r16snorm",
287 Sf::Rg16Unorm => "rg16unorm",
288 Sf::Rg16Snorm => "rg16snorm",
289 Sf::Rgba16Unorm => "rgba16unorm",
290 Sf::Rgba16Snorm => "rgba16snorm",
291 }
292 }
293}
294
295impl TryToWgsl for crate::Scalar {
296 const DESCRIPTION: &'static str = "scalar type";
297
298 fn try_to_wgsl(self) -> Option<&'static str> {
299 use crate::Scalar;
300
301 Some(match self {
302 Scalar::F16 => "f16",
303 Scalar::F32 => "f32",
304 Scalar::F64 => "f64",
305 Scalar::I32 => "i32",
306 Scalar::U32 => "u32",
307 Scalar::I64 => "i64",
308 Scalar::U64 => "u64",
309 Scalar::BOOL => "bool",
310 _ => return None,
311 })
312 }
313
314 fn to_wgsl_for_diagnostics(self) -> String {
315 match self.try_to_wgsl() {
316 Some(static_string) => static_string.to_string(),
317 None => match self.kind {
318 crate::ScalarKind::Sint
319 | crate::ScalarKind::Uint
320 | crate::ScalarKind::Float
321 | crate::ScalarKind::Bool => format!("{{non-WGSL scalar {self:?}}}"),
322 crate::ScalarKind::AbstractInt => "{AbstractInt}".to_string(),
323 crate::ScalarKind::AbstractFloat => "{AbstractFloat}".to_string(),
324 },
325 }
326 }
327}
328
329impl ToWgsl for crate::CooperativeRole {
330 fn to_wgsl(self) -> &'static str {
331 match self {
332 Self::A => "A",
333 Self::B => "B",
334 Self::C => "C",
335 }
336 }
337}
338
339impl ToWgsl for crate::ImageDimension {
340 fn to_wgsl(self) -> &'static str {
341 match self {
342 Self::D1 => "1d",
343 Self::D2 => "2d",
344 Self::D3 => "3d",
345 Self::Cube => "cube",
346 }
347 }
348}
349
350pub const fn address_space_str(
369 space: crate::AddressSpace,
370) -> (Option<&'static str>, Option<&'static str>) {
371 use crate::AddressSpace as As;
372
373 (
374 Some(match space {
375 As::Private => "private",
376 As::Uniform => "uniform",
377 As::Storage { access } => {
378 if access.contains(crate::StorageAccess::ATOMIC) {
379 return (Some("storage"), Some("atomic"));
380 } else if access.contains(crate::StorageAccess::STORE) {
381 return (Some("storage"), Some("read_write"));
382 } else {
383 "storage"
384 }
385 }
386 As::Immediate => "immediate",
387 As::WorkGroup => "workgroup",
388 As::Handle => return (None, None),
389 As::Function => "function",
390 As::TaskPayload => "task_payload",
391 As::IncomingRayPayload => "incoming_ray_payload",
392 As::RayPayload => "ray_payload",
393 }),
394 None,
395 )
396}