Flutter Impeller
blit_pass.cc
Go to the documentation of this file.
1 // Copyright 2013 The Flutter Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
6 #include <memory>
7 #include <utility>
8 
11 #include "impeller/core/formats.h"
12 
13 namespace impeller {
14 
16 
17 BlitPass::~BlitPass() = default;
18 
19 void BlitPass::SetLabel(std::string label) {
20  if (label.empty()) {
21  return;
22  }
23  OnSetLabel(std::move(label));
24 }
25 
26 bool BlitPass::AddCopy(std::shared_ptr<Texture> source,
27  std::shared_ptr<Texture> destination,
28  std::optional<IRect> source_region,
29  IPoint destination_origin,
30  std::string label) {
31  if (!source) {
32  VALIDATION_LOG << "Attempted to add a texture blit with no source.";
33  return false;
34  }
35  if (!destination) {
36  VALIDATION_LOG << "Attempted to add a texture blit with no destination.";
37  return false;
38  }
39 
40  if (source->GetTextureDescriptor().sample_count !=
41  destination->GetTextureDescriptor().sample_count) {
43  "The source sample count (%d) must match the destination sample count "
44  "(%d) for blits.",
45  static_cast<int>(source->GetTextureDescriptor().sample_count),
46  static_cast<int>(destination->GetTextureDescriptor().sample_count));
47  return false;
48  }
49  if (source->GetTextureDescriptor().format !=
50  destination->GetTextureDescriptor().format) {
52  "The source pixel format (%s) must match the destination pixel format "
53  "(%s) "
54  "for blits.",
55  PixelFormatToString(source->GetTextureDescriptor().format),
56  PixelFormatToString(destination->GetTextureDescriptor().format));
57  return false;
58  }
59 
60  if (!source_region.has_value()) {
61  source_region = IRect::MakeSize(source->GetSize());
62  }
63 
64  // Clip the source image.
65  source_region =
66  source_region->Intersection(IRect::MakeSize(source->GetSize()));
67  if (!source_region.has_value()) {
68  return true; // Nothing to blit.
69  }
70 
71  // Clip the destination image.
72  source_region = source_region->Intersection(
73  IRect::MakeOriginSize(-destination_origin, destination->GetSize()));
74  if (!source_region.has_value()) {
75  return true; // Nothing to blit.
76  }
77 
79  std::move(source), std::move(destination), source_region.value(),
80  destination_origin, std::move(label));
81 }
82 
83 bool BlitPass::AddCopy(std::shared_ptr<Texture> source,
84  std::shared_ptr<DeviceBuffer> destination,
85  std::optional<IRect> source_region,
86  size_t destination_offset,
87  std::string label) {
88  if (!source) {
89  VALIDATION_LOG << "Attempted to add a texture blit with no source.";
90  return false;
91  }
92  if (!destination) {
93  VALIDATION_LOG << "Attempted to add a texture blit with no destination.";
94  return false;
95  }
96 
97  if (!source_region.has_value()) {
98  source_region = IRect::MakeSize(source->GetSize());
99  }
100 
101  auto bytes_per_pixel =
102  BytesPerPixelForPixelFormat(source->GetTextureDescriptor().format);
103  auto bytes_per_image = source_region->Area() * bytes_per_pixel;
104  if (destination_offset + bytes_per_image >
105  destination->GetDeviceBufferDescriptor().size) {
107  << "Attempted to add a texture blit with out of bounds access.";
108  return false;
109  }
110 
111  // Clip the source image.
112  source_region =
113  source_region->Intersection(IRect::MakeSize(source->GetSize()));
114  if (!source_region.has_value()) {
115  return true; // Nothing to blit.
116  }
117 
118  return OnCopyTextureToBufferCommand(std::move(source), std::move(destination),
119  source_region.value(), destination_offset,
120  std::move(label));
121 }
122 
124  std::shared_ptr<Texture> destination,
125  std::optional<IRect> destination_region,
126  std::string label,
127  uint32_t slice,
128  bool convert_to_read) {
129  if (!destination) {
130  VALIDATION_LOG << "Attempted to add a texture blit with no destination.";
131  return false;
132  }
133  ISize destination_size = destination->GetSize();
134  IRect destination_region_value =
135  destination_region.value_or(IRect::MakeSize(destination_size));
136  if (destination_region_value.GetX() < 0 ||
137  destination_region_value.GetY() < 0 ||
138  destination_region_value.GetRight() > destination_size.width ||
139  destination_region_value.GetBottom() > destination_size.height) {
140  VALIDATION_LOG << "Blit region cannot be larger than destination texture.";
141  return false;
142  }
143 
144  auto bytes_per_pixel =
145  BytesPerPixelForPixelFormat(destination->GetTextureDescriptor().format);
146  auto bytes_per_region = destination_region_value.Area() * bytes_per_pixel;
147 
148  if (source.range.length != bytes_per_region) {
150  << "Attempted to add a texture blit with out of bounds access.";
151  return false;
152  }
153  if (slice > 5) {
154  VALIDATION_LOG << "Invalid value for slice: " << slice;
155  return false;
156  }
157 
158  return OnCopyBufferToTextureCommand(std::move(source), std::move(destination),
159  destination_region_value,
160  std::move(label), slice, convert_to_read);
161 }
162 
164  const std::shared_ptr<Texture>& texture) {
165  return true;
166 }
167 
168 bool BlitPass::GenerateMipmap(std::shared_ptr<Texture> texture,
169  std::string label) {
170  if (!texture) {
171  VALIDATION_LOG << "Attempted to add an invalid mipmap generation command "
172  "with no texture.";
173  return false;
174  }
175 
176  return OnGenerateMipmapCommand(std::move(texture), std::move(label));
177 }
178 
179 } // namespace impeller
impeller::BlitPass::OnCopyTextureToBufferCommand
virtual bool OnCopyTextureToBufferCommand(std::shared_ptr< Texture > source, std::shared_ptr< DeviceBuffer > destination, IRect source_region, size_t destination_offset, std::string label)=0
impeller::PixelFormatToString
constexpr const char * PixelFormatToString(PixelFormat format)
Definition: formats.h:140
formats.h
impeller::BlitPass::~BlitPass
virtual ~BlitPass()
impeller::BlitPass::AddCopy
bool AddCopy(std::shared_ptr< Texture > source, std::shared_ptr< Texture > destination, std::optional< IRect > source_region=std::nullopt, IPoint destination_origin={}, std::string label="")
Record a command to copy the contents of one texture to another texture. The blit area is limited by ...
Definition: blit_pass.cc:26
impeller::BlitPass::ConvertTextureToShaderRead
virtual bool ConvertTextureToShaderRead(const std::shared_ptr< Texture > &texture)
If the texture is not already in a shader read internal state, then convert it to that state.
Definition: blit_pass.cc:163
impeller::BlitPass::BlitPass
BlitPass()
Definition: blit_pass.cc:15
impeller::TRect::GetX
constexpr Type GetX() const
Returns the X coordinate of the upper left corner, equivalent to |GetOrigin().x|.
Definition: rect.h:323
impeller::BufferView::range
Range range
Definition: buffer_view.h:17
impeller::BlitPass::GenerateMipmap
bool GenerateMipmap(std::shared_ptr< Texture > texture, std::string label="")
Record a command to generate all mip levels for a texture.
Definition: blit_pass.cc:168
impeller::BlitPass::OnCopyTextureToTextureCommand
virtual bool OnCopyTextureToTextureCommand(std::shared_ptr< Texture > source, std::shared_ptr< Texture > destination, IRect source_region, IPoint destination_origin, std::string label)=0
validation.h
impeller::TSize
Definition: size.h:19
impeller::BytesPerPixelForPixelFormat
constexpr size_t BytesPerPixelForPixelFormat(PixelFormat format)
Definition: formats.h:460
impeller::SPrintF
std::string SPrintF(const char *format,...)
Definition: strings.cc:12
impeller::BlitPass::OnCopyBufferToTextureCommand
virtual bool OnCopyBufferToTextureCommand(BufferView source, std::shared_ptr< Texture > destination, IRect destination_region, std::string label, uint32_t slice, bool convert_to_read)=0
impeller::TRect::MakeOriginSize
constexpr static TRect MakeOriginSize(const TPoint< Type > &origin, const TSize< Type > &size)
Definition: rect.h:140
blit_pass.h
impeller::BlitPass::OnGenerateMipmapCommand
virtual bool OnGenerateMipmapCommand(std::shared_ptr< Texture > texture, std::string label)=0
impeller::BlitPass::OnSetLabel
virtual void OnSetLabel(std::string label)=0
strings.h
impeller::TSize::width
Type width
Definition: size.h:22
impeller::BlitPass::SetLabel
void SetLabel(std::string label)
Definition: blit_pass.cc:19
VALIDATION_LOG
#define VALIDATION_LOG
Definition: validation.h:73
impeller::BufferView
Definition: buffer_view.h:15
impeller::TRect::GetRight
constexpr auto GetRight() const
Definition: rect.h:345
impeller::TRect::MakeSize
constexpr static TRect MakeSize(const TSize< U > &size)
Definition: rect.h:146
impeller::TPoint< int64_t >
impeller::TRect::Area
constexpr T Area() const
Get the area of the rectangle, equivalent to |GetSize().Area()|.
Definition: rect.h:366
impeller::TRect::GetBottom
constexpr auto GetBottom() const
Definition: rect.h:347
impeller::TSize::height
Type height
Definition: size.h:23
impeller::Range::length
size_t length
Definition: range.h:14
impeller::TRect::GetY
constexpr Type GetY() const
Returns the Y coordinate of the upper left corner, equivalent to |GetOrigin().y|.
Definition: rect.h:327
impeller
Definition: aiks_blend_unittests.cc:18
impeller::TRect
Definition: rect.h:122