blob: 69b9590d40f61d7c899646be16d03f824f89cbec [file] [log] [blame]
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -07001/*
2 * Copyright 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "BufferHubBufferTest"
18
Fan Xuca70b7b2018-10-31 13:20:12 -070019#include <android/frameworks/bufferhub/1.0/IBufferClient.h>
Jiwen 'Steve' Caid9f2abe2018-10-20 17:03:13 -070020#include <android/frameworks/bufferhub/1.0/IBufferHub.h>
Fan Xu93c94902018-11-01 12:22:05 -070021#include <android/hardware_buffer.h>
Fan Xu467e08f2018-11-09 15:58:51 -080022#include <cutils/native_handle.h>
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -070023#include <gtest/gtest.h>
Jiwen 'Steve' Caid9f2abe2018-10-20 17:03:13 -070024#include <hidl/ServiceManagement.h>
25#include <hwbinder/IPCThreadState.h>
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -070026#include <ui/BufferHubBuffer.h>
27
28namespace android {
29
30namespace {
31
32const int kWidth = 640;
33const int kHeight = 480;
34const int kLayerCount = 1;
35const int kFormat = HAL_PIXEL_FORMAT_RGBA_8888;
36const int kUsage = 0;
37const size_t kUserMetadataSize = 0;
38
Tianyu Jiang7359dc92018-12-13 11:22:28 -080039using BufferHubDefs::AnyClientAcquired;
40using BufferHubDefs::AnyClientGained;
41using BufferHubDefs::AnyClientPosted;
42using BufferHubDefs::IsBufferReleased;
43using BufferHubDefs::IsClientAcquired;
44using BufferHubDefs::IsClientGained;
45using BufferHubDefs::IsClientPosted;
46using BufferHubDefs::IsClientReleased;
47using BufferHubDefs::kFirstClientBitMask;
48using BufferHubDefs::kMetadataHeaderSize;
Fan Xuca70b7b2018-10-31 13:20:12 -070049using frameworks::bufferhub::V1_0::BufferHubStatus;
50using frameworks::bufferhub::V1_0::IBufferClient;
Jiwen 'Steve' Caid9f2abe2018-10-20 17:03:13 -070051using frameworks::bufferhub::V1_0::IBufferHub;
52using hardware::hidl_handle;
Fan Xu93c94902018-11-01 12:22:05 -070053using hardware::graphics::common::V1_2::HardwareBufferDescription;
Jiwen 'Steve' Caid9f2abe2018-10-20 17:03:13 -070054using hidl::base::V1_0::IBase;
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -070055using pdx::LocalChannelHandle;
56
Jiwen 'Steve' Caid9f2abe2018-10-20 17:03:13 -070057class BufferHubBufferTest : public ::testing::Test {
Tianyu Jiangb08b7222018-11-16 17:55:26 -080058protected:
Jiwen 'Steve' Caid9f2abe2018-10-20 17:03:13 -070059 void SetUp() override { android::hardware::ProcessState::self()->startThreadPool(); }
60};
61
Tianyu Jiangb08b7222018-11-16 17:55:26 -080062class BufferHubBufferStateTransitionTest : public BufferHubBufferTest {
63protected:
64 void SetUp() override {
65 BufferHubBufferTest::SetUp();
66 CreateTwoClientsOfABuffer();
67 }
68
69 std::unique_ptr<BufferHubBuffer> b1;
Tianyu Jiangec97b762019-01-07 17:14:02 -080070 uint32_t b1ClientMask = 0U;
Tianyu Jiangb08b7222018-11-16 17:55:26 -080071 std::unique_ptr<BufferHubBuffer> b2;
Tianyu Jiangec97b762019-01-07 17:14:02 -080072 uint32_t b2ClientMask = 0U;
Tianyu Jiangb08b7222018-11-16 17:55:26 -080073
74private:
75 // Creates b1 and b2 as the clients of the same buffer for testing.
76 void CreateTwoClientsOfABuffer();
77};
78
79void BufferHubBufferStateTransitionTest::CreateTwoClientsOfABuffer() {
80 b1 = BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage, kUserMetadataSize);
81 b1ClientMask = b1->client_state_mask();
Tianyu Jianga99f9112018-12-13 18:23:07 -080082 ASSERT_NE(b1ClientMask, 0U);
Tianyu Jiangb08b7222018-11-16 17:55:26 -080083 auto statusOrHandle = b1->Duplicate();
84 ASSERT_TRUE(statusOrHandle);
85 LocalChannelHandle h2 = statusOrHandle.take();
86 b2 = BufferHubBuffer::Import(std::move(h2));
87 b2ClientMask = b2->client_state_mask();
Tianyu Jianga99f9112018-12-13 18:23:07 -080088 ASSERT_NE(b2ClientMask, 0U);
Tianyu Jiangb08b7222018-11-16 17:55:26 -080089 ASSERT_NE(b2ClientMask, b1ClientMask);
90}
91
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -070092TEST_F(BufferHubBufferTest, CreateBufferHubBufferFails) {
93 // Buffer Creation will fail: BLOB format requires height to be 1.
94 auto b1 = BufferHubBuffer::Create(kWidth, /*height=*/2, kLayerCount,
95 /*format=*/HAL_PIXEL_FORMAT_BLOB, kUsage, kUserMetadataSize);
96
97 EXPECT_FALSE(b1->IsConnected());
98 EXPECT_FALSE(b1->IsValid());
99
100 // Buffer Creation will fail: user metadata size too large.
101 auto b2 = BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage,
102 /*userMetadataSize=*/std::numeric_limits<size_t>::max());
103
104 EXPECT_FALSE(b2->IsConnected());
105 EXPECT_FALSE(b2->IsValid());
106
107 // Buffer Creation will fail: user metadata size too large.
108 const size_t userMetadataSize = std::numeric_limits<size_t>::max() - kMetadataHeaderSize;
109 auto b3 = BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage,
110 userMetadataSize);
111
112 EXPECT_FALSE(b3->IsConnected());
113 EXPECT_FALSE(b3->IsValid());
114}
115
116TEST_F(BufferHubBufferTest, CreateBufferHubBuffer) {
117 auto b1 = BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage,
118 kUserMetadataSize);
119 EXPECT_TRUE(b1->IsConnected());
120 EXPECT_TRUE(b1->IsValid());
121 EXPECT_NE(b1->id(), 0);
122}
123
124TEST_F(BufferHubBufferTest, DuplicateBufferHubBuffer) {
125 auto b1 = BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage,
126 kUserMetadataSize);
127 int id1 = b1->id();
Tianyu Jiangec97b762019-01-07 17:14:02 -0800128 uint32_t bufferStateMask1 = b1->client_state_mask();
Tianyu Jianga99f9112018-12-13 18:23:07 -0800129 EXPECT_NE(bufferStateMask1, 0U);
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -0700130 EXPECT_TRUE(b1->IsValid());
131 EXPECT_EQ(b1->user_metadata_size(), kUserMetadataSize);
132
133 auto statusOrHandle = b1->Duplicate();
134 EXPECT_TRUE(statusOrHandle);
135
136 // The detached buffer should still be valid.
137 EXPECT_TRUE(b1->IsConnected());
138 EXPECT_TRUE(b1->IsValid());
139
140 // Gets the channel handle for the duplicated buffer.
141 LocalChannelHandle h2 = statusOrHandle.take();
142 EXPECT_TRUE(h2.valid());
143
144 std::unique_ptr<BufferHubBuffer> b2 = BufferHubBuffer::Import(std::move(h2));
145 EXPECT_FALSE(h2.valid());
146 ASSERT_TRUE(b2 != nullptr);
147 EXPECT_TRUE(b2->IsValid());
148 EXPECT_EQ(b2->user_metadata_size(), kUserMetadataSize);
149
150 int id2 = b2->id();
Tianyu Jiangec97b762019-01-07 17:14:02 -0800151 uint32_t bufferStateMask2 = b2->client_state_mask();
Tianyu Jianga99f9112018-12-13 18:23:07 -0800152 EXPECT_NE(bufferStateMask2, 0U);
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -0700153
154 // These two buffer instances are based on the same physical buffer under the
155 // hood, so they should share the same id.
156 EXPECT_EQ(id1, id2);
Tianyu Jiang7e204b72018-10-26 15:39:18 -0700157 // We use client_state_mask() to tell those two instances apart.
Tianyu Jiang8f10b752018-10-30 17:24:51 -0700158 EXPECT_NE(bufferStateMask1, bufferStateMask2);
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -0700159
Tianyuf669f6a2018-10-10 15:34:32 -0700160 // Both buffer instances should be in released state currently.
161 EXPECT_TRUE(IsBufferReleased(b1->buffer_state()));
162 EXPECT_TRUE(IsBufferReleased(b2->buffer_state()));
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -0700163
164 // TODO(b/112338294): rewrite test after migration
165 return;
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -0700166}
167
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800168TEST_F(BufferHubBufferStateTransitionTest, GainBuffer_fromReleasedState) {
169 ASSERT_TRUE(IsBufferReleased(b1->buffer_state()));
170
171 // Successful gaining the buffer should change the buffer state bit of b1 to
172 // gained state, other client state bits to released state.
173 EXPECT_EQ(b1->Gain(), 0);
174 EXPECT_TRUE(IsClientGained(b1->buffer_state(), b1ClientMask));
175}
176
177TEST_F(BufferHubBufferStateTransitionTest, GainBuffer_fromGainedState) {
178 ASSERT_EQ(b1->Gain(), 0);
179 auto current_buffer_state = b1->buffer_state();
180 ASSERT_TRUE(IsClientGained(current_buffer_state, b1ClientMask));
181
182 // Gaining from gained state by the same client should not return error.
183 EXPECT_EQ(b1->Gain(), 0);
184
185 // Gaining from gained state by another client should return error.
186 EXPECT_EQ(b2->Gain(), -EBUSY);
187}
188
189TEST_F(BufferHubBufferStateTransitionTest, GainBuffer_fromAcquiredState) {
190 ASSERT_EQ(b1->Gain(), 0);
191 ASSERT_EQ(b1->Post(), 0);
192 ASSERT_EQ(b2->Acquire(), 0);
193 ASSERT_TRUE(AnyClientAcquired(b1->buffer_state()));
194
195 // Gaining from acquired state should fail.
196 EXPECT_EQ(b1->Gain(), -EBUSY);
197 EXPECT_EQ(b2->Gain(), -EBUSY);
198}
199
200TEST_F(BufferHubBufferStateTransitionTest, GainBuffer_fromOtherClientInPostedState) {
201 ASSERT_EQ(b1->Gain(), 0);
202 ASSERT_EQ(b1->Post(), 0);
203 ASSERT_TRUE(AnyClientPosted(b1->buffer_state()));
204
205 // Gaining a buffer who has other posted client should succeed.
206 EXPECT_EQ(b1->Gain(), 0);
207}
208
209TEST_F(BufferHubBufferStateTransitionTest, GainBuffer_fromSelfInPostedState) {
210 ASSERT_EQ(b1->Gain(), 0);
211 ASSERT_EQ(b1->Post(), 0);
212 ASSERT_TRUE(AnyClientPosted(b1->buffer_state()));
213
214 // A posted client should be able to gain the buffer when there is no other clients in
215 // acquired state.
216 EXPECT_EQ(b2->Gain(), 0);
217}
218
219TEST_F(BufferHubBufferStateTransitionTest, PostBuffer_fromOtherInGainedState) {
220 ASSERT_EQ(b1->Gain(), 0);
221 ASSERT_TRUE(IsClientGained(b1->buffer_state(), b1ClientMask));
222
223 EXPECT_EQ(b2->Post(), -EBUSY);
224}
225
226TEST_F(BufferHubBufferStateTransitionTest, PostBuffer_fromSelfInGainedState) {
227 ASSERT_EQ(b1->Gain(), 0);
228 ASSERT_TRUE(IsClientGained(b1->buffer_state(), b1ClientMask));
229
230 EXPECT_EQ(b1->Post(), 0);
231 auto current_buffer_state = b1->buffer_state();
232 EXPECT_TRUE(IsClientReleased(current_buffer_state, b1ClientMask));
233 EXPECT_TRUE(IsClientPosted(current_buffer_state, b2ClientMask));
234}
235
236TEST_F(BufferHubBufferStateTransitionTest, PostBuffer_fromPostedState) {
237 ASSERT_EQ(b1->Gain(), 0);
238 ASSERT_EQ(b1->Post(), 0);
239 ASSERT_TRUE(AnyClientPosted(b1->buffer_state()));
240
241 // Post from posted state should fail.
242 EXPECT_EQ(b1->Post(), -EBUSY);
243 EXPECT_EQ(b2->Post(), -EBUSY);
244}
245
246TEST_F(BufferHubBufferStateTransitionTest, PostBuffer_fromAcquiredState) {
247 ASSERT_EQ(b1->Gain(), 0);
248 ASSERT_EQ(b1->Post(), 0);
249 ASSERT_EQ(b2->Acquire(), 0);
250 ASSERT_TRUE(AnyClientAcquired(b1->buffer_state()));
251
252 // Posting from acquired state should fail.
253 EXPECT_EQ(b1->Post(), -EBUSY);
254 EXPECT_EQ(b2->Post(), -EBUSY);
255}
256
257TEST_F(BufferHubBufferStateTransitionTest, PostBuffer_fromReleasedState) {
258 ASSERT_TRUE(IsBufferReleased(b1->buffer_state()));
259
260 // Posting from released state should fail.
261 EXPECT_EQ(b1->Post(), -EBUSY);
262 EXPECT_EQ(b2->Post(), -EBUSY);
263}
264
265TEST_F(BufferHubBufferStateTransitionTest, AcquireBuffer_fromSelfInPostedState) {
266 ASSERT_EQ(b1->Gain(), 0);
267 ASSERT_EQ(b1->Post(), 0);
268 ASSERT_TRUE(IsClientPosted(b1->buffer_state(), b2ClientMask));
269
270 // Acquire from posted state should pass.
271 EXPECT_EQ(b2->Acquire(), 0);
272}
273
274TEST_F(BufferHubBufferStateTransitionTest, AcquireBuffer_fromOtherInPostedState) {
275 ASSERT_EQ(b1->Gain(), 0);
276 ASSERT_EQ(b1->Post(), 0);
277 ASSERT_TRUE(IsClientPosted(b1->buffer_state(), b2ClientMask));
278
279 // Acquire from released state should fail, although there are other clients
280 // in posted state.
281 EXPECT_EQ(b1->Acquire(), -EBUSY);
282}
283
284TEST_F(BufferHubBufferStateTransitionTest, AcquireBuffer_fromSelfInAcquiredState) {
285 ASSERT_EQ(b1->Gain(), 0);
286 ASSERT_EQ(b1->Post(), 0);
287 ASSERT_EQ(b2->Acquire(), 0);
288 auto current_buffer_state = b1->buffer_state();
289 ASSERT_TRUE(IsClientAcquired(current_buffer_state, b2ClientMask));
290
291 // Acquiring from acquired state by the same client should not error out.
292 EXPECT_EQ(b2->Acquire(), 0);
293}
294
295TEST_F(BufferHubBufferStateTransitionTest, AcquireBuffer_fromReleasedState) {
296 ASSERT_TRUE(IsBufferReleased(b1->buffer_state()));
297
298 // Acquiring form released state should fail.
299 EXPECT_EQ(b1->Acquire(), -EBUSY);
300 EXPECT_EQ(b2->Acquire(), -EBUSY);
301}
302
303TEST_F(BufferHubBufferStateTransitionTest, AcquireBuffer_fromGainedState) {
304 ASSERT_EQ(b1->Gain(), 0);
305 ASSERT_TRUE(AnyClientGained(b1->buffer_state()));
306
307 // Acquiring from gained state should fail.
308 EXPECT_EQ(b1->Acquire(), -EBUSY);
309 EXPECT_EQ(b2->Acquire(), -EBUSY);
310}
311
312TEST_F(BufferHubBufferStateTransitionTest, ReleaseBuffer_fromSelfInReleasedState) {
313 ASSERT_TRUE(IsBufferReleased(b1->buffer_state()));
314
315 EXPECT_EQ(b1->Release(), 0);
316}
317
318TEST_F(BufferHubBufferStateTransitionTest, ReleaseBuffer_fromSelfInGainedState) {
319 ASSERT_TRUE(IsBufferReleased(b1->buffer_state()));
320 ASSERT_EQ(b1->Gain(), 0);
321 ASSERT_TRUE(AnyClientGained(b1->buffer_state()));
322
323 EXPECT_EQ(b1->Release(), 0);
324}
325
326TEST_F(BufferHubBufferStateTransitionTest, ReleaseBuffer_fromSelfInPostedState) {
327 ASSERT_EQ(b1->Gain(), 0);
328 ASSERT_EQ(b1->Post(), 0);
329 ASSERT_TRUE(AnyClientPosted(b1->buffer_state()));
330
331 EXPECT_EQ(b2->Release(), 0);
332}
333
334TEST_F(BufferHubBufferStateTransitionTest, ReleaseBuffer_fromSelfInAcquiredState) {
335 ASSERT_EQ(b1->Gain(), 0);
336 ASSERT_EQ(b1->Post(), 0);
337 ASSERT_EQ(b2->Acquire(), 0);
338 ASSERT_TRUE(AnyClientAcquired(b1->buffer_state()));
339
340 EXPECT_EQ(b2->Release(), 0);
341}
342
Tianyu Jiangec97b762019-01-07 17:14:02 -0800343TEST_F(BufferHubBufferStateTransitionTest, BasicUsage) {
344 // 1 producer buffer and 1 consumer buffer initialised in testcase setup.
345 // Test if this set of basic operation succeed:
346 // Producer post three times to the consumer, and released by consumer.
347 for (int i = 0; i < 3; ++i) {
348 ASSERT_EQ(b1->Gain(), 0);
349 ASSERT_EQ(b1->Post(), 0);
350 ASSERT_EQ(b2->Acquire(), 0);
351 ASSERT_EQ(b2->Release(), 0);
352 }
353}
354
355TEST_F(BufferHubBufferTest, createNewConsumerAfterGain) {
356 // Create a poducer buffer and gain.
357 std::unique_ptr<BufferHubBuffer> b1 =
358 BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage,
359 kUserMetadataSize);
360 ASSERT_EQ(b1->Gain(), 0);
361
362 // Create a consumer of the buffer and test if the consumer can acquire the
363 // buffer if producer posts.
364 auto statusOrHandle = b1->Duplicate();
365 ASSERT_TRUE(statusOrHandle);
366 std::unique_ptr<BufferHubBuffer> b2 = BufferHubBuffer::Import(std::move(statusOrHandle.take()));
367 ASSERT_NE(b1->client_state_mask(), b2->client_state_mask());
368
369 ASSERT_EQ(b1->Post(), 0);
370 EXPECT_EQ(b2->Acquire(), 0);
371}
372
373TEST_F(BufferHubBufferTest, createNewConsumerAfterPost) {
374 // Create a poducer buffer and post.
375 std::unique_ptr<BufferHubBuffer> b1 =
376 BufferHubBuffer::Create(kWidth, kHeight, kLayerCount, kFormat, kUsage,
377 kUserMetadataSize);
378 ASSERT_EQ(b1->Gain(), 0);
379 ASSERT_EQ(b1->Post(), 0);
380
381 // Create a consumer of the buffer and test if the consumer can acquire the
382 // buffer if producer posts.
383 auto statusOrHandle = b1->Duplicate();
384 ASSERT_TRUE(statusOrHandle);
385 std::unique_ptr<BufferHubBuffer> b2 = BufferHubBuffer::Import(std::move(statusOrHandle.take()));
386 ASSERT_NE(b1->client_state_mask(), b2->client_state_mask());
387
388 EXPECT_EQ(b2->Acquire(), 0);
389}
390
Fan Xuca70b7b2018-10-31 13:20:12 -0700391} // namespace
Jiwen 'Steve' Cai0e7ee222018-10-12 17:20:40 -0700392} // namespace android