Upload upstream chromium 73.0.3683.0
[platform/framework/web/chromium-efl.git] / cc / tiles / decoded_image_tracker.cc
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
3 // found in the LICENSE file.
4
5 #include "cc/tiles/decoded_image_tracker.h"
6 #include "base/time/default_tick_clock.h"
7 #include "base/trace_event/trace_event.h"
8
9 namespace cc {
10 namespace {
11 // Timeout images after 250ms, whether or not they've been used. This prevents
12 // unbounded cache usage.
13 const int64_t kTimeoutDurationMs = 250;
14 }  // namespace
15
16 DecodedImageTracker::ImageLock::ImageLock(
17     DecodedImageTracker* tracker,
18     ImageController::ImageDecodeRequestId request_id,
19     base::TimeTicks lock_time)
20     : tracker_(tracker), request_id_(request_id), lock_time_(lock_time) {}
21
22 DecodedImageTracker::ImageLock::~ImageLock() {
23   tracker_->image_controller_->UnlockImageDecode(request_id_);
24 }
25
26 DecodedImageTracker::DecodedImageTracker(
27     ImageController* controller,
28     scoped_refptr<base::SequencedTaskRunner> task_runner)
29     : image_controller_(controller),
30       task_runner_(std::move(task_runner)),
31       tick_clock_(base::DefaultTickClock::GetInstance()),
32       weak_ptr_factory_(this) {
33   DCHECK(image_controller_);
34 }
35
36 DecodedImageTracker::~DecodedImageTracker() {
37   UnlockAllImages();
38 }
39
40 void DecodedImageTracker::QueueImageDecode(
41     const PaintImage& image,
42     base::OnceCallback<void(bool)> callback) {
43   size_t frame_index = PaintImage::kDefaultFrameIndex;
44   TRACE_EVENT1(TRACE_DISABLED_BY_DEFAULT("cc.debug"),
45                "DecodedImageTracker::QueueImageDecode", "frame_key",
46                image.GetKeyForFrame(frame_index).ToString());
47   DCHECK(image_controller_);
48   // Queue the decode in the image controller, but switch out the callback for
49   // our own.
50   auto image_bounds = SkIRect::MakeWH(image.width(), image.height());
51   DrawImage draw_image(image, image_bounds, kNone_SkFilterQuality,
52                        SkMatrix::I(), frame_index);
53   image_controller_->QueueImageDecode(
54       draw_image, base::BindOnce(&DecodedImageTracker::ImageDecodeFinished,
55                                  base::Unretained(this), std::move(callback),
56                                  image.stable_id()));
57 }
58
59 void DecodedImageTracker::UnlockAllImages() {
60   locked_images_.clear();
61 }
62
63 void DecodedImageTracker::OnImagesUsedInDraw(
64     const std::vector<DrawImage>& draw_images) {
65   for (const DrawImage& draw_image : draw_images)
66     locked_images_.erase(draw_image.paint_image().stable_id());
67 }
68
69 void DecodedImageTracker::ImageDecodeFinished(
70     base::OnceCallback<void(bool)> callback,
71     PaintImage::Id image_id,
72     ImageController::ImageDecodeRequestId request_id,
73     ImageController::ImageDecodeResult result) {
74   TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"),
75                "DecodedImageTracker::ImageDecodeFinished");
76
77   if (result == ImageController::ImageDecodeResult::SUCCESS) {
78     // If this image already exists, just replace it with the new (latest)
79     // decode.
80     locked_images_.erase(image_id);
81     locked_images_.emplace(
82         image_id,
83         std::make_unique<ImageLock>(this, request_id, tick_clock_->NowTicks()));
84     EnqueueTimeout();
85   }
86   bool decode_succeeded =
87       result == ImageController::ImageDecodeResult::SUCCESS ||
88       result == ImageController::ImageDecodeResult::DECODE_NOT_REQUIRED;
89   std::move(callback).Run(decode_succeeded);
90 }
91
92 void DecodedImageTracker::OnTimeoutImages() {
93   timeout_pending_ = false;
94   if (locked_images_.size() == 0)
95     return;
96
97   auto now = tick_clock_->NowTicks();
98   auto timeout = base::TimeDelta::FromMilliseconds(kTimeoutDurationMs);
99   for (auto it = locked_images_.begin(); it != locked_images_.end();) {
100     auto& image = it->second;
101     if (now - image->lock_time() < timeout) {
102       ++it;
103       continue;
104     }
105     it = locked_images_.erase(it);
106   }
107
108   EnqueueTimeout();
109 }
110
111 void DecodedImageTracker::EnqueueTimeout() {
112   if (timeout_pending_)
113     return;
114   if (locked_images_.size() == 0)
115     return;
116
117   timeout_pending_ = true;
118   task_runner_->PostDelayedTask(
119       FROM_HERE,
120       base::BindOnce(&DecodedImageTracker::OnTimeoutImages,
121                      weak_ptr_factory_.GetWeakPtr()),
122       base::TimeDelta::FromMilliseconds(kTimeoutDurationMs));
123 }
124
125 }  // namespace cc