1 // Copyright 2017 The Dawn Authors 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 15 #ifndef DAWNNATIVE_D3D12_TEXTURECOPYSPLITTER_H_ 16 #define DAWNNATIVE_D3D12_TEXTURECOPYSPLITTER_H_ 17 18 #include "dawn_native/dawn_platform.h" 19 20 #include <array> 21 22 namespace dawn_native { 23 24 struct TexelBlockInfo; 25 26 } // namespace dawn_native 27 28 namespace dawn_native { namespace d3d12 { 29 30 struct TextureCopySubresource { 31 static constexpr unsigned int kMaxTextureCopyRegions = 4; 32 33 struct CopyInfo { 34 uint64_t alignedOffset = 0; 35 Origin3D textureOffset; 36 Origin3D bufferOffset; 37 Extent3D bufferSize; 38 39 Extent3D copySize; 40 }; 41 42 CopyInfo* AddCopy(); 43 44 uint32_t count = 0; 45 std::array<CopyInfo, kMaxTextureCopyRegions> copies; 46 }; 47 48 struct TextureCopySplits { 49 static constexpr uint32_t kMaxTextureCopySubresources = 2; 50 51 std::array<TextureCopySubresource, kMaxTextureCopySubresources> copySubresources; 52 }; 53 54 // This function is shared by 2D and 3D texture copy splitter. But it only knows how to handle 55 // 2D non-arrayed textures correctly, and just forwards "copySize.depthOrArrayLayers". See 56 // details in Compute{2D|3D}TextureCopySplits about how we generate copy regions for 2D array 57 // and 3D textures based on this function. 58 // The resulting copies triggered by API like CopyTextureRegion are equivalent to the copy 59 // regions defines by the arguments of TextureCopySubresource returned by this function and its 60 // counterparts. These arguments should strictly conform to particular invariants. Otherwise, 61 // D3D12 driver may report validation errors when we call CopyTextureRegion. Some important 62 // invariants are listed below. For more details 63 // of these invariants, see src/tests/unittests/d3d12/CopySplitTests.cpp. 64 // - Inside each copy region: 1) its buffer offset plus copy size should be less than its 65 // buffer size, 2) its buffer offset on y-axis should be less than copy format's 66 // blockInfo.height, 3) its buffer offset on z-axis should be 0. 67 // - Each copy region has an offset (aka alignedOffset) aligned to 68 // D3D12_TEXTURE_DATA_PLACEMENT_ALIGNMENT 69 // - The buffer footprint of each copy region should be entirely within the copied buffer, 70 // which means that the last "texel" of the buffer footprint doesn't go past the end of 71 // the buffer even though the last "texel" might not be copied. 72 // - If there are multiple copy regions, each copy region should not overlap with the others. 73 // - Copy region(s) combined should exactly be equivalent to the texture region to be copied. 74 // - Every pixel accessed by every copy region should not be out of the bound of the copied 75 // texture and buffer. 76 TextureCopySubresource Compute2DTextureCopySubresource(Origin3D origin, 77 Extent3D copySize, 78 const TexelBlockInfo& blockInfo, 79 uint64_t offset, 80 uint32_t bytesPerRow); 81 82 TextureCopySplits Compute2DTextureCopySplits(Origin3D origin, 83 Extent3D copySize, 84 const TexelBlockInfo& blockInfo, 85 uint64_t offset, 86 uint32_t bytesPerRow, 87 uint32_t rowsPerImage); 88 89 TextureCopySubresource Compute3DTextureCopySplits(Origin3D origin, 90 Extent3D copySize, 91 const TexelBlockInfo& blockInfo, 92 uint64_t offset, 93 uint32_t bytesPerRow, 94 uint32_t rowsPerImage); 95 }} // namespace dawn_native::d3d12 96 97 #endif // DAWNNATIVE_D3D12_TEXTURECOPYSPLITTER_H_ 98