/* * Copyright 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. */ #ifndef ANDROID_SURFACEREPLAYER_H #define ANDROID_SURFACEREPLAYER_H #include "BufferQueueScheduler.h" #include "Color.h" #include "Event.h" #include <frameworks/native/cmds/surfacereplayer/proto/src/trace.pb.h> #include <gui/SurfaceComposerClient.h> #include <gui/SurfaceControl.h> #include <utils/Errors.h> #include <utils/StrongPointer.h> #include <stdatomic.h> #include <condition_variable> #include <memory> #include <mutex> #include <queue> #include <thread> #include <unordered_map> #include <utility> namespace android { const auto DEFAULT_PATH = "/data/local/tmp/SurfaceTrace.dat"; const auto RAND_COLOR_SEED = 700; const auto DEFAULT_THREADS = 3; typedef int32_t layer_id; typedef int32_t display_id; typedef google::protobuf::RepeatedPtrField<SurfaceChange> SurfaceChanges; typedef google::protobuf::RepeatedPtrField<DisplayChange> DisplayChanges; class Replayer { public: Replayer(const std::string& filename, bool replayManually = false, int numThreads = DEFAULT_THREADS, bool wait = true, nsecs_t stopHere = -1); Replayer(const Trace& trace, bool replayManually = false, int numThreads = DEFAULT_THREADS, bool wait = true, nsecs_t stopHere = -1); status_t replay(); private: status_t initReplay(); void waitForConsoleCommmand(); static void stopAutoReplayHandler(int signal); status_t dispatchEvent(int index); status_t doTransaction(const Transaction& transaction, const std::shared_ptr<Event>& event); status_t createSurfaceControl(const SurfaceCreation& create, const std::shared_ptr<Event>& event); status_t injectVSyncEvent(const VSyncEvent& vsyncEvent, const std::shared_ptr<Event>& event); void createDisplay(const DisplayCreation& create, const std::shared_ptr<Event>& event); void deleteDisplay(const DisplayDeletion& delete_, const std::shared_ptr<Event>& event); void updatePowerMode(const PowerModeUpdate& update, const std::shared_ptr<Event>& event); status_t doSurfaceTransaction(SurfaceComposerClient::Transaction& transaction, const SurfaceChanges& surfaceChange); void doDisplayTransaction(SurfaceComposerClient::Transaction& transaction, const DisplayChanges& displayChange); void setPosition(SurfaceComposerClient::Transaction& t, layer_id id, const PositionChange& pc); void setSize(SurfaceComposerClient::Transaction& t, layer_id id, const SizeChange& sc); void setAlpha(SurfaceComposerClient::Transaction& t, layer_id id, const AlphaChange& ac); void setLayer(SurfaceComposerClient::Transaction& t, layer_id id, const LayerChange& lc); void setCrop(SurfaceComposerClient::Transaction& t, layer_id id, const CropChange& cc); void setCornerRadius(SurfaceComposerClient::Transaction& t, layer_id id, const CornerRadiusChange& cc); void setMatrix(SurfaceComposerClient::Transaction& t, layer_id id, const MatrixChange& mc); void setOverrideScalingMode(SurfaceComposerClient::Transaction& t, layer_id id, const OverrideScalingModeChange& osmc); void setTransparentRegionHint(SurfaceComposerClient::Transaction& t, layer_id id, const TransparentRegionHintChange& trgc); void setLayerStack(SurfaceComposerClient::Transaction& t, layer_id id, const LayerStackChange& lsc); void setHiddenFlag(SurfaceComposerClient::Transaction& t, layer_id id, const HiddenFlagChange& hfc); void setOpaqueFlag(SurfaceComposerClient::Transaction& t, layer_id id, const OpaqueFlagChange& ofc); void setSecureFlag(SurfaceComposerClient::Transaction& t, layer_id id, const SecureFlagChange& sfc); void setDeferredTransaction(SurfaceComposerClient::Transaction& t, layer_id id, const DeferredTransactionChange& dtc); void setDisplaySurface(SurfaceComposerClient::Transaction& t, display_id id, const DispSurfaceChange& dsc); void setDisplayLayerStack(SurfaceComposerClient::Transaction& t, display_id id, const LayerStackChange& lsc); void setDisplaySize(SurfaceComposerClient::Transaction& t, display_id id, const SizeChange& sc); void setDisplayProjection(SurfaceComposerClient::Transaction& t, display_id id, const ProjectionChange& pc); void waitUntilTimestamp(int64_t timestamp); void waitUntilDeferredTransactionLayerExists( const DeferredTransactionChange& dtc, std::unique_lock<std::mutex>& lock); status_t loadSurfaceComposerClient(); Trace mTrace; bool mLoaded = false; int32_t mIncrementIndex = 0; int64_t mCurrentTime = 0; int32_t mNumThreads = DEFAULT_THREADS; Increment mCurrentIncrement; std::string mLastInput; static atomic_bool sReplayingManually; bool mWaitingForNextVSync; bool mWaitForTimeStamps; nsecs_t mStopTimeStamp; bool mHasStopped; std::mutex mLayerLock; std::condition_variable mLayerCond; std::unordered_map<layer_id, sp<SurfaceControl>> mLayers; std::unordered_map<layer_id, HSV> mColors; std::mutex mPendingLayersLock; std::vector<layer_id> mLayersPendingRemoval; std::mutex mBufferQueueSchedulerLock; std::unordered_map<layer_id, std::shared_ptr<BufferQueueScheduler>> mBufferQueueSchedulers; std::mutex mDisplayLock; std::condition_variable mDisplayCond; std::unordered_map<display_id, sp<IBinder>> mDisplays; sp<SurfaceComposerClient> mComposerClient; std::queue<std::shared_ptr<Event>> mPendingIncrements; }; } // namespace android #endif