| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2011 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 |  | 
| Dan Stoza | c6f30bd | 2015-06-08 09:32:50 -0700 | [diff] [blame] | 17 | #include "DummyConsumer.h" | 
|  | 18 |  | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 19 | #include <gtest/gtest.h> | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 20 |  | 
|  | 21 | #include <binder/IMemory.h> | 
| Mathias Agopian | 90ac799 | 2012-02-25 18:48:35 -0800 | [diff] [blame] | 22 | #include <gui/ISurfaceComposer.h> | 
|  | 23 | #include <gui/Surface.h> | 
|  | 24 | #include <gui/SurfaceComposerClient.h> | 
| Eino-Ville Talvala | f7c6087 | 2013-07-30 14:05:02 -0700 | [diff] [blame] | 25 | #include <gui/BufferItemConsumer.h> | 
| Dan Stoza | c187900 | 2014-05-22 15:59:05 -0700 | [diff] [blame] | 26 | #include <ui/Rect.h> | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 27 | #include <utils/String8.h> | 
|  | 28 |  | 
| Mathias Agopian | 41f673c | 2011-11-17 17:48:35 -0800 | [diff] [blame] | 29 | #include <private/gui/ComposerService.h> | 
| Mathias Agopian | 7c1a487 | 2013-03-20 15:56:04 -0700 | [diff] [blame] | 30 | #include <binder/ProcessState.h> | 
| Mathias Agopian | 41f673c | 2011-11-17 17:48:35 -0800 | [diff] [blame] | 31 |  | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 32 | namespace android { | 
|  | 33 |  | 
|  | 34 | class SurfaceTest : public ::testing::Test { | 
|  | 35 | protected: | 
| Mathias Agopian | 7c1a487 | 2013-03-20 15:56:04 -0700 | [diff] [blame] | 36 |  | 
|  | 37 | SurfaceTest() { | 
|  | 38 | ProcessState::self()->startThreadPool(); | 
|  | 39 | } | 
|  | 40 |  | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 41 | virtual void SetUp() { | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 42 | mComposerClient = new SurfaceComposerClient; | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 43 | ASSERT_EQ(NO_ERROR, mComposerClient->initCheck()); | 
|  | 44 |  | 
| Jamie Gennis | fc85012 | 2011-04-25 16:40:05 -0700 | [diff] [blame] | 45 | mSurfaceControl = mComposerClient->createSurface( | 
| Jeff Brown | 9d4e3d2 | 2012-08-24 20:00:51 -0700 | [diff] [blame] | 46 | String8("Test Surface"), 32, 32, PIXEL_FORMAT_RGBA_8888, 0); | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 47 |  | 
|  | 48 | ASSERT_TRUE(mSurfaceControl != NULL); | 
|  | 49 | ASSERT_TRUE(mSurfaceControl->isValid()); | 
|  | 50 |  | 
| Mathias Agopian | 698c087 | 2011-06-28 19:09:31 -0700 | [diff] [blame] | 51 | SurfaceComposerClient::openGlobalTransaction(); | 
| Mathias Agopian | 9303eee | 2011-07-01 15:27:27 -0700 | [diff] [blame] | 52 | ASSERT_EQ(NO_ERROR, mSurfaceControl->setLayer(0x7fffffff)); | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 53 | ASSERT_EQ(NO_ERROR, mSurfaceControl->show()); | 
| Mathias Agopian | 698c087 | 2011-06-28 19:09:31 -0700 | [diff] [blame] | 54 | SurfaceComposerClient::closeGlobalTransaction(); | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 55 |  | 
|  | 56 | mSurface = mSurfaceControl->getSurface(); | 
|  | 57 | ASSERT_TRUE(mSurface != NULL); | 
|  | 58 | } | 
|  | 59 |  | 
|  | 60 | virtual void TearDown() { | 
|  | 61 | mComposerClient->dispose(); | 
|  | 62 | } | 
|  | 63 |  | 
|  | 64 | sp<Surface> mSurface; | 
|  | 65 | sp<SurfaceComposerClient> mComposerClient; | 
|  | 66 | sp<SurfaceControl> mSurfaceControl; | 
|  | 67 | }; | 
|  | 68 |  | 
|  | 69 | TEST_F(SurfaceTest, QueuesToWindowComposerIsTrueWhenVisible) { | 
|  | 70 | sp<ANativeWindow> anw(mSurface); | 
|  | 71 | int result = -123; | 
|  | 72 | int err = anw->query(anw.get(), NATIVE_WINDOW_QUEUES_TO_WINDOW_COMPOSER, | 
|  | 73 | &result); | 
|  | 74 | EXPECT_EQ(NO_ERROR, err); | 
|  | 75 | EXPECT_EQ(1, result); | 
|  | 76 | } | 
|  | 77 |  | 
|  | 78 | TEST_F(SurfaceTest, QueuesToWindowComposerIsTrueWhenPurgatorized) { | 
|  | 79 | mSurfaceControl.clear(); | 
|  | 80 |  | 
|  | 81 | sp<ANativeWindow> anw(mSurface); | 
|  | 82 | int result = -123; | 
|  | 83 | int err = anw->query(anw.get(), NATIVE_WINDOW_QUEUES_TO_WINDOW_COMPOSER, | 
|  | 84 | &result); | 
|  | 85 | EXPECT_EQ(NO_ERROR, err); | 
|  | 86 | EXPECT_EQ(1, result); | 
|  | 87 | } | 
|  | 88 |  | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 89 | // This test probably doesn't belong here. | 
| Jamie Gennis | c901ca0 | 2011-10-11 16:02:31 -0700 | [diff] [blame] | 90 | TEST_F(SurfaceTest, ScreenshotsOfProtectedBuffersSucceed) { | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 91 | sp<ANativeWindow> anw(mSurface); | 
|  | 92 |  | 
|  | 93 | // Verify the screenshot works with no protected buffers. | 
| Dan Stoza | 5603a2f | 2014-04-07 13:41:37 -0700 | [diff] [blame] | 94 | sp<IGraphicBufferProducer> producer; | 
|  | 95 | sp<IGraphicBufferConsumer> consumer; | 
|  | 96 | BufferQueue::createBufferQueue(&producer, &consumer); | 
|  | 97 | sp<CpuConsumer> cpuConsumer = new CpuConsumer(consumer, 1); | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 98 | sp<ISurfaceComposer> sf(ComposerService::getComposerService()); | 
| Jeff Brown | 9d4e3d2 | 2012-08-24 20:00:51 -0700 | [diff] [blame] | 99 | sp<IBinder> display(sf->getBuiltInDisplay(ISurfaceComposer::eDisplayIdMain)); | 
| Dan Stoza | c187900 | 2014-05-22 15:59:05 -0700 | [diff] [blame] | 100 | ASSERT_EQ(NO_ERROR, sf->captureScreen(display, producer, Rect(), | 
| Dan Stoza | 8d75996 | 2014-02-19 18:35:30 -0800 | [diff] [blame] | 101 | 64, 64, 0, 0x7fffffff, false)); | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 102 |  | 
| Pablo Ceballos | 583b1b3 | 2015-09-03 18:23:52 -0700 | [diff] [blame] | 103 | ASSERT_EQ(NO_ERROR, native_window_api_connect(anw.get(), | 
|  | 104 | NATIVE_WINDOW_API_CPU)); | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 105 | // Set the PROTECTED usage bit and verify that the screenshot fails.  Note | 
|  | 106 | // that we need to dequeue a buffer in order for it to actually get | 
|  | 107 | // allocated in SurfaceFlinger. | 
|  | 108 | ASSERT_EQ(NO_ERROR, native_window_set_usage(anw.get(), | 
|  | 109 | GRALLOC_USAGE_PROTECTED)); | 
|  | 110 | ASSERT_EQ(NO_ERROR, native_window_set_buffer_count(anw.get(), 3)); | 
| Iliyan Malchev | 697526b | 2011-05-01 11:33:26 -0700 | [diff] [blame] | 111 | ANativeWindowBuffer* buf = 0; | 
| Mathias Agopian | 9303eee | 2011-07-01 15:27:27 -0700 | [diff] [blame] | 112 |  | 
| Jamie Gennis | d8e812c | 2012-06-13 16:32:25 -0700 | [diff] [blame] | 113 | status_t err = native_window_dequeue_buffer_and_wait(anw.get(), &buf); | 
| Mathias Agopian | 9303eee | 2011-07-01 15:27:27 -0700 | [diff] [blame] | 114 | if (err) { | 
|  | 115 | // we could fail if GRALLOC_USAGE_PROTECTED is not supported. | 
|  | 116 | // that's okay as long as this is the reason for the failure. | 
|  | 117 | // try again without the GRALLOC_USAGE_PROTECTED bit. | 
|  | 118 | ASSERT_EQ(NO_ERROR, native_window_set_usage(anw.get(), 0)); | 
| Jamie Gennis | d8e812c | 2012-06-13 16:32:25 -0700 | [diff] [blame] | 119 | ASSERT_EQ(NO_ERROR, native_window_dequeue_buffer_and_wait(anw.get(), | 
|  | 120 | &buf)); | 
| Mathias Agopian | 9303eee | 2011-07-01 15:27:27 -0700 | [diff] [blame] | 121 | return; | 
|  | 122 | } | 
| Jamie Gennis | d8e812c | 2012-06-13 16:32:25 -0700 | [diff] [blame] | 123 | ASSERT_EQ(NO_ERROR, anw->cancelBuffer(anw.get(), buf, -1)); | 
| Mathias Agopian | 9303eee | 2011-07-01 15:27:27 -0700 | [diff] [blame] | 124 |  | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 125 | for (int i = 0; i < 4; i++) { | 
|  | 126 | // Loop to make sure SurfaceFlinger has retired a protected buffer. | 
| Jamie Gennis | d8e812c | 2012-06-13 16:32:25 -0700 | [diff] [blame] | 127 | ASSERT_EQ(NO_ERROR, native_window_dequeue_buffer_and_wait(anw.get(), | 
|  | 128 | &buf)); | 
|  | 129 | ASSERT_EQ(NO_ERROR, anw->queueBuffer(anw.get(), buf, -1)); | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 130 | } | 
| Dan Stoza | c187900 | 2014-05-22 15:59:05 -0700 | [diff] [blame] | 131 | ASSERT_EQ(NO_ERROR, sf->captureScreen(display, producer, Rect(), | 
| Dan Stoza | 8d75996 | 2014-02-19 18:35:30 -0800 | [diff] [blame] | 132 | 64, 64, 0, 0x7fffffff, false)); | 
| Jamie Gennis | 7a4d0df | 2011-03-09 17:05:02 -0800 | [diff] [blame] | 133 | } | 
|  | 134 |  | 
| Jamie Gennis | 391bbe2 | 2011-03-14 15:00:06 -0700 | [diff] [blame] | 135 | TEST_F(SurfaceTest, ConcreteTypeIsSurface) { | 
|  | 136 | sp<ANativeWindow> anw(mSurface); | 
|  | 137 | int result = -123; | 
|  | 138 | int err = anw->query(anw.get(), NATIVE_WINDOW_CONCRETE_TYPE, &result); | 
|  | 139 | EXPECT_EQ(NO_ERROR, err); | 
|  | 140 | EXPECT_EQ(NATIVE_WINDOW_SURFACE, result); | 
|  | 141 | } | 
|  | 142 |  | 
| Eino-Ville Talvala | f7c6087 | 2013-07-30 14:05:02 -0700 | [diff] [blame] | 143 | TEST_F(SurfaceTest, QueryConsumerUsage) { | 
|  | 144 | const int TEST_USAGE_FLAGS = | 
|  | 145 | GRALLOC_USAGE_SW_READ_OFTEN | GRALLOC_USAGE_HW_RENDER; | 
| Dan Stoza | 5603a2f | 2014-04-07 13:41:37 -0700 | [diff] [blame] | 146 | sp<IGraphicBufferProducer> producer; | 
|  | 147 | sp<IGraphicBufferConsumer> consumer; | 
|  | 148 | BufferQueue::createBufferQueue(&producer, &consumer); | 
|  | 149 | sp<BufferItemConsumer> c = new BufferItemConsumer(consumer, | 
| Eino-Ville Talvala | f7c6087 | 2013-07-30 14:05:02 -0700 | [diff] [blame] | 150 | TEST_USAGE_FLAGS); | 
| Dan Stoza | 5603a2f | 2014-04-07 13:41:37 -0700 | [diff] [blame] | 151 | sp<Surface> s = new Surface(producer); | 
| Eino-Ville Talvala | f7c6087 | 2013-07-30 14:05:02 -0700 | [diff] [blame] | 152 |  | 
|  | 153 | sp<ANativeWindow> anw(s); | 
|  | 154 |  | 
|  | 155 | int flags = -1; | 
|  | 156 | int err = anw->query(anw.get(), NATIVE_WINDOW_CONSUMER_USAGE_BITS, &flags); | 
|  | 157 |  | 
|  | 158 | ASSERT_EQ(NO_ERROR, err); | 
|  | 159 | ASSERT_EQ(TEST_USAGE_FLAGS, flags); | 
|  | 160 | } | 
|  | 161 |  | 
| Eino-Ville Talvala | 5b75a51 | 2015-02-19 16:10:43 -0800 | [diff] [blame] | 162 | TEST_F(SurfaceTest, QueryDefaultBuffersDataSpace) { | 
|  | 163 | const android_dataspace TEST_DATASPACE = HAL_DATASPACE_SRGB; | 
|  | 164 | sp<IGraphicBufferProducer> producer; | 
|  | 165 | sp<IGraphicBufferConsumer> consumer; | 
|  | 166 | BufferQueue::createBufferQueue(&producer, &consumer); | 
|  | 167 | sp<CpuConsumer> cpuConsumer = new CpuConsumer(consumer, 1); | 
|  | 168 |  | 
|  | 169 | cpuConsumer->setDefaultBufferDataSpace(TEST_DATASPACE); | 
|  | 170 |  | 
|  | 171 | sp<Surface> s = new Surface(producer); | 
|  | 172 |  | 
|  | 173 | sp<ANativeWindow> anw(s); | 
|  | 174 |  | 
|  | 175 | android_dataspace dataSpace; | 
|  | 176 |  | 
|  | 177 | int err = anw->query(anw.get(), NATIVE_WINDOW_DEFAULT_DATASPACE, | 
|  | 178 | reinterpret_cast<int*>(&dataSpace)); | 
|  | 179 |  | 
|  | 180 | ASSERT_EQ(NO_ERROR, err); | 
|  | 181 | ASSERT_EQ(TEST_DATASPACE, dataSpace); | 
|  | 182 | } | 
|  | 183 |  | 
| Dan Stoza | 812ed06 | 2015-06-02 15:45:22 -0700 | [diff] [blame] | 184 | TEST_F(SurfaceTest, SettingGenerationNumber) { | 
|  | 185 | sp<IGraphicBufferProducer> producer; | 
|  | 186 | sp<IGraphicBufferConsumer> consumer; | 
|  | 187 | BufferQueue::createBufferQueue(&producer, &consumer); | 
|  | 188 | sp<CpuConsumer> cpuConsumer = new CpuConsumer(consumer, 1); | 
|  | 189 | sp<Surface> surface = new Surface(producer); | 
|  | 190 | sp<ANativeWindow> window(surface); | 
|  | 191 |  | 
|  | 192 | // Allocate a buffer with a generation number of 0 | 
|  | 193 | ANativeWindowBuffer* buffer; | 
|  | 194 | int fenceFd; | 
| Pablo Ceballos | 583b1b3 | 2015-09-03 18:23:52 -0700 | [diff] [blame] | 195 | ASSERT_EQ(NO_ERROR, native_window_api_connect(window.get(), | 
|  | 196 | NATIVE_WINDOW_API_CPU)); | 
| Dan Stoza | 812ed06 | 2015-06-02 15:45:22 -0700 | [diff] [blame] | 197 | ASSERT_EQ(NO_ERROR, window->dequeueBuffer(window.get(), &buffer, &fenceFd)); | 
|  | 198 | ASSERT_EQ(NO_ERROR, window->cancelBuffer(window.get(), buffer, fenceFd)); | 
|  | 199 |  | 
|  | 200 | // Detach the buffer and check its generation number | 
|  | 201 | sp<GraphicBuffer> graphicBuffer; | 
|  | 202 | sp<Fence> fence; | 
|  | 203 | ASSERT_EQ(NO_ERROR, surface->detachNextBuffer(&graphicBuffer, &fence)); | 
|  | 204 | ASSERT_EQ(0U, graphicBuffer->getGenerationNumber()); | 
|  | 205 |  | 
|  | 206 | ASSERT_EQ(NO_ERROR, surface->setGenerationNumber(1)); | 
|  | 207 | buffer = static_cast<ANativeWindowBuffer*>(graphicBuffer.get()); | 
|  | 208 |  | 
|  | 209 | // This should change the generation number of the GraphicBuffer | 
|  | 210 | ASSERT_EQ(NO_ERROR, surface->attachBuffer(buffer)); | 
|  | 211 |  | 
|  | 212 | // Check that the new generation number sticks with the buffer | 
|  | 213 | ASSERT_EQ(NO_ERROR, window->cancelBuffer(window.get(), buffer, -1)); | 
|  | 214 | ASSERT_EQ(NO_ERROR, window->dequeueBuffer(window.get(), &buffer, &fenceFd)); | 
|  | 215 | graphicBuffer = static_cast<GraphicBuffer*>(buffer); | 
|  | 216 | ASSERT_EQ(1U, graphicBuffer->getGenerationNumber()); | 
|  | 217 | } | 
|  | 218 |  | 
| Dan Stoza | c6f30bd | 2015-06-08 09:32:50 -0700 | [diff] [blame] | 219 | TEST_F(SurfaceTest, GetConsumerName) { | 
|  | 220 | sp<IGraphicBufferProducer> producer; | 
|  | 221 | sp<IGraphicBufferConsumer> consumer; | 
|  | 222 | BufferQueue::createBufferQueue(&producer, &consumer); | 
|  | 223 |  | 
|  | 224 | sp<DummyConsumer> dummyConsumer(new DummyConsumer); | 
|  | 225 | consumer->consumerConnect(dummyConsumer, false); | 
|  | 226 | consumer->setConsumerName(String8("TestConsumer")); | 
|  | 227 |  | 
|  | 228 | sp<Surface> surface = new Surface(producer); | 
|  | 229 | sp<ANativeWindow> window(surface); | 
|  | 230 | native_window_api_connect(window.get(), NATIVE_WINDOW_API_CPU); | 
|  | 231 |  | 
|  | 232 | EXPECT_STREQ("TestConsumer", surface->getConsumerName().string()); | 
|  | 233 | } | 
|  | 234 |  | 
| Jamie Gennis | 134f042 | 2011-03-08 12:18:54 -0800 | [diff] [blame] | 235 | } |