OLD | NEW |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 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 <stdint.h> | 5 #include <stdint.h> |
6 | 6 |
7 #include "base/macros.h" | 7 #include "base/macros.h" |
8 #include "base/run_loop.h" | 8 #include "base/run_loop.h" |
9 #include "base/threading/thread_task_runner_handle.h" | 9 #include "base/threading/thread_task_runner_handle.h" |
10 #include "cc/output/begin_frame_args.h" | 10 #include "cc/output/begin_frame_args.h" |
(...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
55 base::SingleThreadTaskRunner* task_runner() { return task_runner_.get(); } | 55 base::SingleThreadTaskRunner* task_runner() { return task_runner_.get(); } |
56 ui::Compositor* compositor() { return compositor_.get(); } | 56 ui::Compositor* compositor() { return compositor_.get(); } |
57 | 57 |
58 private: | 58 private: |
59 scoped_refptr<base::SingleThreadTaskRunner> task_runner_; | 59 scoped_refptr<base::SingleThreadTaskRunner> task_runner_; |
60 std::unique_ptr<ui::Compositor> compositor_; | 60 std::unique_ptr<ui::Compositor> compositor_; |
61 | 61 |
62 DISALLOW_COPY_AND_ASSIGN(CompositorTest); | 62 DISALLOW_COPY_AND_ASSIGN(CompositorTest); |
63 }; | 63 }; |
64 | 64 |
| 65 class MockCompositorLockClient |
| 66 : NON_EXPORTED_BASE(public ui::CompositorLockClient) { |
| 67 public: |
| 68 MOCK_METHOD0(CompositorLockTimedOut, void()); |
| 69 }; |
| 70 |
65 } // namespace | 71 } // namespace |
66 | 72 |
67 TEST_F(CompositorTest, LocksTimeOut) { | 73 TEST_F(CompositorTest, LocksTimeOut) { |
68 scoped_refptr<ui::CompositorLock> lock; | 74 std::unique_ptr<CompositorLock> lock; |
| 75 |
| 76 base::TimeDelta timeout = base::TimeDelta::FromMilliseconds(100); |
| 77 |
69 { | 78 { |
| 79 testing::StrictMock<MockCompositorLockClient> lock_client; |
70 base::RunLoop run_loop; | 80 base::RunLoop run_loop; |
71 // Ensure that the lock times out by default. | 81 // This lock has a timeout. |
72 lock = compositor()->GetCompositorLock(); | 82 lock = compositor()->GetCompositorLock(&lock_client, timeout); |
73 EXPECT_TRUE(compositor()->IsLocked()); | 83 EXPECT_TRUE(compositor()->IsLocked()); |
74 task_runner()->PostDelayedTask( | 84 task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(), timeout); |
75 FROM_HERE, run_loop.QuitClosure(), | 85 EXPECT_CALL(lock_client, CompositorLockTimedOut()).Times(1); |
76 base::TimeDelta::FromMilliseconds(kCompositorLockTimeoutMs)); | |
77 run_loop.Run(); | 86 run_loop.Run(); |
78 EXPECT_FALSE(compositor()->IsLocked()); | 87 EXPECT_FALSE(compositor()->IsLocked()); |
79 } | 88 } |
80 | 89 |
81 { | 90 { |
| 91 testing::StrictMock<MockCompositorLockClient> lock_client; |
82 base::RunLoop run_loop; | 92 base::RunLoop run_loop; |
83 // Ensure that the lock does not time out when set. | 93 // This lock has no timeout. |
84 compositor()->SetLocksWillTimeOut(false); | 94 lock = compositor()->GetCompositorLock(&lock_client, base::TimeDelta()); |
85 lock = compositor()->GetCompositorLock(); | |
86 EXPECT_TRUE(compositor()->IsLocked()); | 95 EXPECT_TRUE(compositor()->IsLocked()); |
87 task_runner()->PostDelayedTask( | 96 task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(), timeout); |
88 FROM_HERE, run_loop.QuitClosure(), | 97 EXPECT_CALL(lock_client, CompositorLockTimedOut()).Times(0); |
89 base::TimeDelta::FromMilliseconds(kCompositorLockTimeoutMs)); | |
90 run_loop.Run(); | 98 run_loop.Run(); |
91 EXPECT_TRUE(compositor()->IsLocked()); | 99 EXPECT_TRUE(compositor()->IsLocked()); |
92 } | 100 } |
93 } | 101 } |
94 | 102 |
| 103 TEST_F(CompositorTest, MultipleLockClients) { |
| 104 testing::StrictMock<MockCompositorLockClient> lock_client1; |
| 105 std::unique_ptr<CompositorLock> lock1; |
| 106 testing::StrictMock<MockCompositorLockClient> lock_client2; |
| 107 std::unique_ptr<CompositorLock> lock2; |
| 108 |
| 109 base::TimeDelta timeout = base::TimeDelta::FromMilliseconds(1); |
| 110 base::RunLoop run_loop; |
| 111 // Both locks are grabbed from the Compositor with a separate client. |
| 112 lock1 = compositor()->GetCompositorLock(&lock_client1, timeout); |
| 113 lock2 = compositor()->GetCompositorLock(&lock_client2, timeout); |
| 114 EXPECT_TRUE(compositor()->IsLocked()); |
| 115 task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(), timeout); |
| 116 // Both clients get notified of timeout. |
| 117 EXPECT_CALL(lock_client1, CompositorLockTimedOut()).Times(1); |
| 118 EXPECT_CALL(lock_client2, CompositorLockTimedOut()).Times(1); |
| 119 run_loop.Run(); |
| 120 EXPECT_FALSE(compositor()->IsLocked()); |
| 121 } |
| 122 |
| 123 TEST_F(CompositorTest, ExtendingLifeOfLockDoesntUseDeadClient) { |
| 124 testing::StrictMock<MockCompositorLockClient> lock_client1; |
| 125 std::unique_ptr<CompositorLock> lock1; |
| 126 testing::StrictMock<MockCompositorLockClient> lock_client2; |
| 127 std::unique_ptr<CompositorLock> lock2; |
| 128 |
| 129 base::TimeDelta timeout = base::TimeDelta::FromMilliseconds(1); |
| 130 base::RunLoop run_loop; |
| 131 |
| 132 // One lock is grabbed from the compositor with a client. The other |
| 133 // extends its lifetime past that of the first. |
| 134 lock1 = compositor()->GetCompositorLock(&lock_client1, timeout); |
| 135 EXPECT_TRUE(compositor()->IsLocked()); |
| 136 |
| 137 // This also locks the compositor and will do so past |lock1| ending. |
| 138 lock2 = compositor()->GetCompositorLock(&lock_client2, timeout); |
| 139 // |lock1| is destroyed, so it won't timeout but |lock2| will. |
| 140 lock1 = nullptr; |
| 141 |
| 142 task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(), timeout); |
| 143 |
| 144 EXPECT_CALL(lock_client1, CompositorLockTimedOut()).Times(0); |
| 145 EXPECT_CALL(lock_client2, CompositorLockTimedOut()).Times(1); |
| 146 run_loop.Run(); |
| 147 EXPECT_FALSE(compositor()->IsLocked()); |
| 148 } |
| 149 |
| 150 TEST_F(CompositorTest, AddingLocksDoesNotExtendTimeout) { |
| 151 testing::StrictMock<MockCompositorLockClient> lock_client1; |
| 152 std::unique_ptr<CompositorLock> lock1; |
| 153 testing::StrictMock<MockCompositorLockClient> lock_client2; |
| 154 std::unique_ptr<CompositorLock> lock2; |
| 155 |
| 156 base::TimeDelta timeout1 = base::TimeDelta::FromMilliseconds(1); |
| 157 base::TimeDelta timeout2 = base::TimeDelta::FromMilliseconds(10); |
| 158 base::RunLoop run_loop; |
| 159 |
| 160 // The first lock has a short timeout. |
| 161 lock1 = compositor()->GetCompositorLock(&lock_client1, timeout1); |
| 162 EXPECT_TRUE(compositor()->IsLocked()); |
| 163 |
| 164 // The second lock has a longer timeout, but since a lock is active, |
| 165 // the first one is used for both. |
| 166 lock2 = compositor()->GetCompositorLock(&lock_client2, timeout2); |
| 167 |
| 168 task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(), timeout1); |
| 169 |
| 170 EXPECT_CALL(lock_client1, CompositorLockTimedOut()).Times(1); |
| 171 EXPECT_CALL(lock_client2, CompositorLockTimedOut()).Times(1); |
| 172 run_loop.Run(); |
| 173 EXPECT_FALSE(compositor()->IsLocked()); |
| 174 } |
| 175 |
| 176 TEST_F(CompositorTest, LockIsDestroyedDoesntTimeout) { |
| 177 base::TimeDelta timeout = base::TimeDelta::FromMilliseconds(1); |
| 178 base::RunLoop run_loop; |
| 179 |
| 180 testing::StrictMock<MockCompositorLockClient> lock_client1; |
| 181 std::unique_ptr<CompositorLock> lock1; |
| 182 lock1 = compositor()->GetCompositorLock(&lock_client1, timeout); |
| 183 EXPECT_TRUE(compositor()->IsLocked()); |
| 184 task_runner()->PostDelayedTask(FROM_HERE, run_loop.QuitClosure(), timeout); |
| 185 // The CompositorLockClient is destroyed when |lock1| is released. |
| 186 lock1 = nullptr; |
| 187 // The client isn't called as a result. |
| 188 EXPECT_CALL(lock_client1, CompositorLockTimedOut()).Times(0); |
| 189 run_loop.Run(); |
| 190 EXPECT_FALSE(compositor()->IsLocked()); |
| 191 } |
| 192 |
95 TEST_F(CompositorTest, ReleaseWidgetWithOutputSurfaceNeverCreated) { | 193 TEST_F(CompositorTest, ReleaseWidgetWithOutputSurfaceNeverCreated) { |
96 compositor()->SetVisible(false); | 194 compositor()->SetVisible(false); |
97 EXPECT_EQ(gfx::kNullAcceleratedWidget, | 195 EXPECT_EQ(gfx::kNullAcceleratedWidget, |
98 compositor()->ReleaseAcceleratedWidget()); | 196 compositor()->ReleaseAcceleratedWidget()); |
99 compositor()->SetAcceleratedWidget(gfx::kNullAcceleratedWidget); | 197 compositor()->SetAcceleratedWidget(gfx::kNullAcceleratedWidget); |
100 compositor()->SetVisible(true); | 198 compositor()->SetVisible(true); |
101 } | 199 } |
102 | 200 |
103 #if defined(OS_WIN) | 201 #if defined(OS_WIN) |
104 // TODO(crbug.com/608436): Flaky on windows trybots | 202 // TODO(crbug.com/608436): Flaky on windows trybots |
(...skipping 14 matching lines...) Expand all Loading... |
119 EXPECT_EQ(gfx::kNullAcceleratedWidget, | 217 EXPECT_EQ(gfx::kNullAcceleratedWidget, |
120 compositor()->ReleaseAcceleratedWidget()); | 218 compositor()->ReleaseAcceleratedWidget()); |
121 compositor()->SetAcceleratedWidget(gfx::kNullAcceleratedWidget); | 219 compositor()->SetAcceleratedWidget(gfx::kNullAcceleratedWidget); |
122 compositor()->SetVisible(true); | 220 compositor()->SetVisible(true); |
123 compositor()->ScheduleDraw(); | 221 compositor()->ScheduleDraw(); |
124 DrawWaiterForTest::WaitForCompositingEnded(compositor()); | 222 DrawWaiterForTest::WaitForCompositingEnded(compositor()); |
125 compositor()->SetRootLayer(nullptr); | 223 compositor()->SetRootLayer(nullptr); |
126 } | 224 } |
127 | 225 |
128 } // namespace ui | 226 } // namespace ui |
OLD | NEW |