/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <frameworks/native/cmds/surfacereplayer/proto/src/trace.pb.h> #include <google/protobuf/io/zero_copy_stream_impl.h> #include <gtest/gtest.h> #include <android/native_window.h> #include <gui/ISurfaceComposer.h> #include <gui/LayerState.h> #include <gui/Surface.h> #include <gui/SurfaceComposerClient.h> #include <private/gui/ComposerService.h> #include <ui/DisplayInfo.h> #include <fstream> #include <random> #include <thread> namespace android { using Transaction = SurfaceComposerClient::Transaction; constexpr int32_t SCALING_UPDATE = 1; constexpr uint32_t BUFFER_UPDATES = 18; constexpr uint32_t LAYER_UPDATE = INT_MAX - 2; constexpr uint32_t SIZE_UPDATE = 134; constexpr uint32_t STACK_UPDATE = 1; constexpr uint64_t DEFERRED_UPDATE = 13; constexpr float ALPHA_UPDATE = 0.29f; constexpr float POSITION_UPDATE = 121; const Rect CROP_UPDATE(16, 16, 32, 32); const String8 DISPLAY_NAME("SurfaceInterceptor Display Test"); constexpr auto LAYER_NAME = "Layer Create and Delete Test"; constexpr auto DEFAULT_FILENAME = "/data/SurfaceTrace.dat"; // Fill an RGBA_8888 formatted surface with a single color. static void fillSurfaceRGBA8(const sp<SurfaceControl>& sc, uint8_t r, uint8_t g, uint8_t b) { ANativeWindow_Buffer outBuffer; sp<Surface> s = sc->getSurface(); ASSERT_TRUE(s != nullptr); ASSERT_EQ(NO_ERROR, s->lock(&outBuffer, nullptr)); uint8_t* img = reinterpret_cast<uint8_t*>(outBuffer.bits); for (int y = 0; y < outBuffer.height; y++) { for (int x = 0; x < outBuffer.width; x++) { uint8_t* pixel = img + (4 * (y*outBuffer.stride + x)); pixel[0] = r; pixel[1] = g; pixel[2] = b; pixel[3] = 255; } } ASSERT_EQ(NO_ERROR, s->unlockAndPost()); } static status_t readProtoFile(Trace* trace) { status_t err = NO_ERROR; int fd = open(DEFAULT_FILENAME, O_RDONLY); { google::protobuf::io::FileInputStream f(fd); if (fd && !trace->ParseFromZeroCopyStream(&f)) { err = PERMISSION_DENIED; } } close(fd); return err; } static void enableInterceptor() { system("service call SurfaceFlinger 1020 i32 1 > /dev/null"); } static void disableInterceptor() { system("service call SurfaceFlinger 1020 i32 0 > /dev/null"); } int32_t getSurfaceId(const std::string& surfaceName) { enableInterceptor(); disableInterceptor(); Trace capturedTrace; readProtoFile(&capturedTrace); int32_t layerId = 0; for (const auto& increment : *capturedTrace.mutable_increment()) { if (increment.increment_case() == increment.kSurfaceCreation) { if (increment.surface_creation().name() == surfaceName) { layerId = increment.surface_creation().id(); break; } } } return layerId; } int32_t getDisplayId(const std::string& displayName) { enableInterceptor(); disableInterceptor(); Trace capturedTrace; readProtoFile(&capturedTrace); int32_t displayId = 0; for (const auto& increment : *capturedTrace.mutable_increment()) { if (increment.increment_case() == increment.kDisplayCreation) { if (increment.display_creation().name() == displayName) { displayId = increment.display_creation().id(); break; } } } return displayId; } class SurfaceInterceptorTest : public ::testing::Test { protected: virtual void SetUp() { // Allow SurfaceInterceptor write to /data system("setenforce 0"); mComposerClient = new SurfaceComposerClient; ASSERT_EQ(NO_ERROR, mComposerClient->initCheck()); sp<IBinder> display(SurfaceComposerClient::getBuiltInDisplay( ISurfaceComposer::eDisplayIdMain)); DisplayInfo info; SurfaceComposerClient::getDisplayInfo(display, &info); ssize_t displayWidth = info.w; ssize_t displayHeight = info.h; // Background surface mBGSurfaceControl = mComposerClient->createSurface( String8("BG Interceptor Test Surface"), displayWidth, displayHeight, PIXEL_FORMAT_RGBA_8888, 0); ASSERT_TRUE(mBGSurfaceControl != nullptr); ASSERT_TRUE(mBGSurfaceControl->isValid()); mBGLayerId = getSurfaceId("BG Interceptor Test Surface"); Transaction t; t.setDisplayLayerStack(display, 0); ASSERT_EQ(NO_ERROR, t.setLayer(mBGSurfaceControl, INT_MAX-3) .show(mBGSurfaceControl) .apply()); } virtual void TearDown() { mComposerClient->dispose(); mBGSurfaceControl.clear(); mComposerClient.clear(); } sp<SurfaceComposerClient> mComposerClient; sp<SurfaceControl> mBGSurfaceControl; int32_t mBGLayerId; // Used to verify creation and destruction of surfaces and displays int32_t mTargetId; public: void captureTest(void (SurfaceInterceptorTest::* action)(Transaction&), bool (SurfaceInterceptorTest::* verification)(Trace *)); void captureTest(void (SurfaceInterceptorTest::* action)(Transaction&), SurfaceChange::SurfaceChangeCase changeCase); void captureTest(void (SurfaceInterceptorTest::* action)(Transaction&), Increment::IncrementCase incrementCase); void runInTransaction(void (SurfaceInterceptorTest::* action)(Transaction&), bool intercepted = false); // Verification of changes to a surface bool positionUpdateFound(const SurfaceChange& change, bool foundPosition); bool sizeUpdateFound(const SurfaceChange& change, bool foundSize); bool alphaUpdateFound(const SurfaceChange& change, bool foundAlpha); bool layerUpdateFound(const SurfaceChange& change, bool foundLayer); bool cropUpdateFound(const SurfaceChange& change, bool foundCrop); bool finalCropUpdateFound(const SurfaceChange& change, bool foundFinalCrop); bool matrixUpdateFound(const SurfaceChange& change, bool foundMatrix); bool scalingModeUpdateFound(const SurfaceChange& change, bool foundScalingMode); bool transparentRegionHintUpdateFound(const SurfaceChange& change, bool foundTransparentRegion); bool layerStackUpdateFound(const SurfaceChange& change, bool foundLayerStack); bool hiddenFlagUpdateFound(const SurfaceChange& change, bool foundHiddenFlag); bool opaqueFlagUpdateFound(const SurfaceChange& change, bool foundOpaqueFlag); bool secureFlagUpdateFound(const SurfaceChange& change, bool foundSecureFlag); bool deferredTransactionUpdateFound(const SurfaceChange& change, bool foundDeferred); bool surfaceUpdateFound(Trace* trace, SurfaceChange::SurfaceChangeCase changeCase); void assertAllUpdatesFound(Trace* trace); // Verification of creation and deletion of a surface bool surfaceCreationFound(const Increment& increment, bool foundSurface); bool surfaceDeletionFound(const Increment& increment, bool foundSurface); bool displayCreationFound(const Increment& increment, bool foundDisplay); bool displayDeletionFound(const Increment& increment, bool foundDisplay); bool singleIncrementFound(Trace* trace, Increment::IncrementCase incrementCase); // Verification of buffer updates bool bufferUpdatesFound(Trace* trace); // Perform each of the possible changes to a surface void positionUpdate(Transaction&); void sizeUpdate(Transaction&); void alphaUpdate(Transaction&); void layerUpdate(Transaction&); void cropUpdate(Transaction&); void finalCropUpdate(Transaction&); void matrixUpdate(Transaction&); void overrideScalingModeUpdate(Transaction&); void transparentRegionHintUpdate(Transaction&); void layerStackUpdate(Transaction&); void hiddenFlagUpdate(Transaction&); void opaqueFlagUpdate(Transaction&); void secureFlagUpdate(Transaction&); void deferredTransactionUpdate(Transaction&); void surfaceCreation(Transaction&); void displayCreation(Transaction&); void displayDeletion(Transaction&); void nBufferUpdates(); void runAllUpdates(); }; void SurfaceInterceptorTest::captureTest(void (SurfaceInterceptorTest::* action)(Transaction&), bool (SurfaceInterceptorTest::* verification)(Trace *)) { runInTransaction(action, true); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); ASSERT_TRUE((this->*verification)(&capturedTrace)); } void SurfaceInterceptorTest::captureTest(void (SurfaceInterceptorTest::* action)(Transaction&), Increment::IncrementCase incrementCase) { runInTransaction(action, true); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); ASSERT_TRUE(singleIncrementFound(&capturedTrace, incrementCase)); } void SurfaceInterceptorTest::captureTest(void (SurfaceInterceptorTest::* action)(Transaction&), SurfaceChange::SurfaceChangeCase changeCase) { runInTransaction(action, true); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); ASSERT_TRUE(surfaceUpdateFound(&capturedTrace, changeCase)); } void SurfaceInterceptorTest::runInTransaction(void (SurfaceInterceptorTest::* action)(Transaction&), bool intercepted) { if (intercepted) { enableInterceptor(); } Transaction t; (this->*action)(t); t.apply(true); if (intercepted) { disableInterceptor(); } } void SurfaceInterceptorTest::positionUpdate(Transaction& t) { t.setPosition(mBGSurfaceControl, POSITION_UPDATE, POSITION_UPDATE); } void SurfaceInterceptorTest::sizeUpdate(Transaction& t) { t.setSize(mBGSurfaceControl, SIZE_UPDATE, SIZE_UPDATE); } void SurfaceInterceptorTest::alphaUpdate(Transaction& t) { t.setAlpha(mBGSurfaceControl, ALPHA_UPDATE); } void SurfaceInterceptorTest::layerUpdate(Transaction& t) { t.setLayer(mBGSurfaceControl, LAYER_UPDATE); } void SurfaceInterceptorTest::cropUpdate(Transaction& t) { t.setCrop(mBGSurfaceControl, CROP_UPDATE); } void SurfaceInterceptorTest::finalCropUpdate(Transaction& t) { t.setFinalCrop(mBGSurfaceControl, CROP_UPDATE); } void SurfaceInterceptorTest::matrixUpdate(Transaction& t) { t.setMatrix(mBGSurfaceControl, M_SQRT1_2, M_SQRT1_2, -M_SQRT1_2, M_SQRT1_2); } void SurfaceInterceptorTest::overrideScalingModeUpdate(Transaction& t) { t.setOverrideScalingMode(mBGSurfaceControl, SCALING_UPDATE); } void SurfaceInterceptorTest::transparentRegionHintUpdate(Transaction& t) { Region region(CROP_UPDATE); t.setTransparentRegionHint(mBGSurfaceControl, region); } void SurfaceInterceptorTest::layerStackUpdate(Transaction& t) { t.setLayerStack(mBGSurfaceControl, STACK_UPDATE); } void SurfaceInterceptorTest::hiddenFlagUpdate(Transaction& t) { t.setFlags(mBGSurfaceControl, layer_state_t::eLayerHidden, layer_state_t::eLayerHidden); } void SurfaceInterceptorTest::opaqueFlagUpdate(Transaction& t) { t.setFlags(mBGSurfaceControl, layer_state_t::eLayerOpaque, layer_state_t::eLayerOpaque); } void SurfaceInterceptorTest::secureFlagUpdate(Transaction& t) { t.setFlags(mBGSurfaceControl, layer_state_t::eLayerSecure, layer_state_t::eLayerSecure); } void SurfaceInterceptorTest::deferredTransactionUpdate(Transaction& t) { t.deferTransactionUntil(mBGSurfaceControl, mBGSurfaceControl->getHandle(), DEFERRED_UPDATE); } void SurfaceInterceptorTest::displayCreation(Transaction&) { sp<IBinder> testDisplay = SurfaceComposerClient::createDisplay(DISPLAY_NAME, true); SurfaceComposerClient::destroyDisplay(testDisplay); } void SurfaceInterceptorTest::displayDeletion(Transaction&) { sp<IBinder> testDisplay = SurfaceComposerClient::createDisplay(DISPLAY_NAME, false); mTargetId = getDisplayId(DISPLAY_NAME.string()); SurfaceComposerClient::destroyDisplay(testDisplay); } void SurfaceInterceptorTest::runAllUpdates() { runInTransaction(&SurfaceInterceptorTest::positionUpdate); runInTransaction(&SurfaceInterceptorTest::sizeUpdate); runInTransaction(&SurfaceInterceptorTest::alphaUpdate); runInTransaction(&SurfaceInterceptorTest::layerUpdate); runInTransaction(&SurfaceInterceptorTest::cropUpdate); runInTransaction(&SurfaceInterceptorTest::finalCropUpdate); runInTransaction(&SurfaceInterceptorTest::matrixUpdate); runInTransaction(&SurfaceInterceptorTest::overrideScalingModeUpdate); runInTransaction(&SurfaceInterceptorTest::transparentRegionHintUpdate); runInTransaction(&SurfaceInterceptorTest::layerStackUpdate); runInTransaction(&SurfaceInterceptorTest::hiddenFlagUpdate); runInTransaction(&SurfaceInterceptorTest::opaqueFlagUpdate); runInTransaction(&SurfaceInterceptorTest::secureFlagUpdate); runInTransaction(&SurfaceInterceptorTest::deferredTransactionUpdate); } void SurfaceInterceptorTest::surfaceCreation(Transaction&) { mComposerClient->createSurface(String8(LAYER_NAME), SIZE_UPDATE, SIZE_UPDATE, PIXEL_FORMAT_RGBA_8888, 0); } void SurfaceInterceptorTest::nBufferUpdates() { std::random_device rd; std::mt19937_64 gen(rd()); // This makes testing fun std::uniform_int_distribution<uint8_t> dis; for (uint32_t i = 0; i < BUFFER_UPDATES; ++i) { fillSurfaceRGBA8(mBGSurfaceControl, dis(gen), dis(gen), dis(gen)); } } bool SurfaceInterceptorTest::positionUpdateFound(const SurfaceChange& change, bool foundPosition) { // There should only be one position transaction with x and y = POSITION_UPDATE bool hasX(change.position().x() == POSITION_UPDATE); bool hasY(change.position().y() == POSITION_UPDATE); if (hasX && hasY && !foundPosition) { foundPosition = true; } // Failed because the position update was found a second time else if (hasX && hasY && foundPosition) { [] () { FAIL(); }(); } return foundPosition; } bool SurfaceInterceptorTest::sizeUpdateFound(const SurfaceChange& change, bool foundSize) { bool hasWidth(change.size().h() == SIZE_UPDATE); bool hasHeight(change.size().w() == SIZE_UPDATE); if (hasWidth && hasHeight && !foundSize) { foundSize = true; } else if (hasWidth && hasHeight && foundSize) { [] () { FAIL(); }(); } return foundSize; } bool SurfaceInterceptorTest::alphaUpdateFound(const SurfaceChange& change, bool foundAlpha) { bool hasAlpha(change.alpha().alpha() == ALPHA_UPDATE); if (hasAlpha && !foundAlpha) { foundAlpha = true; } else if (hasAlpha && foundAlpha) { [] () { FAIL(); }(); } return foundAlpha; } bool SurfaceInterceptorTest::layerUpdateFound(const SurfaceChange& change, bool foundLayer) { bool hasLayer(change.layer().layer() == LAYER_UPDATE); if (hasLayer && !foundLayer) { foundLayer = true; } else if (hasLayer && foundLayer) { [] () { FAIL(); }(); } return foundLayer; } bool SurfaceInterceptorTest::cropUpdateFound(const SurfaceChange& change, bool foundCrop) { bool hasLeft(change.crop().rectangle().left() == CROP_UPDATE.left); bool hasTop(change.crop().rectangle().top() == CROP_UPDATE.top); bool hasRight(change.crop().rectangle().right() == CROP_UPDATE.right); bool hasBottom(change.crop().rectangle().bottom() == CROP_UPDATE.bottom); if (hasLeft && hasRight && hasTop && hasBottom && !foundCrop) { foundCrop = true; } else if (hasLeft && hasRight && hasTop && hasBottom && foundCrop) { [] () { FAIL(); }(); } return foundCrop; } bool SurfaceInterceptorTest::finalCropUpdateFound(const SurfaceChange& change, bool foundFinalCrop) { bool hasLeft(change.final_crop().rectangle().left() == CROP_UPDATE.left); bool hasTop(change.final_crop().rectangle().top() == CROP_UPDATE.top); bool hasRight(change.final_crop().rectangle().right() == CROP_UPDATE.right); bool hasBottom(change.final_crop().rectangle().bottom() == CROP_UPDATE.bottom); if (hasLeft && hasRight && hasTop && hasBottom && !foundFinalCrop) { foundFinalCrop = true; } else if (hasLeft && hasRight && hasTop && hasBottom && foundFinalCrop) { [] () { FAIL(); }(); } return foundFinalCrop; } bool SurfaceInterceptorTest::matrixUpdateFound(const SurfaceChange& change, bool foundMatrix) { bool hasSx((float)change.matrix().dsdx() == (float)M_SQRT1_2); bool hasTx((float)change.matrix().dtdx() == (float)M_SQRT1_2); bool hasSy((float)change.matrix().dsdy() == (float)-M_SQRT1_2); bool hasTy((float)change.matrix().dtdy() == (float)M_SQRT1_2); if (hasSx && hasTx && hasSy && hasTy && !foundMatrix) { foundMatrix = true; } else if (hasSx && hasTx && hasSy && hasTy && foundMatrix) { [] () { FAIL(); }(); } return foundMatrix; } bool SurfaceInterceptorTest::scalingModeUpdateFound(const SurfaceChange& change, bool foundScalingMode) { bool hasScalingUpdate(change.override_scaling_mode().override_scaling_mode() == SCALING_UPDATE); if (hasScalingUpdate && !foundScalingMode) { foundScalingMode = true; } else if (hasScalingUpdate && foundScalingMode) { [] () { FAIL(); }(); } return foundScalingMode; } bool SurfaceInterceptorTest::transparentRegionHintUpdateFound(const SurfaceChange& change, bool foundTransparentRegion) { auto traceRegion = change.transparent_region_hint().region(0); bool hasLeft(traceRegion.left() == CROP_UPDATE.left); bool hasTop(traceRegion.top() == CROP_UPDATE.top); bool hasRight(traceRegion.right() == CROP_UPDATE.right); bool hasBottom(traceRegion.bottom() == CROP_UPDATE.bottom); if (hasLeft && hasRight && hasTop && hasBottom && !foundTransparentRegion) { foundTransparentRegion = true; } else if (hasLeft && hasRight && hasTop && hasBottom && foundTransparentRegion) { [] () { FAIL(); }(); } return foundTransparentRegion; } bool SurfaceInterceptorTest::layerStackUpdateFound(const SurfaceChange& change, bool foundLayerStack) { bool hasLayerStackUpdate(change.layer_stack().layer_stack() == STACK_UPDATE); if (hasLayerStackUpdate && !foundLayerStack) { foundLayerStack = true; } else if (hasLayerStackUpdate && foundLayerStack) { [] () { FAIL(); }(); } return foundLayerStack; } bool SurfaceInterceptorTest::hiddenFlagUpdateFound(const SurfaceChange& change, bool foundHiddenFlag) { bool hasHiddenFlag(change.hidden_flag().hidden_flag()); if (hasHiddenFlag && !foundHiddenFlag) { foundHiddenFlag = true; } else if (hasHiddenFlag && foundHiddenFlag) { [] () { FAIL(); }(); } return foundHiddenFlag; } bool SurfaceInterceptorTest::opaqueFlagUpdateFound(const SurfaceChange& change, bool foundOpaqueFlag) { bool hasOpaqueFlag(change.opaque_flag().opaque_flag()); if (hasOpaqueFlag && !foundOpaqueFlag) { foundOpaqueFlag = true; } else if (hasOpaqueFlag && foundOpaqueFlag) { [] () { FAIL(); }(); } return foundOpaqueFlag; } bool SurfaceInterceptorTest::secureFlagUpdateFound(const SurfaceChange& change, bool foundSecureFlag) { bool hasSecureFlag(change.secure_flag().secure_flag()); if (hasSecureFlag && !foundSecureFlag) { foundSecureFlag = true; } else if (hasSecureFlag && foundSecureFlag) { [] () { FAIL(); }(); } return foundSecureFlag; } bool SurfaceInterceptorTest::deferredTransactionUpdateFound(const SurfaceChange& change, bool foundDeferred) { bool hasId(change.deferred_transaction().layer_id() == mBGLayerId); bool hasFrameNumber(change.deferred_transaction().frame_number() == DEFERRED_UPDATE); if (hasId && hasFrameNumber && !foundDeferred) { foundDeferred = true; } else if (hasId && hasFrameNumber && foundDeferred) { [] () { FAIL(); }(); } return foundDeferred; } bool SurfaceInterceptorTest::surfaceUpdateFound(Trace* trace, SurfaceChange::SurfaceChangeCase changeCase) { bool foundUpdate = false; for (const auto& increment : *trace->mutable_increment()) { if (increment.increment_case() == increment.kTransaction) { for (const auto& change : increment.transaction().surface_change()) { if (change.id() == mBGLayerId && change.SurfaceChange_case() == changeCase) { switch (changeCase) { case SurfaceChange::SurfaceChangeCase::kPosition: // foundUpdate is sent for the tests to fail on duplicated increments foundUpdate = positionUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kSize: foundUpdate = sizeUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kAlpha: foundUpdate = alphaUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kLayer: foundUpdate = layerUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kCrop: foundUpdate = cropUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kFinalCrop: foundUpdate = finalCropUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kMatrix: foundUpdate = matrixUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kOverrideScalingMode: foundUpdate = scalingModeUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kTransparentRegionHint: foundUpdate = transparentRegionHintUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kLayerStack: foundUpdate = layerStackUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kHiddenFlag: foundUpdate = hiddenFlagUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kOpaqueFlag: foundUpdate = opaqueFlagUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kSecureFlag: foundUpdate = secureFlagUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::kDeferredTransaction: foundUpdate = deferredTransactionUpdateFound(change, foundUpdate); break; case SurfaceChange::SurfaceChangeCase::SURFACECHANGE_NOT_SET: break; } } } } } return foundUpdate; } void SurfaceInterceptorTest::assertAllUpdatesFound(Trace* trace) { ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kPosition)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kSize)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kAlpha)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kLayer)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kCrop)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kFinalCrop)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kMatrix)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kOverrideScalingMode)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kTransparentRegionHint)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kLayerStack)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kHiddenFlag)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kOpaqueFlag)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kSecureFlag)); ASSERT_TRUE(surfaceUpdateFound(trace, SurfaceChange::SurfaceChangeCase::kDeferredTransaction)); } bool SurfaceInterceptorTest::surfaceCreationFound(const Increment& increment, bool foundSurface) { bool isMatch(increment.surface_creation().name() == LAYER_NAME && increment.surface_creation().w() == SIZE_UPDATE && increment.surface_creation().h() == SIZE_UPDATE); if (isMatch && !foundSurface) { foundSurface = true; } else if (isMatch && foundSurface) { [] () { FAIL(); }(); } return foundSurface; } bool SurfaceInterceptorTest::surfaceDeletionFound(const Increment& increment, bool foundSurface) { bool isMatch(increment.surface_deletion().id() == mTargetId); if (isMatch && !foundSurface) { foundSurface = true; } else if (isMatch && foundSurface) { [] () { FAIL(); }(); } return foundSurface; } bool SurfaceInterceptorTest::displayCreationFound(const Increment& increment, bool foundDisplay) { bool isMatch(increment.display_creation().name() == DISPLAY_NAME.string() && increment.display_creation().is_secure()); if (isMatch && !foundDisplay) { foundDisplay = true; } else if (isMatch && foundDisplay) { [] () { FAIL(); }(); } return foundDisplay; } bool SurfaceInterceptorTest::displayDeletionFound(const Increment& increment, bool foundDisplay) { bool isMatch(increment.display_deletion().id() == mTargetId); if (isMatch && !foundDisplay) { foundDisplay = true; } else if (isMatch && foundDisplay) { [] () { FAIL(); }(); } return foundDisplay; } bool SurfaceInterceptorTest::singleIncrementFound(Trace* trace, Increment::IncrementCase incrementCase) { bool foundIncrement = false; for (const auto& increment : *trace->mutable_increment()) { if (increment.increment_case() == incrementCase) { switch (incrementCase) { case Increment::IncrementCase::kSurfaceCreation: foundIncrement = surfaceCreationFound(increment, foundIncrement); break; case Increment::IncrementCase::kSurfaceDeletion: foundIncrement = surfaceDeletionFound(increment, foundIncrement); break; case Increment::IncrementCase::kDisplayCreation: foundIncrement = displayCreationFound(increment, foundIncrement); break; case Increment::IncrementCase::kDisplayDeletion: foundIncrement = displayDeletionFound(increment, foundIncrement); break; default: /* code */ break; } } } return foundIncrement; } bool SurfaceInterceptorTest::bufferUpdatesFound(Trace* trace) { uint32_t updates = 0; for (const auto& inc : *trace->mutable_increment()) { if (inc.increment_case() == inc.kBufferUpdate && inc.buffer_update().id() == mBGLayerId) { updates++; } } return updates == BUFFER_UPDATES; } TEST_F(SurfaceInterceptorTest, InterceptPositionUpdateWorks) { captureTest(&SurfaceInterceptorTest::positionUpdate, SurfaceChange::SurfaceChangeCase::kPosition); } TEST_F(SurfaceInterceptorTest, InterceptSizeUpdateWorks) { captureTest(&SurfaceInterceptorTest::sizeUpdate, SurfaceChange::SurfaceChangeCase::kSize); } TEST_F(SurfaceInterceptorTest, InterceptAlphaUpdateWorks) { captureTest(&SurfaceInterceptorTest::alphaUpdate, SurfaceChange::SurfaceChangeCase::kAlpha); } TEST_F(SurfaceInterceptorTest, InterceptLayerUpdateWorks) { captureTest(&SurfaceInterceptorTest::layerUpdate, SurfaceChange::SurfaceChangeCase::kLayer); } TEST_F(SurfaceInterceptorTest, InterceptCropUpdateWorks) { captureTest(&SurfaceInterceptorTest::cropUpdate, SurfaceChange::SurfaceChangeCase::kCrop); } TEST_F(SurfaceInterceptorTest, InterceptFinalCropUpdateWorks) { captureTest(&SurfaceInterceptorTest::finalCropUpdate, SurfaceChange::SurfaceChangeCase::kFinalCrop); } TEST_F(SurfaceInterceptorTest, InterceptMatrixUpdateWorks) { captureTest(&SurfaceInterceptorTest::matrixUpdate, SurfaceChange::SurfaceChangeCase::kMatrix); } TEST_F(SurfaceInterceptorTest, InterceptOverrideScalingModeUpdateWorks) { captureTest(&SurfaceInterceptorTest::overrideScalingModeUpdate, SurfaceChange::SurfaceChangeCase::kOverrideScalingMode); } TEST_F(SurfaceInterceptorTest, InterceptTransparentRegionHintUpdateWorks) { captureTest(&SurfaceInterceptorTest::transparentRegionHintUpdate, SurfaceChange::SurfaceChangeCase::kTransparentRegionHint); } TEST_F(SurfaceInterceptorTest, InterceptLayerStackUpdateWorks) { captureTest(&SurfaceInterceptorTest::layerStackUpdate, SurfaceChange::SurfaceChangeCase::kLayerStack); } TEST_F(SurfaceInterceptorTest, InterceptHiddenFlagUpdateWorks) { captureTest(&SurfaceInterceptorTest::hiddenFlagUpdate, SurfaceChange::SurfaceChangeCase::kHiddenFlag); } TEST_F(SurfaceInterceptorTest, InterceptOpaqueFlagUpdateWorks) { captureTest(&SurfaceInterceptorTest::opaqueFlagUpdate, SurfaceChange::SurfaceChangeCase::kOpaqueFlag); } TEST_F(SurfaceInterceptorTest, InterceptSecureFlagUpdateWorks) { captureTest(&SurfaceInterceptorTest::secureFlagUpdate, SurfaceChange::SurfaceChangeCase::kSecureFlag); } TEST_F(SurfaceInterceptorTest, InterceptDeferredTransactionUpdateWorks) { captureTest(&SurfaceInterceptorTest::deferredTransactionUpdate, SurfaceChange::SurfaceChangeCase::kDeferredTransaction); } TEST_F(SurfaceInterceptorTest, InterceptAllUpdatesWorks) { enableInterceptor(); runAllUpdates(); disableInterceptor(); // Find all of the updates in the single trace Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); assertAllUpdatesFound(&capturedTrace); } TEST_F(SurfaceInterceptorTest, InterceptSurfaceCreationWorks) { captureTest(&SurfaceInterceptorTest::surfaceCreation, Increment::IncrementCase::kSurfaceCreation); } TEST_F(SurfaceInterceptorTest, InterceptSurfaceDeletionWorks) { sp<SurfaceControl> layerToDelete = mComposerClient->createSurface(String8(LAYER_NAME), SIZE_UPDATE, SIZE_UPDATE, PIXEL_FORMAT_RGBA_8888, 0); this->mTargetId = getSurfaceId(LAYER_NAME); enableInterceptor(); mComposerClient->destroySurface(layerToDelete->getHandle()); disableInterceptor(); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); ASSERT_TRUE(singleIncrementFound(&capturedTrace, Increment::IncrementCase::kSurfaceDeletion)); } TEST_F(SurfaceInterceptorTest, InterceptDisplayCreationWorks) { captureTest(&SurfaceInterceptorTest::displayCreation, Increment::IncrementCase::kDisplayCreation); } TEST_F(SurfaceInterceptorTest, InterceptDisplayDeletionWorks) { captureTest(&SurfaceInterceptorTest::displayDeletion, Increment::IncrementCase::kDisplayDeletion); } TEST_F(SurfaceInterceptorTest, InterceptBufferUpdateWorks) { nBufferUpdates(); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); ASSERT_TRUE(bufferUpdatesFound(&capturedTrace)); } // If the interceptor is enabled while buffer updates are being pushed, the interceptor should // first create a snapshot of the existing displays and surfaces and then start capturing // the buffer updates TEST_F(SurfaceInterceptorTest, InterceptWhileBufferUpdatesWorks) { std::thread bufferUpdates(&SurfaceInterceptorTest::nBufferUpdates, this); enableInterceptor(); disableInterceptor(); bufferUpdates.join(); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); const auto& firstIncrement = capturedTrace.mutable_increment(0); ASSERT_EQ(firstIncrement->increment_case(), Increment::IncrementCase::kDisplayCreation); } TEST_F(SurfaceInterceptorTest, InterceptSimultaneousUpdatesWorks) { enableInterceptor(); std::thread bufferUpdates(&SurfaceInterceptorTest::nBufferUpdates, this); std::thread surfaceUpdates(&SurfaceInterceptorTest::runAllUpdates, this); runInTransaction(&SurfaceInterceptorTest::surfaceCreation); bufferUpdates.join(); surfaceUpdates.join(); disableInterceptor(); Trace capturedTrace; ASSERT_EQ(NO_ERROR, readProtoFile(&capturedTrace)); assertAllUpdatesFound(&capturedTrace); ASSERT_TRUE(bufferUpdatesFound(&capturedTrace)); ASSERT_TRUE(singleIncrementFound(&capturedTrace, Increment::IncrementCase::kSurfaceCreation)); } }