1use alloc::{format, string::String, sync::Arc, vec::Vec};
2
3use arrayvec::ArrayVec;
4use thiserror::Error;
5use wgt::{
6 error::{ErrorType, WebGpuError},
7 BufferAddress, BufferTextureCopyInfoError, BufferUsages, Extent3d, TextureSelector,
8 TextureUsages,
9};
10
11use crate::{
12 api_log,
13 command::{
14 clear_texture, encoder::EncodingState, ArcCommand, CommandEncoderError, EncoderStateError,
15 },
16 device::MissingDownlevelFlags,
17 global::Global,
18 id::{BufferId, CommandEncoderId, TextureId},
19 init_tracker::{
20 has_copy_partial_init_tracker_coverage, MemoryInitKind, TextureInitRange,
21 TextureInitTrackerAction,
22 },
23 resource::{
24 Buffer, MissingBufferUsageError, MissingTextureUsageError, ParentDevice, RawResourceAccess,
25 Texture, TextureErrorDimension,
26 },
27};
28
29use super::ClearError;
30
31type TexelCopyBufferInfo = wgt::TexelCopyBufferInfo<BufferId>;
32type TexelCopyTextureInfo = wgt::TexelCopyTextureInfo<Arc<Texture>>;
33
34#[derive(Clone, Copy, Debug, Eq, PartialEq)]
35pub enum CopySide {
36 Source,
37 Destination,
38}
39
40#[derive(Clone, Debug, Error)]
42#[non_exhaustive]
43pub enum TransferError {
44 #[error("Source and destination cannot be the same buffer")]
45 SameSourceDestinationBuffer,
46 #[error(transparent)]
47 MissingBufferUsage(#[from] MissingBufferUsageError),
48 #[error(transparent)]
49 MissingTextureUsage(#[from] MissingTextureUsageError),
50 #[error(
51 "Copy at offset {start_offset} bytes would end up overrunning the bounds of the {side:?} buffer of size {buffer_size}"
52 )]
53 BufferStartOffsetOverrun {
54 start_offset: BufferAddress,
55 buffer_size: BufferAddress,
56 side: CopySide,
57 },
58 #[error(
59 "Copy at offset {start_offset} for {size} bytes would end up overrunning the bounds of the {side:?} buffer of size {buffer_size}"
60 )]
61 BufferEndOffsetOverrun {
62 start_offset: BufferAddress,
63 size: BufferAddress,
64 buffer_size: BufferAddress,
65 side: CopySide,
66 },
67 #[error("Copy of {dimension:?} {start_offset}..{end_offset} would end up overrunning the bounds of the {side:?} texture of {dimension:?} size {texture_size}")]
68 TextureOverrun {
69 start_offset: u32,
70 end_offset: u32,
71 texture_size: u32,
72 dimension: TextureErrorDimension,
73 side: CopySide,
74 },
75 #[error("Partial copy of {start_offset}..{end_offset} on {dimension:?} dimension with size {texture_size} \
76 is not supported for the {side:?} texture format {format:?} with {sample_count} samples")]
77 UnsupportedPartialTransfer {
78 format: wgt::TextureFormat,
79 sample_count: u32,
80 start_offset: u32,
81 end_offset: u32,
82 texture_size: u32,
83 dimension: TextureErrorDimension,
84 side: CopySide,
85 },
86 #[error(
87 "Copying{} layers {}..{} to{} layers {}..{} of the same texture is not allowed",
88 if *src_aspects == wgt::TextureAspect::All { String::new() } else { format!(" {src_aspects:?}") },
89 src_origin_z,
90 src_origin_z + array_layer_count,
91 if *dst_aspects == wgt::TextureAspect::All { String::new() } else { format!(" {dst_aspects:?}") },
92 dst_origin_z,
93 dst_origin_z + array_layer_count,
94 )]
95 InvalidCopyWithinSameTexture {
96 src_aspects: wgt::TextureAspect,
97 dst_aspects: wgt::TextureAspect,
98 src_origin_z: u32,
99 dst_origin_z: u32,
100 array_layer_count: u32,
101 },
102 #[error("Unable to select texture aspect {aspect:?} from format {format:?}")]
103 InvalidTextureAspect {
104 format: wgt::TextureFormat,
105 aspect: wgt::TextureAspect,
106 },
107 #[error("Unable to select texture mip level {level} out of {total}")]
108 InvalidTextureMipLevel { level: u32, total: u32 },
109 #[error("Texture dimension must be 2D when copying from an external texture")]
110 InvalidDimensionExternal,
111 #[error("Buffer offset {0} is not aligned to block size or `COPY_BUFFER_ALIGNMENT`")]
112 UnalignedBufferOffset(BufferAddress),
113 #[error("Copy size {0} does not respect `COPY_BUFFER_ALIGNMENT`")]
114 UnalignedCopySize(BufferAddress),
115 #[error("Copy width is not a multiple of block width")]
116 UnalignedCopyWidth,
117 #[error("Copy height is not a multiple of block height")]
118 UnalignedCopyHeight,
119 #[error("Copy origin's x component is not a multiple of block width")]
120 UnalignedCopyOriginX,
121 #[error("Copy origin's y component is not a multiple of block height")]
122 UnalignedCopyOriginY,
123 #[error("Bytes per row does not respect `COPY_BYTES_PER_ROW_ALIGNMENT`")]
124 UnalignedBytesPerRow,
125 #[error("Number of bytes per row needs to be specified since more than one row is copied")]
126 UnspecifiedBytesPerRow,
127 #[error("Number of rows per image needs to be specified since more than one image is copied")]
128 UnspecifiedRowsPerImage,
129 #[error("Number of bytes per row is less than the number of bytes in a complete row")]
130 InvalidBytesPerRow,
131 #[error("Number of rows per image is invalid")]
132 InvalidRowsPerImage,
133 #[error("Overflow while computing the size of the copy")]
134 SizeOverflow,
135 #[error("Copy source aspects must refer to all aspects of the source texture format")]
136 CopySrcMissingAspects,
137 #[error(
138 "Copy destination aspects must refer to all aspects of the destination texture format"
139 )]
140 CopyDstMissingAspects,
141 #[error("Copy aspect must refer to a single aspect of texture format")]
142 CopyAspectNotOne,
143 #[error("Copying from textures with format {0:?} is forbidden")]
144 CopyFromForbiddenTextureFormat(wgt::TextureFormat),
145 #[error("Copying from textures with format {format:?} and aspect {aspect:?} is forbidden")]
146 CopyFromForbiddenTextureFormatAspect {
147 format: wgt::TextureFormat,
148 aspect: wgt::TextureAspect,
149 },
150 #[error("Copying to textures with format {0:?} is forbidden")]
151 CopyToForbiddenTextureFormat(wgt::TextureFormat),
152 #[error("Copying to textures with format {format:?} and aspect {aspect:?} is forbidden")]
153 CopyToForbiddenTextureFormatAspect {
154 format: wgt::TextureFormat,
155 aspect: wgt::TextureAspect,
156 },
157 #[error(
158 "Copying to textures with format {0:?} is forbidden when copying from external texture"
159 )]
160 ExternalCopyToForbiddenTextureFormat(wgt::TextureFormat),
161 #[error(
162 "Source format ({src_format:?}) and destination format ({dst_format:?}) are not copy-compatible (they may only differ in srgb-ness)"
163 )]
164 TextureFormatsNotCopyCompatible {
165 src_format: wgt::TextureFormat,
166 dst_format: wgt::TextureFormat,
167 },
168 #[error(transparent)]
169 MemoryInitFailure(#[from] ClearError),
170 #[error("Cannot encode this copy because of a missing downelevel flag")]
171 MissingDownlevelFlags(#[from] MissingDownlevelFlags),
172 #[error("Source texture sample count must be 1, got {sample_count}")]
173 InvalidSampleCount { sample_count: u32 },
174 #[error(
175 "Source sample count ({src_sample_count:?}) and destination sample count ({dst_sample_count:?}) are not equal"
176 )]
177 SampleCountNotEqual {
178 src_sample_count: u32,
179 dst_sample_count: u32,
180 },
181 #[error("Requested mip level {requested} does not exist (count: {count})")]
182 InvalidMipLevel { requested: u32, count: u32 },
183 #[error("Buffer is expected to be unmapped, but was not")]
184 BufferNotAvailable,
185}
186
187impl WebGpuError for TransferError {
188 fn webgpu_error_type(&self) -> ErrorType {
189 match self {
190 Self::MissingBufferUsage(e) => e.webgpu_error_type(),
191 Self::MissingTextureUsage(e) => e.webgpu_error_type(),
192 Self::MemoryInitFailure(e) => e.webgpu_error_type(),
193
194 Self::BufferEndOffsetOverrun { .. }
195 | Self::TextureOverrun { .. }
196 | Self::BufferStartOffsetOverrun { .. }
197 | Self::UnsupportedPartialTransfer { .. }
198 | Self::InvalidCopyWithinSameTexture { .. }
199 | Self::InvalidTextureAspect { .. }
200 | Self::InvalidTextureMipLevel { .. }
201 | Self::InvalidDimensionExternal
202 | Self::UnalignedBufferOffset(..)
203 | Self::UnalignedCopySize(..)
204 | Self::UnalignedCopyWidth
205 | Self::UnalignedCopyHeight
206 | Self::UnalignedCopyOriginX
207 | Self::UnalignedCopyOriginY
208 | Self::UnalignedBytesPerRow
209 | Self::UnspecifiedBytesPerRow
210 | Self::UnspecifiedRowsPerImage
211 | Self::InvalidBytesPerRow
212 | Self::InvalidRowsPerImage
213 | Self::SizeOverflow
214 | Self::CopySrcMissingAspects
215 | Self::CopyDstMissingAspects
216 | Self::CopyAspectNotOne
217 | Self::CopyFromForbiddenTextureFormat(..)
218 | Self::CopyFromForbiddenTextureFormatAspect { .. }
219 | Self::CopyToForbiddenTextureFormat(..)
220 | Self::CopyToForbiddenTextureFormatAspect { .. }
221 | Self::ExternalCopyToForbiddenTextureFormat(..)
222 | Self::TextureFormatsNotCopyCompatible { .. }
223 | Self::MissingDownlevelFlags(..)
224 | Self::InvalidSampleCount { .. }
225 | Self::SampleCountNotEqual { .. }
226 | Self::InvalidMipLevel { .. }
227 | Self::SameSourceDestinationBuffer
228 | Self::BufferNotAvailable => ErrorType::Validation,
229 }
230 }
231}
232
233impl From<BufferTextureCopyInfoError> for TransferError {
234 fn from(value: BufferTextureCopyInfoError) -> Self {
235 match value {
236 BufferTextureCopyInfoError::InvalidBytesPerRow => Self::InvalidBytesPerRow,
237 BufferTextureCopyInfoError::InvalidRowsPerImage => Self::InvalidRowsPerImage,
238 BufferTextureCopyInfoError::ImageStrideOverflow
239 | BufferTextureCopyInfoError::ImageBytesOverflow(_)
240 | BufferTextureCopyInfoError::ArraySizeOverflow(_) => Self::SizeOverflow,
241 }
242 }
243}
244
245pub(crate) fn extract_texture_selector<T>(
246 copy_texture: &wgt::TexelCopyTextureInfo<T>,
247 copy_size: &Extent3d,
248 texture: &Texture,
249) -> Result<(TextureSelector, hal::TextureCopyBase), TransferError> {
250 let format = texture.desc.format;
251 let copy_aspect = hal::FormatAspects::new(format, copy_texture.aspect);
252 if copy_aspect.is_empty() {
253 return Err(TransferError::InvalidTextureAspect {
254 format,
255 aspect: copy_texture.aspect,
256 });
257 }
258
259 let (layers, origin_z) = match texture.desc.dimension {
260 wgt::TextureDimension::D1 => (0..1, 0),
261 wgt::TextureDimension::D2 => (
262 copy_texture.origin.z..copy_texture.origin.z + copy_size.depth_or_array_layers,
263 0,
264 ),
265 wgt::TextureDimension::D3 => (0..1, copy_texture.origin.z),
266 };
267 let base = hal::TextureCopyBase {
268 origin: wgt::Origin3d {
269 x: copy_texture.origin.x,
270 y: copy_texture.origin.y,
271 z: origin_z,
272 },
273 array_layer: layers.start,
275 mip_level: copy_texture.mip_level,
276 aspect: copy_aspect,
277 };
278 let selector = TextureSelector {
279 mips: copy_texture.mip_level..copy_texture.mip_level + 1,
280 layers,
281 };
282
283 Ok((selector, base))
284}
285
286pub(crate) fn validate_linear_texture_data(
298 layout: &wgt::TexelCopyBufferLayout,
299 format: wgt::TextureFormat,
300 aspect: wgt::TextureAspect,
301 buffer_size: BufferAddress,
302 buffer_side: CopySide,
303 copy_size: &Extent3d,
304) -> Result<(BufferAddress, BufferAddress, bool), TransferError> {
305 let wgt::BufferTextureCopyInfo {
306 copy_width,
307 copy_height,
308 depth_or_array_layers,
309
310 offset,
311
312 block_size_bytes: _,
313 block_width_texels,
314 block_height_texels,
315
316 width_blocks: _,
317 height_blocks,
318
319 row_bytes_dense,
320 row_stride_bytes,
321
322 image_stride_rows: _,
323 image_stride_bytes,
324
325 image_rows_dense: _,
326 image_bytes_dense,
327
328 bytes_in_copy,
329 } = layout.get_buffer_texture_copy_info(format, aspect, copy_size)?;
330
331 if !copy_width.is_multiple_of(block_width_texels) {
332 return Err(TransferError::UnalignedCopyWidth);
333 }
334 if !copy_height.is_multiple_of(block_height_texels) {
335 return Err(TransferError::UnalignedCopyHeight);
336 }
337
338 let requires_multiple_rows = depth_or_array_layers > 1 || height_blocks > 1;
339 let requires_multiple_images = depth_or_array_layers > 1;
340
341 if layout.bytes_per_row.is_none() && requires_multiple_rows {
346 return Err(TransferError::UnspecifiedBytesPerRow);
347 }
348
349 if layout.rows_per_image.is_none() && requires_multiple_images {
350 return Err(TransferError::UnspecifiedRowsPerImage);
351 };
352
353 if offset > buffer_size {
354 return Err(TransferError::BufferStartOffsetOverrun {
355 start_offset: offset,
356 buffer_size,
357 side: buffer_side,
358 });
359 }
360 if bytes_in_copy > buffer_size - offset {
362 return Err(TransferError::BufferEndOffsetOverrun {
363 start_offset: offset,
364 size: bytes_in_copy,
365 buffer_size,
366 side: buffer_side,
367 });
368 }
369
370 let is_contiguous = (row_stride_bytes == row_bytes_dense || !requires_multiple_rows)
371 && (image_stride_bytes == image_bytes_dense || !requires_multiple_images);
372
373 Ok((bytes_in_copy, image_stride_bytes, is_contiguous))
374}
375
376pub(crate) fn validate_texture_copy_src_format(
385 format: wgt::TextureFormat,
386 aspect: wgt::TextureAspect,
387) -> Result<(), TransferError> {
388 use wgt::TextureAspect as Ta;
389 use wgt::TextureFormat as Tf;
390 match (format, aspect) {
391 (Tf::Depth24Plus, _) => Err(TransferError::CopyFromForbiddenTextureFormat(format)),
392 (Tf::Depth24PlusStencil8, Ta::DepthOnly) => {
393 Err(TransferError::CopyFromForbiddenTextureFormatAspect { format, aspect })
394 }
395 _ => Ok(()),
396 }
397}
398
399pub(crate) fn validate_texture_copy_dst_format(
408 format: wgt::TextureFormat,
409 aspect: wgt::TextureAspect,
410) -> Result<(), TransferError> {
411 use wgt::TextureAspect as Ta;
412 use wgt::TextureFormat as Tf;
413 match (format, aspect) {
414 (Tf::Depth24Plus | Tf::Depth32Float, _) => {
415 Err(TransferError::CopyToForbiddenTextureFormat(format))
416 }
417 (Tf::Depth24PlusStencil8 | Tf::Depth32FloatStencil8, Ta::DepthOnly) => {
418 Err(TransferError::CopyToForbiddenTextureFormatAspect { format, aspect })
419 }
420 _ => Ok(()),
421 }
422}
423
424pub(crate) fn validate_texture_buffer_copy<T>(
452 texture_copy_view: &wgt::TexelCopyTextureInfo<T>,
453 aspect: hal::FormatAspects,
454 desc: &wgt::TextureDescriptor<String, Vec<wgt::TextureFormat>>,
455 layout: &wgt::TexelCopyBufferLayout,
456 aligned: bool,
457) -> Result<(), TransferError> {
458 if desc.sample_count != 1 {
459 return Err(TransferError::InvalidSampleCount {
460 sample_count: desc.sample_count,
461 });
462 }
463
464 if !aspect.is_one() {
465 return Err(TransferError::CopyAspectNotOne);
466 }
467
468 let offset_alignment = if desc.format.is_depth_stencil_format() {
469 4
470 } else {
471 desc.format
476 .block_copy_size(Some(texture_copy_view.aspect))
477 .expect("non-copyable formats should have been rejected previously")
478 };
479
480 if aligned && !layout.offset.is_multiple_of(u64::from(offset_alignment)) {
481 return Err(TransferError::UnalignedBufferOffset(layout.offset));
482 }
483
484 if let Some(bytes_per_row) = layout.bytes_per_row {
485 if aligned && bytes_per_row % wgt::COPY_BYTES_PER_ROW_ALIGNMENT != 0 {
486 return Err(TransferError::UnalignedBytesPerRow);
487 }
488 }
489
490 Ok(())
491}
492
493pub(crate) fn validate_texture_copy_range<T>(
504 texture_copy_view: &wgt::TexelCopyTextureInfo<T>,
505 desc: &wgt::TextureDescriptor<String, Vec<wgt::TextureFormat>>,
506 texture_side: CopySide,
507 copy_size: &Extent3d,
508) -> Result<(hal::CopyExtent, u32), TransferError> {
509 let (block_width, block_height) = desc.format.block_dimensions();
510
511 let extent_virtual = desc.mip_level_size(texture_copy_view.mip_level).ok_or(
512 TransferError::InvalidTextureMipLevel {
513 level: texture_copy_view.mip_level,
514 total: desc.mip_level_count,
515 },
516 )?;
517 let extent = extent_virtual.physical_size(desc.format);
519
520 let requires_exact_size = desc.format.is_depth_stencil_format() || desc.sample_count > 1;
523
524 let check_dimension = |dimension: TextureErrorDimension,
527 start_offset: u32,
528 size: u32,
529 texture_size: u32,
530 requires_exact_size: bool|
531 -> Result<(), TransferError> {
532 if requires_exact_size && (start_offset != 0 || size != texture_size) {
533 Err(TransferError::UnsupportedPartialTransfer {
534 format: desc.format,
535 sample_count: desc.sample_count,
536 start_offset,
537 end_offset: start_offset.wrapping_add(size),
538 texture_size,
539 dimension,
540 side: texture_side,
541 })
542 } else if start_offset > texture_size || texture_size - start_offset < size {
545 Err(TransferError::TextureOverrun {
546 start_offset,
547 end_offset: start_offset.wrapping_add(size),
548 texture_size,
549 dimension,
550 side: texture_side,
551 })
552 } else {
553 Ok(())
554 }
555 };
556
557 check_dimension(
558 TextureErrorDimension::X,
559 texture_copy_view.origin.x,
560 copy_size.width,
561 extent.width,
562 requires_exact_size,
563 )?;
564 check_dimension(
565 TextureErrorDimension::Y,
566 texture_copy_view.origin.y,
567 copy_size.height,
568 extent.height,
569 requires_exact_size,
570 )?;
571 check_dimension(
572 TextureErrorDimension::Z,
573 texture_copy_view.origin.z,
574 copy_size.depth_or_array_layers,
575 extent.depth_or_array_layers,
576 false, )?;
578
579 if !texture_copy_view.origin.x.is_multiple_of(block_width) {
580 return Err(TransferError::UnalignedCopyOriginX);
581 }
582 if !texture_copy_view.origin.y.is_multiple_of(block_height) {
583 return Err(TransferError::UnalignedCopyOriginY);
584 }
585 if !copy_size.width.is_multiple_of(block_width) {
586 return Err(TransferError::UnalignedCopyWidth);
587 }
588 if !copy_size.height.is_multiple_of(block_height) {
589 return Err(TransferError::UnalignedCopyHeight);
590 }
591
592 let (depth, array_layer_count) = match desc.dimension {
593 wgt::TextureDimension::D1 => (1, 1),
594 wgt::TextureDimension::D2 => (1, copy_size.depth_or_array_layers),
595 wgt::TextureDimension::D3 => (copy_size.depth_or_array_layers, 1),
596 };
597
598 let copy_extent = hal::CopyExtent {
599 width: copy_size.width,
600 height: copy_size.height,
601 depth,
602 };
603 Ok((copy_extent, array_layer_count))
604}
605
606pub(crate) fn validate_copy_within_same_texture<T>(
617 src: &wgt::TexelCopyTextureInfo<T>,
618 dst: &wgt::TexelCopyTextureInfo<T>,
619 format: wgt::TextureFormat,
620 array_layer_count: u32,
621) -> Result<(), TransferError> {
622 let src_aspects = hal::FormatAspects::new(format, src.aspect);
623 let dst_aspects = hal::FormatAspects::new(format, dst.aspect);
624 if (src_aspects & dst_aspects).is_empty() {
625 return Ok(());
627 }
628
629 if src.origin.z >= dst.origin.z + array_layer_count
630 || dst.origin.z >= src.origin.z + array_layer_count
631 {
632 return Ok(());
634 }
635
636 if src.mip_level != dst.mip_level {
637 return Ok(());
639 }
640
641 Err(TransferError::InvalidCopyWithinSameTexture {
642 src_aspects: src.aspect,
643 dst_aspects: dst.aspect,
644 src_origin_z: src.origin.z,
645 dst_origin_z: dst.origin.z,
646 array_layer_count,
647 })
648}
649
650fn handle_texture_init(
651 state: &mut EncodingState,
652 init_kind: MemoryInitKind,
653 copy_texture: &TexelCopyTextureInfo,
654 copy_size: &Extent3d,
655 texture: &Arc<Texture>,
656) -> Result<(), ClearError> {
657 let init_layer_range = if texture.desc.dimension == wgt::TextureDimension::D3 {
658 0..1
660 } else {
661 copy_texture.origin.z..copy_texture.origin.z + copy_size.depth_or_array_layers
662 };
663 let init_action = TextureInitTrackerAction {
664 texture: texture.clone(),
665 range: TextureInitRange {
666 mip_range: copy_texture.mip_level..copy_texture.mip_level + 1,
667 layer_range: init_layer_range,
668 },
669 kind: init_kind,
670 };
671
672 let immediate_inits = state
674 .texture_memory_actions
675 .register_init_action(&{ init_action });
676
677 if !immediate_inits.is_empty() {
679 for init in immediate_inits {
680 clear_texture(
681 &init.texture,
682 TextureInitRange {
683 mip_range: init.mip_level..(init.mip_level + 1),
684 layer_range: init.layer..(init.layer + 1),
685 },
686 state.raw_encoder,
687 &mut state.tracker.textures,
688 &state.device.alignments,
689 state.device.zero_buffer.as_ref(),
690 state.snatch_guard,
691 state.device.instance_flags,
692 )?;
693 }
694 }
695
696 Ok(())
697}
698
699fn handle_src_texture_init(
704 state: &mut EncodingState,
705 source: &TexelCopyTextureInfo,
706 copy_size: &Extent3d,
707 texture: &Arc<Texture>,
708) -> Result<(), TransferError> {
709 handle_texture_init(
710 state,
711 MemoryInitKind::NeedsInitializedMemory,
712 source,
713 copy_size,
714 texture,
715 )?;
716 Ok(())
717}
718
719fn handle_dst_texture_init(
724 state: &mut EncodingState,
725 destination: &wgt::TexelCopyTextureInfo<Arc<Texture>>,
726 copy_size: &Extent3d,
727 texture: &Arc<Texture>,
728) -> Result<(), TransferError> {
729 let dst_init_kind =
734 if has_copy_partial_init_tracker_coverage(copy_size, destination, &texture.desc) {
735 MemoryInitKind::NeedsInitializedMemory
736 } else {
737 MemoryInitKind::ImplicitlyInitialized
738 };
739
740 handle_texture_init(state, dst_init_kind, destination, copy_size, texture)?;
741 Ok(())
742}
743
744fn handle_buffer_init(
749 state: &mut EncodingState,
750 info: &wgt::TexelCopyBufferInfo<Arc<Buffer>>,
751 direction: CopySide,
752 required_buffer_bytes_in_copy: BufferAddress,
753 is_contiguous: bool,
754) {
755 const ALIGN_SIZE: BufferAddress = wgt::COPY_BUFFER_ALIGNMENT;
756 const ALIGN_MASK: BufferAddress = wgt::COPY_BUFFER_ALIGNMENT - 1;
757
758 let buffer = &info.buffer;
759 let start = info.layout.offset;
760 let end = info.layout.offset + required_buffer_bytes_in_copy;
761 if !is_contiguous || direction == CopySide::Source {
762 let aligned_start = start & !ALIGN_MASK;
773 let aligned_end = (end + ALIGN_MASK) & !ALIGN_MASK;
774 state
775 .buffer_memory_init_actions
776 .extend(buffer.initialization_status.read().create_action(
777 buffer,
778 aligned_start..aligned_end,
779 MemoryInitKind::NeedsInitializedMemory,
780 ));
781 } else {
782 let aligned_start = (start + ALIGN_MASK) & !ALIGN_MASK;
792 let aligned_end = end & !ALIGN_MASK;
793 if aligned_start != start {
794 state.buffer_memory_init_actions.extend(
795 buffer.initialization_status.read().create_action(
796 buffer,
797 aligned_start - ALIGN_SIZE..aligned_start,
798 MemoryInitKind::NeedsInitializedMemory,
799 ),
800 );
801 }
802 if aligned_start != aligned_end {
803 state.buffer_memory_init_actions.extend(
804 buffer.initialization_status.read().create_action(
805 buffer,
806 aligned_start..aligned_end,
807 MemoryInitKind::ImplicitlyInitialized,
808 ),
809 );
810 }
811 if aligned_end != end {
812 state.buffer_memory_init_actions.extend(
818 buffer.initialization_status.read().create_action(
819 buffer,
820 aligned_end..aligned_end + ALIGN_SIZE,
821 MemoryInitKind::NeedsInitializedMemory,
822 ),
823 );
824 }
825 }
826}
827
828impl Global {
829 pub fn command_encoder_copy_buffer_to_buffer(
830 &self,
831 command_encoder_id: CommandEncoderId,
832 source: BufferId,
833 source_offset: BufferAddress,
834 destination: BufferId,
835 destination_offset: BufferAddress,
836 size: Option<BufferAddress>,
837 ) -> Result<(), EncoderStateError> {
838 profiling::scope!("CommandEncoder::copy_buffer_to_buffer");
839 api_log!(
840 "CommandEncoder::copy_buffer_to_buffer {source:?} -> {destination:?} {size:?}bytes"
841 );
842
843 let hub = &self.hub;
844
845 let cmd_enc = hub.command_encoders.get(command_encoder_id);
846 let mut cmd_buf_data = cmd_enc.data.lock();
847
848 cmd_buf_data.push_with(|| -> Result<_, CommandEncoderError> {
849 Ok(ArcCommand::CopyBufferToBuffer {
850 src: self.resolve_buffer_id(source)?,
851 src_offset: source_offset,
852 dst: self.resolve_buffer_id(destination)?,
853 dst_offset: destination_offset,
854 size,
855 })
856 })
857 }
858
859 pub fn command_encoder_copy_buffer_to_texture(
860 &self,
861 command_encoder_id: CommandEncoderId,
862 source: &TexelCopyBufferInfo,
863 destination: &wgt::TexelCopyTextureInfo<TextureId>,
864 copy_size: &Extent3d,
865 ) -> Result<(), EncoderStateError> {
866 profiling::scope!("CommandEncoder::copy_buffer_to_texture");
867 api_log!(
868 "CommandEncoder::copy_buffer_to_texture {:?} -> {:?} {copy_size:?}",
869 source.buffer,
870 destination.texture
871 );
872
873 let cmd_enc = self.hub.command_encoders.get(command_encoder_id);
874
875 let mut cmd_buf_data = cmd_enc.data.lock();
876
877 cmd_buf_data.push_with(|| -> Result<_, CommandEncoderError> {
878 let texture = self.resolve_texture_id(destination.texture);
879 texture.check_valid(&cmd_enc.device.snatchable_lock.read())?;
880 Ok(ArcCommand::CopyBufferToTexture {
881 src: wgt::TexelCopyBufferInfo::<Arc<Buffer>> {
882 buffer: self.resolve_buffer_id(source.buffer)?,
883 layout: source.layout,
884 },
885 dst: wgt::TexelCopyTextureInfo::<Arc<Texture>> {
886 texture,
887 mip_level: destination.mip_level,
888 origin: destination.origin,
889 aspect: destination.aspect,
890 },
891 size: *copy_size,
892 })
893 })
894 }
895
896 pub fn command_encoder_copy_texture_to_buffer(
897 &self,
898 command_encoder_id: CommandEncoderId,
899 source: &wgt::TexelCopyTextureInfo<TextureId>,
900 destination: &TexelCopyBufferInfo,
901 copy_size: &Extent3d,
902 ) -> Result<(), EncoderStateError> {
903 profiling::scope!("CommandEncoder::copy_texture_to_buffer");
904 api_log!(
905 "CommandEncoder::copy_texture_to_buffer {:?} -> {:?} {copy_size:?}",
906 source.texture,
907 destination.buffer
908 );
909
910 let cmd_enc = self.hub.command_encoders.get(command_encoder_id);
911
912 let mut cmd_buf_data = cmd_enc.data.lock();
913
914 cmd_buf_data.push_with(|| -> Result<_, CommandEncoderError> {
915 let texture = self.resolve_texture_id(source.texture);
916 texture.check_valid(&cmd_enc.device.snatchable_lock.read())?;
917 Ok(ArcCommand::CopyTextureToBuffer {
918 src: wgt::TexelCopyTextureInfo::<Arc<Texture>> {
919 texture,
920 mip_level: source.mip_level,
921 origin: source.origin,
922 aspect: source.aspect,
923 },
924 dst: wgt::TexelCopyBufferInfo::<Arc<Buffer>> {
925 buffer: self.resolve_buffer_id(destination.buffer)?,
926 layout: destination.layout,
927 },
928 size: *copy_size,
929 })
930 })
931 }
932
933 pub fn command_encoder_copy_texture_to_texture(
934 &self,
935 command_encoder_id: CommandEncoderId,
936 source: &wgt::TexelCopyTextureInfo<TextureId>,
937 destination: &wgt::TexelCopyTextureInfo<TextureId>,
938 copy_size: &Extent3d,
939 ) -> Result<(), EncoderStateError> {
940 profiling::scope!("CommandEncoder::copy_texture_to_texture");
941 api_log!(
942 "CommandEncoder::copy_texture_to_texture {:?} -> {:?} {copy_size:?}",
943 source.texture,
944 destination.texture
945 );
946
947 let cmd_enc = self.hub.command_encoders.get(command_encoder_id);
948
949 let mut cmd_buf_data = cmd_enc.data.lock();
950
951 cmd_buf_data.push_with(|| -> Result<_, CommandEncoderError> {
952 let src_texture = self.resolve_texture_id(source.texture);
953 let dst_texture = self.resolve_texture_id(destination.texture);
954 {
955 let snatch_guard = cmd_enc.device.snatchable_lock.read();
956 src_texture.check_valid(&snatch_guard)?;
957 dst_texture.check_valid(&snatch_guard)?;
958 }
959 Ok(ArcCommand::CopyTextureToTexture {
960 src: wgt::TexelCopyTextureInfo {
961 texture: src_texture,
962 mip_level: source.mip_level,
963 origin: source.origin,
964 aspect: source.aspect,
965 },
966 dst: wgt::TexelCopyTextureInfo {
967 texture: dst_texture,
968 mip_level: destination.mip_level,
969 origin: destination.origin,
970 aspect: destination.aspect,
971 },
972 size: *copy_size,
973 })
974 })
975 }
976}
977
978pub(super) fn copy_buffer_to_buffer(
979 state: &mut EncodingState,
980 src_buffer: &Arc<Buffer>,
981 source_offset: BufferAddress,
982 dst_buffer: &Arc<Buffer>,
983 destination_offset: BufferAddress,
984 size: Option<BufferAddress>,
985) -> Result<(), CommandEncoderError> {
986 if src_buffer.is_equal(dst_buffer) {
987 return Err(TransferError::SameSourceDestinationBuffer.into());
988 }
989
990 src_buffer.same_device(state.device)?;
991
992 let src_pending = state
993 .tracker
994 .buffers
995 .set_single(src_buffer, wgt::BufferUses::COPY_SRC);
996
997 let src_raw = src_buffer.try_raw(state.snatch_guard)?;
998 src_buffer
999 .check_usage(BufferUsages::COPY_SRC)
1000 .map_err(TransferError::MissingBufferUsage)?;
1001 let src_barrier = src_pending.map(|pending| pending.into_hal(src_buffer, state.snatch_guard));
1003
1004 dst_buffer.same_device(state.device)?;
1005
1006 let dst_pending = state
1007 .tracker
1008 .buffers
1009 .set_single(dst_buffer, wgt::BufferUses::COPY_DST);
1010
1011 let dst_raw = dst_buffer.try_raw(state.snatch_guard)?;
1012 dst_buffer
1013 .check_usage(BufferUsages::COPY_DST)
1014 .map_err(TransferError::MissingBufferUsage)?;
1015 let dst_barrier = dst_pending.map(|pending| pending.into_hal(dst_buffer, state.snatch_guard));
1016
1017 if source_offset > src_buffer.size {
1018 return Err(TransferError::BufferStartOffsetOverrun {
1019 start_offset: source_offset,
1020 buffer_size: src_buffer.size,
1021 side: CopySide::Source,
1022 }
1023 .into());
1024 }
1025 let size = size.unwrap_or_else(|| {
1026 src_buffer.size - source_offset
1028 });
1029
1030 if !size.is_multiple_of(wgt::COPY_BUFFER_ALIGNMENT) {
1031 return Err(TransferError::UnalignedCopySize(size).into());
1032 }
1033 if !source_offset.is_multiple_of(wgt::COPY_BUFFER_ALIGNMENT) {
1034 return Err(TransferError::UnalignedBufferOffset(source_offset).into());
1035 }
1036 if !destination_offset.is_multiple_of(wgt::COPY_BUFFER_ALIGNMENT) {
1037 return Err(TransferError::UnalignedBufferOffset(destination_offset).into());
1038 }
1039 if !state
1040 .device
1041 .downlevel
1042 .flags
1043 .contains(wgt::DownlevelFlags::UNRESTRICTED_INDEX_BUFFER)
1044 && (src_buffer.usage.contains(BufferUsages::INDEX)
1045 || dst_buffer.usage.contains(BufferUsages::INDEX))
1046 {
1047 let forbidden_usages = BufferUsages::VERTEX
1048 | BufferUsages::UNIFORM
1049 | BufferUsages::INDIRECT
1050 | BufferUsages::STORAGE;
1051 if src_buffer.usage.intersects(forbidden_usages)
1052 || dst_buffer.usage.intersects(forbidden_usages)
1053 {
1054 return Err(TransferError::MissingDownlevelFlags(MissingDownlevelFlags(
1055 wgt::DownlevelFlags::UNRESTRICTED_INDEX_BUFFER,
1056 ))
1057 .into());
1058 }
1059 }
1060
1061 if size > src_buffer.size - source_offset {
1062 return Err(TransferError::BufferEndOffsetOverrun {
1063 start_offset: source_offset,
1064 size,
1065 buffer_size: src_buffer.size,
1066 side: CopySide::Source,
1067 }
1068 .into());
1069 }
1070 let source_end_offset = source_offset + size;
1072
1073 if destination_offset > dst_buffer.size {
1074 return Err(TransferError::BufferStartOffsetOverrun {
1075 start_offset: destination_offset,
1076 buffer_size: dst_buffer.size,
1077 side: CopySide::Destination,
1078 }
1079 .into());
1080 }
1081 if size > dst_buffer.size - destination_offset {
1083 return Err(TransferError::BufferEndOffsetOverrun {
1084 start_offset: destination_offset,
1085 size,
1086 buffer_size: dst_buffer.size,
1087 side: CopySide::Destination,
1088 }
1089 .into());
1090 }
1091 let destination_end_offset = destination_offset + size;
1093
1094 if size == 0 {
1097 log::trace!("Ignoring copy_buffer_to_buffer of size 0");
1098 return Ok(());
1099 }
1100
1101 state
1103 .buffer_memory_init_actions
1104 .extend(dst_buffer.initialization_status.read().create_action(
1105 dst_buffer,
1106 destination_offset..destination_end_offset,
1107 MemoryInitKind::ImplicitlyInitialized,
1108 ));
1109 state
1110 .buffer_memory_init_actions
1111 .extend(src_buffer.initialization_status.read().create_action(
1112 src_buffer,
1113 source_offset..source_end_offset,
1114 MemoryInitKind::NeedsInitializedMemory,
1115 ));
1116
1117 let region = hal::BufferCopy {
1118 src_offset: source_offset,
1119 dst_offset: destination_offset,
1120 size: wgt::BufferSize::new(size).unwrap(),
1121 };
1122 let barriers = src_barrier
1123 .into_iter()
1124 .chain(dst_barrier)
1125 .collect::<Vec<_>>();
1126 unsafe {
1127 state.raw_encoder.transition_buffers(&barriers);
1128 state
1129 .raw_encoder
1130 .copy_buffer_to_buffer(src_raw, dst_raw, &[region]);
1131 }
1132
1133 Ok(())
1134}
1135
1136pub(super) fn copy_buffer_to_texture(
1137 state: &mut EncodingState,
1138 source: &wgt::TexelCopyBufferInfo<Arc<Buffer>>,
1139 destination: &wgt::TexelCopyTextureInfo<Arc<Texture>>,
1140 copy_size: &Extent3d,
1141) -> Result<(), CommandEncoderError> {
1142 let dst_texture = &destination.texture;
1143 let src_buffer = &source.buffer;
1144
1145 dst_texture.same_device(state.device)?;
1146 src_buffer.same_device(state.device)?;
1147
1148 let (hal_copy_size, array_layer_count) = validate_texture_copy_range(
1149 destination,
1150 &dst_texture.desc,
1151 CopySide::Destination,
1152 copy_size,
1153 )?;
1154
1155 let (dst_range, dst_base) = extract_texture_selector(destination, copy_size, dst_texture)?;
1156
1157 let src_raw = src_buffer.try_raw(state.snatch_guard)?;
1158 src_buffer
1159 .check_usage(BufferUsages::COPY_SRC)
1160 .map_err(TransferError::MissingBufferUsage)?;
1161
1162 let dst_raw = dst_texture.try_inner(state.snatch_guard)?.raw();
1163 dst_texture
1164 .check_usage(TextureUsages::COPY_DST)
1165 .map_err(TransferError::MissingTextureUsage)?;
1166
1167 validate_texture_copy_dst_format(dst_texture.desc.format, destination.aspect)?;
1168
1169 validate_texture_buffer_copy(
1170 destination,
1171 dst_base.aspect,
1172 &dst_texture.desc,
1173 &source.layout,
1174 true, )?;
1176
1177 let (required_buffer_bytes_in_copy, bytes_per_array_layer, is_contiguous) =
1178 validate_linear_texture_data(
1179 &source.layout,
1180 dst_texture.desc.format,
1181 destination.aspect,
1182 src_buffer.size,
1183 CopySide::Source,
1184 copy_size,
1185 )?;
1186
1187 if dst_texture.desc.format.is_depth_stencil_format() {
1188 state
1189 .device
1190 .require_downlevel_flags(wgt::DownlevelFlags::DEPTH_TEXTURE_AND_BUFFER_COPIES)
1191 .map_err(TransferError::from)?;
1192 }
1193
1194 if copy_size.width == 0 || copy_size.height == 0 || copy_size.depth_or_array_layers == 0 {
1197 log::trace!("Ignoring copy_buffer_to_texture of size 0");
1198 return Ok(());
1199 }
1200
1201 handle_dst_texture_init(state, destination, copy_size, dst_texture)?;
1205
1206 let src_pending = state
1207 .tracker
1208 .buffers
1209 .set_single(src_buffer, wgt::BufferUses::COPY_SRC);
1210 let src_barrier = src_pending.map(|pending| pending.into_hal(src_buffer, state.snatch_guard));
1211
1212 let dst_pending =
1213 state
1214 .tracker
1215 .textures
1216 .set_single(dst_texture, dst_range, wgt::TextureUses::COPY_DST);
1217 let dst_barrier = dst_pending
1218 .map(|pending| pending.into_hal(dst_raw))
1219 .collect::<Vec<_>>();
1220
1221 handle_buffer_init(
1222 state,
1223 source,
1224 CopySide::Source,
1225 required_buffer_bytes_in_copy,
1226 is_contiguous,
1227 );
1228
1229 let regions = (0..array_layer_count)
1230 .map(|rel_array_layer| {
1231 let mut texture_base = dst_base.clone();
1232 texture_base.array_layer += rel_array_layer;
1233 let mut buffer_layout = source.layout;
1234 buffer_layout.offset += rel_array_layer as u64 * bytes_per_array_layer;
1235 hal::BufferTextureCopy {
1236 buffer_layout,
1237 texture_base,
1238 size: hal_copy_size,
1239 }
1240 })
1241 .collect::<Vec<_>>();
1242
1243 unsafe {
1244 state.raw_encoder.transition_textures(&dst_barrier);
1245 state.raw_encoder.transition_buffers(src_barrier.as_slice());
1246 state
1247 .raw_encoder
1248 .copy_buffer_to_texture(src_raw, dst_raw, ®ions);
1249 }
1250
1251 Ok(())
1252}
1253
1254pub(super) fn copy_texture_to_buffer(
1255 state: &mut EncodingState,
1256 source: &TexelCopyTextureInfo,
1257 destination: &wgt::TexelCopyBufferInfo<Arc<Buffer>>,
1258 copy_size: &Extent3d,
1259) -> Result<(), CommandEncoderError> {
1260 let src_texture = &source.texture;
1261 let dst_buffer = &destination.buffer;
1262
1263 src_texture.same_device(state.device)?;
1264 dst_buffer.same_device(state.device)?;
1265
1266 let (hal_copy_size, array_layer_count) =
1267 validate_texture_copy_range(source, &src_texture.desc, CopySide::Source, copy_size)?;
1268
1269 let (src_range, src_base) = extract_texture_selector(source, copy_size, src_texture)?;
1270
1271 let src_raw = src_texture.try_inner(state.snatch_guard)?.raw();
1272 src_texture
1273 .check_usage(TextureUsages::COPY_SRC)
1274 .map_err(TransferError::MissingTextureUsage)?;
1275
1276 if source.mip_level >= src_texture.desc.mip_level_count {
1277 return Err(TransferError::InvalidMipLevel {
1278 requested: source.mip_level,
1279 count: src_texture.desc.mip_level_count,
1280 }
1281 .into());
1282 }
1283
1284 validate_texture_copy_src_format(src_texture.desc.format, source.aspect)?;
1285
1286 validate_texture_buffer_copy(
1287 source,
1288 src_base.aspect,
1289 &src_texture.desc,
1290 &destination.layout,
1291 true, )?;
1293
1294 let (required_buffer_bytes_in_copy, bytes_per_array_layer, is_contiguous) =
1295 validate_linear_texture_data(
1296 &destination.layout,
1297 src_texture.desc.format,
1298 source.aspect,
1299 dst_buffer.size,
1300 CopySide::Destination,
1301 copy_size,
1302 )?;
1303
1304 if src_texture.desc.format.is_depth_stencil_format() {
1305 state
1306 .device
1307 .require_downlevel_flags(wgt::DownlevelFlags::DEPTH_TEXTURE_AND_BUFFER_COPIES)
1308 .map_err(TransferError::from)?;
1309 }
1310
1311 let dst_raw = dst_buffer.try_raw(state.snatch_guard)?;
1312 dst_buffer
1313 .check_usage(BufferUsages::COPY_DST)
1314 .map_err(TransferError::MissingBufferUsage)?;
1315
1316 if copy_size.width == 0 || copy_size.height == 0 || copy_size.depth_or_array_layers == 0 {
1319 log::trace!("Ignoring copy_texture_to_buffer of size 0");
1320 return Ok(());
1321 }
1322
1323 handle_src_texture_init(state, source, copy_size, src_texture)?;
1327
1328 let src_pending =
1329 state
1330 .tracker
1331 .textures
1332 .set_single(src_texture, src_range, wgt::TextureUses::COPY_SRC);
1333 let src_barrier = src_pending
1334 .map(|pending| pending.into_hal(src_raw))
1335 .collect::<Vec<_>>();
1336
1337 let dst_pending = state
1338 .tracker
1339 .buffers
1340 .set_single(dst_buffer, wgt::BufferUses::COPY_DST);
1341
1342 let dst_barrier = dst_pending.map(|pending| pending.into_hal(dst_buffer, state.snatch_guard));
1343
1344 handle_buffer_init(
1345 state,
1346 destination,
1347 CopySide::Destination,
1348 required_buffer_bytes_in_copy,
1349 is_contiguous,
1350 );
1351
1352 let regions = (0..array_layer_count)
1353 .map(|rel_array_layer| {
1354 let mut texture_base = src_base.clone();
1355 texture_base.array_layer += rel_array_layer;
1356 let mut buffer_layout = destination.layout;
1357 buffer_layout.offset += rel_array_layer as u64 * bytes_per_array_layer;
1358 hal::BufferTextureCopy {
1359 buffer_layout,
1360 texture_base,
1361 size: hal_copy_size,
1362 }
1363 })
1364 .collect::<Vec<_>>();
1365 unsafe {
1366 state.raw_encoder.transition_buffers(dst_barrier.as_slice());
1367 state.raw_encoder.transition_textures(&src_barrier);
1368 state.raw_encoder.copy_texture_to_buffer(
1369 src_raw,
1370 wgt::TextureUses::COPY_SRC,
1371 dst_raw,
1372 ®ions,
1373 );
1374 }
1375
1376 Ok(())
1377}
1378
1379pub(super) fn copy_texture_to_texture(
1380 state: &mut EncodingState,
1381 source: &TexelCopyTextureInfo,
1382 destination: &TexelCopyTextureInfo,
1383 copy_size: &Extent3d,
1384) -> Result<(), CommandEncoderError> {
1385 let src_texture = &source.texture;
1386 let dst_texture = &destination.texture;
1387
1388 src_texture.same_device(state.device)?;
1389 dst_texture.same_device(state.device)?;
1390
1391 let src_fmt_no_srgb = src_texture.desc.format.remove_srgb_suffix();
1405 let dst_fmt_no_srgb = dst_texture.desc.format.remove_srgb_suffix();
1406 let planar_split_ok = src_fmt_no_srgb.is_multi_planar_format()
1407 && src_fmt_no_srgb.aspect_specific_format(source.aspect) == Some(dst_fmt_no_srgb);
1408 if src_fmt_no_srgb != dst_fmt_no_srgb && !planar_split_ok {
1409 return Err(TransferError::TextureFormatsNotCopyCompatible {
1410 src_format: src_texture.desc.format,
1411 dst_format: dst_texture.desc.format,
1412 }
1413 .into());
1414 }
1415
1416 let (src_copy_size, array_layer_count) =
1417 validate_texture_copy_range(source, &src_texture.desc, CopySide::Source, copy_size)?;
1418 let (dst_copy_size, _) = validate_texture_copy_range(
1419 destination,
1420 &dst_texture.desc,
1421 CopySide::Destination,
1422 copy_size,
1423 )?;
1424
1425 if planar_split_ok {
1431 let plane = source.aspect.to_plane().expect("planar_split_ok aspect");
1434 let plane_extent = src_texture
1435 .desc
1436 .compute_render_extent(source.mip_level, Some(plane));
1437 let check = |dimension, start: u32, size: u32, plane_size: u32| {
1438 if start > plane_size || plane_size - start < size {
1439 Err(TransferError::TextureOverrun {
1440 start_offset: start,
1441 end_offset: start.wrapping_add(size),
1442 texture_size: plane_size,
1443 dimension,
1444 side: CopySide::Source,
1445 })
1446 } else {
1447 Ok(())
1448 }
1449 };
1450 check(
1451 TextureErrorDimension::X,
1452 source.origin.x,
1453 copy_size.width,
1454 plane_extent.width,
1455 )?;
1456 check(
1457 TextureErrorDimension::Y,
1458 source.origin.y,
1459 copy_size.height,
1460 plane_extent.height,
1461 )?;
1462 }
1463
1464 if Arc::as_ptr(src_texture) == Arc::as_ptr(dst_texture) {
1465 validate_copy_within_same_texture(
1466 source,
1467 destination,
1468 src_texture.desc.format,
1469 array_layer_count,
1470 )?;
1471 }
1472
1473 let (src_range, src_tex_base) = extract_texture_selector(source, copy_size, src_texture)?;
1474 let (dst_range, dst_tex_base) = extract_texture_selector(destination, copy_size, dst_texture)?;
1475 let src_texture_aspects = hal::FormatAspects::from(src_texture.desc.format);
1476 let dst_texture_aspects = hal::FormatAspects::from(dst_texture.desc.format);
1477 if src_tex_base.aspect != src_texture_aspects && !planar_split_ok {
1479 return Err(TransferError::CopySrcMissingAspects.into());
1480 }
1481 if dst_tex_base.aspect != dst_texture_aspects {
1482 return Err(TransferError::CopyDstMissingAspects.into());
1483 }
1484
1485 if src_texture.desc.sample_count != dst_texture.desc.sample_count {
1486 return Err(TransferError::SampleCountNotEqual {
1487 src_sample_count: src_texture.desc.sample_count,
1488 dst_sample_count: dst_texture.desc.sample_count,
1489 }
1490 .into());
1491 }
1492
1493 let src_raw = src_texture.try_inner(state.snatch_guard)?.raw();
1494 src_texture
1495 .check_usage(TextureUsages::COPY_SRC)
1496 .map_err(TransferError::MissingTextureUsage)?;
1497 let dst_raw = dst_texture.try_inner(state.snatch_guard)?.raw();
1498 dst_texture
1499 .check_usage(TextureUsages::COPY_DST)
1500 .map_err(TransferError::MissingTextureUsage)?;
1501
1502 if copy_size.width == 0 || copy_size.height == 0 || copy_size.depth_or_array_layers == 0 {
1505 log::trace!("Ignoring copy_texture_to_texture of size 0");
1506 return Ok(());
1507 }
1508
1509 handle_src_texture_init(state, source, copy_size, src_texture)?;
1513 handle_dst_texture_init(state, destination, copy_size, dst_texture)?;
1514
1515 let src_pending =
1516 state
1517 .tracker
1518 .textures
1519 .set_single(src_texture, src_range, wgt::TextureUses::COPY_SRC);
1520
1521 let mut barriers: ArrayVec<_, 2> = src_pending
1524 .map(|pending| pending.into_hal(src_raw))
1525 .collect();
1526
1527 let dst_pending =
1528 state
1529 .tracker
1530 .textures
1531 .set_single(dst_texture, dst_range, wgt::TextureUses::COPY_DST);
1532 barriers.extend(dst_pending.map(|pending| pending.into_hal(dst_raw)));
1533
1534 let hal_copy_size = hal::CopyExtent {
1535 width: src_copy_size.width.min(dst_copy_size.width),
1536 height: src_copy_size.height.min(dst_copy_size.height),
1537 depth: src_copy_size.depth.min(dst_copy_size.depth),
1538 };
1539
1540 let dst_format = dst_texture.desc.format;
1541
1542 let regions = (0..array_layer_count).map(|rel_array_layer| {
1543 let mut src_base = src_tex_base.clone();
1544 let mut dst_base = dst_tex_base.clone();
1545 src_base.array_layer += rel_array_layer;
1546 dst_base.array_layer += rel_array_layer;
1547 hal::TextureCopy {
1548 src_base,
1549 dst_base,
1550 size: hal_copy_size,
1551 }
1552 });
1553
1554 let regions = if dst_tex_base.aspect == hal::FormatAspects::DEPTH_STENCIL {
1555 regions
1556 .flat_map(|region| {
1557 let (mut depth, mut stencil) = (region.clone(), region);
1558 depth.src_base.aspect = hal::FormatAspects::DEPTH;
1559 depth.dst_base.aspect = hal::FormatAspects::DEPTH;
1560 stencil.src_base.aspect = hal::FormatAspects::STENCIL;
1561 stencil.dst_base.aspect = hal::FormatAspects::STENCIL;
1562 [depth, stencil]
1563 })
1564 .collect::<Vec<_>>()
1565 } else if let Some(plane_count) = dst_format.planes() {
1566 regions
1567 .into_iter()
1568 .flat_map(|region| {
1569 (0..plane_count).map(move |plane| {
1570 let mut plane_region = region.clone();
1571
1572 let plane_aspect = wgt::TextureAspect::from_plane(plane)
1573 .expect("expected texture aspect to exist for the plane");
1574 let plane_aspect = hal::FormatAspects::new(dst_format, plane_aspect);
1575 plane_region.src_base.aspect = plane_aspect;
1576 plane_region.dst_base.aspect = plane_aspect;
1577
1578 let (w_subsampling, h_subsampling) =
1579 dst_format.subsampling_factors(Some(plane));
1580 plane_region.src_base.origin.x /= w_subsampling;
1581 plane_region.src_base.origin.y /= h_subsampling;
1582 plane_region.dst_base.origin.x /= w_subsampling;
1583 plane_region.dst_base.origin.y /= h_subsampling;
1584
1585 plane_region.size.width /= w_subsampling;
1586 plane_region.size.height /= h_subsampling;
1587
1588 plane_region
1589 })
1590 })
1591 .collect::<Vec<_>>()
1592 } else {
1593 regions.collect::<Vec<_>>()
1594 };
1595 unsafe {
1596 state.raw_encoder.transition_textures(&barriers);
1597 state.raw_encoder.copy_texture_to_texture(
1598 src_raw,
1599 wgt::TextureUses::COPY_SRC,
1600 dst_raw,
1601 ®ions,
1602 );
1603 }
1604
1605 Ok(())
1606}