OLD | NEW |
1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "cc/raster/image_hijack_canvas.h" | 5 #include "cc/raster/image_hijack_canvas.h" |
6 | 6 |
7 #include "base/optional.h" | 7 #include "base/optional.h" |
8 #include "base/trace_event/trace_event.h" | 8 #include "base/trace_event/trace_event.h" |
9 #include "cc/paint/discardable_image_map.h" | 9 #include "cc/paint/discardable_image_map.h" |
10 #include "cc/tiles/image_decode_cache.h" | 10 #include "cc/tiles/image_decode_cache.h" |
11 #include "third_party/skia/include/core/SkPath.h" | 11 #include "third_party/skia/include/core/SkPath.h" |
12 | 12 |
13 namespace cc { | 13 namespace cc { |
14 namespace { | 14 namespace { |
15 | 15 |
16 SkIRect RoundOutRect(const SkRect& rect) { | 16 SkIRect RoundOutRect(const SkRect& rect) { |
17 SkIRect result; | 17 SkIRect result; |
18 rect.roundOut(&result); | 18 rect.roundOut(&result); |
19 return result; | 19 return result; |
20 } | 20 } |
21 | 21 |
22 class ScopedDecodedImageLock { | 22 class ScopedDecodedImageLock { |
23 public: | 23 public: |
24 ScopedDecodedImageLock(ImageDecodeCache* image_decode_cache, | 24 ScopedDecodedImageLock(ImageDecodeCache* image_decode_cache, |
25 sk_sp<const SkImage> image, | 25 sk_sp<const SkImage> image, |
26 const SkRect& src_rect, | 26 const SkRect& src_rect, |
27 const SkMatrix& matrix, | 27 const SkMatrix& matrix, |
28 const SkPaint* paint) | 28 const SkPaint* paint, |
| 29 const gfx::ColorSpace& target_color_space) |
29 : image_decode_cache_(image_decode_cache), | 30 : image_decode_cache_(image_decode_cache), |
30 draw_image_(std::move(image), | 31 draw_image_(std::move(image), |
31 RoundOutRect(src_rect), | 32 RoundOutRect(src_rect), |
32 paint ? paint->getFilterQuality() : kNone_SkFilterQuality, | 33 paint ? paint->getFilterQuality() : kNone_SkFilterQuality, |
33 matrix), | 34 matrix, |
| 35 target_color_space), |
34 decoded_draw_image_( | 36 decoded_draw_image_( |
35 image_decode_cache_->GetDecodedImageForDraw(draw_image_)) { | 37 image_decode_cache_->GetDecodedImageForDraw(draw_image_)) { |
36 DCHECK(draw_image_.image()->isLazyGenerated()); | 38 DCHECK(draw_image_.image()->isLazyGenerated()); |
37 if (paint) { | 39 if (paint) { |
38 decoded_paint_ = *paint; | 40 decoded_paint_ = *paint; |
39 decoded_paint_->setFilterQuality(decoded_draw_image_.filter_quality()); | 41 decoded_paint_->setFilterQuality(decoded_draw_image_.filter_quality()); |
40 } | 42 } |
41 } | 43 } |
42 | 44 |
43 ScopedDecodedImageLock(ScopedDecodedImageLock&& from) | 45 ScopedDecodedImageLock(ScopedDecodedImageLock&& from) |
(...skipping 25 matching lines...) Expand all Loading... |
69 // Encapsulates a ScopedDecodedImageLock and an SkPaint. Use of this class | 71 // Encapsulates a ScopedDecodedImageLock and an SkPaint. Use of this class |
70 // ensures that the ScopedDecodedImageLock outlives the dependent SkPaint. | 72 // ensures that the ScopedDecodedImageLock outlives the dependent SkPaint. |
71 class ScopedImagePaint { | 73 class ScopedImagePaint { |
72 public: | 74 public: |
73 // Tries to create a ScopedImagePaint for the provided SkPaint. If a | 75 // Tries to create a ScopedImagePaint for the provided SkPaint. If a |
74 // the SkPaint does not contain an image that we support replacing, | 76 // the SkPaint does not contain an image that we support replacing, |
75 // an empty base::Optional will be returned. | 77 // an empty base::Optional will be returned. |
76 static base::Optional<ScopedImagePaint> TryCreate( | 78 static base::Optional<ScopedImagePaint> TryCreate( |
77 ImageDecodeCache* image_decode_cache, | 79 ImageDecodeCache* image_decode_cache, |
78 const SkMatrix& ctm, | 80 const SkMatrix& ctm, |
79 const SkPaint& paint) { | 81 const SkPaint& paint, |
| 82 const gfx::ColorSpace& target_color_space) { |
80 SkShader* shader = paint.getShader(); | 83 SkShader* shader = paint.getShader(); |
81 if (!shader) | 84 if (!shader) |
82 return base::Optional<ScopedImagePaint>(); | 85 return base::Optional<ScopedImagePaint>(); |
83 | 86 |
84 SkMatrix matrix; | 87 SkMatrix matrix; |
85 SkShader::TileMode xy[2]; | 88 SkShader::TileMode xy[2]; |
86 SkImage* image = shader->isAImage(&matrix, xy); | 89 SkImage* image = shader->isAImage(&matrix, xy); |
87 if (!image || !image->isLazyGenerated()) | 90 if (!image || !image->isLazyGenerated()) |
88 return base::Optional<ScopedImagePaint>(); | 91 return base::Optional<ScopedImagePaint>(); |
89 | 92 |
90 SkMatrix total_image_matrix = matrix; | 93 SkMatrix total_image_matrix = matrix; |
91 total_image_matrix.preConcat(ctm); | 94 total_image_matrix.preConcat(ctm); |
92 | 95 |
93 ScopedDecodedImageLock scoped_lock( | 96 ScopedDecodedImageLock scoped_lock( |
94 image_decode_cache, sk_ref_sp(image), | 97 image_decode_cache, sk_ref_sp(image), |
95 SkRect::MakeIWH(image->width(), image->height()), total_image_matrix, | 98 SkRect::MakeIWH(image->width(), image->height()), total_image_matrix, |
96 &paint); | 99 &paint, target_color_space); |
97 const DecodedDrawImage& decoded_image = scoped_lock.decoded_image(); | 100 const DecodedDrawImage& decoded_image = scoped_lock.decoded_image(); |
98 if (!decoded_image.image()) | 101 if (!decoded_image.image()) |
99 return base::Optional<ScopedImagePaint>(); | 102 return base::Optional<ScopedImagePaint>(); |
100 | 103 |
101 bool need_scale = !decoded_image.is_scale_adjustment_identity(); | 104 bool need_scale = !decoded_image.is_scale_adjustment_identity(); |
102 if (need_scale) { | 105 if (need_scale) { |
103 matrix.preScale(1.f / decoded_image.scale_adjustment().width(), | 106 matrix.preScale(1.f / decoded_image.scale_adjustment().width(), |
104 1.f / decoded_image.scale_adjustment().height()); | 107 1.f / decoded_image.scale_adjustment().height()); |
105 } | 108 } |
106 SkPaint scratch_paint = paint; | 109 SkPaint scratch_paint = paint; |
(...skipping 16 matching lines...) Expand all Loading... |
123 const SkImage* GetImageInPaint(const SkPaint& paint) { | 126 const SkImage* GetImageInPaint(const SkPaint& paint) { |
124 SkShader* shader = paint.getShader(); | 127 SkShader* shader = paint.getShader(); |
125 return shader ? shader->isAImage(nullptr, nullptr) : nullptr; | 128 return shader ? shader->isAImage(nullptr, nullptr) : nullptr; |
126 } | 129 } |
127 | 130 |
128 } // namespace | 131 } // namespace |
129 | 132 |
130 ImageHijackCanvas::ImageHijackCanvas(int width, | 133 ImageHijackCanvas::ImageHijackCanvas(int width, |
131 int height, | 134 int height, |
132 ImageDecodeCache* image_decode_cache, | 135 ImageDecodeCache* image_decode_cache, |
133 const ImageIdFlatSet* images_to_skip) | 136 const ImageIdFlatSet* images_to_skip, |
| 137 const gfx::ColorSpace& target_color_space) |
134 : SkNWayCanvas(width, height), | 138 : SkNWayCanvas(width, height), |
135 image_decode_cache_(image_decode_cache), | 139 image_decode_cache_(image_decode_cache), |
136 images_to_skip_(images_to_skip) {} | 140 images_to_skip_(images_to_skip), |
| 141 target_color_space_(target_color_space) {} |
137 | 142 |
138 void ImageHijackCanvas::onDrawPicture(const SkPicture* picture, | 143 void ImageHijackCanvas::onDrawPicture(const SkPicture* picture, |
139 const SkMatrix* matrix, | 144 const SkMatrix* matrix, |
140 const SkPaint* paint) { | 145 const SkPaint* paint) { |
141 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), | 146 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), |
142 "ImageHijackCanvas::onDrawPicture"); | 147 "ImageHijackCanvas::onDrawPicture"); |
143 // Ensure that pictures are unpacked by this canvas, instead of being | 148 // Ensure that pictures are unpacked by this canvas, instead of being |
144 // forwarded to the raster canvas. | 149 // forwarded to the raster canvas. |
145 SkCanvas::onDrawPicture(picture, matrix, paint); | 150 SkCanvas::onDrawPicture(picture, matrix, paint); |
146 } | 151 } |
(...skipping 10 matching lines...) Expand all Loading... |
157 return; | 162 return; |
158 } | 163 } |
159 | 164 |
160 if (ShouldSkipImage(image)) | 165 if (ShouldSkipImage(image)) |
161 return; | 166 return; |
162 | 167 |
163 SkMatrix ctm = getTotalMatrix(); | 168 SkMatrix ctm = getTotalMatrix(); |
164 | 169 |
165 ScopedDecodedImageLock scoped_lock( | 170 ScopedDecodedImageLock scoped_lock( |
166 image_decode_cache_, sk_ref_sp(image), | 171 image_decode_cache_, sk_ref_sp(image), |
167 SkRect::MakeIWH(image->width(), image->height()), ctm, paint); | 172 SkRect::MakeIWH(image->width(), image->height()), ctm, paint, |
| 173 target_color_space_); |
168 const DecodedDrawImage& decoded_image = scoped_lock.decoded_image(); | 174 const DecodedDrawImage& decoded_image = scoped_lock.decoded_image(); |
169 if (!decoded_image.image()) | 175 if (!decoded_image.image()) |
170 return; | 176 return; |
171 | 177 |
172 DCHECK_EQ(0, static_cast<int>(decoded_image.src_rect_offset().width())); | 178 DCHECK_EQ(0, static_cast<int>(decoded_image.src_rect_offset().width())); |
173 DCHECK_EQ(0, static_cast<int>(decoded_image.src_rect_offset().height())); | 179 DCHECK_EQ(0, static_cast<int>(decoded_image.src_rect_offset().height())); |
174 const SkPaint* decoded_paint = scoped_lock.decoded_paint(); | 180 const SkPaint* decoded_paint = scoped_lock.decoded_paint(); |
175 | 181 |
176 bool need_scale = !decoded_image.is_scale_adjustment_identity(); | 182 bool need_scale = !decoded_image.is_scale_adjustment_identity(); |
177 if (need_scale) { | 183 if (need_scale) { |
(...skipping 25 matching lines...) Expand all Loading... |
203 SkRect src_storage; | 209 SkRect src_storage; |
204 if (!src) { | 210 if (!src) { |
205 src_storage = SkRect::MakeIWH(image->width(), image->height()); | 211 src_storage = SkRect::MakeIWH(image->width(), image->height()); |
206 src = &src_storage; | 212 src = &src_storage; |
207 } | 213 } |
208 SkMatrix matrix; | 214 SkMatrix matrix; |
209 matrix.setRectToRect(*src, dst, SkMatrix::kFill_ScaleToFit); | 215 matrix.setRectToRect(*src, dst, SkMatrix::kFill_ScaleToFit); |
210 matrix.postConcat(getTotalMatrix()); | 216 matrix.postConcat(getTotalMatrix()); |
211 | 217 |
212 ScopedDecodedImageLock scoped_lock(image_decode_cache_, sk_ref_sp(image), | 218 ScopedDecodedImageLock scoped_lock(image_decode_cache_, sk_ref_sp(image), |
213 *src, matrix, paint); | 219 *src, matrix, paint, target_color_space_); |
214 const DecodedDrawImage& decoded_image = scoped_lock.decoded_image(); | 220 const DecodedDrawImage& decoded_image = scoped_lock.decoded_image(); |
215 if (!decoded_image.image()) | 221 if (!decoded_image.image()) |
216 return; | 222 return; |
217 | 223 |
218 const SkPaint* decoded_paint = scoped_lock.decoded_paint(); | 224 const SkPaint* decoded_paint = scoped_lock.decoded_paint(); |
219 | 225 |
220 SkRect adjusted_src = | 226 SkRect adjusted_src = |
221 src->makeOffset(decoded_image.src_rect_offset().width(), | 227 src->makeOffset(decoded_image.src_rect_offset().width(), |
222 decoded_image.src_rect_offset().height()); | 228 decoded_image.src_rect_offset().height()); |
223 if (!decoded_image.is_scale_adjustment_identity()) { | 229 if (!decoded_image.is_scale_adjustment_identity()) { |
224 float x_scale = decoded_image.scale_adjustment().width(); | 230 float x_scale = decoded_image.scale_adjustment().width(); |
225 float y_scale = decoded_image.scale_adjustment().height(); | 231 float y_scale = decoded_image.scale_adjustment().height(); |
226 adjusted_src = SkRect::MakeXYWH( | 232 adjusted_src = SkRect::MakeXYWH( |
227 adjusted_src.x() * x_scale, adjusted_src.y() * y_scale, | 233 adjusted_src.x() * x_scale, adjusted_src.y() * y_scale, |
228 adjusted_src.width() * x_scale, adjusted_src.height() * y_scale); | 234 adjusted_src.width() * x_scale, adjusted_src.height() * y_scale); |
229 } | 235 } |
230 SkNWayCanvas::onDrawImageRect(decoded_image.image().get(), &adjusted_src, dst, | 236 SkNWayCanvas::onDrawImageRect(decoded_image.image().get(), &adjusted_src, dst, |
231 decoded_paint, constraint); | 237 decoded_paint, constraint); |
232 } | 238 } |
233 | 239 |
234 void ImageHijackCanvas::onDrawRect(const SkRect& r, const SkPaint& paint) { | 240 void ImageHijackCanvas::onDrawRect(const SkRect& r, const SkPaint& paint) { |
235 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), | 241 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), |
236 "ImageHijackCanvas::onDrawRect"); | 242 "ImageHijackCanvas::onDrawRect"); |
237 if (ShouldSkipImageInPaint(paint)) | 243 if (ShouldSkipImageInPaint(paint)) |
238 return; | 244 return; |
239 | 245 |
240 base::Optional<ScopedImagePaint> image_paint = | 246 base::Optional<ScopedImagePaint> image_paint = ScopedImagePaint::TryCreate( |
241 ScopedImagePaint::TryCreate(image_decode_cache_, getTotalMatrix(), paint); | 247 image_decode_cache_, getTotalMatrix(), paint, target_color_space_); |
242 if (!image_paint.has_value()) { | 248 if (!image_paint.has_value()) { |
243 SkNWayCanvas::onDrawRect(r, paint); | 249 SkNWayCanvas::onDrawRect(r, paint); |
244 return; | 250 return; |
245 } | 251 } |
246 SkNWayCanvas::onDrawRect(r, image_paint.value().paint()); | 252 SkNWayCanvas::onDrawRect(r, image_paint.value().paint()); |
247 } | 253 } |
248 | 254 |
249 void ImageHijackCanvas::onDrawPath(const SkPath& path, const SkPaint& paint) { | 255 void ImageHijackCanvas::onDrawPath(const SkPath& path, const SkPaint& paint) { |
250 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), | 256 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), |
251 "ImageHijackCanvas::onDrawPath"); | 257 "ImageHijackCanvas::onDrawPath"); |
252 if (ShouldSkipImageInPaint(paint)) | 258 if (ShouldSkipImageInPaint(paint)) |
253 return; | 259 return; |
254 | 260 |
255 base::Optional<ScopedImagePaint> image_paint = | 261 base::Optional<ScopedImagePaint> image_paint = ScopedImagePaint::TryCreate( |
256 ScopedImagePaint::TryCreate(image_decode_cache_, getTotalMatrix(), paint); | 262 image_decode_cache_, getTotalMatrix(), paint, target_color_space_); |
257 if (!image_paint.has_value()) { | 263 if (!image_paint.has_value()) { |
258 SkNWayCanvas::onDrawPath(path, paint); | 264 SkNWayCanvas::onDrawPath(path, paint); |
259 return; | 265 return; |
260 } | 266 } |
261 SkNWayCanvas::onDrawPath(path, image_paint.value().paint()); | 267 SkNWayCanvas::onDrawPath(path, image_paint.value().paint()); |
262 } | 268 } |
263 | 269 |
264 void ImageHijackCanvas::onDrawOval(const SkRect& r, const SkPaint& paint) { | 270 void ImageHijackCanvas::onDrawOval(const SkRect& r, const SkPaint& paint) { |
265 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), | 271 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), |
266 "ImageHijackCanvas::onDrawOval"); | 272 "ImageHijackCanvas::onDrawOval"); |
267 if (ShouldSkipImageInPaint(paint)) | 273 if (ShouldSkipImageInPaint(paint)) |
268 return; | 274 return; |
269 | 275 |
270 base::Optional<ScopedImagePaint> image_paint = | 276 base::Optional<ScopedImagePaint> image_paint = ScopedImagePaint::TryCreate( |
271 ScopedImagePaint::TryCreate(image_decode_cache_, getTotalMatrix(), paint); | 277 image_decode_cache_, getTotalMatrix(), paint, target_color_space_); |
272 if (!image_paint.has_value()) { | 278 if (!image_paint.has_value()) { |
273 SkNWayCanvas::onDrawOval(r, paint); | 279 SkNWayCanvas::onDrawOval(r, paint); |
274 return; | 280 return; |
275 } | 281 } |
276 SkNWayCanvas::onDrawOval(r, image_paint.value().paint()); | 282 SkNWayCanvas::onDrawOval(r, image_paint.value().paint()); |
277 } | 283 } |
278 | 284 |
279 void ImageHijackCanvas::onDrawArc(const SkRect& r, | 285 void ImageHijackCanvas::onDrawArc(const SkRect& r, |
280 SkScalar start_angle, | 286 SkScalar start_angle, |
281 SkScalar sweep_angle, | 287 SkScalar sweep_angle, |
282 bool use_center, | 288 bool use_center, |
283 const SkPaint& paint) { | 289 const SkPaint& paint) { |
284 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), | 290 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), |
285 "ImageHijackCanvas::onDrawArc"); | 291 "ImageHijackCanvas::onDrawArc"); |
286 if (ShouldSkipImageInPaint(paint)) | 292 if (ShouldSkipImageInPaint(paint)) |
287 return; | 293 return; |
288 | 294 |
289 base::Optional<ScopedImagePaint> image_paint = | 295 base::Optional<ScopedImagePaint> image_paint = ScopedImagePaint::TryCreate( |
290 ScopedImagePaint::TryCreate(image_decode_cache_, getTotalMatrix(), paint); | 296 image_decode_cache_, getTotalMatrix(), paint, target_color_space_); |
291 if (!image_paint.has_value()) { | 297 if (!image_paint.has_value()) { |
292 SkNWayCanvas::onDrawArc(r, start_angle, sweep_angle, use_center, paint); | 298 SkNWayCanvas::onDrawArc(r, start_angle, sweep_angle, use_center, paint); |
293 return; | 299 return; |
294 } | 300 } |
295 SkNWayCanvas::onDrawArc(r, start_angle, sweep_angle, use_center, | 301 SkNWayCanvas::onDrawArc(r, start_angle, sweep_angle, use_center, |
296 image_paint.value().paint()); | 302 image_paint.value().paint()); |
297 } | 303 } |
298 | 304 |
299 void ImageHijackCanvas::onDrawRRect(const SkRRect& rr, const SkPaint& paint) { | 305 void ImageHijackCanvas::onDrawRRect(const SkRRect& rr, const SkPaint& paint) { |
300 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), | 306 TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"), |
301 "ImageHijackCanvas::onDrawRRect"); | 307 "ImageHijackCanvas::onDrawRRect"); |
302 if (ShouldSkipImageInPaint(paint)) | 308 if (ShouldSkipImageInPaint(paint)) |
303 return; | 309 return; |
304 | 310 |
305 base::Optional<ScopedImagePaint> image_paint = | 311 base::Optional<ScopedImagePaint> image_paint = ScopedImagePaint::TryCreate( |
306 ScopedImagePaint::TryCreate(image_decode_cache_, getTotalMatrix(), paint); | 312 image_decode_cache_, getTotalMatrix(), paint, target_color_space_); |
307 if (!image_paint.has_value()) { | 313 if (!image_paint.has_value()) { |
308 SkNWayCanvas::onDrawRRect(rr, paint); | 314 SkNWayCanvas::onDrawRRect(rr, paint); |
309 return; | 315 return; |
310 } | 316 } |
311 SkNWayCanvas::onDrawRRect(rr, image_paint.value().paint()); | 317 SkNWayCanvas::onDrawRRect(rr, image_paint.value().paint()); |
312 } | 318 } |
313 | 319 |
314 void ImageHijackCanvas::onDrawImageNine(const SkImage* image, | 320 void ImageHijackCanvas::onDrawImageNine(const SkImage* image, |
315 const SkIRect& center, | 321 const SkIRect& center, |
316 const SkRect& dst, | 322 const SkRect& dst, |
(...skipping 10 matching lines...) Expand all Loading... |
327 image->uniqueID(), "skip", skip); | 333 image->uniqueID(), "skip", skip); |
328 return skip; | 334 return skip; |
329 } | 335 } |
330 | 336 |
331 bool ImageHijackCanvas::ShouldSkipImageInPaint(const SkPaint& paint) const { | 337 bool ImageHijackCanvas::ShouldSkipImageInPaint(const SkPaint& paint) const { |
332 const SkImage* image = GetImageInPaint(paint); | 338 const SkImage* image = GetImageInPaint(paint); |
333 return image ? ShouldSkipImage(image) : false; | 339 return image ? ShouldSkipImage(image) : false; |
334 } | 340 } |
335 | 341 |
336 } // namespace cc | 342 } // namespace cc |
OLD | NEW |