|  | /* | 
|  | * Copyright (C) 2007 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. | 
|  | */ | 
|  |  | 
|  | #define LOG_TAG "SurfaceComposerClient" | 
|  |  | 
|  | #include <semaphore.h> | 
|  | #include <stdint.h> | 
|  | #include <sys/types.h> | 
|  |  | 
|  | #include <android/gui/BnWindowInfosReportedListener.h> | 
|  | #include <android/gui/DisplayState.h> | 
|  | #include <android/gui/ISurfaceComposerClient.h> | 
|  | #include <android/gui/IWindowInfosListener.h> | 
|  | #include <android/gui/TrustedPresentationThresholds.h> | 
|  | #include <android/os/IInputConstants.h> | 
|  | #include <gui/FrameRateUtils.h> | 
|  | #include <gui/TraceUtils.h> | 
|  | #include <utils/Errors.h> | 
|  | #include <utils/Log.h> | 
|  | #include <utils/SortedVector.h> | 
|  | #include <utils/String8.h> | 
|  | #include <utils/threads.h> | 
|  |  | 
|  | #include <binder/IPCThreadState.h> | 
|  | #include <binder/IServiceManager.h> | 
|  | #include <binder/ProcessState.h> | 
|  |  | 
|  | #include <system/graphics.h> | 
|  |  | 
|  | #include <gui/AidlStatusUtil.h> | 
|  | #include <gui/BufferItemConsumer.h> | 
|  | #include <gui/CpuConsumer.h> | 
|  | #include <gui/IGraphicBufferProducer.h> | 
|  | #include <gui/ISurfaceComposer.h> | 
|  | #include <gui/LayerState.h> | 
|  | #include <gui/Surface.h> | 
|  | #include <gui/SurfaceComposerClient.h> | 
|  | #include <gui/WindowInfo.h> | 
|  | #include <private/gui/ParcelUtils.h> | 
|  | #include <ui/DisplayMode.h> | 
|  | #include <ui/DisplayState.h> | 
|  | #include <ui/DynamicDisplayInfo.h> | 
|  |  | 
|  | #include <android-base/thread_annotations.h> | 
|  | #include <gui/LayerStatePermissions.h> | 
|  | #include <private/gui/ComposerService.h> | 
|  | #include <private/gui/ComposerServiceAIDL.h> | 
|  |  | 
|  | // This server size should always be smaller than the server cache size | 
|  | #define BUFFER_CACHE_MAX_SIZE 4096 | 
|  |  | 
|  | namespace android { | 
|  |  | 
|  | using aidl::android::hardware::graphics::common::DisplayDecorationSupport; | 
|  | using gui::FocusRequest; | 
|  | using gui::IRegionSamplingListener; | 
|  | using gui::TrustedPresentationThresholds; | 
|  | using gui::WindowInfo; | 
|  | using gui::WindowInfoHandle; | 
|  | using gui::WindowInfosListener; | 
|  | using gui::aidl_utils::statusTFromBinderStatus; | 
|  | using ui::ColorMode; | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | ANDROID_SINGLETON_STATIC_INSTANCE(ComposerService); | 
|  | ANDROID_SINGLETON_STATIC_INSTANCE(ComposerServiceAIDL); | 
|  |  | 
|  | namespace { | 
|  | // Initialize transaction id counter used to generate transaction ids | 
|  | std::atomic<uint32_t> idCounter = 0; | 
|  | int64_t generateId() { | 
|  | return (((int64_t)getpid()) << 32) | ++idCounter; | 
|  | } | 
|  |  | 
|  | void emptyCallback(nsecs_t, const sp<Fence>&, const std::vector<SurfaceControlStats>&) {} | 
|  | } // namespace | 
|  |  | 
|  | ComposerService::ComposerService() | 
|  | : Singleton<ComposerService>() { | 
|  | Mutex::Autolock _l(mLock); | 
|  | connectLocked(); | 
|  | } | 
|  |  | 
|  | bool ComposerService::connectLocked() { | 
|  | const String16 name("SurfaceFlinger"); | 
|  | mComposerService = waitForService<ISurfaceComposer>(name); | 
|  | if (mComposerService == nullptr) { | 
|  | return false; // fatal error or permission problem | 
|  | } | 
|  |  | 
|  | // Create the death listener. | 
|  | class DeathObserver : public IBinder::DeathRecipient { | 
|  | ComposerService& mComposerService; | 
|  | virtual void binderDied(const wp<IBinder>& who) { | 
|  | ALOGW("ComposerService remote (surfaceflinger) died [%p]", | 
|  | who.unsafe_get()); | 
|  | mComposerService.composerServiceDied(); | 
|  | } | 
|  | public: | 
|  | explicit DeathObserver(ComposerService& mgr) : mComposerService(mgr) { } | 
|  | }; | 
|  |  | 
|  | mDeathObserver = new DeathObserver(*const_cast<ComposerService*>(this)); | 
|  | IInterface::asBinder(mComposerService)->linkToDeath(mDeathObserver); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /*static*/ sp<ISurfaceComposer> ComposerService::getComposerService() { | 
|  | ComposerService& instance = ComposerService::getInstance(); | 
|  | Mutex::Autolock _l(instance.mLock); | 
|  | if (instance.mComposerService == nullptr) { | 
|  | if (ComposerService::getInstance().connectLocked()) { | 
|  | ALOGD("ComposerService reconnected"); | 
|  | } | 
|  | } | 
|  | return instance.mComposerService; | 
|  | } | 
|  |  | 
|  | void ComposerService::composerServiceDied() | 
|  | { | 
|  | Mutex::Autolock _l(mLock); | 
|  | mComposerService = nullptr; | 
|  | mDeathObserver = nullptr; | 
|  | } | 
|  |  | 
|  | ComposerServiceAIDL::ComposerServiceAIDL() : Singleton<ComposerServiceAIDL>() { | 
|  | std::scoped_lock lock(mMutex); | 
|  | connectLocked(); | 
|  | } | 
|  |  | 
|  | bool ComposerServiceAIDL::connectLocked() { | 
|  | const String16 name("SurfaceFlingerAIDL"); | 
|  | mComposerService = waitForService<gui::ISurfaceComposer>(name); | 
|  | if (mComposerService == nullptr) { | 
|  | return false; // fatal error or permission problem | 
|  | } | 
|  |  | 
|  | // Create the death listener. | 
|  | class DeathObserver : public IBinder::DeathRecipient { | 
|  | ComposerServiceAIDL& mComposerService; | 
|  | virtual void binderDied(const wp<IBinder>& who) { | 
|  | ALOGW("ComposerService aidl remote (surfaceflinger) died [%p]", who.unsafe_get()); | 
|  | mComposerService.composerServiceDied(); | 
|  | } | 
|  |  | 
|  | public: | 
|  | explicit DeathObserver(ComposerServiceAIDL& mgr) : mComposerService(mgr) {} | 
|  | }; | 
|  |  | 
|  | mDeathObserver = new DeathObserver(*const_cast<ComposerServiceAIDL*>(this)); | 
|  | IInterface::asBinder(mComposerService)->linkToDeath(mDeathObserver); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /*static*/ sp<gui::ISurfaceComposer> ComposerServiceAIDL::getComposerService() { | 
|  | ComposerServiceAIDL& instance = ComposerServiceAIDL::getInstance(); | 
|  | std::scoped_lock lock(instance.mMutex); | 
|  | if (instance.mComposerService == nullptr) { | 
|  | if (ComposerServiceAIDL::getInstance().connectLocked()) { | 
|  | ALOGD("ComposerServiceAIDL reconnected"); | 
|  | WindowInfosListenerReporter::getInstance()->reconnect(instance.mComposerService); | 
|  | } | 
|  | } | 
|  | return instance.mComposerService; | 
|  | } | 
|  |  | 
|  | void ComposerServiceAIDL::composerServiceDied() { | 
|  | std::scoped_lock lock(mMutex); | 
|  | mComposerService = nullptr; | 
|  | mDeathObserver = nullptr; | 
|  | } | 
|  |  | 
|  | class DefaultComposerClient: public Singleton<DefaultComposerClient> { | 
|  | Mutex mLock; | 
|  | sp<SurfaceComposerClient> mClient; | 
|  | friend class Singleton<ComposerService>; | 
|  | public: | 
|  | static sp<SurfaceComposerClient> getComposerClient() { | 
|  | DefaultComposerClient& dc = DefaultComposerClient::getInstance(); | 
|  | Mutex::Autolock _l(dc.mLock); | 
|  | if (dc.mClient == nullptr) { | 
|  | dc.mClient = new SurfaceComposerClient; | 
|  | } | 
|  | return dc.mClient; | 
|  | } | 
|  | }; | 
|  | ANDROID_SINGLETON_STATIC_INSTANCE(DefaultComposerClient); | 
|  |  | 
|  |  | 
|  | sp<SurfaceComposerClient> SurfaceComposerClient::getDefault() { | 
|  | return DefaultComposerClient::getComposerClient(); | 
|  | } | 
|  |  | 
|  | JankDataListener::~JankDataListener() { | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | // TransactionCompletedListener does not use ANDROID_SINGLETON_STATIC_INSTANCE because it needs | 
|  | // to be able to return a sp<> to its instance to pass to SurfaceFlinger. | 
|  | // ANDROID_SINGLETON_STATIC_INSTANCE only allows a reference to an instance. | 
|  |  | 
|  | // 0 is an invalid callback id | 
|  | TransactionCompletedListener::TransactionCompletedListener() : mCallbackIdCounter(1) {} | 
|  |  | 
|  | int64_t TransactionCompletedListener::getNextIdLocked() { | 
|  | return mCallbackIdCounter++; | 
|  | } | 
|  |  | 
|  | sp<TransactionCompletedListener> TransactionCompletedListener::sInstance = nullptr; | 
|  | static std::mutex sListenerInstanceMutex; | 
|  |  | 
|  | void TransactionCompletedListener::setInstance(const sp<TransactionCompletedListener>& listener) { | 
|  | sInstance = listener; | 
|  | } | 
|  |  | 
|  | sp<TransactionCompletedListener> TransactionCompletedListener::getInstance() { | 
|  | std::lock_guard<std::mutex> lock(sListenerInstanceMutex); | 
|  | if (sInstance == nullptr) { | 
|  | sInstance = new TransactionCompletedListener; | 
|  | } | 
|  | return sInstance; | 
|  | } | 
|  |  | 
|  | sp<ITransactionCompletedListener> TransactionCompletedListener::getIInstance() { | 
|  | return static_cast<sp<ITransactionCompletedListener>>(getInstance()); | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::startListeningLocked() { | 
|  | if (mListening) { | 
|  | return; | 
|  | } | 
|  | ProcessState::self()->startThreadPool(); | 
|  | mListening = true; | 
|  | } | 
|  |  | 
|  | CallbackId TransactionCompletedListener::addCallbackFunction( | 
|  | const TransactionCompletedCallback& callbackFunction, | 
|  | const std::unordered_set<sp<SurfaceControl>, SurfaceComposerClient::SCHash>& | 
|  | surfaceControls, | 
|  | CallbackId::Type callbackType) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  | return addCallbackFunctionLocked(callbackFunction, surfaceControls, callbackType); | 
|  | } | 
|  |  | 
|  | CallbackId TransactionCompletedListener::addCallbackFunctionLocked( | 
|  | const TransactionCompletedCallback& callbackFunction, | 
|  | const std::unordered_set<sp<SurfaceControl>, SurfaceComposerClient::SCHash>& | 
|  | surfaceControls, | 
|  | CallbackId::Type callbackType) { | 
|  | startListeningLocked(); | 
|  |  | 
|  | CallbackId callbackId(getNextIdLocked(), callbackType); | 
|  | mCallbacks[callbackId].callbackFunction = callbackFunction; | 
|  | auto& callbackSurfaceControls = mCallbacks[callbackId].surfaceControls; | 
|  |  | 
|  | for (const auto& surfaceControl : surfaceControls) { | 
|  | callbackSurfaceControls[surfaceControl->getHandle()] = surfaceControl; | 
|  |  | 
|  | if (callbackType == CallbackId::Type::ON_COMPLETE && | 
|  | mJankListeners.count(surfaceControl->getLayerId()) != 0) { | 
|  | callbackId.includeJankData = true; | 
|  | } | 
|  | } | 
|  |  | 
|  | return callbackId; | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::addJankListener(const sp<JankDataListener>& listener, | 
|  | sp<SurfaceControl> surfaceControl) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  | mJankListeners.insert({surfaceControl->getLayerId(), listener}); | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::removeJankListener(const sp<JankDataListener>& listener) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  | for (auto it = mJankListeners.begin(); it != mJankListeners.end();) { | 
|  | if (it->second == listener) { | 
|  | it = mJankListeners.erase(it); | 
|  | } else { | 
|  | it++; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::setReleaseBufferCallback(const ReleaseCallbackId& callbackId, | 
|  | ReleaseBufferCallback listener) { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | mReleaseBufferCallbacks[callbackId] = listener; | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::addSurfaceStatsListener(void* context, void* cookie, | 
|  | sp<SurfaceControl> surfaceControl, SurfaceStatsCallback listener) { | 
|  | std::scoped_lock<std::recursive_mutex> lock(mSurfaceStatsListenerMutex); | 
|  | mSurfaceStatsListeners.insert( | 
|  | {surfaceControl->getLayerId(), SurfaceStatsCallbackEntry(context, cookie, listener)}); | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::removeSurfaceStatsListener(void* context, void* cookie) { | 
|  | std::scoped_lock<std::recursive_mutex> lock(mSurfaceStatsListenerMutex); | 
|  | for (auto it = mSurfaceStatsListeners.begin(); it != mSurfaceStatsListeners.end();) { | 
|  | auto [itContext, itCookie, itListener] = it->second; | 
|  | if (itContext == context && itCookie == cookie) { | 
|  | it = mSurfaceStatsListeners.erase(it); | 
|  | } else { | 
|  | it++; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::addSurfaceControlToCallbacks( | 
|  | SurfaceComposerClient::CallbackInfo& callbackInfo, | 
|  | const sp<SurfaceControl>& surfaceControl) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  |  | 
|  | bool includingJankData = false; | 
|  | for (auto callbackId : callbackInfo.callbackIds) { | 
|  | mCallbacks[callbackId].surfaceControls.emplace(std::piecewise_construct, | 
|  | std::forward_as_tuple( | 
|  | surfaceControl->getHandle()), | 
|  | std::forward_as_tuple(surfaceControl)); | 
|  | includingJankData = includingJankData || callbackId.includeJankData; | 
|  | } | 
|  |  | 
|  | // If no registered callback is requesting jank data, but there is a jank listener registered | 
|  | // on the new surface control, add a synthetic callback that requests the jank data. | 
|  | if (!includingJankData && mJankListeners.count(surfaceControl->getLayerId()) != 0) { | 
|  | CallbackId callbackId = | 
|  | addCallbackFunctionLocked(&emptyCallback, callbackInfo.surfaceControls, | 
|  | CallbackId::Type::ON_COMPLETE); | 
|  | callbackInfo.callbackIds.emplace(callbackId); | 
|  | } | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::onTransactionCompleted(ListenerStats listenerStats) { | 
|  | std::unordered_map<CallbackId, CallbackTranslation, CallbackIdHash> callbacksMap; | 
|  | std::multimap<int32_t, sp<JankDataListener>> jankListenersMap; | 
|  | { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  |  | 
|  | /* This listener knows all the sp<IBinder> to sp<SurfaceControl> for all its registered | 
|  | * callbackIds, except for when Transactions are merged together. This probably cannot be | 
|  | * solved before this point because the Transactions could be merged together and applied in | 
|  | * a different process. | 
|  | * | 
|  | * Fortunately, we get all the callbacks for this listener for the same frame together at | 
|  | * the same time. This means if any Transactions were merged together, we will get their | 
|  | * callbacks at the same time. We can combine all the sp<IBinder> to sp<SurfaceControl> maps | 
|  | * for all the callbackIds to generate one super map that contains all the sp<IBinder> to | 
|  | * sp<SurfaceControl> that could possibly exist for the callbacks. | 
|  | */ | 
|  | callbacksMap = mCallbacks; | 
|  | jankListenersMap = mJankListeners; | 
|  | for (const auto& transactionStats : listenerStats.transactionStats) { | 
|  | for (auto& callbackId : transactionStats.callbackIds) { | 
|  | mCallbacks.erase(callbackId); | 
|  | } | 
|  | } | 
|  | } | 
|  | for (const auto& transactionStats : listenerStats.transactionStats) { | 
|  | // handle on commit callbacks | 
|  | for (auto callbackId : transactionStats.callbackIds) { | 
|  | if (callbackId.type != CallbackId::Type::ON_COMMIT) { | 
|  | continue; | 
|  | } | 
|  | auto& [callbackFunction, callbackSurfaceControls] = callbacksMap[callbackId]; | 
|  | if (!callbackFunction) { | 
|  | continue; | 
|  | } | 
|  | std::vector<SurfaceControlStats> surfaceControlStats; | 
|  | for (const auto& surfaceStats : transactionStats.surfaceStats) { | 
|  | surfaceControlStats | 
|  | .emplace_back(callbacksMap[callbackId] | 
|  | .surfaceControls[surfaceStats.surfaceControl], | 
|  | transactionStats.latchTime, surfaceStats.acquireTimeOrFence, | 
|  | transactionStats.presentFence, | 
|  | surfaceStats.previousReleaseFence, surfaceStats.transformHint, | 
|  | surfaceStats.eventStats, | 
|  | surfaceStats.currentMaxAcquiredBufferCount); | 
|  | } | 
|  |  | 
|  | callbackFunction(transactionStats.latchTime, transactionStats.presentFence, | 
|  | surfaceControlStats); | 
|  |  | 
|  | // More than one transaction may contain the same callback id. Erase the callback from | 
|  | // the map to ensure that it is only called once. This can happen if transactions are | 
|  | // parcelled out of process and applied in both processes. | 
|  | callbacksMap.erase(callbackId); | 
|  | } | 
|  |  | 
|  | // handle on complete callbacks | 
|  | for (auto callbackId : transactionStats.callbackIds) { | 
|  | if (callbackId.type != CallbackId::Type::ON_COMPLETE) { | 
|  | continue; | 
|  | } | 
|  | auto& [callbackFunction, callbackSurfaceControls] = callbacksMap[callbackId]; | 
|  | if (!callbackFunction) { | 
|  | ALOGE("cannot call null callback function, skipping"); | 
|  | continue; | 
|  | } | 
|  | std::vector<SurfaceControlStats> surfaceControlStats; | 
|  | for (const auto& surfaceStats : transactionStats.surfaceStats) { | 
|  | surfaceControlStats | 
|  | .emplace_back(callbacksMap[callbackId] | 
|  | .surfaceControls[surfaceStats.surfaceControl], | 
|  | transactionStats.latchTime, surfaceStats.acquireTimeOrFence, | 
|  | transactionStats.presentFence, | 
|  | surfaceStats.previousReleaseFence, surfaceStats.transformHint, | 
|  | surfaceStats.eventStats, | 
|  | surfaceStats.currentMaxAcquiredBufferCount); | 
|  | if (callbacksMap[callbackId].surfaceControls[surfaceStats.surfaceControl] && | 
|  | surfaceStats.transformHint.has_value()) { | 
|  | callbacksMap[callbackId] | 
|  | .surfaceControls[surfaceStats.surfaceControl] | 
|  | ->setTransformHint(*surfaceStats.transformHint); | 
|  | } | 
|  | // If there is buffer id set, we look up any pending client release buffer callbacks | 
|  | // and call them. This is a performance optimization when we have a transaction | 
|  | // callback and a release buffer callback happening at the same time to avoid an | 
|  | // additional ipc call from the server. | 
|  | if (surfaceStats.previousReleaseCallbackId != ReleaseCallbackId::INVALID_ID) { | 
|  | ReleaseBufferCallback callback; | 
|  | { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | callback = popReleaseBufferCallbackLocked( | 
|  | surfaceStats.previousReleaseCallbackId); | 
|  | } | 
|  | if (callback) { | 
|  | callback(surfaceStats.previousReleaseCallbackId, | 
|  | surfaceStats.previousReleaseFence | 
|  | ? surfaceStats.previousReleaseFence | 
|  | : Fence::NO_FENCE, | 
|  | surfaceStats.currentMaxAcquiredBufferCount); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | callbackFunction(transactionStats.latchTime, transactionStats.presentFence, | 
|  | surfaceControlStats); | 
|  | } | 
|  | } | 
|  |  | 
|  | for (const auto& transactionStats : listenerStats.transactionStats) { | 
|  | for (const auto& surfaceStats : transactionStats.surfaceStats) { | 
|  | // The callbackMap contains the SurfaceControl object, which we need to look up the | 
|  | // layerId. Since we don't know which callback contains the SurfaceControl, iterate | 
|  | // through all until the SC is found. | 
|  | int32_t layerId = -1; | 
|  | for (auto callbackId : transactionStats.callbackIds) { | 
|  | if (callbackId.type != CallbackId::Type::ON_COMPLETE) { | 
|  | // We only want to run the stats callback for ON_COMPLETE | 
|  | continue; | 
|  | } | 
|  | sp<SurfaceControl> sc = | 
|  | callbacksMap[callbackId].surfaceControls[surfaceStats.surfaceControl]; | 
|  | if (sc != nullptr) { | 
|  | layerId = sc->getLayerId(); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (layerId != -1) { | 
|  | // Acquire surface stats listener lock such that we guarantee that after calling | 
|  | // unregister, there won't be any further callback. | 
|  | std::scoped_lock<std::recursive_mutex> lock(mSurfaceStatsListenerMutex); | 
|  | auto listenerRange = mSurfaceStatsListeners.equal_range(layerId); | 
|  | for (auto it = listenerRange.first; it != listenerRange.second; it++) { | 
|  | auto entry = it->second; | 
|  | entry.callback(entry.context, transactionStats.latchTime, | 
|  | transactionStats.presentFence, surfaceStats); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (surfaceStats.jankData.empty()) continue; | 
|  | auto jankRange = jankListenersMap.equal_range(layerId); | 
|  | for (auto it = jankRange.first; it != jankRange.second; it++) { | 
|  | it->second->onJankDataAvailable(surfaceStats.jankData); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::onTransactionQueueStalled(const String8& reason) { | 
|  | std::unordered_map<void*, std::function<void(const std::string&)>> callbackCopy; | 
|  | { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | callbackCopy = mQueueStallListeners; | 
|  | } | 
|  | for (auto const& it : callbackCopy) { | 
|  | it.second(reason.c_str()); | 
|  | } | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::addQueueStallListener( | 
|  | std::function<void(const std::string&)> stallListener, void* id) { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | mQueueStallListeners[id] = stallListener; | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::removeQueueStallListener(void* id) { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | mQueueStallListeners.erase(id); | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::onReleaseBuffer(ReleaseCallbackId callbackId, | 
|  | sp<Fence> releaseFence, | 
|  | uint32_t currentMaxAcquiredBufferCount) { | 
|  | ReleaseBufferCallback callback; | 
|  | { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | callback = popReleaseBufferCallbackLocked(callbackId); | 
|  | } | 
|  | if (!callback) { | 
|  | ALOGE("Could not call release buffer callback, buffer not found %s", | 
|  | callbackId.to_string().c_str()); | 
|  | return; | 
|  | } | 
|  | std::optional<uint32_t> optionalMaxAcquiredBufferCount = | 
|  | currentMaxAcquiredBufferCount == UINT_MAX | 
|  | ? std::nullopt | 
|  | : std::make_optional<uint32_t>(currentMaxAcquiredBufferCount); | 
|  | callback(callbackId, releaseFence, optionalMaxAcquiredBufferCount); | 
|  | } | 
|  |  | 
|  | ReleaseBufferCallback TransactionCompletedListener::popReleaseBufferCallbackLocked( | 
|  | const ReleaseCallbackId& callbackId) { | 
|  | ReleaseBufferCallback callback; | 
|  | auto itr = mReleaseBufferCallbacks.find(callbackId); | 
|  | if (itr == mReleaseBufferCallbacks.end()) { | 
|  | return nullptr; | 
|  | } | 
|  | callback = itr->second; | 
|  | mReleaseBufferCallbacks.erase(itr); | 
|  | return callback; | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::removeReleaseBufferCallback( | 
|  | const ReleaseCallbackId& callbackId) { | 
|  | { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | popReleaseBufferCallbackLocked(callbackId); | 
|  | } | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::PresentationCallbackRAII::PresentationCallbackRAII( | 
|  | TransactionCompletedListener* tcl, int id) { | 
|  | mTcl = tcl; | 
|  | mId = id; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::PresentationCallbackRAII::~PresentationCallbackRAII() { | 
|  | mTcl->clearTrustedPresentationCallback(mId); | 
|  | } | 
|  |  | 
|  | sp<SurfaceComposerClient::PresentationCallbackRAII> | 
|  | TransactionCompletedListener::addTrustedPresentationCallback(TrustedPresentationCallback tpc, | 
|  | int id, void* context) { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | mTrustedPresentationCallbacks[id] = | 
|  | std::tuple<TrustedPresentationCallback, void*>(tpc, context); | 
|  | return new SurfaceComposerClient::PresentationCallbackRAII(this, id); | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::clearTrustedPresentationCallback(int id) { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | mTrustedPresentationCallbacks.erase(id); | 
|  | } | 
|  |  | 
|  | void TransactionCompletedListener::onTrustedPresentationChanged(int id, | 
|  | bool presentedWithinThresholds) { | 
|  | TrustedPresentationCallback tpc; | 
|  | void* context; | 
|  | { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | auto it = mTrustedPresentationCallbacks.find(id); | 
|  | if (it == mTrustedPresentationCallbacks.end()) { | 
|  | return; | 
|  | } | 
|  | std::tie(tpc, context) = it->second; | 
|  | } | 
|  | tpc(context, presentedWithinThresholds); | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | void removeDeadBufferCallback(void* /*context*/, uint64_t graphicBufferId); | 
|  |  | 
|  | /** | 
|  | * We use the BufferCache to reduce the overhead of exchanging GraphicBuffers with | 
|  | * the server. If we were to simply parcel the GraphicBuffer we would pay two overheads | 
|  | *     1. Cost of sending the FD | 
|  | *     2. Cost of importing the GraphicBuffer with the mapper in the receiving process. | 
|  | * To ease this cost we implement the following scheme of caching buffers to integers, | 
|  | * or said-otherwise, naming them with integers. This is the scheme known as slots in | 
|  | * the legacy BufferQueue system. | 
|  | *     1. When sending Buffers to SurfaceFlinger we look up the Buffer in the cache. | 
|  | *     2. If there is a cache-hit we remove the Buffer from the Transaction and instead | 
|  | *        send the cached integer. | 
|  | *     3. If there is a cache miss, we cache the new buffer and send the integer | 
|  | *        along with the Buffer, SurfaceFlinger on it's side creates a new cache | 
|  | *        entry, and we use the integer for further communication. | 
|  | * A few details about lifetime: | 
|  | *     1. The cache evicts by LRU. The server side cache is keyed by BufferCache::getToken | 
|  | *        which is per process Unique. The server side cache is larger than the client side | 
|  | *        cache so that the server will never evict entries before the client. | 
|  | *     2. When the client evicts an entry it notifies the server via an uncacheBuffer | 
|  | *        transaction. | 
|  | *     3. The client only references the Buffers by ID, and uses buffer->addDeathCallback | 
|  | *        to auto-evict destroyed buffers. | 
|  | */ | 
|  | class BufferCache : public Singleton<BufferCache> { | 
|  | public: | 
|  | BufferCache() : token(new BBinder()) {} | 
|  |  | 
|  | sp<IBinder> getToken() { | 
|  | return IInterface::asBinder(TransactionCompletedListener::getIInstance()); | 
|  | } | 
|  |  | 
|  | status_t getCacheId(const sp<GraphicBuffer>& buffer, uint64_t* cacheId) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  |  | 
|  | auto itr = mBuffers.find(buffer->getId()); | 
|  | if (itr == mBuffers.end()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | itr->second = getCounter(); | 
|  | *cacheId = buffer->getId(); | 
|  | return NO_ERROR; | 
|  | } | 
|  |  | 
|  | uint64_t cache(const sp<GraphicBuffer>& buffer, | 
|  | std::optional<client_cache_t>& outUncacheBuffer) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  |  | 
|  | if (mBuffers.size() >= BUFFER_CACHE_MAX_SIZE) { | 
|  | outUncacheBuffer = findLeastRecentlyUsedBuffer(); | 
|  | mBuffers.erase(outUncacheBuffer->id); | 
|  | } | 
|  |  | 
|  | buffer->addDeathCallback(removeDeadBufferCallback, nullptr); | 
|  |  | 
|  | mBuffers[buffer->getId()] = getCounter(); | 
|  | return buffer->getId(); | 
|  | } | 
|  |  | 
|  | void uncache(uint64_t cacheId) { | 
|  | std::lock_guard<std::mutex> lock(mMutex); | 
|  | if (mBuffers.erase(cacheId)) { | 
|  | SurfaceComposerClient::doUncacheBufferTransaction(cacheId); | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | client_cache_t findLeastRecentlyUsedBuffer() REQUIRES(mMutex) { | 
|  | auto itr = mBuffers.begin(); | 
|  | uint64_t minCounter = itr->second; | 
|  | auto minBuffer = itr; | 
|  | itr++; | 
|  |  | 
|  | while (itr != mBuffers.end()) { | 
|  | uint64_t counter = itr->second; | 
|  | if (counter < minCounter) { | 
|  | minCounter = counter; | 
|  | minBuffer = itr; | 
|  | } | 
|  | itr++; | 
|  | } | 
|  |  | 
|  | return {.token = getToken(), .id = minBuffer->first}; | 
|  | } | 
|  |  | 
|  | uint64_t getCounter() REQUIRES(mMutex) { | 
|  | static uint64_t counter = 0; | 
|  | return counter++; | 
|  | } | 
|  |  | 
|  | std::mutex mMutex; | 
|  | std::map<uint64_t /*Cache id*/, uint64_t /*counter*/> mBuffers GUARDED_BY(mMutex); | 
|  |  | 
|  | // Used by ISurfaceComposer to identify which process is sending the cached buffer. | 
|  | sp<IBinder> token; | 
|  | }; | 
|  |  | 
|  | ANDROID_SINGLETON_STATIC_INSTANCE(BufferCache); | 
|  |  | 
|  | void removeDeadBufferCallback(void* /*context*/, uint64_t graphicBufferId) { | 
|  | // GraphicBuffer id's are used as the cache ids. | 
|  | BufferCache::getInstance().uncache(graphicBufferId); | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | SurfaceComposerClient::Transaction::Transaction() { | 
|  | mId = generateId(); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction::Transaction(const Transaction& other) | 
|  | : mId(other.mId), | 
|  | mTransactionNestCount(other.mTransactionNestCount), | 
|  | mAnimation(other.mAnimation), | 
|  | mEarlyWakeupStart(other.mEarlyWakeupStart), | 
|  | mEarlyWakeupEnd(other.mEarlyWakeupEnd), | 
|  | mMayContainBuffer(other.mMayContainBuffer), | 
|  | mDesiredPresentTime(other.mDesiredPresentTime), | 
|  | mIsAutoTimestamp(other.mIsAutoTimestamp), | 
|  | mFrameTimelineInfo(other.mFrameTimelineInfo), | 
|  | mApplyToken(other.mApplyToken) { | 
|  | mDisplayStates = other.mDisplayStates; | 
|  | mComposerStates = other.mComposerStates; | 
|  | mInputWindowCommands = other.mInputWindowCommands; | 
|  | mListenerCallbacks = other.mListenerCallbacks; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::sanitize(int pid, int uid) { | 
|  | uint32_t permissions = LayerStatePermissions::getTransactionPermissions(pid, uid); | 
|  | for (auto & [handle, composerState] : mComposerStates) { | 
|  | composerState.state.sanitize(permissions); | 
|  | } | 
|  | if (!mInputWindowCommands.empty() && | 
|  | (permissions & layer_state_t::Permission::ACCESS_SURFACE_FLINGER) == 0) { | 
|  | ALOGE("Only privileged callers are allowed to send input commands."); | 
|  | mInputWindowCommands.clear(); | 
|  | } | 
|  | } | 
|  |  | 
|  | std::unique_ptr<SurfaceComposerClient::Transaction> | 
|  | SurfaceComposerClient::Transaction::createFromParcel(const Parcel* parcel) { | 
|  | auto transaction = std::make_unique<Transaction>(); | 
|  | if (transaction->readFromParcel(parcel) == NO_ERROR) { | 
|  | return transaction; | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  |  | 
|  | status_t SurfaceComposerClient::Transaction::readFromParcel(const Parcel* parcel) { | 
|  | const uint64_t transactionId = parcel->readUint64(); | 
|  | const uint32_t transactionNestCount = parcel->readUint32(); | 
|  | const bool animation = parcel->readBool(); | 
|  | const bool earlyWakeupStart = parcel->readBool(); | 
|  | const bool earlyWakeupEnd = parcel->readBool(); | 
|  | const int64_t desiredPresentTime = parcel->readInt64(); | 
|  | const bool isAutoTimestamp = parcel->readBool(); | 
|  | FrameTimelineInfo frameTimelineInfo; | 
|  | frameTimelineInfo.readFromParcel(parcel); | 
|  |  | 
|  | sp<IBinder> applyToken; | 
|  | parcel->readNullableStrongBinder(&applyToken); | 
|  | size_t count = static_cast<size_t>(parcel->readUint32()); | 
|  | if (count > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | SortedVector<DisplayState> displayStates; | 
|  | displayStates.setCapacity(count); | 
|  | for (size_t i = 0; i < count; i++) { | 
|  | DisplayState displayState; | 
|  | if (displayState.read(*parcel) == BAD_VALUE) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | displayStates.add(displayState); | 
|  | } | 
|  |  | 
|  | count = static_cast<size_t>(parcel->readUint32()); | 
|  | if (count > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | std::unordered_map<sp<ITransactionCompletedListener>, CallbackInfo, TCLHash> listenerCallbacks; | 
|  | listenerCallbacks.reserve(count); | 
|  | for (size_t i = 0; i < count; i++) { | 
|  | sp<ITransactionCompletedListener> listener = | 
|  | interface_cast<ITransactionCompletedListener>(parcel->readStrongBinder()); | 
|  | size_t numCallbackIds = parcel->readUint32(); | 
|  | if (numCallbackIds > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | for (size_t j = 0; j < numCallbackIds; j++) { | 
|  | CallbackId id; | 
|  | parcel->readParcelable(&id); | 
|  | listenerCallbacks[listener].callbackIds.insert(id); | 
|  | } | 
|  | size_t numSurfaces = parcel->readUint32(); | 
|  | if (numSurfaces > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | for (size_t j = 0; j < numSurfaces; j++) { | 
|  | sp<SurfaceControl> surface; | 
|  | SAFE_PARCEL(SurfaceControl::readFromParcel, *parcel, &surface); | 
|  | listenerCallbacks[listener].surfaceControls.insert(surface); | 
|  | } | 
|  | } | 
|  |  | 
|  | count = static_cast<size_t>(parcel->readUint32()); | 
|  | if (count > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | std::unordered_map<sp<IBinder>, ComposerState, IBinderHash> composerStates; | 
|  | composerStates.reserve(count); | 
|  | for (size_t i = 0; i < count; i++) { | 
|  | sp<IBinder> surfaceControlHandle; | 
|  | SAFE_PARCEL(parcel->readStrongBinder, &surfaceControlHandle); | 
|  |  | 
|  | ComposerState composerState; | 
|  | if (composerState.read(*parcel) == BAD_VALUE) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | composerStates[surfaceControlHandle] = composerState; | 
|  | } | 
|  |  | 
|  | InputWindowCommands inputWindowCommands; | 
|  | inputWindowCommands.read(*parcel); | 
|  |  | 
|  | count = static_cast<size_t>(parcel->readUint32()); | 
|  | if (count > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | std::vector<client_cache_t> uncacheBuffers(count); | 
|  | for (size_t i = 0; i < count; i++) { | 
|  | sp<IBinder> tmpBinder; | 
|  | SAFE_PARCEL(parcel->readStrongBinder, &tmpBinder); | 
|  | uncacheBuffers[i].token = tmpBinder; | 
|  | SAFE_PARCEL(parcel->readUint64, &uncacheBuffers[i].id); | 
|  | } | 
|  |  | 
|  | count = static_cast<size_t>(parcel->readUint32()); | 
|  | if (count > parcel->dataSize()) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | std::vector<uint64_t> mergedTransactionIds(count); | 
|  | for (size_t i = 0; i < count; i++) { | 
|  | SAFE_PARCEL(parcel->readUint64, &mergedTransactionIds[i]); | 
|  | } | 
|  |  | 
|  | // Parsing was successful. Update the object. | 
|  | mId = transactionId; | 
|  | mTransactionNestCount = transactionNestCount; | 
|  | mAnimation = animation; | 
|  | mEarlyWakeupStart = earlyWakeupStart; | 
|  | mEarlyWakeupEnd = earlyWakeupEnd; | 
|  | mDesiredPresentTime = desiredPresentTime; | 
|  | mIsAutoTimestamp = isAutoTimestamp; | 
|  | mFrameTimelineInfo = frameTimelineInfo; | 
|  | mDisplayStates = displayStates; | 
|  | mListenerCallbacks = listenerCallbacks; | 
|  | mComposerStates = composerStates; | 
|  | mInputWindowCommands = inputWindowCommands; | 
|  | mApplyToken = applyToken; | 
|  | mUncacheBuffers = std::move(uncacheBuffers); | 
|  | mMergedTransactionIds = std::move(mergedTransactionIds); | 
|  | return NO_ERROR; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::Transaction::writeToParcel(Parcel* parcel) const { | 
|  | // If we write the Transaction to a parcel, we want to ensure the Buffers are cached | 
|  | // before crossing the IPC boundary. Otherwise the receiving party will cache the buffers | 
|  | // but is unlikely to use them again as they are owned by the other process. | 
|  | // You may be asking yourself, is this const cast safe? Const cast is safe up | 
|  | // until the point where you try and write to an object that was originally const at which | 
|  | // point we enter undefined behavior. In this case we are safe though, because there are | 
|  | // two possibilities: | 
|  | //    1. The SurfaceComposerClient::Transaction was originally non-const. Safe. | 
|  | //    2. It was originall const! In this case not only was it useless, but it by definition | 
|  | //       contains no composer states and so cacheBuffers will not perform any writes. | 
|  |  | 
|  | const_cast<SurfaceComposerClient::Transaction*>(this)->cacheBuffers(); | 
|  |  | 
|  | parcel->writeUint64(mId); | 
|  | parcel->writeUint32(mTransactionNestCount); | 
|  | parcel->writeBool(mAnimation); | 
|  | parcel->writeBool(mEarlyWakeupStart); | 
|  | parcel->writeBool(mEarlyWakeupEnd); | 
|  | parcel->writeInt64(mDesiredPresentTime); | 
|  | parcel->writeBool(mIsAutoTimestamp); | 
|  | mFrameTimelineInfo.writeToParcel(parcel); | 
|  | parcel->writeStrongBinder(mApplyToken); | 
|  | parcel->writeUint32(static_cast<uint32_t>(mDisplayStates.size())); | 
|  | for (auto const& displayState : mDisplayStates) { | 
|  | displayState.write(*parcel); | 
|  | } | 
|  |  | 
|  | parcel->writeUint32(static_cast<uint32_t>(mListenerCallbacks.size())); | 
|  | for (auto const& [listener, callbackInfo] : mListenerCallbacks) { | 
|  | parcel->writeStrongBinder(ITransactionCompletedListener::asBinder(listener)); | 
|  | parcel->writeUint32(static_cast<uint32_t>(callbackInfo.callbackIds.size())); | 
|  | for (auto callbackId : callbackInfo.callbackIds) { | 
|  | parcel->writeParcelable(callbackId); | 
|  | } | 
|  | parcel->writeUint32(static_cast<uint32_t>(callbackInfo.surfaceControls.size())); | 
|  | for (auto surfaceControl : callbackInfo.surfaceControls) { | 
|  | SAFE_PARCEL(surfaceControl->writeToParcel, *parcel); | 
|  | } | 
|  | } | 
|  |  | 
|  | parcel->writeUint32(static_cast<uint32_t>(mComposerStates.size())); | 
|  | for (auto const& [handle, composerState] : mComposerStates) { | 
|  | SAFE_PARCEL(parcel->writeStrongBinder, handle); | 
|  | composerState.write(*parcel); | 
|  | } | 
|  |  | 
|  | mInputWindowCommands.write(*parcel); | 
|  |  | 
|  | SAFE_PARCEL(parcel->writeUint32, static_cast<uint32_t>(mUncacheBuffers.size())); | 
|  | for (const client_cache_t& uncacheBuffer : mUncacheBuffers) { | 
|  | SAFE_PARCEL(parcel->writeStrongBinder, uncacheBuffer.token.promote()); | 
|  | SAFE_PARCEL(parcel->writeUint64, uncacheBuffer.id); | 
|  | } | 
|  |  | 
|  | SAFE_PARCEL(parcel->writeUint32, static_cast<uint32_t>(mMergedTransactionIds.size())); | 
|  | for (auto mergedTransactionId : mMergedTransactionIds) { | 
|  | SAFE_PARCEL(parcel->writeUint64, mergedTransactionId); | 
|  | } | 
|  |  | 
|  | return NO_ERROR; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::releaseBufferIfOverwriting(const layer_state_t& state) { | 
|  | if (!(state.what & layer_state_t::eBufferChanged) || !state.bufferData->hasBuffer()) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | auto listener = state.bufferData->releaseBufferListener; | 
|  | sp<Fence> fence = | 
|  | state.bufferData->acquireFence ? state.bufferData->acquireFence : Fence::NO_FENCE; | 
|  | if (state.bufferData->releaseBufferEndpoint == | 
|  | IInterface::asBinder(TransactionCompletedListener::getIInstance())) { | 
|  | // if the callback is in process, run on a different thread to avoid any lock contigency | 
|  | // issues in the client. | 
|  | SurfaceComposerClient::getDefault() | 
|  | ->mReleaseCallbackThread | 
|  | .addReleaseCallback(state.bufferData->generateReleaseCallbackId(), fence); | 
|  | } else { | 
|  | listener->onReleaseBuffer(state.bufferData->generateReleaseCallbackId(), fence, UINT_MAX); | 
|  | } | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::merge(Transaction&& other) { | 
|  | while (mMergedTransactionIds.size() + other.mMergedTransactionIds.size() > | 
|  | MAX_MERGE_HISTORY_LENGTH - 1 && | 
|  | mMergedTransactionIds.size() > 0) { | 
|  | mMergedTransactionIds.pop_back(); | 
|  | } | 
|  | if (other.mMergedTransactionIds.size() == MAX_MERGE_HISTORY_LENGTH) { | 
|  | mMergedTransactionIds.insert(mMergedTransactionIds.begin(), | 
|  | other.mMergedTransactionIds.begin(), | 
|  | other.mMergedTransactionIds.end() - 1); | 
|  | } else if (other.mMergedTransactionIds.size() > 0u) { | 
|  | mMergedTransactionIds.insert(mMergedTransactionIds.begin(), | 
|  | other.mMergedTransactionIds.begin(), | 
|  | other.mMergedTransactionIds.end()); | 
|  | } | 
|  | mMergedTransactionIds.insert(mMergedTransactionIds.begin(), other.mId); | 
|  |  | 
|  | for (auto const& [handle, composerState] : other.mComposerStates) { | 
|  | if (mComposerStates.count(handle) == 0) { | 
|  | mComposerStates[handle] = composerState; | 
|  | } else { | 
|  | if (composerState.state.what & layer_state_t::eBufferChanged) { | 
|  | releaseBufferIfOverwriting(mComposerStates[handle].state); | 
|  | } | 
|  | mComposerStates[handle].state.merge(composerState.state); | 
|  | } | 
|  | } | 
|  |  | 
|  | for (auto const& state : other.mDisplayStates) { | 
|  | ssize_t index = mDisplayStates.indexOf(state); | 
|  | if (index < 0) { | 
|  | mDisplayStates.add(state); | 
|  | } else { | 
|  | mDisplayStates.editItemAt(static_cast<size_t>(index)).merge(state); | 
|  | } | 
|  | } | 
|  |  | 
|  | for (const auto& [listener, callbackInfo] : other.mListenerCallbacks) { | 
|  | auto& [callbackIds, surfaceControls] = callbackInfo; | 
|  | mListenerCallbacks[listener].callbackIds.insert(std::make_move_iterator( | 
|  | callbackIds.begin()), | 
|  | std::make_move_iterator(callbackIds.end())); | 
|  |  | 
|  | mListenerCallbacks[listener].surfaceControls.insert(surfaceControls.begin(), | 
|  | surfaceControls.end()); | 
|  |  | 
|  | auto& currentProcessCallbackInfo = | 
|  | mListenerCallbacks[TransactionCompletedListener::getIInstance()]; | 
|  | currentProcessCallbackInfo.surfaceControls | 
|  | .insert(std::make_move_iterator(surfaceControls.begin()), | 
|  | std::make_move_iterator(surfaceControls.end())); | 
|  |  | 
|  | // register all surface controls for all callbackIds for this listener that is merging | 
|  | for (const auto& surfaceControl : currentProcessCallbackInfo.surfaceControls) { | 
|  | TransactionCompletedListener::getInstance() | 
|  | ->addSurfaceControlToCallbacks(currentProcessCallbackInfo, surfaceControl); | 
|  | } | 
|  | } | 
|  |  | 
|  | for (const auto& cacheId : other.mUncacheBuffers) { | 
|  | mUncacheBuffers.push_back(cacheId); | 
|  | } | 
|  |  | 
|  | mInputWindowCommands.merge(other.mInputWindowCommands); | 
|  |  | 
|  | mMayContainBuffer |= other.mMayContainBuffer; | 
|  | mEarlyWakeupStart = mEarlyWakeupStart || other.mEarlyWakeupStart; | 
|  | mEarlyWakeupEnd = mEarlyWakeupEnd || other.mEarlyWakeupEnd; | 
|  | mApplyToken = other.mApplyToken; | 
|  |  | 
|  | mergeFrameTimelineInfo(mFrameTimelineInfo, other.mFrameTimelineInfo); | 
|  |  | 
|  | other.clear(); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::clear() { | 
|  | mComposerStates.clear(); | 
|  | mDisplayStates.clear(); | 
|  | mListenerCallbacks.clear(); | 
|  | mInputWindowCommands.clear(); | 
|  | mUncacheBuffers.clear(); | 
|  | mMayContainBuffer = false; | 
|  | mTransactionNestCount = 0; | 
|  | mAnimation = false; | 
|  | mEarlyWakeupStart = false; | 
|  | mEarlyWakeupEnd = false; | 
|  | mDesiredPresentTime = 0; | 
|  | mIsAutoTimestamp = true; | 
|  | mFrameTimelineInfo = {}; | 
|  | mApplyToken = nullptr; | 
|  | mMergedTransactionIds.clear(); | 
|  | } | 
|  |  | 
|  | uint64_t SurfaceComposerClient::Transaction::getId() { | 
|  | return mId; | 
|  | } | 
|  |  | 
|  | std::vector<uint64_t> SurfaceComposerClient::Transaction::getMergedTransactionIds() { | 
|  | return mMergedTransactionIds; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::doUncacheBufferTransaction(uint64_t cacheId) { | 
|  | sp<ISurfaceComposer> sf(ComposerService::getComposerService()); | 
|  |  | 
|  | client_cache_t uncacheBuffer; | 
|  | uncacheBuffer.token = BufferCache::getInstance().getToken(); | 
|  | uncacheBuffer.id = cacheId; | 
|  | Vector<ComposerState> composerStates; | 
|  | status_t status = sf->setTransactionState(FrameTimelineInfo{}, composerStates, {}, | 
|  | ISurfaceComposer::eOneWay, | 
|  | Transaction::getDefaultApplyToken(), {}, systemTime(), | 
|  | true, {uncacheBuffer}, false, {}, generateId(), {}); | 
|  | if (status != NO_ERROR) { | 
|  | ALOGE_AND_TRACE("SurfaceComposerClient::doUncacheBufferTransaction - %s", | 
|  | strerror(-status)); | 
|  | } | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::cacheBuffers() { | 
|  | if (!mMayContainBuffer) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | size_t count = 0; | 
|  | for (auto& [handle, cs] : mComposerStates) { | 
|  | layer_state_t* s = &(mComposerStates[handle].state); | 
|  | if (!(s->what & layer_state_t::eBufferChanged)) { | 
|  | continue; | 
|  | } else if (s->bufferData && | 
|  | s->bufferData->flags.test(BufferData::BufferDataChange::cachedBufferChanged)) { | 
|  | // If eBufferChanged and eCachedBufferChanged are both trued then that means | 
|  | // we already cached the buffer in a previous call to cacheBuffers, perhaps | 
|  | // from writeToParcel on a Transaction that was merged in to this one. | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // Don't try to cache a null buffer. Sending null buffers is cheap so we shouldn't waste | 
|  | // time trying to cache them. | 
|  | if (!s->bufferData || !s->bufferData->buffer) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | uint64_t cacheId = 0; | 
|  | status_t ret = BufferCache::getInstance().getCacheId(s->bufferData->buffer, &cacheId); | 
|  | if (ret == NO_ERROR) { | 
|  | // Cache-hit. Strip the buffer and send only the id. | 
|  | s->bufferData->buffer = nullptr; | 
|  | } else { | 
|  | // Cache-miss. Include the buffer and send the new cacheId. | 
|  | std::optional<client_cache_t> uncacheBuffer; | 
|  | cacheId = BufferCache::getInstance().cache(s->bufferData->buffer, uncacheBuffer); | 
|  | if (uncacheBuffer) { | 
|  | mUncacheBuffers.push_back(*uncacheBuffer); | 
|  | } | 
|  | } | 
|  | s->bufferData->flags |= BufferData::BufferDataChange::cachedBufferChanged; | 
|  | s->bufferData->cachedBuffer.token = BufferCache::getInstance().getToken(); | 
|  | s->bufferData->cachedBuffer.id = cacheId; | 
|  |  | 
|  | // If we have more buffers than the size of the cache, we should stop caching so we don't | 
|  | // evict other buffers in this transaction | 
|  | count++; | 
|  | if (count >= BUFFER_CACHE_MAX_SIZE) { | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | class SyncCallback { | 
|  | public: | 
|  | static auto getCallback(std::shared_ptr<SyncCallback>& callbackContext) { | 
|  | return [callbackContext](void* /* unused context */, nsecs_t /* latchTime */, | 
|  | const sp<Fence>& /* presentFence */, | 
|  | const std::vector<SurfaceControlStats>& /* stats */) { | 
|  | if (!callbackContext) { | 
|  | ALOGE("failed to get callback context for SyncCallback"); | 
|  | return; | 
|  | } | 
|  | LOG_ALWAYS_FATAL_IF(sem_post(&callbackContext->mSemaphore), "sem_post failed"); | 
|  | }; | 
|  | } | 
|  | ~SyncCallback() { | 
|  | if (mInitialized) { | 
|  | LOG_ALWAYS_FATAL_IF(sem_destroy(&mSemaphore), "sem_destroy failed"); | 
|  | } | 
|  | } | 
|  | void init() { | 
|  | LOG_ALWAYS_FATAL_IF(clock_gettime(CLOCK_MONOTONIC, &mTimeoutTimespec) == -1, | 
|  | "clock_gettime() fail! in SyncCallback::init"); | 
|  | mTimeoutTimespec.tv_sec += 4; | 
|  | LOG_ALWAYS_FATAL_IF(sem_init(&mSemaphore, 0, 0), "sem_init failed"); | 
|  | mInitialized = true; | 
|  | } | 
|  | void wait() { | 
|  | int result = sem_clockwait(&mSemaphore, CLOCK_MONOTONIC, &mTimeoutTimespec); | 
|  | if (result && errno != ETIMEDOUT && errno != EINTR) { | 
|  | LOG_ALWAYS_FATAL("sem_clockwait failed(%d)", errno); | 
|  | } else if (errno == ETIMEDOUT) { | 
|  | ALOGW("Sync transaction timed out waiting for commit callback."); | 
|  | } | 
|  | } | 
|  | void* getContext() { return static_cast<void*>(this); } | 
|  |  | 
|  | private: | 
|  | sem_t mSemaphore; | 
|  | bool mInitialized = false; | 
|  | timespec mTimeoutTimespec; | 
|  | }; | 
|  |  | 
|  | status_t SurfaceComposerClient::Transaction::apply(bool synchronous, bool oneWay) { | 
|  | if (mStatus != NO_ERROR) { | 
|  | return mStatus; | 
|  | } | 
|  |  | 
|  | std::shared_ptr<SyncCallback> syncCallback = std::make_shared<SyncCallback>(); | 
|  | if (synchronous) { | 
|  | syncCallback->init(); | 
|  | addTransactionCommittedCallback(SyncCallback::getCallback(syncCallback), | 
|  | /*callbackContext=*/nullptr); | 
|  | } | 
|  |  | 
|  | bool hasListenerCallbacks = !mListenerCallbacks.empty(); | 
|  | std::vector<ListenerCallbacks> listenerCallbacks; | 
|  | // For every listener with registered callbacks | 
|  | for (const auto& [listener, callbackInfo] : mListenerCallbacks) { | 
|  | auto& [callbackIds, surfaceControls] = callbackInfo; | 
|  | if (callbackIds.empty()) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (surfaceControls.empty()) { | 
|  | listenerCallbacks.emplace_back(IInterface::asBinder(listener), std::move(callbackIds)); | 
|  | } else { | 
|  | // If the listener has any SurfaceControls set on this Transaction update the surface | 
|  | // state | 
|  | for (const auto& surfaceControl : surfaceControls) { | 
|  | layer_state_t* s = getLayerState(surfaceControl); | 
|  | if (!s) { | 
|  | ALOGE("failed to get layer state"); | 
|  | continue; | 
|  | } | 
|  | std::vector<CallbackId> callbacks(callbackIds.begin(), callbackIds.end()); | 
|  | s->what |= layer_state_t::eHasListenerCallbacksChanged; | 
|  | s->listeners.emplace_back(IInterface::asBinder(listener), callbacks); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | cacheBuffers(); | 
|  |  | 
|  | Vector<ComposerState> composerStates; | 
|  | Vector<DisplayState> displayStates; | 
|  | uint32_t flags = 0; | 
|  |  | 
|  | for (auto const& kv : mComposerStates) { | 
|  | composerStates.add(kv.second); | 
|  | } | 
|  |  | 
|  | displayStates = std::move(mDisplayStates); | 
|  |  | 
|  | if (mAnimation) { | 
|  | flags |= ISurfaceComposer::eAnimation; | 
|  | } | 
|  | if (oneWay) { | 
|  | if (synchronous) { | 
|  | ALOGE("Transaction attempted to set synchronous and one way at the same time" | 
|  | " this is an invalid request. Synchronous will win for safety"); | 
|  | } else { | 
|  | flags |= ISurfaceComposer::eOneWay; | 
|  | } | 
|  | } | 
|  |  | 
|  | // If both mEarlyWakeupStart and mEarlyWakeupEnd are set | 
|  | // it is equivalent for none | 
|  | if (mEarlyWakeupStart && !mEarlyWakeupEnd) { | 
|  | flags |= ISurfaceComposer::eEarlyWakeupStart; | 
|  | } | 
|  | if (mEarlyWakeupEnd && !mEarlyWakeupStart) { | 
|  | flags |= ISurfaceComposer::eEarlyWakeupEnd; | 
|  | } | 
|  |  | 
|  | sp<IBinder> applyToken = mApplyToken ? mApplyToken : sApplyToken; | 
|  |  | 
|  | sp<ISurfaceComposer> sf(ComposerService::getComposerService()); | 
|  | sf->setTransactionState(mFrameTimelineInfo, composerStates, displayStates, flags, applyToken, | 
|  | mInputWindowCommands, mDesiredPresentTime, mIsAutoTimestamp, | 
|  | mUncacheBuffers, hasListenerCallbacks, listenerCallbacks, mId, | 
|  | mMergedTransactionIds); | 
|  | mId = generateId(); | 
|  |  | 
|  | // Clear the current states and flags | 
|  | clear(); | 
|  |  | 
|  | if (synchronous) { | 
|  | syncCallback->wait(); | 
|  | } | 
|  |  | 
|  | mStatus = NO_ERROR; | 
|  | return NO_ERROR; | 
|  | } | 
|  |  | 
|  | sp<IBinder> SurfaceComposerClient::Transaction::sApplyToken = new BBinder(); | 
|  |  | 
|  | sp<IBinder> SurfaceComposerClient::Transaction::getDefaultApplyToken() { | 
|  | return sApplyToken; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setDefaultApplyToken(sp<IBinder> applyToken) { | 
|  | sApplyToken = applyToken; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::Transaction::sendSurfaceFlushJankDataTransaction( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | Transaction t; | 
|  | layer_state_t* s = t.getLayerState(sc); | 
|  | if (!s) { | 
|  | return BAD_INDEX; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eFlushJankData; | 
|  | t.registerSurfaceControlForCallback(sc); | 
|  | return t.apply(/*sync=*/false, /* oneWay=*/true); | 
|  | } | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | sp<IBinder> SurfaceComposerClient::createDisplay(const String8& displayName, bool secure, | 
|  | float requestedRefereshRate) { | 
|  | sp<IBinder> display = nullptr; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->createDisplay(std::string( | 
|  | displayName.c_str()), | 
|  | secure, requestedRefereshRate, | 
|  | &display); | 
|  | return status.isOk() ? display : nullptr; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::destroyDisplay(const sp<IBinder>& display) { | 
|  | ComposerServiceAIDL::getComposerService()->destroyDisplay(display); | 
|  | } | 
|  |  | 
|  | std::vector<PhysicalDisplayId> SurfaceComposerClient::getPhysicalDisplayIds() { | 
|  | std::vector<int64_t> displayIds; | 
|  | std::vector<PhysicalDisplayId> physicalDisplayIds; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getPhysicalDisplayIds(&displayIds); | 
|  | if (status.isOk()) { | 
|  | physicalDisplayIds.reserve(displayIds.size()); | 
|  | for (auto item : displayIds) { | 
|  | auto id = DisplayId::fromValue<PhysicalDisplayId>(static_cast<uint64_t>(item)); | 
|  | physicalDisplayIds.push_back(*id); | 
|  | } | 
|  | } | 
|  | return physicalDisplayIds; | 
|  | } | 
|  |  | 
|  | sp<IBinder> SurfaceComposerClient::getPhysicalDisplayToken(PhysicalDisplayId displayId) { | 
|  | sp<IBinder> display = nullptr; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getPhysicalDisplayToken(displayId.value, | 
|  | &display); | 
|  | return status.isOk() ? display : nullptr; | 
|  | } | 
|  |  | 
|  | std::optional<gui::StalledTransactionInfo> SurfaceComposerClient::getStalledTransactionInfo( | 
|  | pid_t pid) { | 
|  | std::optional<gui::StalledTransactionInfo> result; | 
|  | ComposerServiceAIDL::getComposerService()->getStalledTransactionInfo(pid, &result); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setAnimationTransaction() { | 
|  | mAnimation = true; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setEarlyWakeupStart() { | 
|  | mEarlyWakeupStart = true; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setEarlyWakeupEnd() { | 
|  | mEarlyWakeupEnd = true; | 
|  | } | 
|  |  | 
|  | layer_state_t* SurfaceComposerClient::Transaction::getLayerState(const sp<SurfaceControl>& sc) { | 
|  | auto handle = sc->getLayerStateHandle(); | 
|  |  | 
|  | if (mComposerStates.count(handle) == 0) { | 
|  | // we don't have it, add an initialized layer_state to our list | 
|  | ComposerState s; | 
|  |  | 
|  | s.state.surface = handle; | 
|  | s.state.layerId = sc->getLayerId(); | 
|  |  | 
|  | mComposerStates[handle] = s; | 
|  | } | 
|  |  | 
|  | return &(mComposerStates[handle].state); | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::registerSurfaceControlForCallback( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | auto& callbackInfo = mListenerCallbacks[TransactionCompletedListener::getIInstance()]; | 
|  | callbackInfo.surfaceControls.insert(sc); | 
|  |  | 
|  | TransactionCompletedListener::getInstance()->addSurfaceControlToCallbacks(callbackInfo, sc); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setPosition( | 
|  | const sp<SurfaceControl>& sc, float x, float y) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::ePositionChanged; | 
|  | s->x = x; | 
|  | s->y = y; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::show( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | return setFlags(sc, 0, layer_state_t::eLayerHidden); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::hide( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | return setFlags(sc, layer_state_t::eLayerHidden, layer_state_t::eLayerHidden); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setLayer( | 
|  | const sp<SurfaceControl>& sc, int32_t z) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eLayerChanged; | 
|  | s->what &= ~layer_state_t::eRelativeLayerChanged; | 
|  | s->z = z; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setRelativeLayer( | 
|  | const sp<SurfaceControl>& sc, const sp<SurfaceControl>& relativeTo, int32_t z) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eRelativeLayerChanged; | 
|  | s->what &= ~layer_state_t::eLayerChanged; | 
|  | s->relativeLayerSurfaceControl = relativeTo; | 
|  | s->z = z; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setFlags( | 
|  | const sp<SurfaceControl>& sc, uint32_t flags, | 
|  | uint32_t mask) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | if ((mask & layer_state_t::eLayerOpaque) || (mask & layer_state_t::eLayerHidden) || | 
|  | (mask & layer_state_t::eLayerSecure) || (mask & layer_state_t::eLayerSkipScreenshot) || | 
|  | (mask & layer_state_t::eEnableBackpressure) || | 
|  | (mask & layer_state_t::eIgnoreDestinationFrame) || | 
|  | (mask & layer_state_t::eLayerIsDisplayDecoration) || | 
|  | (mask & layer_state_t::eLayerIsRefreshRateIndicator)) { | 
|  | s->what |= layer_state_t::eFlagsChanged; | 
|  | } | 
|  | s->flags &= ~mask; | 
|  | s->flags |= (flags & mask); | 
|  | s->mask |= mask; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setTransparentRegionHint( | 
|  | const sp<SurfaceControl>& sc, | 
|  | const Region& transparentRegion) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eTransparentRegionChanged; | 
|  | s->transparentRegion = transparentRegion; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setDimmingEnabled( | 
|  | const sp<SurfaceControl>& sc, bool dimmingEnabled) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eDimmingEnabledChanged; | 
|  | s->dimmingEnabled = dimmingEnabled; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setAlpha( | 
|  | const sp<SurfaceControl>& sc, float alpha) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | if (alpha < 0.0f || alpha > 1.0f) { | 
|  | ALOGE("SurfaceComposerClient::Transaction::setAlpha: invalid alpha %f, clamping", alpha); | 
|  | } | 
|  | s->what |= layer_state_t::eAlphaChanged; | 
|  | s->color.a = std::clamp(alpha, 0.f, 1.f); | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setLayerStack( | 
|  | const sp<SurfaceControl>& sc, ui::LayerStack layerStack) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eLayerStackChanged; | 
|  | s->layerStack = layerStack; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setMetadata( | 
|  | const sp<SurfaceControl>& sc, uint32_t key, const Parcel& p) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eMetadataChanged; | 
|  |  | 
|  | s->metadata.mMap[key] = {p.data(), p.data() + p.dataSize()}; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setMatrix( | 
|  | const sp<SurfaceControl>& sc, float dsdx, float dtdx, | 
|  | float dtdy, float dsdy) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eMatrixChanged; | 
|  | layer_state_t::matrix22_t matrix; | 
|  | matrix.dsdx = dsdx; | 
|  | matrix.dtdx = dtdx; | 
|  | matrix.dsdy = dsdy; | 
|  | matrix.dtdy = dtdy; | 
|  | s->matrix = matrix; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setCrop( | 
|  | const sp<SurfaceControl>& sc, const Rect& crop) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eCropChanged; | 
|  | s->crop = crop; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setCornerRadius( | 
|  | const sp<SurfaceControl>& sc, float cornerRadius) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eCornerRadiusChanged; | 
|  | s->cornerRadius = cornerRadius; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setBackgroundBlurRadius( | 
|  | const sp<SurfaceControl>& sc, int backgroundBlurRadius) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eBackgroundBlurRadiusChanged; | 
|  | s->backgroundBlurRadius = backgroundBlurRadius; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setBlurRegions( | 
|  | const sp<SurfaceControl>& sc, const std::vector<BlurRegion>& blurRegions) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eBlurRegionsChanged; | 
|  | s->blurRegions = blurRegions; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::reparent( | 
|  | const sp<SurfaceControl>& sc, const sp<SurfaceControl>& newParent) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | if (SurfaceControl::isSameSurface(sc, newParent)) { | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eReparent; | 
|  | s->parentSurfaceControlForChild = newParent ? newParent->getParentingLayer() : nullptr; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setColor( | 
|  | const sp<SurfaceControl>& sc, | 
|  | const half3& color) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eColorChanged; | 
|  | s->color.rgb = color; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setBackgroundColor( | 
|  | const sp<SurfaceControl>& sc, const half3& color, float alpha, ui::Dataspace dataspace) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eBackgroundColorChanged; | 
|  | s->bgColor.rgb = color; | 
|  | s->bgColor.a = alpha; | 
|  | s->bgColorDataspace = dataspace; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setTransform( | 
|  | const sp<SurfaceControl>& sc, uint32_t transform) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eBufferTransformChanged; | 
|  | s->bufferTransform = transform; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::setTransformToDisplayInverse(const sp<SurfaceControl>& sc, | 
|  | bool transformToDisplayInverse) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eTransformToDisplayInverseChanged; | 
|  | s->transformToDisplayInverse = transformToDisplayInverse; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | std::shared_ptr<BufferData> SurfaceComposerClient::Transaction::getAndClearBuffer( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | return nullptr; | 
|  | } | 
|  | if (!(s->what & layer_state_t::eBufferChanged)) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | std::shared_ptr<BufferData> bufferData = std::move(s->bufferData); | 
|  |  | 
|  | TransactionCompletedListener::getInstance()->removeReleaseBufferCallback( | 
|  | bufferData->generateReleaseCallbackId()); | 
|  | s->what &= ~layer_state_t::eBufferChanged; | 
|  | s->bufferData = nullptr; | 
|  |  | 
|  | return bufferData; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setBufferHasBarrier( | 
|  | const sp<SurfaceControl>& sc, uint64_t barrierFrameNumber) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->bufferData->hasBarrier = true; | 
|  | s->bufferData->barrierFrameNumber = barrierFrameNumber; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setBuffer( | 
|  | const sp<SurfaceControl>& sc, const sp<GraphicBuffer>& buffer, | 
|  | const std::optional<sp<Fence>>& fence, const std::optional<uint64_t>& optFrameNumber, | 
|  | uint32_t producerId, ReleaseBufferCallback callback) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | releaseBufferIfOverwriting(*s); | 
|  |  | 
|  | std::shared_ptr<BufferData> bufferData = std::make_shared<BufferData>(); | 
|  | bufferData->buffer = buffer; | 
|  | if (buffer) { | 
|  | uint64_t frameNumber = sc->resolveFrameNumber(optFrameNumber); | 
|  | bufferData->frameNumber = frameNumber; | 
|  | bufferData->producerId = producerId; | 
|  | bufferData->flags |= BufferData::BufferDataChange::frameNumberChanged; | 
|  | if (fence) { | 
|  | bufferData->acquireFence = *fence; | 
|  | bufferData->flags |= BufferData::BufferDataChange::fenceChanged; | 
|  | } | 
|  | bufferData->releaseBufferEndpoint = | 
|  | IInterface::asBinder(TransactionCompletedListener::getIInstance()); | 
|  | setReleaseBufferCallback(bufferData.get(), callback); | 
|  | } | 
|  |  | 
|  | if (mIsAutoTimestamp) { | 
|  | mDesiredPresentTime = systemTime(); | 
|  | } | 
|  | s->what |= layer_state_t::eBufferChanged; | 
|  | s->bufferData = std::move(bufferData); | 
|  | registerSurfaceControlForCallback(sc); | 
|  |  | 
|  | // With the current infrastructure, a release callback will not be invoked if there's no | 
|  | // transaction callback in the case when a buffer is latched and not released early. This is | 
|  | // because the legacy implementation didn't have a release callback and sent releases in the | 
|  | // transaction callback. Because of this, we need to make sure to have a transaction callback | 
|  | // set up when a buffer is sent in a transaction to ensure the caller gets the release | 
|  | // callback, regardless if they set up a transaction callback. | 
|  | // | 
|  | // TODO (b/230380821): Remove when release callbacks are separated from transaction callbacks | 
|  | addTransactionCompletedCallback([](void*, nsecs_t, const sp<Fence>&, | 
|  | const std::vector<SurfaceControlStats>&) {}, | 
|  | nullptr); | 
|  |  | 
|  | mMayContainBuffer = true; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::unsetBuffer( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | if (!(s->what & layer_state_t::eBufferChanged)) { | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | releaseBufferIfOverwriting(*s); | 
|  |  | 
|  | s->what &= ~layer_state_t::eBufferChanged; | 
|  | s->bufferData = nullptr; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setReleaseBufferCallback(BufferData* bufferData, | 
|  | ReleaseBufferCallback callback) { | 
|  | if (!callback) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (!bufferData->buffer) { | 
|  | ALOGW("Transaction::setReleaseBufferCallback" | 
|  | "ignored trying to set a callback on a null buffer."); | 
|  | return; | 
|  | } | 
|  |  | 
|  | bufferData->releaseBufferListener = TransactionCompletedListener::getIInstance(); | 
|  | auto listener = TransactionCompletedListener::getInstance(); | 
|  | listener->setReleaseBufferCallback(bufferData->generateReleaseCallbackId(), callback); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setDataspace( | 
|  | const sp<SurfaceControl>& sc, ui::Dataspace dataspace) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eDataspaceChanged; | 
|  | s->dataspace = dataspace; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setExtendedRangeBrightness( | 
|  | const sp<SurfaceControl>& sc, float currentBufferRatio, float desiredRatio) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eExtendedRangeBrightnessChanged; | 
|  | s->currentHdrSdrRatio = currentBufferRatio; | 
|  | s->desiredHdrSdrRatio = desiredRatio; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setCachingHint( | 
|  | const sp<SurfaceControl>& sc, gui::CachingHint cachingHint) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eCachingHintChanged; | 
|  | s->cachingHint = cachingHint; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setHdrMetadata( | 
|  | const sp<SurfaceControl>& sc, const HdrMetadata& hdrMetadata) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eHdrMetadataChanged; | 
|  | s->hdrMetadata = hdrMetadata; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setSurfaceDamageRegion( | 
|  | const sp<SurfaceControl>& sc, const Region& surfaceDamageRegion) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eSurfaceDamageRegionChanged; | 
|  | s->surfaceDamageRegion = surfaceDamageRegion; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setApi( | 
|  | const sp<SurfaceControl>& sc, int32_t api) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eApiChanged; | 
|  | s->api = api; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setSidebandStream( | 
|  | const sp<SurfaceControl>& sc, const sp<NativeHandle>& sidebandStream) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eSidebandStreamChanged; | 
|  | s->sidebandStream = sidebandStream; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setDesiredPresentTime( | 
|  | nsecs_t desiredPresentTime) { | 
|  | mDesiredPresentTime = desiredPresentTime; | 
|  | mIsAutoTimestamp = false; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setColorSpaceAgnostic( | 
|  | const sp<SurfaceControl>& sc, const bool agnostic) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eColorSpaceAgnosticChanged; | 
|  | s->colorSpaceAgnostic = agnostic; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::setFrameRateSelectionPriority(const sp<SurfaceControl>& sc, | 
|  | int32_t priority) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eFrameRateSelectionPriority; | 
|  | s->frameRateSelectionPriority = priority; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::addTransactionCallback( | 
|  | TransactionCompletedCallbackTakesContext callback, void* callbackContext, | 
|  | CallbackId::Type callbackType) { | 
|  | auto listener = TransactionCompletedListener::getInstance(); | 
|  |  | 
|  | auto callbackWithContext = std::bind(callback, callbackContext, std::placeholders::_1, | 
|  | std::placeholders::_2, std::placeholders::_3); | 
|  | const auto& surfaceControls = | 
|  | mListenerCallbacks[TransactionCompletedListener::getIInstance()].surfaceControls; | 
|  |  | 
|  | CallbackId callbackId = | 
|  | listener->addCallbackFunction(callbackWithContext, surfaceControls, callbackType); | 
|  |  | 
|  | mListenerCallbacks[TransactionCompletedListener::getIInstance()].callbackIds.emplace( | 
|  | callbackId); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::addTransactionCompletedCallback( | 
|  | TransactionCompletedCallbackTakesContext callback, void* callbackContext) { | 
|  | return addTransactionCallback(callback, callbackContext, CallbackId::Type::ON_COMPLETE); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::addTransactionCommittedCallback( | 
|  | TransactionCompletedCallbackTakesContext callback, void* callbackContext) { | 
|  | return addTransactionCallback(callback, callbackContext, CallbackId::Type::ON_COMMIT); | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::notifyProducerDisconnect( | 
|  | const sp<SurfaceControl>& sc) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eProducerDisconnect; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setInputWindowInfo( | 
|  | const sp<SurfaceControl>& sc, const WindowInfo& info) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->windowInfoHandle = new WindowInfoHandle(info); | 
|  | s->what |= layer_state_t::eInputInfoChanged; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setFocusedWindow( | 
|  | const FocusRequest& request) { | 
|  | mInputWindowCommands.focusRequests.push_back(request); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::addWindowInfosReportedListener( | 
|  | sp<gui::IWindowInfosReportedListener> windowInfosReportedListener) { | 
|  | mInputWindowCommands.windowInfosReportedListeners.insert(windowInfosReportedListener); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setColorTransform( | 
|  | const sp<SurfaceControl>& sc, const mat3& matrix, const vec3& translation) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eColorTransformChanged; | 
|  | s->colorTransform = mat4(matrix, translation); | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setGeometry( | 
|  | const sp<SurfaceControl>& sc, const Rect& source, const Rect& dst, int transform) { | 
|  | setCrop(sc, source); | 
|  |  | 
|  | int x = dst.left; | 
|  | int y = dst.top; | 
|  |  | 
|  | float sourceWidth = source.getWidth(); | 
|  | float sourceHeight = source.getHeight(); | 
|  |  | 
|  | float xScale = sourceWidth < 0 ? 1.0f : dst.getWidth() / sourceWidth; | 
|  | float yScale = sourceHeight < 0 ? 1.0f : dst.getHeight() / sourceHeight; | 
|  | float matrix[4] = {1, 0, 0, 1}; | 
|  |  | 
|  | switch (transform) { | 
|  | case NATIVE_WINDOW_TRANSFORM_FLIP_H: | 
|  | matrix[0] = -xScale; matrix[1] = 0; | 
|  | matrix[2] = 0; matrix[3] = yScale; | 
|  | x += source.getWidth(); | 
|  | break; | 
|  | case NATIVE_WINDOW_TRANSFORM_FLIP_V: | 
|  | matrix[0] = xScale; matrix[1] = 0; | 
|  | matrix[2] = 0; matrix[3] = -yScale; | 
|  | y += source.getHeight(); | 
|  | break; | 
|  | case NATIVE_WINDOW_TRANSFORM_ROT_90: | 
|  | matrix[0] = 0; matrix[1] = -yScale; | 
|  | matrix[2] = xScale; matrix[3] = 0; | 
|  | x += source.getHeight(); | 
|  | break; | 
|  | case NATIVE_WINDOW_TRANSFORM_ROT_180: | 
|  | matrix[0] = -xScale; matrix[1] = 0; | 
|  | matrix[2] = 0; matrix[3] = -yScale; | 
|  | x += source.getWidth(); | 
|  | y += source.getHeight(); | 
|  | break; | 
|  | case NATIVE_WINDOW_TRANSFORM_ROT_270: | 
|  | matrix[0] = 0; matrix[1] = yScale; | 
|  | matrix[2] = -xScale; matrix[3] = 0; | 
|  | y += source.getWidth(); | 
|  | break; | 
|  | default: | 
|  | matrix[0] = xScale; matrix[1] = 0; | 
|  | matrix[2] = 0; matrix[3] = yScale; | 
|  | break; | 
|  | } | 
|  | setMatrix(sc, matrix[0], matrix[1], matrix[2], matrix[3]); | 
|  | float offsetX = xScale * source.left; | 
|  | float offsetY = yScale * source.top; | 
|  | setPosition(sc, x - offsetX, y - offsetY); | 
|  |  | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setShadowRadius( | 
|  | const sp<SurfaceControl>& sc, float shadowRadius) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eShadowRadiusChanged; | 
|  | s->shadowRadius = shadowRadius; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setFrameRate( | 
|  | const sp<SurfaceControl>& sc, float frameRate, int8_t compatibility, | 
|  | int8_t changeFrameRateStrategy) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | // Allow privileged values as well here, those will be ignored by SF if | 
|  | // the caller is not privileged | 
|  | if (!ValidateFrameRate(frameRate, compatibility, changeFrameRateStrategy, | 
|  | "Transaction::setFrameRate", | 
|  | /*privileged=*/true)) { | 
|  | mStatus = BAD_VALUE; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eFrameRateChanged; | 
|  | s->frameRate = frameRate; | 
|  | s->frameRateCompatibility = compatibility; | 
|  | s->changeFrameRateStrategy = changeFrameRateStrategy; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::setDefaultFrameRateCompatibility(const sp<SurfaceControl>& sc, | 
|  | int8_t compatibility) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eDefaultFrameRateCompatibilityChanged; | 
|  | s->defaultFrameRateCompatibility = compatibility; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setFrameRateCategory( | 
|  | const sp<SurfaceControl>& sc, int8_t category) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eFrameRateCategoryChanged; | 
|  | s->frameRateCategory = category; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setFixedTransformHint( | 
|  | const sp<SurfaceControl>& sc, int32_t fixedTransformHint) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | const ui::Transform::RotationFlags transform = fixedTransformHint == -1 | 
|  | ? ui::Transform::ROT_INVALID | 
|  | : ui::Transform::toRotationFlags(static_cast<ui::Rotation>(fixedTransformHint)); | 
|  | s->what |= layer_state_t::eFixedTransformHintChanged; | 
|  | s->fixedTransformHint = transform; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setFrameTimelineInfo( | 
|  | const FrameTimelineInfo& frameTimelineInfo) { | 
|  | mergeFrameTimelineInfo(mFrameTimelineInfo, frameTimelineInfo); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setAutoRefresh( | 
|  | const sp<SurfaceControl>& sc, bool autoRefresh) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eAutoRefreshChanged; | 
|  | s->autoRefresh = autoRefresh; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setTrustedOverlay( | 
|  | const sp<SurfaceControl>& sc, bool isTrustedOverlay) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eTrustedOverlayChanged; | 
|  | s->isTrustedOverlay = isTrustedOverlay; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setApplyToken( | 
|  | const sp<IBinder>& applyToken) { | 
|  | mApplyToken = applyToken; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setStretchEffect( | 
|  | const sp<SurfaceControl>& sc, const StretchEffect& stretchEffect) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eStretchChanged; | 
|  | s->stretchEffect = stretchEffect; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setBufferCrop( | 
|  | const sp<SurfaceControl>& sc, const Rect& bufferCrop) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eBufferCropChanged; | 
|  | s->bufferCrop = bufferCrop; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setDestinationFrame( | 
|  | const sp<SurfaceControl>& sc, const Rect& destinationFrame) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eDestinationFrameChanged; | 
|  | s->destinationFrame = destinationFrame; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::setDropInputMode( | 
|  | const sp<SurfaceControl>& sc, gui::DropInputMode mode) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eDropInputModeChanged; | 
|  | s->dropInputMode = mode; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& SurfaceComposerClient::Transaction::enableBorder( | 
|  | const sp<SurfaceControl>& sc, bool shouldEnable, float width, const half4& color) { | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | s->what |= layer_state_t::eRenderBorderChanged; | 
|  | s->borderEnabled = shouldEnable; | 
|  | s->borderWidth = width; | 
|  | s->borderColor = color; | 
|  |  | 
|  | registerSurfaceControlForCallback(sc); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | DisplayState& SurfaceComposerClient::Transaction::getDisplayState(const sp<IBinder>& token) { | 
|  | DisplayState s; | 
|  | s.token = token; | 
|  | ssize_t index = mDisplayStates.indexOf(s); | 
|  | if (index < 0) { | 
|  | // we don't have it, add an initialized layer_state to our list | 
|  | s.what = 0; | 
|  | index = mDisplayStates.add(s); | 
|  | } | 
|  | return mDisplayStates.editItemAt(static_cast<size_t>(index)); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::Transaction::setDisplaySurface(const sp<IBinder>& token, | 
|  | const sp<IGraphicBufferProducer>& bufferProducer) { | 
|  | if (bufferProducer.get() != nullptr) { | 
|  | // Make sure that composition can never be stalled by a virtual display | 
|  | // consumer that isn't processing buffers fast enough. | 
|  | status_t err = bufferProducer->setAsyncMode(true); | 
|  | if (err != NO_ERROR) { | 
|  | ALOGE("Composer::setDisplaySurface Failed to enable async mode on the " | 
|  | "BufferQueue. This BufferQueue cannot be used for virtual " | 
|  | "display. (%d)", err); | 
|  | return err; | 
|  | } | 
|  | } | 
|  | DisplayState& s(getDisplayState(token)); | 
|  | s.surface = bufferProducer; | 
|  | s.what |= DisplayState::eSurfaceChanged; | 
|  | return NO_ERROR; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setDisplayLayerStack(const sp<IBinder>& token, | 
|  | ui::LayerStack layerStack) { | 
|  | DisplayState& s(getDisplayState(token)); | 
|  | s.layerStack = layerStack; | 
|  | s.what |= DisplayState::eLayerStackChanged; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setDisplayFlags(const sp<IBinder>& token, uint32_t flags) { | 
|  | DisplayState& s(getDisplayState(token)); | 
|  | s.flags = flags; | 
|  | s.what |= DisplayState::eFlagsChanged; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setDisplayProjection(const sp<IBinder>& token, | 
|  | ui::Rotation orientation, | 
|  | const Rect& layerStackRect, | 
|  | const Rect& displayRect) { | 
|  | DisplayState& s(getDisplayState(token)); | 
|  | s.orientation = orientation; | 
|  | s.layerStackSpaceRect = layerStackRect; | 
|  | s.orientedDisplaySpaceRect = displayRect; | 
|  | s.what |= DisplayState::eDisplayProjectionChanged; | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::Transaction::setDisplaySize(const sp<IBinder>& token, uint32_t width, uint32_t height) { | 
|  | DisplayState& s(getDisplayState(token)); | 
|  | s.width = width; | 
|  | s.height = height; | 
|  | s.what |= DisplayState::eDisplaySizeChanged; | 
|  | } | 
|  |  | 
|  | // copied from FrameTimelineInfo::merge() | 
|  | void SurfaceComposerClient::Transaction::mergeFrameTimelineInfo(FrameTimelineInfo& t, | 
|  | const FrameTimelineInfo& other) { | 
|  | // When merging vsync Ids we take the oldest valid one | 
|  | if (t.vsyncId != FrameTimelineInfo::INVALID_VSYNC_ID && | 
|  | other.vsyncId != FrameTimelineInfo::INVALID_VSYNC_ID) { | 
|  | if (other.vsyncId > t.vsyncId) { | 
|  | t = other; | 
|  | } | 
|  | } else if (t.vsyncId == FrameTimelineInfo::INVALID_VSYNC_ID) { | 
|  | t = other; | 
|  | } | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::setTrustedPresentationCallback( | 
|  | const sp<SurfaceControl>& sc, TrustedPresentationCallback cb, | 
|  | const TrustedPresentationThresholds& thresholds, void* context, | 
|  | sp<SurfaceComposerClient::PresentationCallbackRAII>& outCallbackRef) { | 
|  | auto listener = TransactionCompletedListener::getInstance(); | 
|  | outCallbackRef = listener->addTrustedPresentationCallback(cb, sc->getLayerId(), context); | 
|  |  | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eTrustedPresentationInfoChanged; | 
|  | s->trustedPresentationThresholds = thresholds; | 
|  | s->trustedPresentationListener.callbackInterface = TransactionCompletedListener::getIInstance(); | 
|  | s->trustedPresentationListener.callbackId = sc->getLayerId(); | 
|  |  | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::Transaction& | 
|  | SurfaceComposerClient::Transaction::clearTrustedPresentationCallback(const sp<SurfaceControl>& sc) { | 
|  | auto listener = TransactionCompletedListener::getInstance(); | 
|  | listener->clearTrustedPresentationCallback(sc->getLayerId()); | 
|  |  | 
|  | layer_state_t* s = getLayerState(sc); | 
|  | if (!s) { | 
|  | mStatus = BAD_INDEX; | 
|  | return *this; | 
|  | } | 
|  | s->what |= layer_state_t::eTrustedPresentationInfoChanged; | 
|  | s->trustedPresentationThresholds = TrustedPresentationThresholds(); | 
|  | s->trustedPresentationListener.callbackInterface = nullptr; | 
|  | s->trustedPresentationListener.callbackId = -1; | 
|  |  | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  |  | 
|  | SurfaceComposerClient::SurfaceComposerClient() : mStatus(NO_INIT) {} | 
|  |  | 
|  | SurfaceComposerClient::SurfaceComposerClient(const sp<ISurfaceComposerClient>& client) | 
|  | : mStatus(NO_ERROR), mClient(client) {} | 
|  |  | 
|  | void SurfaceComposerClient::onFirstRef() { | 
|  | sp<gui::ISurfaceComposer> sf(ComposerServiceAIDL::getComposerService()); | 
|  | if (sf != nullptr && mStatus == NO_INIT) { | 
|  | sp<ISurfaceComposerClient> conn; | 
|  | binder::Status status = sf->createConnection(&conn); | 
|  | if (status.isOk() && conn != nullptr) { | 
|  | mClient = conn; | 
|  | mStatus = NO_ERROR; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | SurfaceComposerClient::~SurfaceComposerClient() { | 
|  | dispose(); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::initCheck() const { | 
|  | return mStatus; | 
|  | } | 
|  |  | 
|  | sp<IBinder> SurfaceComposerClient::connection() const { | 
|  | return IInterface::asBinder(mClient); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::linkToComposerDeath( | 
|  | const sp<IBinder::DeathRecipient>& recipient, | 
|  | void* cookie, uint32_t flags) { | 
|  | sp<ISurfaceComposer> sf(ComposerService::getComposerService()); | 
|  | return IInterface::asBinder(sf)->linkToDeath(recipient, cookie, flags); | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::dispose() { | 
|  | // this can be called more than once. | 
|  | sp<ISurfaceComposerClient> client; | 
|  | Mutex::Autolock _lm(mLock); | 
|  | if (mClient != nullptr) { | 
|  | client = mClient; // hold ref while lock is held | 
|  | mClient.clear(); | 
|  | } | 
|  | mStatus = NO_INIT; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::bootFinished() { | 
|  | sp<gui::ISurfaceComposer> sf(ComposerServiceAIDL::getComposerService()); | 
|  | binder::Status status = sf->bootFinished(); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | sp<SurfaceControl> SurfaceComposerClient::createSurface(const String8& name, uint32_t w, uint32_t h, | 
|  | PixelFormat format, int32_t flags, | 
|  | const sp<IBinder>& parentHandle, | 
|  | LayerMetadata metadata, | 
|  | uint32_t* outTransformHint) { | 
|  | sp<SurfaceControl> s; | 
|  | createSurfaceChecked(name, w, h, format, &s, flags, parentHandle, std::move(metadata), | 
|  | outTransformHint); | 
|  | return s; | 
|  | } | 
|  |  | 
|  | static std::string toString(const String16& string) { | 
|  | return std::string(String8(string).c_str()); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::createSurfaceChecked(const String8& name, uint32_t w, uint32_t h, | 
|  | PixelFormat format, | 
|  | sp<SurfaceControl>* outSurface, int32_t flags, | 
|  | const sp<IBinder>& parentHandle, | 
|  | LayerMetadata metadata, | 
|  | uint32_t* outTransformHint) { | 
|  | sp<SurfaceControl> sur; | 
|  | status_t err = mStatus; | 
|  |  | 
|  | if (mStatus == NO_ERROR) { | 
|  | gui::CreateSurfaceResult result; | 
|  | binder::Status status = mClient->createSurface(std::string(name.c_str()), flags, | 
|  | parentHandle, std::move(metadata), &result); | 
|  | err = statusTFromBinderStatus(status); | 
|  | if (outTransformHint) { | 
|  | *outTransformHint = result.transformHint; | 
|  | } | 
|  | ALOGE_IF(err, "SurfaceComposerClient::createSurface error %s", strerror(-err)); | 
|  | if (err == NO_ERROR) { | 
|  | *outSurface = new SurfaceControl(this, result.handle, result.layerId, | 
|  | toString(result.layerName), w, h, format, | 
|  | result.transformHint, flags); | 
|  | } | 
|  | } | 
|  | return err; | 
|  | } | 
|  |  | 
|  | sp<SurfaceControl> SurfaceComposerClient::mirrorSurface(SurfaceControl* mirrorFromSurface) { | 
|  | if (mirrorFromSurface == nullptr) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | sp<IBinder> mirrorFromHandle = mirrorFromSurface->getHandle(); | 
|  | gui::CreateSurfaceResult result; | 
|  | const binder::Status status = mClient->mirrorSurface(mirrorFromHandle, &result); | 
|  | const status_t err = statusTFromBinderStatus(status); | 
|  | if (err == NO_ERROR) { | 
|  | return new SurfaceControl(this, result.handle, result.layerId, toString(result.layerName)); | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | sp<SurfaceControl> SurfaceComposerClient::mirrorDisplay(DisplayId displayId) { | 
|  | gui::CreateSurfaceResult result; | 
|  | const binder::Status status = mClient->mirrorDisplay(displayId.value, &result); | 
|  | const status_t err = statusTFromBinderStatus(status); | 
|  | if (err == NO_ERROR) { | 
|  | return new SurfaceControl(this, result.handle, result.layerId, toString(result.layerName)); | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::clearLayerFrameStats(const sp<IBinder>& token) const { | 
|  | if (mStatus != NO_ERROR) { | 
|  | return mStatus; | 
|  | } | 
|  | const binder::Status status = mClient->clearLayerFrameStats(token); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getLayerFrameStats(const sp<IBinder>& token, | 
|  | FrameStats* outStats) const { | 
|  | if (mStatus != NO_ERROR) { | 
|  | return mStatus; | 
|  | } | 
|  | gui::FrameStats stats; | 
|  | const binder::Status status = mClient->getLayerFrameStats(token, &stats); | 
|  | if (status.isOk()) { | 
|  | outStats->refreshPeriodNano = stats.refreshPeriodNano; | 
|  | outStats->desiredPresentTimesNano.setCapacity(stats.desiredPresentTimesNano.size()); | 
|  | for (const auto& t : stats.desiredPresentTimesNano) { | 
|  | outStats->desiredPresentTimesNano.add(t); | 
|  | } | 
|  | outStats->actualPresentTimesNano.setCapacity(stats.actualPresentTimesNano.size()); | 
|  | for (const auto& t : stats.actualPresentTimesNano) { | 
|  | outStats->actualPresentTimesNano.add(t); | 
|  | } | 
|  | outStats->frameReadyTimesNano.setCapacity(stats.frameReadyTimesNano.size()); | 
|  | for (const auto& t : stats.frameReadyTimesNano) { | 
|  | outStats->frameReadyTimesNano.add(t); | 
|  | } | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | // ---------------------------------------------------------------------------- | 
|  |  | 
|  | status_t SurfaceComposerClient::getDisplayState(const sp<IBinder>& display, | 
|  | ui::DisplayState* state) { | 
|  | gui::DisplayState ds; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDisplayState(display, &ds); | 
|  | if (status.isOk()) { | 
|  | state->layerStack = ui::LayerStack::fromValue(ds.layerStack); | 
|  | state->orientation = static_cast<ui::Rotation>(ds.orientation); | 
|  | state->layerStackSpaceRect = | 
|  | ui::Size(ds.layerStackSpaceRect.width, ds.layerStackSpaceRect.height); | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getStaticDisplayInfo(int64_t displayId, | 
|  | ui::StaticDisplayInfo* outInfo) { | 
|  | using Tag = android::gui::DeviceProductInfo::ManufactureOrModelDate::Tag; | 
|  | gui::StaticDisplayInfo ginfo; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getStaticDisplayInfo(displayId, &ginfo); | 
|  | if (status.isOk()) { | 
|  | // convert gui::StaticDisplayInfo to ui::StaticDisplayInfo | 
|  | outInfo->connectionType = static_cast<ui::DisplayConnectionType>(ginfo.connectionType); | 
|  | outInfo->density = ginfo.density; | 
|  | outInfo->secure = ginfo.secure; | 
|  | outInfo->installOrientation = static_cast<ui::Rotation>(ginfo.installOrientation); | 
|  |  | 
|  | if (const std::optional<gui::DeviceProductInfo> dpi = ginfo.deviceProductInfo) { | 
|  | DeviceProductInfo info; | 
|  | info.name = dpi->name; | 
|  | if (dpi->manufacturerPnpId.size() > 0) { | 
|  | // copid from PnpId = std::array<char, 4> in ui/DeviceProductInfo.h | 
|  | constexpr int kMaxPnpIdSize = 4; | 
|  | size_t count = std::max<size_t>(kMaxPnpIdSize, dpi->manufacturerPnpId.size()); | 
|  | std::copy_n(dpi->manufacturerPnpId.begin(), count, info.manufacturerPnpId.begin()); | 
|  | } | 
|  | if (dpi->relativeAddress.size() > 0) { | 
|  | std::copy(dpi->relativeAddress.begin(), dpi->relativeAddress.end(), | 
|  | std::back_inserter(info.relativeAddress)); | 
|  | } | 
|  | info.productId = dpi->productId; | 
|  |  | 
|  | const gui::DeviceProductInfo::ManufactureOrModelDate& date = | 
|  | dpi->manufactureOrModelDate; | 
|  | if (date.getTag() == Tag::modelYear) { | 
|  | DeviceProductInfo::ModelYear modelYear; | 
|  | modelYear.year = static_cast<uint32_t>(date.get<Tag::modelYear>().year); | 
|  | info.manufactureOrModelDate = modelYear; | 
|  | } else if (date.getTag() == Tag::manufactureYear) { | 
|  | DeviceProductInfo::ManufactureYear manufactureYear; | 
|  | manufactureYear.year = date.get<Tag::manufactureYear>().modelYear.year; | 
|  | info.manufactureOrModelDate = manufactureYear; | 
|  | } else if (date.getTag() == Tag::manufactureWeekAndYear) { | 
|  | DeviceProductInfo::ManufactureWeekAndYear weekAndYear; | 
|  | weekAndYear.year = | 
|  | date.get<Tag::manufactureWeekAndYear>().manufactureYear.modelYear.year; | 
|  | weekAndYear.week = date.get<Tag::manufactureWeekAndYear>().week; | 
|  | info.manufactureOrModelDate = weekAndYear; | 
|  | } | 
|  |  | 
|  | outInfo->deviceProductInfo = info; | 
|  | } | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::getDynamicDisplayInfoInternal(gui::DynamicDisplayInfo& ginfo, | 
|  | ui::DynamicDisplayInfo*& outInfo) { | 
|  | // convert gui::DynamicDisplayInfo to ui::DynamicDisplayInfo | 
|  | outInfo->supportedDisplayModes.clear(); | 
|  | outInfo->supportedDisplayModes.reserve(ginfo.supportedDisplayModes.size()); | 
|  | for (const auto& mode : ginfo.supportedDisplayModes) { | 
|  | ui::DisplayMode outMode; | 
|  | outMode.id = mode.id; | 
|  | outMode.resolution.width = mode.resolution.width; | 
|  | outMode.resolution.height = mode.resolution.height; | 
|  | outMode.xDpi = mode.xDpi; | 
|  | outMode.yDpi = mode.yDpi; | 
|  | outMode.refreshRate = mode.refreshRate; | 
|  | outMode.appVsyncOffset = mode.appVsyncOffset; | 
|  | outMode.sfVsyncOffset = mode.sfVsyncOffset; | 
|  | outMode.presentationDeadline = mode.presentationDeadline; | 
|  | outMode.group = mode.group; | 
|  | std::transform(mode.supportedHdrTypes.begin(), mode.supportedHdrTypes.end(), | 
|  | std::back_inserter(outMode.supportedHdrTypes), | 
|  | [](const int32_t& value) { return static_cast<ui::Hdr>(value); }); | 
|  | outInfo->supportedDisplayModes.push_back(outMode); | 
|  | } | 
|  |  | 
|  | outInfo->activeDisplayModeId = ginfo.activeDisplayModeId; | 
|  | outInfo->renderFrameRate = ginfo.renderFrameRate; | 
|  |  | 
|  | outInfo->supportedColorModes.clear(); | 
|  | outInfo->supportedColorModes.reserve(ginfo.supportedColorModes.size()); | 
|  | for (const auto& cmode : ginfo.supportedColorModes) { | 
|  | outInfo->supportedColorModes.push_back(static_cast<ui::ColorMode>(cmode)); | 
|  | } | 
|  |  | 
|  | outInfo->activeColorMode = static_cast<ui::ColorMode>(ginfo.activeColorMode); | 
|  |  | 
|  | std::vector<ui::Hdr> types; | 
|  | types.reserve(ginfo.hdrCapabilities.supportedHdrTypes.size()); | 
|  | for (const auto& hdr : ginfo.hdrCapabilities.supportedHdrTypes) { | 
|  | types.push_back(static_cast<ui::Hdr>(hdr)); | 
|  | } | 
|  | outInfo->hdrCapabilities = HdrCapabilities(types, ginfo.hdrCapabilities.maxLuminance, | 
|  | ginfo.hdrCapabilities.maxAverageLuminance, | 
|  | ginfo.hdrCapabilities.minLuminance); | 
|  |  | 
|  | outInfo->autoLowLatencyModeSupported = ginfo.autoLowLatencyModeSupported; | 
|  | outInfo->gameContentTypeSupported = ginfo.gameContentTypeSupported; | 
|  | outInfo->preferredBootDisplayMode = ginfo.preferredBootDisplayMode; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getDynamicDisplayInfoFromId(int64_t displayId, | 
|  | ui::DynamicDisplayInfo* outInfo) { | 
|  | gui::DynamicDisplayInfo ginfo; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDynamicDisplayInfoFromId(displayId, | 
|  | &ginfo); | 
|  | if (status.isOk()) { | 
|  | getDynamicDisplayInfoInternal(ginfo, outInfo); | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getDynamicDisplayInfoFromToken(const sp<IBinder>& display, | 
|  | ui::DynamicDisplayInfo* outInfo) { | 
|  | gui::DynamicDisplayInfo ginfo; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDynamicDisplayInfoFromToken(display, | 
|  | &ginfo); | 
|  | if (status.isOk()) { | 
|  | getDynamicDisplayInfoInternal(ginfo, outInfo); | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getActiveDisplayMode(const sp<IBinder>& display, | 
|  | ui::DisplayMode* mode) { | 
|  | ui::DynamicDisplayInfo info; | 
|  |  | 
|  | status_t result = getDynamicDisplayInfoFromToken(display, &info); | 
|  | if (result != NO_ERROR) { | 
|  | return result; | 
|  | } | 
|  |  | 
|  | if (const auto activeMode = info.getActiveDisplayMode()) { | 
|  | *mode = *activeMode; | 
|  | return NO_ERROR; | 
|  | } | 
|  |  | 
|  | ALOGE("Active display mode not found."); | 
|  | return NAME_NOT_FOUND; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setDesiredDisplayModeSpecs(const sp<IBinder>& displayToken, | 
|  | const gui::DisplayModeSpecs& specs) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->setDesiredDisplayModeSpecs(displayToken, | 
|  | specs); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getDesiredDisplayModeSpecs(const sp<IBinder>& displayToken, | 
|  | gui::DisplayModeSpecs* outSpecs) { | 
|  | if (!outSpecs) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDesiredDisplayModeSpecs(displayToken, | 
|  | outSpecs); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getDisplayNativePrimaries(const sp<IBinder>& display, | 
|  | ui::DisplayPrimaries& outPrimaries) { | 
|  | gui::DisplayPrimaries primaries; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDisplayNativePrimaries(display, | 
|  | &primaries); | 
|  | if (status.isOk()) { | 
|  | outPrimaries.red.X = primaries.red.X; | 
|  | outPrimaries.red.Y = primaries.red.Y; | 
|  | outPrimaries.red.Z = primaries.red.Z; | 
|  |  | 
|  | outPrimaries.green.X = primaries.green.X; | 
|  | outPrimaries.green.Y = primaries.green.Y; | 
|  | outPrimaries.green.Z = primaries.green.Z; | 
|  |  | 
|  | outPrimaries.blue.X = primaries.blue.X; | 
|  | outPrimaries.blue.Y = primaries.blue.Y; | 
|  | outPrimaries.blue.Z = primaries.blue.Z; | 
|  |  | 
|  | outPrimaries.white.X = primaries.white.X; | 
|  | outPrimaries.white.Y = primaries.white.Y; | 
|  | outPrimaries.white.Z = primaries.white.Z; | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setActiveColorMode(const sp<IBinder>& display, | 
|  | ColorMode colorMode) { | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService() | 
|  | ->setActiveColorMode(display, static_cast<int>(colorMode)); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getBootDisplayModeSupport(bool* support) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getBootDisplayModeSupport(support); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getOverlaySupport(gui::OverlayProperties* outProperties) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getOverlaySupport(outProperties); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setBootDisplayMode(const sp<IBinder>& display, | 
|  | ui::DisplayModeId displayModeId) { | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService() | 
|  | ->setBootDisplayMode(display, static_cast<int>(displayModeId)); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::clearBootDisplayMode(const sp<IBinder>& display) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->clearBootDisplayMode(display); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getHdrConversionCapabilities( | 
|  | std::vector<gui::HdrConversionCapability>* hdrConversionCapabilities) { | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService()->getHdrConversionCapabilities( | 
|  | hdrConversionCapabilities); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setHdrConversionStrategy( | 
|  | gui::HdrConversionStrategy hdrConversionStrategy, ui::Hdr* outPreferredHdrOutputType) { | 
|  | int hdrType; | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService() | 
|  | ->setHdrConversionStrategy(hdrConversionStrategy, &hdrType); | 
|  | *outPreferredHdrOutputType = static_cast<ui::Hdr>(hdrType); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getHdrOutputConversionSupport(bool* isSupported) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getHdrOutputConversionSupport(isSupported); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setOverrideFrameRate(uid_t uid, float frameRate) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->setOverrideFrameRate(uid, frameRate); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::updateSmallAreaDetection(std::vector<int32_t>& uids, | 
|  | std::vector<float>& thresholds) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->updateSmallAreaDetection(uids, thresholds); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setSmallAreaDetectionThreshold(uid_t uid, float threshold) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->setSmallAreaDetectionThreshold(uid, | 
|  | threshold); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::setAutoLowLatencyMode(const sp<IBinder>& display, bool on) { | 
|  | ComposerServiceAIDL::getComposerService()->setAutoLowLatencyMode(display, on); | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::setGameContentType(const sp<IBinder>& display, bool on) { | 
|  | ComposerServiceAIDL::getComposerService()->setGameContentType(display, on); | 
|  | } | 
|  |  | 
|  | void SurfaceComposerClient::setDisplayPowerMode(const sp<IBinder>& token, | 
|  | int mode) { | 
|  | ComposerServiceAIDL::getComposerService()->setPowerMode(token, mode); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getCompositionPreference( | 
|  | ui::Dataspace* defaultDataspace, ui::PixelFormat* defaultPixelFormat, | 
|  | ui::Dataspace* wideColorGamutDataspace, ui::PixelFormat* wideColorGamutPixelFormat) { | 
|  | gui::CompositionPreference pref; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getCompositionPreference(&pref); | 
|  | if (status.isOk()) { | 
|  | *defaultDataspace = static_cast<ui::Dataspace>(pref.defaultDataspace); | 
|  | *defaultPixelFormat = static_cast<ui::PixelFormat>(pref.defaultPixelFormat); | 
|  | *wideColorGamutDataspace = static_cast<ui::Dataspace>(pref.wideColorGamutDataspace); | 
|  | *wideColorGamutPixelFormat = static_cast<ui::PixelFormat>(pref.wideColorGamutPixelFormat); | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | bool SurfaceComposerClient::getProtectedContentSupport() { | 
|  | bool supported = false; | 
|  | ComposerServiceAIDL::getComposerService()->getProtectedContentSupport(&supported); | 
|  | return supported; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::clearAnimationFrameStats() { | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService()->clearAnimationFrameStats(); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getAnimationFrameStats(FrameStats* outStats) { | 
|  | gui::FrameStats stats; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getAnimationFrameStats(&stats); | 
|  | if (status.isOk()) { | 
|  | outStats->refreshPeriodNano = stats.refreshPeriodNano; | 
|  | outStats->desiredPresentTimesNano.setCapacity(stats.desiredPresentTimesNano.size()); | 
|  | for (const auto& t : stats.desiredPresentTimesNano) { | 
|  | outStats->desiredPresentTimesNano.add(t); | 
|  | } | 
|  | outStats->actualPresentTimesNano.setCapacity(stats.actualPresentTimesNano.size()); | 
|  | for (const auto& t : stats.actualPresentTimesNano) { | 
|  | outStats->actualPresentTimesNano.add(t); | 
|  | } | 
|  | outStats->frameReadyTimesNano.setCapacity(stats.frameReadyTimesNano.size()); | 
|  | for (const auto& t : stats.frameReadyTimesNano) { | 
|  | outStats->frameReadyTimesNano.add(t); | 
|  | } | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::overrideHdrTypes(const sp<IBinder>& display, | 
|  | const std::vector<ui::Hdr>& hdrTypes) { | 
|  | std::vector<int32_t> hdrTypesVector; | 
|  | hdrTypesVector.reserve(hdrTypes.size()); | 
|  | for (auto t : hdrTypes) { | 
|  | hdrTypesVector.push_back(static_cast<int32_t>(t)); | 
|  | } | 
|  |  | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->overrideHdrTypes(display, hdrTypesVector); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::onPullAtom(const int32_t atomId, std::string* outData, | 
|  | bool* success) { | 
|  | gui::PullAtomData pad; | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService()->onPullAtom(atomId, &pad); | 
|  | if (status.isOk()) { | 
|  | outData->assign(pad.data.begin(), pad.data.end()); | 
|  | *success = pad.success; | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getDisplayedContentSamplingAttributes(const sp<IBinder>& display, | 
|  | ui::PixelFormat* outFormat, | 
|  | ui::Dataspace* outDataspace, | 
|  | uint8_t* outComponentMask) { | 
|  | if (!outFormat || !outDataspace || !outComponentMask) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  |  | 
|  | gui::ContentSamplingAttributes attrs; | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService() | 
|  | ->getDisplayedContentSamplingAttributes(display, &attrs); | 
|  | if (status.isOk()) { | 
|  | *outFormat = static_cast<ui::PixelFormat>(attrs.format); | 
|  | *outDataspace = static_cast<ui::Dataspace>(attrs.dataspace); | 
|  | *outComponentMask = static_cast<uint8_t>(attrs.componentMask); | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setDisplayContentSamplingEnabled(const sp<IBinder>& display, | 
|  | bool enable, uint8_t componentMask, | 
|  | uint64_t maxFrames) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService() | 
|  | ->setDisplayContentSamplingEnabled(display, enable, | 
|  | static_cast<int8_t>(componentMask), | 
|  | static_cast<int64_t>(maxFrames)); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::getDisplayedContentSample(const sp<IBinder>& display, | 
|  | uint64_t maxFrames, uint64_t timestamp, | 
|  | DisplayedFrameStats* outStats) { | 
|  | if (!outStats) { | 
|  | return BAD_VALUE; | 
|  | } | 
|  |  | 
|  | gui::DisplayedFrameStats stats; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDisplayedContentSample(display, maxFrames, | 
|  | timestamp, &stats); | 
|  | if (status.isOk()) { | 
|  | // convert gui::DisplayedFrameStats to ui::DisplayedFrameStats | 
|  | outStats->numFrames = static_cast<uint64_t>(stats.numFrames); | 
|  | outStats->component_0_sample.reserve(stats.component_0_sample.size()); | 
|  | for (const auto& s : stats.component_0_sample) { | 
|  | outStats->component_0_sample.push_back(static_cast<uint64_t>(s)); | 
|  | } | 
|  | outStats->component_1_sample.reserve(stats.component_1_sample.size()); | 
|  | for (const auto& s : stats.component_1_sample) { | 
|  | outStats->component_1_sample.push_back(static_cast<uint64_t>(s)); | 
|  | } | 
|  | outStats->component_2_sample.reserve(stats.component_2_sample.size()); | 
|  | for (const auto& s : stats.component_2_sample) { | 
|  | outStats->component_2_sample.push_back(static_cast<uint64_t>(s)); | 
|  | } | 
|  | outStats->component_3_sample.reserve(stats.component_3_sample.size()); | 
|  | for (const auto& s : stats.component_3_sample) { | 
|  | outStats->component_3_sample.push_back(static_cast<uint64_t>(s)); | 
|  | } | 
|  | } | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::isWideColorDisplay(const sp<IBinder>& display, | 
|  | bool* outIsWideColorDisplay) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->isWideColorDisplay(display, | 
|  | outIsWideColorDisplay); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::addRegionSamplingListener( | 
|  | const Rect& samplingArea, const sp<IBinder>& stopLayerHandle, | 
|  | const sp<IRegionSamplingListener>& listener) { | 
|  | gui::ARect rect; | 
|  | rect.left = samplingArea.left; | 
|  | rect.top = samplingArea.top; | 
|  | rect.right = samplingArea.right; | 
|  | rect.bottom = samplingArea.bottom; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->addRegionSamplingListener(rect, | 
|  | stopLayerHandle, | 
|  | listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::removeRegionSamplingListener( | 
|  | const sp<IRegionSamplingListener>& listener) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->removeRegionSamplingListener(listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::addFpsListener(int32_t taskId, | 
|  | const sp<gui::IFpsListener>& listener) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->addFpsListener(taskId, listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::removeFpsListener(const sp<gui::IFpsListener>& listener) { | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService()->removeFpsListener(listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::addTunnelModeEnabledListener( | 
|  | const sp<gui::ITunnelModeEnabledListener>& listener) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->addTunnelModeEnabledListener(listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::removeTunnelModeEnabledListener( | 
|  | const sp<gui::ITunnelModeEnabledListener>& listener) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->removeTunnelModeEnabledListener(listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | bool SurfaceComposerClient::getDisplayBrightnessSupport(const sp<IBinder>& displayToken) { | 
|  | bool support = false; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDisplayBrightnessSupport(displayToken, | 
|  | &support); | 
|  | return status.isOk() ? support : false; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setDisplayBrightness(const sp<IBinder>& displayToken, | 
|  | const gui::DisplayBrightness& brightness) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->setDisplayBrightness(displayToken, | 
|  | brightness); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::addHdrLayerInfoListener( | 
|  | const sp<IBinder>& displayToken, const sp<gui::IHdrLayerInfoListener>& listener) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->addHdrLayerInfoListener(displayToken, | 
|  | listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::removeHdrLayerInfoListener( | 
|  | const sp<IBinder>& displayToken, const sp<gui::IHdrLayerInfoListener>& listener) { | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->removeHdrLayerInfoListener(displayToken, | 
|  | listener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::notifyPowerBoost(int32_t boostId) { | 
|  | binder::Status status = ComposerServiceAIDL::getComposerService()->notifyPowerBoost(boostId); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::setGlobalShadowSettings(const half4& ambientColor, | 
|  | const half4& spotColor, float lightPosY, | 
|  | float lightPosZ, float lightRadius) { | 
|  | gui::Color ambientColorG, spotColorG; | 
|  | ambientColorG.r = ambientColor.r; | 
|  | ambientColorG.g = ambientColor.g; | 
|  | ambientColorG.b = ambientColor.b; | 
|  | ambientColorG.a = ambientColor.a; | 
|  | spotColorG.r = spotColor.r; | 
|  | spotColorG.g = spotColor.g; | 
|  | spotColorG.b = spotColor.b; | 
|  | spotColorG.a = spotColor.a; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->setGlobalShadowSettings(ambientColorG, | 
|  | spotColorG, | 
|  | lightPosY, lightPosZ, | 
|  | lightRadius); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | std::optional<DisplayDecorationSupport> SurfaceComposerClient::getDisplayDecorationSupport( | 
|  | const sp<IBinder>& displayToken) { | 
|  | std::optional<gui::DisplayDecorationSupport> gsupport; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getDisplayDecorationSupport(displayToken, | 
|  | &gsupport); | 
|  | std::optional<DisplayDecorationSupport> support; | 
|  | if (status.isOk() && gsupport.has_value()) { | 
|  | support.emplace(DisplayDecorationSupport{ | 
|  | .format = | 
|  | static_cast<aidl::android::hardware::graphics::common::PixelFormat>( | 
|  | gsupport->format), | 
|  | .alphaInterpretation = | 
|  | static_cast<aidl::android::hardware::graphics::common::AlphaInterpretation>( | 
|  | gsupport->alphaInterpretation) | 
|  | }); | 
|  | } | 
|  | return support; | 
|  | } | 
|  |  | 
|  | int SurfaceComposerClient::getGpuContextPriority() { | 
|  | int priority; | 
|  | binder::Status status = | 
|  | ComposerServiceAIDL::getComposerService()->getGpuContextPriority(&priority); | 
|  | if (!status.isOk()) { | 
|  | status_t err = statusTFromBinderStatus(status); | 
|  | ALOGE("getGpuContextPriority failed to read data:  %s (%d)", strerror(-err), err); | 
|  | return 0; | 
|  | } | 
|  | return priority; | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::addWindowInfosListener( | 
|  | const sp<WindowInfosListener>& windowInfosListener, | 
|  | std::pair<std::vector<gui::WindowInfo>, std::vector<gui::DisplayInfo>>* outInitialInfo) { | 
|  | return WindowInfosListenerReporter::getInstance() | 
|  | ->addWindowInfosListener(windowInfosListener, ComposerServiceAIDL::getComposerService(), | 
|  | outInitialInfo); | 
|  | } | 
|  |  | 
|  | status_t SurfaceComposerClient::removeWindowInfosListener( | 
|  | const sp<WindowInfosListener>& windowInfosListener) { | 
|  | return WindowInfosListenerReporter::getInstance() | 
|  | ->removeWindowInfosListener(windowInfosListener, | 
|  | ComposerServiceAIDL::getComposerService()); | 
|  | } | 
|  |  | 
|  | // ---------------------------------------------------------------------------- | 
|  |  | 
|  | status_t ScreenshotClient::captureDisplay(const DisplayCaptureArgs& captureArgs, | 
|  | const sp<IScreenCaptureListener>& captureListener) { | 
|  | sp<gui::ISurfaceComposer> s(ComposerServiceAIDL::getComposerService()); | 
|  | if (s == nullptr) return NO_INIT; | 
|  |  | 
|  | binder::Status status = s->captureDisplay(captureArgs, captureListener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t ScreenshotClient::captureDisplay(DisplayId displayId, | 
|  | const sp<IScreenCaptureListener>& captureListener) { | 
|  | sp<gui::ISurfaceComposer> s(ComposerServiceAIDL::getComposerService()); | 
|  | if (s == nullptr) return NO_INIT; | 
|  |  | 
|  | binder::Status status = s->captureDisplayById(displayId.value, captureListener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | status_t ScreenshotClient::captureLayers(const LayerCaptureArgs& captureArgs, | 
|  | const sp<IScreenCaptureListener>& captureListener) { | 
|  | sp<gui::ISurfaceComposer> s(ComposerServiceAIDL::getComposerService()); | 
|  | if (s == nullptr) return NO_INIT; | 
|  |  | 
|  | binder::Status status = s->captureLayers(captureArgs, captureListener); | 
|  | return statusTFromBinderStatus(status); | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------------- | 
|  |  | 
|  | void ReleaseCallbackThread::addReleaseCallback(const ReleaseCallbackId callbackId, | 
|  | sp<Fence> releaseFence) { | 
|  | std::scoped_lock<std::mutex> lock(mMutex); | 
|  | if (!mStarted) { | 
|  | mThread = std::thread(&ReleaseCallbackThread::threadMain, this); | 
|  | mStarted = true; | 
|  | } | 
|  |  | 
|  | mCallbackInfos.emplace(callbackId, std::move(releaseFence)); | 
|  | mReleaseCallbackPending.notify_one(); | 
|  | } | 
|  |  | 
|  | void ReleaseCallbackThread::threadMain() { | 
|  | const auto listener = TransactionCompletedListener::getInstance(); | 
|  | std::queue<std::tuple<const ReleaseCallbackId, const sp<Fence>>> callbackInfos; | 
|  | while (true) { | 
|  | { | 
|  | std::unique_lock<std::mutex> lock(mMutex); | 
|  | base::ScopedLockAssertion assumeLocked(mMutex); | 
|  | callbackInfos = std::move(mCallbackInfos); | 
|  | mCallbackInfos = {}; | 
|  | } | 
|  |  | 
|  | while (!callbackInfos.empty()) { | 
|  | auto [callbackId, releaseFence] = callbackInfos.front(); | 
|  | listener->onReleaseBuffer(callbackId, std::move(releaseFence), UINT_MAX); | 
|  | callbackInfos.pop(); | 
|  | } | 
|  |  | 
|  | { | 
|  | std::unique_lock<std::mutex> lock(mMutex); | 
|  | base::ScopedLockAssertion assumeLocked(mMutex); | 
|  | if (mCallbackInfos.size() == 0) { | 
|  | mReleaseCallbackPending.wait(lock); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | } // namespace android |