[M108 Migration][VD] Avoid pending frame counter becoming negative
[platform/framework/web/chromium-efl.git] / cc / tiles / decoded_image_tracker.cc
1 // Copyright 2016 The Chromium Authors
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   DCHECK(image_controller_);
33 }
34
35 DecodedImageTracker::~DecodedImageTracker() {
36   UnlockAllImages();
37 }
38
39 void DecodedImageTracker::QueueImageDecode(
40     const PaintImage& image,
41     const TargetColorParams& target_color_params,
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, false, image_bounds,
52                        PaintFlags::FilterQuality::kNone, SkM44(), frame_index,
53                        target_color_params);
54   image_controller_->QueueImageDecode(
55       draw_image, base::BindOnce(&DecodedImageTracker::ImageDecodeFinished,
56                                  base::Unretained(this), std::move(callback),
57                                  image.stable_id()));
58 }
59
60 void DecodedImageTracker::UnlockAllImages() {
61   locked_images_.clear();
62 }
63
64 void DecodedImageTracker::OnImagesUsedInDraw(
65     const std::vector<DrawImage>& draw_images) {
66   for (const DrawImage& draw_image : draw_images)
67     locked_images_.erase(draw_image.paint_image().stable_id());
68 }
69
70 void DecodedImageTracker::ImageDecodeFinished(
71     base::OnceCallback<void(bool)> callback,
72     PaintImage::Id image_id,
73     ImageController::ImageDecodeRequestId request_id,
74     ImageController::ImageDecodeResult result) {
75   TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("cc.debug"),
76                "DecodedImageTracker::ImageDecodeFinished");
77
78   if (result == ImageController::ImageDecodeResult::SUCCESS) {
79     // If this image already exists, just replace it with the new (latest)
80     // decode.
81     locked_images_.erase(image_id);
82     locked_images_.emplace(
83         image_id,
84         std::make_unique<ImageLock>(this, request_id, tick_clock_->NowTicks()));
85     EnqueueTimeout();
86   }
87   bool decode_succeeded =
88       result == ImageController::ImageDecodeResult::SUCCESS ||
89       result == ImageController::ImageDecodeResult::DECODE_NOT_REQUIRED;
90   std::move(callback).Run(decode_succeeded);
91 }
92
93 void DecodedImageTracker::OnTimeoutImages() {
94   timeout_pending_ = false;
95   if (locked_images_.size() == 0)
96     return;
97
98   auto now = tick_clock_->NowTicks();
99   auto timeout = base::Milliseconds(kTimeoutDurationMs);
100   for (auto it = locked_images_.begin(); it != locked_images_.end();) {
101     auto& image = it->second;
102     if (now - image->lock_time() < timeout) {
103       ++it;
104       continue;
105     }
106     it = locked_images_.erase(it);
107   }
108
109   EnqueueTimeout();
110 }
111
112 void DecodedImageTracker::EnqueueTimeout() {
113   if (timeout_pending_)
114     return;
115   if (locked_images_.size() == 0)
116     return;
117
118   timeout_pending_ = true;
119   task_runner_->PostDelayedTask(
120       FROM_HERE,
121       base::BindOnce(&DecodedImageTracker::OnTimeoutImages,
122                      weak_ptr_factory_.GetWeakPtr()),
123       base::Milliseconds(kTimeoutDurationMs));
124 }
125
126 }  // namespace cc