blob: 06d12128aca6b25a001bab4ff3b26f851ef64827 [file] [log] [blame]
dnicoara9372a7912014-12-11 01:29:061// Copyright 2014 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
Joone Hurd3ae8732018-04-17 18:05:095#include "ui/display/manager/configure_displays_task.h"
dnicoara9372a7912014-12-11 01:29:066
7#include "base/auto_reset.h"
8#include "base/bind.h"
Brett Wilsonb02c0a22017-09-25 22:34:429#include "base/containers/queue.h"
Joone Hurd3ae8732018-04-17 18:05:0910#include "ui/display/manager/display_util.h"
dnicoara9372a7912014-12-11 01:29:0611#include "ui/display/types/display_snapshot.h"
12#include "ui/display/types/native_display_delegate.h"
13
kylechar7a067ec2017-01-07 01:16:2814namespace display {
dnicoara9372a7912014-12-11 01:29:0615
16namespace {
17
18// Find the next best mode after |display_mode|. If none can be found return
19// nullptr.
20const DisplayMode* FindNextMode(const DisplaySnapshot& display_state,
dnicoaraa89c2082015-01-05 16:49:0621 const DisplayMode* display_mode) {
22 if (!display_mode)
23 return nullptr;
24
dnicoara9372a7912014-12-11 01:29:0625 int best_mode_pixels = 0;
26 const DisplayMode* best_mode = nullptr;
dnicoaraa89c2082015-01-05 16:49:0627 int current_mode_pixels = display_mode->size().GetArea();
dbasehore01e90042016-05-27 06:16:5128 for (const std::unique_ptr<const DisplayMode>& mode : display_state.modes()) {
dnicoara9372a7912014-12-11 01:29:0629 int pixel_count = mode->size().GetArea();
30 if (pixel_count < current_mode_pixels && pixel_count > best_mode_pixels) {
dbasehore01e90042016-05-27 06:16:5131 best_mode = mode.get();
dnicoara9372a7912014-12-11 01:29:0632 best_mode_pixels = pixel_count;
33 }
34 }
35
36 return best_mode;
37}
38
39} // namespace
40
41DisplayConfigureRequest::DisplayConfigureRequest(DisplaySnapshot* display,
42 const DisplayMode* mode,
43 const gfx::Point& origin)
kylechar731f85f92016-12-01 20:50:4644 : display(display), mode(mode), origin(origin) {}
dnicoara9372a7912014-12-11 01:29:0645
46ConfigureDisplaysTask::ConfigureDisplaysTask(
47 NativeDisplayDelegate* delegate,
48 const std::vector<DisplayConfigureRequest>& requests,
Sylvain Defresne23395c7a2019-10-02 10:07:4549 ResponseCallback callback)
dnicoara9372a7912014-12-11 01:29:0650 : delegate_(delegate),
51 requests_(requests),
Sylvain Defresne23395c7a2019-10-02 10:07:4552 callback_(std::move(callback)),
dnicoara9372a7912014-12-11 01:29:0653 is_configuring_(false),
54 num_displays_configured_(0),
Jeremy Roman47d432e2019-08-20 14:24:0055 task_status_(SUCCESS) {
dnicoara9372a7912014-12-11 01:29:0656 for (size_t i = 0; i < requests_.size(); ++i)
57 pending_request_indexes_.push(i);
afakhry4e92e8c2017-04-20 17:04:5958 delegate_->AddObserver(this);
dnicoara9372a7912014-12-11 01:29:0659}
60
afakhry4e92e8c2017-04-20 17:04:5961ConfigureDisplaysTask::~ConfigureDisplaysTask() {
62 delegate_->RemoveObserver(this);
63}
dnicoara9372a7912014-12-11 01:29:0664
65void ConfigureDisplaysTask::Run() {
66 // Synchronous configurators will recursively call Run(). In that case just
67 // defer their call to the next iteration in the while-loop. This is done to
68 // guard against stack overflows if the display has a large list of broken
69 // modes.
70 if (is_configuring_)
71 return;
72
dnicoara65d9d312014-12-13 02:30:4773 {
74 base::AutoReset<bool> recursivity_guard(&is_configuring_, true);
75 while (!pending_request_indexes_.empty()) {
76 size_t index = pending_request_indexes_.front();
77 DisplayConfigureRequest* request = &requests_[index];
78 pending_request_indexes_.pop();
Sylvain Defresne23395c7a2019-10-02 10:07:4579 delegate_->Configure(
80 *request->display, request->mode, request->origin,
81 base::BindOnce(&ConfigureDisplaysTask::OnConfigured,
82 weak_ptr_factory_.GetWeakPtr(), index));
dnicoara65d9d312014-12-13 02:30:4783 }
dnicoara9372a7912014-12-11 01:29:0684 }
85
dnicoara65d9d312014-12-13 02:30:4786 // Nothing should be modified after the |callback_| is called since the
dnicoara9372a7912014-12-11 01:29:0687 // task may be deleted in the callback.
88 if (num_displays_configured_ == requests_.size())
Sylvain Defresne23395c7a2019-10-02 10:07:4589 std::move(callback_).Run(task_status_);
dnicoara9372a7912014-12-11 01:29:0690}
91
afakhry4e92e8c2017-04-20 17:04:5992void ConfigureDisplaysTask::OnConfigurationChanged() {}
93
94void ConfigureDisplaysTask::OnDisplaySnapshotsInvalidated() {
Brett Wilsonb02c0a22017-09-25 22:34:4295 base::queue<size_t> empty_queue;
afakhry4e92e8c2017-04-20 17:04:5996 pending_request_indexes_.swap(empty_queue);
97 // From now on, don't access |requests_[index]->display|; they're invalid.
98 task_status_ = ERROR;
99 weak_ptr_factory_.InvalidateWeakPtrs();
100 Run();
101}
102
dnicoara9372a7912014-12-11 01:29:06103void ConfigureDisplaysTask::OnConfigured(size_t index, bool success) {
104 DisplayConfigureRequest* request = &requests_[index];
105 VLOG(2) << "Configured status=" << success
106 << " display=" << request->display->display_id()
107 << " origin=" << request->origin.ToString()
108 << " mode=" << (request->mode ? request->mode->ToString() : "null");
109 if (!success) {
dnicoaraa89c2082015-01-05 16:49:06110 request->mode = FindNextMode(*request->display, request->mode);
dnicoara9372a7912014-12-11 01:29:06111 if (request->mode) {
112 pending_request_indexes_.push(index);
113 if (task_status_ == SUCCESS)
114 task_status_ = PARTIAL_SUCCESS;
115
116 Run();
117 return;
118 }
119 } else {
120 request->display->set_current_mode(request->mode);
121 request->display->set_origin(request->origin);
122 }
123
124 num_displays_configured_++;
125 if (!success)
126 task_status_ = ERROR;
127
128 Run();
129}
130
kylechar7a067ec2017-01-07 01:16:28131} // namespace display