3149 lines
121 KiB
C++
3149 lines
121 KiB
C++
/*
|
|
* 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/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) {
|
|
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);
|
|
}
|
|
|
|
callbackFunction(transactionStats.latchTime, transactionStats.presentFence,
|
|
surfaceControlStats);
|
|
}
|
|
|
|
// 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& 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;
|
|
clearFrameTimelineInfo(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;
|
|
Vector<DisplayState> displayStates;
|
|
status_t status = sf->setTransactionState(FrameTimelineInfo{}, composerStates, displayStates,
|
|
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;
|
|
|
|
#ifdef MTK_LIBGUI_DEBUG_SUPPORT
|
|
// To debug no-focus window, we log each sent InputWindowCommands.focusRequests
|
|
for (const auto& focusRequest: mInputWindowCommands.focusRequests) {
|
|
ALOGD("Transaction::apply InputWindowCommands.focusRequests timestamp=%" PRId64 ", windowName=%s",
|
|
focusRequest.timestamp,
|
|
focusRequest.windowName.empty() ? "n/a" : focusRequest.windowName.c_str());
|
|
}
|
|
#endif
|
|
|
|
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.string()),
|
|
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;
|
|
}
|
|
|
|
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::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.vsyncId = other.vsyncId;
|
|
t.inputEventId = other.inputEventId;
|
|
t.startTimeNanos = other.startTimeNanos;
|
|
t.useForRefreshRateSelection = other.useForRefreshRateSelection;
|
|
}
|
|
} else if (t.vsyncId == FrameTimelineInfo::INVALID_VSYNC_ID) {
|
|
t.vsyncId = other.vsyncId;
|
|
t.inputEventId = other.inputEventId;
|
|
t.startTimeNanos = other.startTimeNanos;
|
|
t.useForRefreshRateSelection = other.useForRefreshRateSelection;
|
|
}
|
|
}
|
|
|
|
// copied from FrameTimelineInfo::clear()
|
|
void SurfaceComposerClient::Transaction::clearFrameTimelineInfo(FrameTimelineInfo& t) {
|
|
t.vsyncId = FrameTimelineInfo::INVALID_VSYNC_ID;
|
|
t.inputEventId = os::IInputConstants::INVALID_INPUT_EVENT_ID;
|
|
t.startTimeNanos = 0;
|
|
t.useForRefreshRateSelection = false;
|
|
}
|
|
|
|
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.string()), 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);
|
|
}
|
|
|
|
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
|