blob: 64a8ae7fcd12cfad5e1b83d500bb708ff3ebceb9 [file] [log] [blame]
/*
* Copyright 2021 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.
*/
#undef LOG_TAG
#define LOG_TAG "HwcComposer"
#define ATRACE_TAG ATRACE_TAG_GRAPHICS
#include "AidlComposerHal.h"
#include <SurfaceFlingerProperties.h>
#include <android-base/file.h>
#include <android/binder_ibinder_platform.h>
#include <android/binder_manager.h>
#include <common/FlagManager.h>
#include <gui/TraceUtils.h>
#include <log/log.h>
#include <utils/Trace.h>
#include <aidl/android/hardware/graphics/composer3/BnComposerCallback.h>
#include <algorithm>
#include <cinttypes>
#include "HWC2.h"
namespace android {
using hardware::hidl_handle;
using hardware::hidl_vec;
using hardware::Return;
using aidl::android::hardware::graphics::composer3::BnComposerCallback;
using aidl::android::hardware::graphics::composer3::Capability;
using aidl::android::hardware::graphics::composer3::ClientTargetPropertyWithBrightness;
using aidl::android::hardware::graphics::composer3::PowerMode;
using aidl::android::hardware::graphics::composer3::VirtualDisplay;
using aidl::android::hardware::graphics::composer3::CommandResultPayload;
using AidlColorMode = aidl::android::hardware::graphics::composer3::ColorMode;
using AidlContentType = aidl::android::hardware::graphics::composer3::ContentType;
using AidlDisplayIdentification =
aidl::android::hardware::graphics::composer3::DisplayIdentification;
using AidlDisplayContentSample = aidl::android::hardware::graphics::composer3::DisplayContentSample;
using AidlDisplayAttribute = aidl::android::hardware::graphics::composer3::DisplayAttribute;
using AidlDisplayCapability = aidl::android::hardware::graphics::composer3::DisplayCapability;
using AidlHdrCapabilities = aidl::android::hardware::graphics::composer3::HdrCapabilities;
using AidlHdrConversionCapability =
aidl::android::hardware::graphics::common::HdrConversionCapability;
using AidlHdrConversionStrategy = aidl::android::hardware::graphics::common::HdrConversionStrategy;
using AidlOverlayProperties = aidl::android::hardware::graphics::composer3::OverlayProperties;
using AidlPerFrameMetadata = aidl::android::hardware::graphics::composer3::PerFrameMetadata;
using AidlPerFrameMetadataKey = aidl::android::hardware::graphics::composer3::PerFrameMetadataKey;
using AidlPerFrameMetadataBlob = aidl::android::hardware::graphics::composer3::PerFrameMetadataBlob;
using AidlRenderIntent = aidl::android::hardware::graphics::composer3::RenderIntent;
using AidlVsyncPeriodChangeConstraints =
aidl::android::hardware::graphics::composer3::VsyncPeriodChangeConstraints;
using AidlVsyncPeriodChangeTimeline =
aidl::android::hardware::graphics::composer3::VsyncPeriodChangeTimeline;
using AidlDisplayContentSamplingAttributes =
aidl::android::hardware::graphics::composer3::DisplayContentSamplingAttributes;
using AidlFormatColorComponent = aidl::android::hardware::graphics::composer3::FormatColorComponent;
using AidlDisplayConnectionType =
aidl::android::hardware::graphics::composer3::DisplayConnectionType;
using AidlColorTransform = aidl::android::hardware::graphics::common::ColorTransform;
using AidlDataspace = aidl::android::hardware::graphics::common::Dataspace;
using AidlDisplayHotplugEvent = aidl::android::hardware::graphics::common::DisplayHotplugEvent;
using AidlFRect = aidl::android::hardware::graphics::common::FRect;
using AidlRect = aidl::android::hardware::graphics::common::Rect;
using AidlTransform = aidl::android::hardware::graphics::common::Transform;
namespace Hwc2 {
namespace {
template <typename To, typename From>
To translate(From x) {
return static_cast<To>(x);
}
template <typename To, typename From>
std::vector<To> translate(const std::vector<From>& in) {
std::vector<To> out;
out.reserve(in.size());
std::transform(in.begin(), in.end(), std::back_inserter(out),
[](From x) { return translate<To>(x); });
return out;
}
template <>
AidlRect translate(IComposerClient::Rect x) {
return AidlRect{
.left = x.left,
.top = x.top,
.right = x.right,
.bottom = x.bottom,
};
}
template <>
AidlFRect translate(IComposerClient::FRect x) {
return AidlFRect{
.left = x.left,
.top = x.top,
.right = x.right,
.bottom = x.bottom,
};
}
template <>
AidlPerFrameMetadataBlob translate(IComposerClient::PerFrameMetadataBlob x) {
AidlPerFrameMetadataBlob blob;
blob.key = translate<AidlPerFrameMetadataKey>(x.key),
std::copy(x.blob.begin(), x.blob.end(), std::inserter(blob.blob, blob.blob.end()));
return blob;
}
template <>
AidlPerFrameMetadata translate(IComposerClient::PerFrameMetadata x) {
return AidlPerFrameMetadata{
.key = translate<AidlPerFrameMetadataKey>(x.key),
.value = x.value,
};
}
template <>
DisplayedFrameStats translate(AidlDisplayContentSample x) {
return DisplayedFrameStats{
.numFrames = static_cast<uint64_t>(x.frameCount),
.component_0_sample = translate<uint64_t>(x.sampleComponent0),
.component_1_sample = translate<uint64_t>(x.sampleComponent1),
.component_2_sample = translate<uint64_t>(x.sampleComponent2),
.component_3_sample = translate<uint64_t>(x.sampleComponent3),
};
}
template <>
AidlVsyncPeriodChangeConstraints translate(IComposerClient::VsyncPeriodChangeConstraints x) {
return AidlVsyncPeriodChangeConstraints{
.desiredTimeNanos = x.desiredTimeNanos,
.seamlessRequired = x.seamlessRequired,
};
}
template <>
VsyncPeriodChangeTimeline translate(AidlVsyncPeriodChangeTimeline x) {
return VsyncPeriodChangeTimeline{
.newVsyncAppliedTimeNanos = x.newVsyncAppliedTimeNanos,
.refreshRequired = x.refreshRequired,
.refreshTimeNanos = x.refreshTimeNanos,
};
}
mat4 makeMat4(std::vector<float> in) {
return mat4(static_cast<const float*>(in.data()));
}
} // namespace
class AidlIComposerCallbackWrapper : public BnComposerCallback {
public:
AidlIComposerCallbackWrapper(HWC2::ComposerCallback& callback) : mCallback(callback) {}
::ndk::ScopedAStatus onHotplug(int64_t in_display, bool in_connected) override {
const auto event = in_connected ? AidlDisplayHotplugEvent::CONNECTED
: AidlDisplayHotplugEvent::DISCONNECTED;
mCallback.onComposerHalHotplugEvent(translate<Display>(in_display), event);
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onRefresh(int64_t in_display) override {
mCallback.onComposerHalRefresh(translate<Display>(in_display));
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onSeamlessPossible(int64_t in_display) override {
mCallback.onComposerHalSeamlessPossible(translate<Display>(in_display));
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onVsync(int64_t in_display, int64_t in_timestamp,
int32_t in_vsyncPeriodNanos) override {
mCallback.onComposerHalVsync(translate<Display>(in_display), in_timestamp,
static_cast<uint32_t>(in_vsyncPeriodNanos));
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onVsyncPeriodTimingChanged(
int64_t in_display, const AidlVsyncPeriodChangeTimeline& in_updatedTimeline) override {
mCallback.onComposerHalVsyncPeriodTimingChanged(translate<Display>(in_display),
translate<V2_4::VsyncPeriodChangeTimeline>(
in_updatedTimeline));
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onVsyncIdle(int64_t in_display) override {
mCallback.onComposerHalVsyncIdle(translate<Display>(in_display));
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onRefreshRateChangedDebug(
const RefreshRateChangedDebugData& refreshRateChangedDebugData) override {
mCallback.onRefreshRateChangedDebug(refreshRateChangedDebugData);
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus onHotplugEvent(int64_t in_display,
AidlDisplayHotplugEvent event) override {
mCallback.onComposerHalHotplugEvent(translate<Display>(in_display), event);
return ::ndk::ScopedAStatus::ok();
}
private:
HWC2::ComposerCallback& mCallback;
};
std::string AidlComposer::instance(const std::string& serviceName) {
return std::string(AidlIComposer::descriptor) + "/" + serviceName;
}
bool AidlComposer::isDeclared(const std::string& serviceName) {
return AServiceManager_isDeclared(instance(serviceName).c_str());
}
AidlComposer::AidlComposer(const std::string& serviceName) {
// This only waits if the service is actually declared
mAidlComposer = AidlIComposer::fromBinder(
ndk::SpAIBinder(AServiceManager_waitForService(instance(serviceName).c_str())));
if (!mAidlComposer) {
LOG_ALWAYS_FATAL("Failed to get AIDL composer service");
return;
}
if (!mAidlComposer->createClient(&mAidlComposerClient).isOk()) {
LOG_ALWAYS_FATAL("Can't create AidlComposerClient, fallback to HIDL");
return;
}
addReader(translate<Display>(kSingleReaderKey));
// If unable to read interface version, then become backwards compatible.
const auto status = mAidlComposerClient->getInterfaceVersion(&mComposerInterfaceVersion);
if (!status.isOk()) {
ALOGE("getInterfaceVersion for AidlComposer constructor failed %s",
status.getDescription().c_str());
}
if (mComposerInterfaceVersion <= 1) {
if (sysprop::clear_slots_with_set_layer_buffer(false)) {
mClearSlotBuffer = sp<GraphicBuffer>::make(1, 1, PIXEL_FORMAT_RGBX_8888,
GraphicBuffer::USAGE_HW_COMPOSER |
GraphicBuffer::USAGE_SW_READ_OFTEN |
GraphicBuffer::USAGE_SW_WRITE_OFTEN,
"AidlComposer");
if (!mClearSlotBuffer || mClearSlotBuffer->initCheck() != ::android::OK) {
LOG_ALWAYS_FATAL("Failed to allocate a buffer for clearing layer buffer slots");
return;
}
}
}
if (getLayerLifecycleBatchCommand()) {
mEnableLayerCommandBatchingFlag =
FlagManager::getInstance().enable_layer_command_batching();
}
ALOGI("Loaded AIDL composer3 HAL service");
}
AidlComposer::~AidlComposer() = default;
bool AidlComposer::isSupported(OptionalFeature feature) const {
switch (feature) {
case OptionalFeature::RefreshRateSwitching:
case OptionalFeature::ExpectedPresentTime:
case OptionalFeature::DisplayBrightnessCommand:
case OptionalFeature::KernelIdleTimer:
case OptionalFeature::PhysicalDisplayOrientation:
return true;
}
}
bool AidlComposer::isVrrSupported() const {
return mComposerInterfaceVersion >= 3 && FlagManager::getInstance().vrr_config();
}
std::vector<Capability> AidlComposer::getCapabilities() {
std::vector<Capability> capabilities;
const auto status = mAidlComposer->getCapabilities(&capabilities);
if (!status.isOk()) {
ALOGE("getCapabilities failed %s", status.getDescription().c_str());
return {};
}
return capabilities;
}
std::string AidlComposer::dumpDebugInfo() {
int pipefds[2];
int result = pipe(pipefds);
if (result < 0) {
ALOGE("dumpDebugInfo: pipe failed: %s", strerror(errno));
return {};
}
std::string str;
// Use other thread to read pipe to prevent
// pipe is full, making HWC be blocked in writing.
std::thread t([&]() {
base::ReadFdToString(pipefds[0], &str);
});
const auto status = mAidlComposer->dump(pipefds[1], /*args*/ nullptr, /*numArgs*/ 0);
// Close the write-end of the pipe to make sure that when reading from the
// read-end we will get eof instead of blocking forever
close(pipefds[1]);
if (status != STATUS_OK) {
ALOGE("dumpDebugInfo: dump failed: %d", status);
}
t.join();
close(pipefds[0]);
return str;
}
void AidlComposer::registerCallback(HWC2::ComposerCallback& callback) {
if (mAidlComposerCallback) {
ALOGE("Callback already registered");
}
mAidlComposerCallback = ndk::SharedRefBase::make<AidlIComposerCallbackWrapper>(callback);
AIBinder_setMinSchedulerPolicy(mAidlComposerCallback->asBinder().get(), SCHED_FIFO, 2);
const auto status = mAidlComposerClient->registerCallback(mAidlComposerCallback);
if (!status.isOk()) {
ALOGE("registerCallback failed %s", status.getDescription().c_str());
}
}
Error AidlComposer::executeCommands(Display display) {
mMutex.lock_shared();
auto error = execute(display);
mMutex.unlock_shared();
return error;
}
uint32_t AidlComposer::getMaxVirtualDisplayCount() {
int32_t count = 0;
const auto status = mAidlComposerClient->getMaxVirtualDisplayCount(&count);
if (!status.isOk()) {
ALOGE("getMaxVirtualDisplayCount failed %s", status.getDescription().c_str());
return 0;
}
return static_cast<uint32_t>(count);
}
Error AidlComposer::createVirtualDisplay(uint32_t width, uint32_t height, PixelFormat* format,
Display* outDisplay) {
using AidlPixelFormat = aidl::android::hardware::graphics::common::PixelFormat;
const int32_t bufferSlotCount = 1;
VirtualDisplay virtualDisplay;
const auto status =
mAidlComposerClient->createVirtualDisplay(static_cast<int32_t>(width),
static_cast<int32_t>(height),
static_cast<AidlPixelFormat>(*format),
bufferSlotCount, &virtualDisplay);
if (!status.isOk()) {
ALOGE("createVirtualDisplay failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outDisplay = translate<Display>(virtualDisplay.display);
*format = static_cast<PixelFormat>(virtualDisplay.format);
addDisplay(translate<Display>(virtualDisplay.display));
return Error::NONE;
}
Error AidlComposer::destroyVirtualDisplay(Display display) {
const auto status = mAidlComposerClient->destroyVirtualDisplay(translate<int64_t>(display));
if (!status.isOk()) {
ALOGE("destroyVirtualDisplay failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
removeDisplay(display);
return Error::NONE;
}
Error AidlComposer::acceptDisplayChanges(Display display) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().acceptDisplayChanges(translate<int64_t>(display));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::createLayer(Display display, Layer* outLayer) {
int64_t layer;
Error error = Error::NONE;
if (!mEnableLayerCommandBatchingFlag) {
const auto status = mAidlComposerClient->createLayer(translate<int64_t>(display),
kMaxLayerBufferCount, &layer);
if (!status.isOk()) {
ALOGE("createLayer failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
} else {
// generate a unique layerID. map in AidlComposer with <SF_layerID, HWC_layerID>
// Add this as a new displayCommand in execute command.
// return the SF generated layerID instead of calling HWC
layer = mLayerID++;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerLifecycleBatchCommandType(translate<int64_t>(display),
translate<int64_t>(layer),
LayerLifecycleBatchCommandType::CREATE);
writer->get().setNewBufferSlotCount(translate<int64_t>(display),
translate<int64_t>(layer), kMaxLayerBufferCount);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
}
*outLayer = translate<Layer>(layer);
return error;
}
Error AidlComposer::destroyLayer(Display display, Layer layer) {
Error error = Error::NONE;
if (!mEnableLayerCommandBatchingFlag) {
const auto status = mAidlComposerClient->destroyLayer(translate<int64_t>(display),
translate<int64_t>(layer));
if (!status.isOk()) {
ALOGE("destroyLayer failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
} else {
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get()
.setLayerLifecycleBatchCommandType(translate<int64_t>(display),
translate<int64_t>(layer),
LayerLifecycleBatchCommandType::DESTROY);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
}
return error;
}
Error AidlComposer::getActiveConfig(Display display, Config* outConfig) {
int32_t config;
const auto status = mAidlComposerClient->getActiveConfig(translate<int64_t>(display), &config);
if (!status.isOk()) {
ALOGE("getActiveConfig failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outConfig = translate<Config>(config);
return Error::NONE;
}
Error AidlComposer::getChangedCompositionTypes(
Display display, std::vector<Layer>* outLayers,
std::vector<aidl::android::hardware::graphics::composer3::Composition>* outTypes) {
std::vector<ChangedCompositionLayer> changedLayers;
Error error = Error::NONE;
{
mMutex.lock_shared();
if (auto reader = getReader(display)) {
changedLayers = reader->get().takeChangedCompositionTypes(translate<int64_t>(display));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
}
outLayers->reserve(changedLayers.size());
outTypes->reserve(changedLayers.size());
for (const auto& layer : changedLayers) {
outLayers->emplace_back(translate<Layer>(layer.layer));
outTypes->emplace_back(layer.composition);
}
return error;
}
Error AidlComposer::getColorModes(Display display, std::vector<ColorMode>* outModes) {
std::vector<AidlColorMode> modes;
const auto status = mAidlComposerClient->getColorModes(translate<int64_t>(display), &modes);
if (!status.isOk()) {
ALOGE("getColorModes failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outModes = translate<ColorMode>(modes);
return Error::NONE;
}
Error AidlComposer::getDisplayAttribute(Display display, Config config,
IComposerClient::Attribute attribute, int32_t* outValue) {
const auto status =
mAidlComposerClient->getDisplayAttribute(translate<int64_t>(display),
translate<int32_t>(config),
static_cast<AidlDisplayAttribute>(attribute),
outValue);
if (!status.isOk()) {
ALOGE("getDisplayAttribute failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getDisplayConfigs(Display display, std::vector<Config>* outConfigs) {
std::vector<int32_t> configs;
const auto status =
mAidlComposerClient->getDisplayConfigs(translate<int64_t>(display), &configs);
if (!status.isOk()) {
ALOGE("getDisplayConfigs failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outConfigs = translate<Config>(configs);
return Error::NONE;
}
Error AidlComposer::getDisplayConfigurations(Display display, int32_t maxFrameIntervalNs,
std::vector<DisplayConfiguration>* outConfigs) {
const auto status =
mAidlComposerClient->getDisplayConfigurations(translate<int64_t>(display),
maxFrameIntervalNs, outConfigs);
if (!status.isOk()) {
ALOGE("getDisplayConfigurations failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getDisplayName(Display display, std::string* outName) {
const auto status = mAidlComposerClient->getDisplayName(translate<int64_t>(display), outName);
if (!status.isOk()) {
ALOGE("getDisplayName failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getDisplayRequests(Display display, uint32_t* outDisplayRequestMask,
std::vector<Layer>* outLayers,
std::vector<uint32_t>* outLayerRequestMasks) {
Error error = Error::NONE;
DisplayRequest displayRequests;
{
mMutex.lock_shared();
if (auto reader = getReader(display)) {
displayRequests = reader->get().takeDisplayRequests(translate<int64_t>(display));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
}
*outDisplayRequestMask = translate<uint32_t>(displayRequests.mask);
outLayers->reserve(displayRequests.layerRequests.size());
outLayerRequestMasks->reserve(displayRequests.layerRequests.size());
for (const auto& layer : displayRequests.layerRequests) {
outLayers->emplace_back(translate<Layer>(layer.layer));
outLayerRequestMasks->emplace_back(translate<uint32_t>(layer.mask));
}
return error;
}
Error AidlComposer::getDozeSupport(Display display, bool* outSupport) {
std::vector<AidlDisplayCapability> capabilities;
const auto status =
mAidlComposerClient->getDisplayCapabilities(translate<int64_t>(display), &capabilities);
if (!status.isOk()) {
ALOGE("getDisplayCapabilities failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outSupport = std::find(capabilities.begin(), capabilities.end(),
AidlDisplayCapability::DOZE) != capabilities.end();
return Error::NONE;
}
Error AidlComposer::hasDisplayIdleTimerCapability(Display display, bool* outSupport) {
std::vector<AidlDisplayCapability> capabilities;
const auto status =
mAidlComposerClient->getDisplayCapabilities(translate<int64_t>(display), &capabilities);
if (!status.isOk()) {
ALOGE("getDisplayCapabilities failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outSupport = std::find(capabilities.begin(), capabilities.end(),
AidlDisplayCapability::DISPLAY_IDLE_TIMER) != capabilities.end();
return Error::NONE;
}
Error AidlComposer::getHdrCapabilities(Display display, std::vector<Hdr>* outTypes,
float* outMaxLuminance, float* outMaxAverageLuminance,
float* outMinLuminance) {
AidlHdrCapabilities capabilities;
const auto status =
mAidlComposerClient->getHdrCapabilities(translate<int64_t>(display), &capabilities);
if (!status.isOk()) {
ALOGE("getHdrCapabilities failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outTypes = capabilities.types;
*outMaxLuminance = capabilities.maxLuminance;
*outMaxAverageLuminance = capabilities.maxAverageLuminance;
*outMinLuminance = capabilities.minLuminance;
return Error::NONE;
}
bool AidlComposer::getLayerLifecycleBatchCommand() {
std::vector<Capability> capabilities = getCapabilities();
bool hasCapability = std::find(capabilities.begin(), capabilities.end(),
Capability::LAYER_LIFECYCLE_BATCH_COMMAND) != capabilities.end();
return hasCapability;
}
Error AidlComposer::getOverlaySupport(AidlOverlayProperties* outProperties) {
const auto status = mAidlComposerClient->getOverlaySupport(outProperties);
if (!status.isOk()) {
ALOGE("getOverlaySupport failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getReleaseFences(Display display, std::vector<Layer>* outLayers,
std::vector<int>* outReleaseFences) {
Error error = Error::NONE;
std::vector<ReleaseFences::Layer> fences;
{
mMutex.lock_shared();
if (auto reader = getReader(display)) {
fences = reader->get().takeReleaseFences(translate<int64_t>(display));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
}
outLayers->reserve(fences.size());
outReleaseFences->reserve(fences.size());
for (auto& fence : fences) {
outLayers->emplace_back(translate<Layer>(fence.layer));
// take ownership
const int fenceOwner = fence.fence.get();
*fence.fence.getR() = -1;
outReleaseFences->emplace_back(fenceOwner);
}
return error;
}
Error AidlComposer::presentDisplay(Display display, int* outPresentFence) {
const auto displayId = translate<int64_t>(display);
ATRACE_FORMAT("HwcPresentDisplay %" PRId64, displayId);
Error error = Error::NONE;
mMutex.lock_shared();
auto writer = getWriter(display);
auto reader = getReader(display);
if (writer && reader) {
writer->get().presentDisplay(displayId);
error = execute(display);
} else {
error = Error::BAD_DISPLAY;
}
if (error != Error::NONE) {
mMutex.unlock_shared();
return error;
}
auto fence = reader->get().takePresentFence(displayId);
mMutex.unlock_shared();
// take ownership
*outPresentFence = fence.get();
*fence.getR() = -1;
return Error::NONE;
}
Error AidlComposer::setActiveConfig(Display display, Config config) {
const auto status = mAidlComposerClient->setActiveConfig(translate<int64_t>(display),
translate<int32_t>(config));
if (!status.isOk()) {
ALOGE("setActiveConfig failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setClientTarget(Display display, uint32_t slot, const sp<GraphicBuffer>& target,
int acquireFence, Dataspace dataspace,
const std::vector<IComposerClient::Rect>& damage,
float hdrSdrRatio) {
const native_handle_t* handle = nullptr;
if (target.get()) {
handle = target->getNativeBuffer()->handle;
}
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get()
.setClientTarget(translate<int64_t>(display), slot, handle, acquireFence,
translate<aidl::android::hardware::graphics::common::Dataspace>(
dataspace),
translate<AidlRect>(damage), hdrSdrRatio);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setColorMode(Display display, ColorMode mode, RenderIntent renderIntent) {
const auto status =
mAidlComposerClient->setColorMode(translate<int64_t>(display),
translate<AidlColorMode>(mode),
translate<AidlRenderIntent>(renderIntent));
if (!status.isOk()) {
ALOGE("setColorMode failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setColorTransform(Display display, const float* matrix) {
auto error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setColorTransform(translate<int64_t>(display), matrix);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setOutputBuffer(Display display, const native_handle_t* buffer,
int releaseFence) {
auto error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setOutputBuffer(translate<int64_t>(display), 0, buffer, dup(releaseFence));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setPowerMode(Display display, IComposerClient::PowerMode mode) {
const auto status = mAidlComposerClient->setPowerMode(translate<int64_t>(display),
translate<PowerMode>(mode));
if (!status.isOk()) {
ALOGE("setPowerMode failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setVsyncEnabled(Display display, IComposerClient::Vsync enabled) {
const bool enableVsync = enabled == IComposerClient::Vsync::ENABLE;
const auto status =
mAidlComposerClient->setVsyncEnabled(translate<int64_t>(display), enableVsync);
if (!status.isOk()) {
ALOGE("setVsyncEnabled failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setClientTargetSlotCount(Display display) {
const int32_t bufferSlotCount = BufferQueue::NUM_BUFFER_SLOTS;
const auto status = mAidlComposerClient->setClientTargetSlotCount(translate<int64_t>(display),
bufferSlotCount);
if (!status.isOk()) {
ALOGE("setClientTargetSlotCount failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::validateDisplay(Display display, nsecs_t expectedPresentTime,
int32_t frameIntervalNs, uint32_t* outNumTypes,
uint32_t* outNumRequests) {
const auto displayId = translate<int64_t>(display);
ATRACE_FORMAT("HwcValidateDisplay %" PRId64, displayId);
Error error = Error::NONE;
mMutex.lock_shared();
auto writer = getWriter(display);
auto reader = getReader(display);
if (writer && reader) {
writer->get().validateDisplay(displayId, ClockMonotonicTimestamp{expectedPresentTime},
frameIntervalNs);
error = execute(display);
} else {
error = Error::BAD_DISPLAY;
}
if (error != Error::NONE) {
mMutex.unlock_shared();
return error;
}
reader->get().hasChanges(displayId, outNumTypes, outNumRequests);
mMutex.unlock_shared();
return Error::NONE;
}
Error AidlComposer::presentOrValidateDisplay(Display display, nsecs_t expectedPresentTime,
int32_t frameIntervalNs, uint32_t* outNumTypes,
uint32_t* outNumRequests, int* outPresentFence,
uint32_t* state) {
const auto displayId = translate<int64_t>(display);
ATRACE_FORMAT("HwcPresentOrValidateDisplay %" PRId64, displayId);
Error error = Error::NONE;
mMutex.lock_shared();
auto writer = getWriter(display);
auto reader = getReader(display);
if (writer && reader) {
writer->get().presentOrvalidateDisplay(displayId,
ClockMonotonicTimestamp{expectedPresentTime},
frameIntervalNs);
error = execute(display);
} else {
error = Error::BAD_DISPLAY;
}
if (error != Error::NONE) {
mMutex.unlock_shared();
return error;
}
const auto result = reader->get().takePresentOrValidateStage(displayId);
if (!result.has_value()) {
*state = translate<uint32_t>(-1);
mMutex.unlock_shared();
return Error::NO_RESOURCES;
}
*state = translate<uint32_t>(*result);
if (*result == PresentOrValidate::Result::Presented) {
auto fence = reader->get().takePresentFence(displayId);
// take ownership
*outPresentFence = fence.get();
*fence.getR() = -1;
}
if (*result == PresentOrValidate::Result::Validated) {
reader->get().hasChanges(displayId, outNumTypes, outNumRequests);
}
mMutex.unlock_shared();
return Error::NONE;
}
Error AidlComposer::setCursorPosition(Display display, Layer layer, int32_t x, int32_t y) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerCursorPosition(translate<int64_t>(display), translate<int64_t>(layer),
x, y);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerBuffer(Display display, Layer layer, uint32_t slot,
const sp<GraphicBuffer>& buffer, int acquireFence) {
const native_handle_t* handle = nullptr;
if (buffer.get()) {
handle = buffer->getNativeBuffer()->handle;
}
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerBuffer(translate<int64_t>(display), translate<int64_t>(layer), slot,
handle, acquireFence);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerBufferSlotsToClear(Display display, Layer layer,
const std::vector<uint32_t>& slotsToClear,
uint32_t activeBufferSlot) {
if (slotsToClear.empty()) {
return Error::NONE;
}
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
if (mComposerInterfaceVersion > 1) {
writer->get().setLayerBufferSlotsToClear(translate<int64_t>(display),
translate<int64_t>(layer), slotsToClear);
// Backwards compatible way of clearing buffer slots is to set the layer buffer with a
// placeholder buffer, using the slot that needs to cleared... tricky.
} else if (mClearSlotBuffer != nullptr) {
for (uint32_t slot : slotsToClear) {
// Don't clear the active buffer slot because we need to restore the active buffer
// after clearing the requested buffer slots with a placeholder buffer.
if (slot != activeBufferSlot) {
writer->get().setLayerBufferWithNewCommand(translate<int64_t>(display),
translate<int64_t>(layer), slot,
mClearSlotBuffer->handle,
/*fence*/ -1);
}
}
// Since we clear buffers by setting them to a placeholder buffer, we want to make
// sure that the last setLayerBuffer command is sent with the currently active
// buffer, not the placeholder buffer, so that there is no perceptual change when
// buffers are discarded.
writer->get().setLayerBufferWithNewCommand(translate<int64_t>(display),
translate<int64_t>(layer), activeBufferSlot,
// The active buffer is still cached in
// its slot and doesn't need a fence.
/*buffer*/ nullptr, /*fence*/ -1);
}
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerSurfaceDamage(Display display, Layer layer,
const std::vector<IComposerClient::Rect>& damage) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerSurfaceDamage(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlRect>(damage));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerBlendMode(Display display, Layer layer,
IComposerClient::BlendMode mode) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerBlendMode(translate<int64_t>(display), translate<int64_t>(layer),
translate<BlendMode>(mode));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerColor(Display display, Layer layer, const Color& color) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerColor(translate<int64_t>(display), translate<int64_t>(layer), color);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerCompositionType(
Display display, Layer layer,
aidl::android::hardware::graphics::composer3::Composition type) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerCompositionType(translate<int64_t>(display),
translate<int64_t>(layer), type);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerDataspace(Display display, Layer layer, Dataspace dataspace) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerDataspace(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlDataspace>(dataspace));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerDisplayFrame(Display display, Layer layer,
const IComposerClient::Rect& frame) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerDisplayFrame(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlRect>(frame));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerPlaneAlpha(Display display, Layer layer, float alpha) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerPlaneAlpha(translate<int64_t>(display), translate<int64_t>(layer),
alpha);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerSidebandStream(Display display, Layer layer,
const native_handle_t* stream) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerSidebandStream(translate<int64_t>(display), translate<int64_t>(layer),
stream);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerSourceCrop(Display display, Layer layer,
const IComposerClient::FRect& crop) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerSourceCrop(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlFRect>(crop));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerTransform(Display display, Layer layer, Transform transform) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerTransform(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlTransform>(transform));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerVisibleRegion(Display display, Layer layer,
const std::vector<IComposerClient::Rect>& visible) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerVisibleRegion(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlRect>(visible));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerZOrder(Display display, Layer layer, uint32_t z) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerZOrder(translate<int64_t>(display), translate<int64_t>(layer), z);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::execute(Display display) {
auto writer = getWriter(display);
auto reader = getReader(display);
if (!writer || !reader) {
return Error::BAD_DISPLAY;
}
auto commands = writer->get().takePendingCommands();
if (commands.empty()) {
return Error::NONE;
}
{ // scope for results
std::vector<CommandResultPayload> results;
auto status = mAidlComposerClient->executeCommands(commands, &results);
if (!status.isOk()) {
ALOGE("executeCommands failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
reader->get().parse(std::move(results));
}
const auto commandErrors = reader->get().takeErrors();
Error error = Error::NONE;
for (const auto& cmdErr : commandErrors) {
const auto index = static_cast<size_t>(cmdErr.commandIndex);
if (index < 0 || index >= commands.size()) {
ALOGE("invalid command index %zu", index);
return Error::BAD_PARAMETER;
}
const auto& command = commands[index];
if (command.validateDisplay || command.presentDisplay || command.presentOrValidateDisplay) {
error = translate<Error>(cmdErr.errorCode);
} else {
ALOGW("command '%s' generated error %" PRId32, command.toString().c_str(),
cmdErr.errorCode);
}
}
return error;
}
Error AidlComposer::setLayerPerFrameMetadata(
Display display, Layer layer,
const std::vector<IComposerClient::PerFrameMetadata>& perFrameMetadatas) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerPerFrameMetadata(translate<int64_t>(display),
translate<int64_t>(layer),
translate<AidlPerFrameMetadata>(perFrameMetadatas));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
std::vector<IComposerClient::PerFrameMetadataKey> AidlComposer::getPerFrameMetadataKeys(
Display display) {
std::vector<AidlPerFrameMetadataKey> keys;
const auto status =
mAidlComposerClient->getPerFrameMetadataKeys(translate<int64_t>(display), &keys);
if (!status.isOk()) {
ALOGE("getPerFrameMetadataKeys failed %s", status.getDescription().c_str());
return {};
}
return translate<IComposerClient::PerFrameMetadataKey>(keys);
}
Error AidlComposer::getRenderIntents(Display display, ColorMode colorMode,
std::vector<RenderIntent>* outRenderIntents) {
std::vector<AidlRenderIntent> renderIntents;
const auto status = mAidlComposerClient->getRenderIntents(translate<int64_t>(display),
translate<AidlColorMode>(colorMode),
&renderIntents);
if (!status.isOk()) {
ALOGE("getRenderIntents failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outRenderIntents = translate<RenderIntent>(renderIntents);
return Error::NONE;
}
Error AidlComposer::getDataspaceSaturationMatrix(Dataspace dataspace, mat4* outMatrix) {
std::vector<float> matrix;
const auto status =
mAidlComposerClient->getDataspaceSaturationMatrix(translate<AidlDataspace>(dataspace),
&matrix);
if (!status.isOk()) {
ALOGE("getDataspaceSaturationMatrix failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outMatrix = makeMat4(matrix);
return Error::NONE;
}
Error AidlComposer::getDisplayIdentificationData(Display display, uint8_t* outPort,
std::vector<uint8_t>* outData) {
AidlDisplayIdentification displayIdentification;
const auto status =
mAidlComposerClient->getDisplayIdentificationData(translate<int64_t>(display),
&displayIdentification);
if (!status.isOk()) {
ALOGE("getDisplayIdentificationData failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outPort = static_cast<uint8_t>(displayIdentification.port);
*outData = displayIdentification.data;
return Error::NONE;
}
Error AidlComposer::setLayerColorTransform(Display display, Layer layer, const float* matrix) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerColorTransform(translate<int64_t>(display), translate<int64_t>(layer),
matrix);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::getDisplayedContentSamplingAttributes(Display display, PixelFormat* outFormat,
Dataspace* outDataspace,
uint8_t* outComponentMask) {
if (!outFormat || !outDataspace || !outComponentMask) {
return Error::BAD_PARAMETER;
}
AidlDisplayContentSamplingAttributes attributes;
const auto status =
mAidlComposerClient->getDisplayedContentSamplingAttributes(translate<int64_t>(display),
&attributes);
if (!status.isOk()) {
ALOGE("getDisplayedContentSamplingAttributes failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outFormat = translate<PixelFormat>(attributes.format);
*outDataspace = translate<Dataspace>(attributes.dataspace);
*outComponentMask = static_cast<uint8_t>(attributes.componentMask);
return Error::NONE;
}
Error AidlComposer::setDisplayContentSamplingEnabled(Display display, bool enabled,
uint8_t componentMask, uint64_t maxFrames) {
const auto status =
mAidlComposerClient
->setDisplayedContentSamplingEnabled(translate<int64_t>(display), enabled,
static_cast<AidlFormatColorComponent>(
componentMask),
static_cast<int64_t>(maxFrames));
if (!status.isOk()) {
ALOGE("setDisplayedContentSamplingEnabled failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getDisplayedContentSample(Display display, uint64_t maxFrames,
uint64_t timestamp, DisplayedFrameStats* outStats) {
if (!outStats) {
return Error::BAD_PARAMETER;
}
AidlDisplayContentSample sample;
const auto status =
mAidlComposerClient->getDisplayedContentSample(translate<int64_t>(display),
static_cast<int64_t>(maxFrames),
static_cast<int64_t>(timestamp),
&sample);
if (!status.isOk()) {
ALOGE("getDisplayedContentSample failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*outStats = translate<DisplayedFrameStats>(sample);
return Error::NONE;
}
Error AidlComposer::setLayerPerFrameMetadataBlobs(
Display display, Layer layer,
const std::vector<IComposerClient::PerFrameMetadataBlob>& metadata) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerPerFrameMetadataBlobs(translate<int64_t>(display),
translate<int64_t>(layer),
translate<AidlPerFrameMetadataBlob>(metadata));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setDisplayBrightness(Display display, float brightness, float brightnessNits,
const DisplayBrightnessOptions& options) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setDisplayBrightness(translate<int64_t>(display), brightness, brightnessNits);
if (options.applyImmediately) {
error = execute(display);
mMutex.unlock_shared();
return error;
}
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::getDisplayCapabilities(Display display,
std::vector<AidlDisplayCapability>* outCapabilities) {
const auto status = mAidlComposerClient->getDisplayCapabilities(translate<int64_t>(display),
outCapabilities);
if (!status.isOk()) {
ALOGE("getDisplayCapabilities failed %s", status.getDescription().c_str());
outCapabilities->clear();
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
V2_4::Error AidlComposer::getDisplayConnectionType(
Display display, IComposerClient::DisplayConnectionType* outType) {
AidlDisplayConnectionType type;
const auto status =
mAidlComposerClient->getDisplayConnectionType(translate<int64_t>(display), &type);
if (!status.isOk()) {
ALOGE("getDisplayConnectionType failed %s", status.getDescription().c_str());
return static_cast<V2_4::Error>(status.getServiceSpecificError());
}
*outType = translate<IComposerClient::DisplayConnectionType>(type);
return V2_4::Error::NONE;
}
V2_4::Error AidlComposer::getDisplayVsyncPeriod(Display display, VsyncPeriodNanos* outVsyncPeriod) {
int32_t vsyncPeriod;
const auto status =
mAidlComposerClient->getDisplayVsyncPeriod(translate<int64_t>(display), &vsyncPeriod);
if (!status.isOk()) {
ALOGE("getDisplayVsyncPeriod failed %s", status.getDescription().c_str());
return static_cast<V2_4::Error>(status.getServiceSpecificError());
}
*outVsyncPeriod = translate<VsyncPeriodNanos>(vsyncPeriod);
return V2_4::Error::NONE;
}
V2_4::Error AidlComposer::setActiveConfigWithConstraints(
Display display, Config config,
const IComposerClient::VsyncPeriodChangeConstraints& vsyncPeriodChangeConstraints,
VsyncPeriodChangeTimeline* outTimeline) {
AidlVsyncPeriodChangeTimeline timeline;
const auto status =
mAidlComposerClient
->setActiveConfigWithConstraints(translate<int64_t>(display),
translate<int32_t>(config),
translate<AidlVsyncPeriodChangeConstraints>(
vsyncPeriodChangeConstraints),
&timeline);
if (!status.isOk()) {
ALOGE("setActiveConfigWithConstraints failed %s", status.getDescription().c_str());
return static_cast<V2_4::Error>(status.getServiceSpecificError());
}
*outTimeline = translate<VsyncPeriodChangeTimeline>(timeline);
return V2_4::Error::NONE;
}
V2_4::Error AidlComposer::setAutoLowLatencyMode(Display display, bool on) {
const auto status = mAidlComposerClient->setAutoLowLatencyMode(translate<int64_t>(display), on);
if (!status.isOk()) {
ALOGE("setAutoLowLatencyMode failed %s", status.getDescription().c_str());
return static_cast<V2_4::Error>(status.getServiceSpecificError());
}
return V2_4::Error::NONE;
}
V2_4::Error AidlComposer::getSupportedContentTypes(
Display displayId, std::vector<IComposerClient::ContentType>* outSupportedContentTypes) {
std::vector<AidlContentType> types;
const auto status =
mAidlComposerClient->getSupportedContentTypes(translate<int64_t>(displayId), &types);
if (!status.isOk()) {
ALOGE("getSupportedContentTypes failed %s", status.getDescription().c_str());
return static_cast<V2_4::Error>(status.getServiceSpecificError());
}
*outSupportedContentTypes = translate<IComposerClient::ContentType>(types);
return V2_4::Error::NONE;
}
V2_4::Error AidlComposer::setContentType(Display display,
IComposerClient::ContentType contentType) {
const auto status =
mAidlComposerClient->setContentType(translate<int64_t>(display),
translate<AidlContentType>(contentType));
if (!status.isOk()) {
ALOGE("setContentType failed %s", status.getDescription().c_str());
return static_cast<V2_4::Error>(status.getServiceSpecificError());
}
return V2_4::Error::NONE;
}
V2_4::Error AidlComposer::setLayerGenericMetadata(Display, Layer, const std::string&, bool,
const std::vector<uint8_t>&) {
// There are no users for this API. See b/209691612.
return V2_4::Error::UNSUPPORTED;
}
V2_4::Error AidlComposer::getLayerGenericMetadataKeys(
std::vector<IComposerClient::LayerGenericMetadataKey>*) {
// There are no users for this API. See b/209691612.
return V2_4::Error::UNSUPPORTED;
}
Error AidlComposer::setBootDisplayConfig(Display display, Config config) {
const auto status = mAidlComposerClient->setBootDisplayConfig(translate<int64_t>(display),
translate<int32_t>(config));
if (!status.isOk()) {
ALOGE("setBootDisplayConfig failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::clearBootDisplayConfig(Display display) {
const auto status = mAidlComposerClient->clearBootDisplayConfig(translate<int64_t>(display));
if (!status.isOk()) {
ALOGE("clearBootDisplayConfig failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getPreferredBootDisplayConfig(Display display, Config* config) {
int32_t displayConfig;
const auto status =
mAidlComposerClient->getPreferredBootDisplayConfig(translate<int64_t>(display),
&displayConfig);
if (!status.isOk()) {
ALOGE("getPreferredBootDisplayConfig failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
*config = translate<uint32_t>(displayConfig);
return Error::NONE;
}
Error AidlComposer::getHdrConversionCapabilities(
std::vector<AidlHdrConversionCapability>* hdrConversionCapabilities) {
const auto status =
mAidlComposerClient->getHdrConversionCapabilities(hdrConversionCapabilities);
if (!status.isOk()) {
hdrConversionCapabilities = {};
ALOGE("getHdrConversionCapabilities failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setHdrConversionStrategy(AidlHdrConversionStrategy hdrConversionStrategy,
Hdr* outPreferredHdrOutputType) {
const auto status = mAidlComposerClient->setHdrConversionStrategy(hdrConversionStrategy,
outPreferredHdrOutputType);
if (!status.isOk()) {
ALOGE("setHdrConversionStrategy failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setRefreshRateChangedCallbackDebugEnabled(Display displayId, bool enabled) {
const auto status =
mAidlComposerClient->setRefreshRateChangedCallbackDebugEnabled(translate<int64_t>(
displayId),
enabled);
if (!status.isOk()) {
ALOGE("setRefreshRateChangedCallbackDebugEnabled failed %s",
status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::notifyExpectedPresent(Display displayId, nsecs_t expectedPresentTime,
int32_t frameIntervalNs) {
const auto status =
mAidlComposerClient->notifyExpectedPresent(translate<int64_t>(displayId),
ClockMonotonicTimestamp{expectedPresentTime},
frameIntervalNs);
if (!status.isOk()) {
ALOGE("notifyExpectedPresent failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getClientTargetProperty(
Display display, ClientTargetPropertyWithBrightness* outClientTargetProperty) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto reader = getReader(display)) {
*outClientTargetProperty =
reader->get().takeClientTargetProperty(translate<int64_t>(display));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerBrightness(Display display, Layer layer, float brightness) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerBrightness(translate<int64_t>(display), translate<int64_t>(layer),
brightness);
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::setLayerBlockingRegion(Display display, Layer layer,
const std::vector<IComposerClient::Rect>& blocking) {
Error error = Error::NONE;
mMutex.lock_shared();
if (auto writer = getWriter(display)) {
writer->get().setLayerBlockingRegion(translate<int64_t>(display), translate<int64_t>(layer),
translate<AidlRect>(blocking));
} else {
error = Error::BAD_DISPLAY;
}
mMutex.unlock_shared();
return error;
}
Error AidlComposer::getDisplayDecorationSupport(Display display,
std::optional<DisplayDecorationSupport>* support) {
const auto status =
mAidlComposerClient->getDisplayDecorationSupport(translate<int64_t>(display), support);
if (!status.isOk()) {
ALOGE("getDisplayDecorationSupport failed %s", status.getDescription().c_str());
support->reset();
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::setIdleTimerEnabled(Display displayId, std::chrono::milliseconds timeout) {
const auto status =
mAidlComposerClient->setIdleTimerEnabled(translate<int64_t>(displayId),
translate<int32_t>(timeout.count()));
if (!status.isOk()) {
ALOGE("setIdleTimerEnabled failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
Error AidlComposer::getPhysicalDisplayOrientation(Display displayId,
AidlTransform* outDisplayOrientation) {
const auto status =
mAidlComposerClient->getDisplayPhysicalOrientation(translate<int64_t>(displayId),
outDisplayOrientation);
if (!status.isOk()) {
ALOGE("getPhysicalDisplayOrientation failed %s", status.getDescription().c_str());
return static_cast<Error>(status.getServiceSpecificError());
}
return Error::NONE;
}
ftl::Optional<std::reference_wrapper<ComposerClientWriter>> AidlComposer::getWriter(Display display)
REQUIRES_SHARED(mMutex) {
return mWriters.get(display);
}
ftl::Optional<std::reference_wrapper<ComposerClientReader>> AidlComposer::getReader(Display display)
REQUIRES_SHARED(mMutex) {
if (mSingleReader) {
display = translate<Display>(kSingleReaderKey);
}
return mReaders.get(display);
}
void AidlComposer::removeDisplay(Display display) {
mMutex.lock();
bool wasErased = mWriters.erase(display);
ALOGW_IF(!wasErased,
"Attempting to remove writer for display %" PRId64 " which is not connected",
translate<int64_t>(display));
if (!mSingleReader) {
removeReader(display);
}
mMutex.unlock();
}
void AidlComposer::onHotplugDisconnect(Display display) {
removeDisplay(display);
}
bool AidlComposer::hasMultiThreadedPresentSupport(Display display) {
if (!FlagManager::getInstance().multithreaded_present()) return false;
const auto displayId = translate<int64_t>(display);
std::vector<AidlDisplayCapability> capabilities;
const auto status = mAidlComposerClient->getDisplayCapabilities(displayId, &capabilities);
if (!status.isOk()) {
ALOGE("getDisplayCapabilities failed %s", status.getDescription().c_str());
return false;
}
return std::find(capabilities.begin(), capabilities.end(),
AidlDisplayCapability::MULTI_THREADED_PRESENT) != capabilities.end();
}
void AidlComposer::addReader(Display display) {
const auto displayId = translate<int64_t>(display);
std::optional<int64_t> displayOpt;
if (displayId != kSingleReaderKey) {
displayOpt.emplace(displayId);
}
auto [it, added] = mReaders.try_emplace(display, std::move(displayOpt));
ALOGW_IF(!added, "Attempting to add writer for display %" PRId64 " which is already connected",
displayId);
}
void AidlComposer::removeReader(Display display) {
bool wasErased = mReaders.erase(display);
ALOGW_IF(!wasErased,
"Attempting to remove reader for display %" PRId64 " which is not connected",
translate<int64_t>(display));
}
void AidlComposer::addDisplay(Display display) {
const auto displayId = translate<int64_t>(display);
mMutex.lock();
auto [it, added] = mWriters.try_emplace(display, displayId);
ALOGW_IF(!added, "Attempting to add writer for display %" PRId64 " which is already connected",
displayId);
if (mSingleReader) {
if (hasMultiThreadedPresentSupport(display)) {
mSingleReader = false;
removeReader(translate<Display>(kSingleReaderKey));
// Note that this includes the new display.
for (const auto& [existingDisplay, _] : mWriters) {
addReader(existingDisplay);
}
}
} else {
addReader(display);
}
mMutex.unlock();
}
void AidlComposer::onHotplugConnect(Display display) {
addDisplay(display);
}
} // namespace Hwc2
} // namespace android