blob: 63fd1e8ebea6ee1da444b098f2ce68be39723dac [file] [log] [blame]
/*
* Copyright 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#undef LOG_TAG
#define LOG_TAG "HwcComposer"
#include <inttypes.h>
#include <log/log.h>
#include "ComposerHal.h"
namespace android {
using hardware::Return;
using hardware::hidl_vec;
using hardware::hidl_handle;
namespace Hwc2 {
namespace {
class BufferHandle {
public:
BufferHandle(const native_handle_t* buffer)
{
// nullptr is not a valid handle to HIDL
mHandle = (buffer) ? buffer : native_handle_init(mStorage, 0, 0);
}
operator const hidl_handle&() const
{
return mHandle;
}
private:
NATIVE_HANDLE_DECLARE_STORAGE(mStorage, 0, 0);
hidl_handle mHandle;
};
class FenceHandle
{
public:
FenceHandle(int fd, bool owned)
: mOwned(owned)
{
native_handle_t* handle;
if (fd >= 0) {
handle = native_handle_init(mStorage, 1, 0);
handle->data[0] = fd;
} else {
// nullptr is not a valid handle to HIDL
handle = native_handle_init(mStorage, 0, 0);
}
mHandle = handle;
}
~FenceHandle()
{
if (mOwned) {
native_handle_close(mHandle);
}
}
operator const hidl_handle&() const
{
return mHandle;
}
private:
bool mOwned;
NATIVE_HANDLE_DECLARE_STORAGE(mStorage, 1, 0);
hidl_handle mHandle;
};
// assume NO_RESOURCES when Status::isOk returns false
constexpr Error kDefaultError = Error::NO_RESOURCES;
template<typename T, typename U>
T unwrapRet(Return<T>& ret, const U& default_val)
{
return (ret.getStatus().isOk()) ? static_cast<T>(ret) :
static_cast<T>(default_val);
}
Error unwrapRet(Return<Error>& ret)
{
return unwrapRet(ret, kDefaultError);
}
template<typename T>
void assignFromHidlVec(std::vector<T>& vec, const hidl_vec<T>& data)
{
vec.clear();
vec.insert(vec.begin(), &data[0], &data[data.size()]);
}
} // anonymous namespace
Composer::Composer()
: mWriter(kWriterInitialSize)
{
mComposer = IComposer::getService("hwcomposer");
if (mComposer == nullptr) {
LOG_ALWAYS_FATAL("failed to get hwcomposer service");
}
mComposer->createClient(
[&](const auto& tmpError, const auto& tmpClient)
{
if (tmpError == Error::NONE) {
mClient = tmpClient;
}
});
if (mClient == nullptr) {
LOG_ALWAYS_FATAL("failed to create composer client");
}
}
std::vector<IComposer::Capability> Composer::getCapabilities()
{
std::vector<IComposer::Capability> capabilities;
mComposer->getCapabilities(
[&](const auto& tmpCapabilities) {
assignFromHidlVec(capabilities, tmpCapabilities);
});
return capabilities;
}
std::string Composer::dumpDebugInfo()
{
std::string info;
mComposer->dumpDebugInfo([&](const auto& tmpInfo) {
info = tmpInfo.c_str();
});
return info;
}
void Composer::registerCallback(const sp<IComposerCallback>& callback)
{
auto ret = mClient->registerCallback(callback);
if (!ret.getStatus().isOk()) {
ALOGE("failed to register IComposerCallback");
}
}
uint32_t Composer::getMaxVirtualDisplayCount()
{
auto ret = mClient->getMaxVirtualDisplayCount();
return unwrapRet(ret, 0);
}
Error Composer::createVirtualDisplay(uint32_t width, uint32_t height,
PixelFormat& format, Display& display)
{
const uint32_t bufferSlotCount = 1;
Error error = kDefaultError;
mClient->createVirtualDisplay(width, height, format, bufferSlotCount,
[&](const auto& tmpError, const auto& tmpDisplay,
const auto& tmpFormat) {
error = tmpError;
if (error != Error::NONE) {
return;
}
display = tmpDisplay;
format = tmpFormat;
});
return error;
}
Error Composer::destroyVirtualDisplay(Display display)
{
auto ret = mClient->destroyVirtualDisplay(display);
return unwrapRet(ret);
}
Error Composer::acceptDisplayChanges(Display display)
{
mWriter.selectDisplay(display);
mWriter.acceptDisplayChanges();
return Error::NONE;
}
Error Composer::createLayer(Display display, Layer& layer)
{
const uint32_t bufferSlotCount = 1;
Error error = kDefaultError;
mClient->createLayer(display, bufferSlotCount,
[&](const auto& tmpError, const auto& tmpLayer) {
error = tmpError;
if (error != Error::NONE) {
return;
}
layer = tmpLayer;
});
return error;
}
Error Composer::destroyLayer(Display display, Layer layer)
{
auto ret = mClient->destroyLayer(display, layer);
return unwrapRet(ret);
}
Error Composer::getActiveConfig(Display display, Config& config)
{
Error error = kDefaultError;
mClient->getActiveConfig(display,
[&](const auto& tmpError, const auto& tmpConfig) {
error = tmpError;
if (error != Error::NONE) {
return;
}
config = tmpConfig;
});
return error;
}
Error Composer::getChangedCompositionTypes(Display display,
std::vector<Layer>& layers,
std::vector<IComposerClient::Composition>& types)
{
mReader.takeChangedCompositionTypes(display, layers, types);
return Error::NONE;
}
Error Composer::getColorModes(Display display,
std::vector<ColorMode>& modes)
{
Error error = kDefaultError;
mClient->getColorModes(display,
[&](const auto& tmpError, const auto& tmpModes) {
error = tmpError;
if (error != Error::NONE) {
return;
}
assignFromHidlVec(modes, tmpModes);
});
return error;
}
Error Composer::getDisplayAttribute(Display display, Config config,
IComposerClient::Attribute attribute, int32_t& value)
{
Error error = kDefaultError;
mClient->getDisplayAttribute(display, config, attribute,
[&](const auto& tmpError, const auto& tmpValue) {
error = tmpError;
if (error != Error::NONE) {
return;
}
value = tmpValue;
});
return error;
}
Error Composer::getDisplayConfigs(Display display,
std::vector<Config>& configs)
{
Error error = kDefaultError;
mClient->getDisplayConfigs(display,
[&](const auto& tmpError, const auto& tmpConfigs) {
error = tmpError;
if (error != Error::NONE) {
return;
}
assignFromHidlVec(configs, tmpConfigs);
});
return error;
}
Error Composer::getDisplayName(Display display, std::string& name)
{
Error error = kDefaultError;
mClient->getDisplayName(display,
[&](const auto& tmpError, const auto& tmpName) {
error = tmpError;
if (error != Error::NONE) {
return;
}
name = tmpName.c_str();
});
return error;
}
Error Composer::getDisplayRequests(Display display,
uint32_t& displayRequestMask, std::vector<Layer>& layers,
std::vector<uint32_t>& layerRequestMasks)
{
mReader.takeDisplayRequests(display, displayRequestMask,
layers, layerRequestMasks);
return Error::NONE;
}
Error Composer::getDisplayType(Display display, IComposerClient::DisplayType& type)
{
Error error = kDefaultError;
mClient->getDisplayType(display,
[&](const auto& tmpError, const auto& tmpType) {
error = tmpError;
if (error != Error::NONE) {
return;
}
type = tmpType;
});
return error;
}
Error Composer::getDozeSupport(Display display, bool& support)
{
Error error = kDefaultError;
mClient->getDozeSupport(display,
[&](const auto& tmpError, const auto& tmpSupport) {
error = tmpError;
if (error != Error::NONE) {
return;
}
support = tmpSupport;
});
return error;
}
Error Composer::getHdrCapabilities(Display display, std::vector<Hdr>& types,
float& maxLuminance, float& maxAverageLuminance,
float& minLuminance)
{
Error error = kDefaultError;
mClient->getHdrCapabilities(display,
[&](const auto& tmpError, const auto& tmpTypes,
const auto& tmpMaxLuminance,
const auto& tmpMaxAverageLuminance,
const auto& tmpMinLuminance) {
error = tmpError;
if (error != Error::NONE) {
return;
}
assignFromHidlVec(types, tmpTypes);
maxLuminance = tmpMaxLuminance;
maxAverageLuminance = tmpMaxAverageLuminance;
minLuminance = tmpMinLuminance;
});
return error;
}
Error Composer::getReleaseFences(Display display, std::vector<Layer>& layers,
std::vector<int>& releaseFences)
{
mReader.takeReleaseFences(display, layers, releaseFences);
return Error::NONE;
}
Error Composer::presentDisplay(Display display, int& presentFence)
{
mWriter.selectDisplay(display);
mWriter.presentDisplay();
Error error = execute();
if (error != Error::NONE) {
return error;
}
mReader.takePresentFence(display, presentFence);
return Error::NONE;
}
Error Composer::setActiveConfig(Display display, Config config)
{
auto ret = mClient->setActiveConfig(display, config);
return unwrapRet(ret);
}
Error Composer::setClientTarget(Display display, const native_handle_t* target,
int acquireFence, Dataspace dataspace,
const std::vector<IComposerClient::Rect>& damage)
{
mWriter.selectDisplay(display);
mWriter.setClientTarget(0, target, acquireFence, dataspace, damage);
return Error::NONE;
}
Error Composer::setColorMode(Display display, ColorMode mode)
{
auto ret = mClient->setColorMode(display, mode);
return unwrapRet(ret);
}
Error Composer::setColorTransform(Display display, const float* matrix,
ColorTransform hint)
{
mWriter.selectDisplay(display);
mWriter.setColorTransform(matrix, hint);
return Error::NONE;
}
Error Composer::setOutputBuffer(Display display, const native_handle_t* buffer,
int releaseFence)
{
mWriter.selectDisplay(display);
mWriter.setOutputBuffer(0, buffer, dup(releaseFence));
return Error::NONE;
}
Error Composer::setPowerMode(Display display, IComposerClient::PowerMode mode)
{
auto ret = mClient->setPowerMode(display, mode);
return unwrapRet(ret);
}
Error Composer::setVsyncEnabled(Display display, IComposerClient::Vsync enabled)
{
auto ret = mClient->setVsyncEnabled(display, enabled);
return unwrapRet(ret);
}
Error Composer::setClientTargetSlotCount(Display display)
{
const uint32_t bufferSlotCount = 1;
auto ret = mClient->setClientTargetSlotCount(display, bufferSlotCount);
return unwrapRet(ret);
}
Error Composer::validateDisplay(Display display, uint32_t& numTypes,
uint32_t& numRequests)
{
mWriter.selectDisplay(display);
mWriter.validateDisplay();
Error error = execute();
if (error != Error::NONE) {
return error;
}
mReader.hasChanges(display, numTypes, numRequests);
return Error::NONE;
}
Error Composer::setCursorPosition(Display display, Layer layer,
int32_t x, int32_t y)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerCursorPosition(x, y);
return Error::NONE;
}
Error Composer::setLayerBuffer(Display display, Layer layer,
const native_handle_t* buffer, int acquireFence)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerBuffer(0, buffer, acquireFence);
return Error::NONE;
}
Error Composer::setLayerSurfaceDamage(Display display, Layer layer,
const std::vector<IComposerClient::Rect>& damage)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerSurfaceDamage(damage);
return Error::NONE;
}
Error Composer::setLayerBlendMode(Display display, Layer layer,
IComposerClient::BlendMode mode)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerBlendMode(mode);
return Error::NONE;
}
Error Composer::setLayerColor(Display display, Layer layer,
const IComposerClient::Color& color)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerColor(color);
return Error::NONE;
}
Error Composer::setLayerCompositionType(Display display, Layer layer,
IComposerClient::Composition type)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerCompositionType(type);
return Error::NONE;
}
Error Composer::setLayerDataspace(Display display, Layer layer,
Dataspace dataspace)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerDataspace(dataspace);
return Error::NONE;
}
Error Composer::setLayerDisplayFrame(Display display, Layer layer,
const IComposerClient::Rect& frame)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerDisplayFrame(frame);
return Error::NONE;
}
Error Composer::setLayerPlaneAlpha(Display display, Layer layer,
float alpha)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerPlaneAlpha(alpha);
return Error::NONE;
}
Error Composer::setLayerSidebandStream(Display display, Layer layer,
const native_handle_t* stream)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerSidebandStream(stream);
return Error::NONE;
}
Error Composer::setLayerSourceCrop(Display display, Layer layer,
const IComposerClient::FRect& crop)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerSourceCrop(crop);
return Error::NONE;
}
Error Composer::setLayerTransform(Display display, Layer layer,
Transform transform)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerTransform(transform);
return Error::NONE;
}
Error Composer::setLayerVisibleRegion(Display display, Layer layer,
const std::vector<IComposerClient::Rect>& visible)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerVisibleRegion(visible);
return Error::NONE;
}
Error Composer::setLayerZOrder(Display display, Layer layer, uint32_t z)
{
mWriter.selectDisplay(display);
mWriter.selectLayer(layer);
mWriter.setLayerZOrder(z);
return Error::NONE;
}
Error Composer::execute()
{
// prepare input command queue
bool queueChanged = false;
uint32_t commandLength = 0;
hidl_vec<hidl_handle> commandHandles;
if (!mWriter.writeQueue(queueChanged, commandLength, commandHandles)) {
mWriter.reset();
return Error::NO_RESOURCES;
}
// set up new input command queue if necessary
if (queueChanged) {
auto ret = mClient->setInputCommandQueue(*mWriter.getMQDescriptor());
auto error = unwrapRet(ret);
if (error != Error::NONE) {
mWriter.reset();
return error;
}
}
Error error = kDefaultError;
mClient->executeCommands(commandLength, commandHandles,
[&](const auto& tmpError, const auto& tmpOutChanged,
const auto& tmpOutLength, const auto& tmpOutHandles)
{
error = tmpError;
// set up new output command queue if necessary
if (error == Error::NONE && tmpOutChanged) {
error = kDefaultError;
mClient->getOutputCommandQueue(
[&](const auto& tmpError,
const auto& tmpDescriptor)
{
error = tmpError;
if (error != Error::NONE) {
return;
}
mReader.setMQDescriptor(tmpDescriptor);
});
}
if (error != Error::NONE) {
return;
}
if (mReader.readQueue(tmpOutLength, tmpOutHandles)) {
error = mReader.parse();
mReader.reset();
} else {
error = Error::NO_RESOURCES;
}
});
if (error == Error::NONE) {
std::vector<CommandReader::CommandError> commandErrors =
mReader.takeErrors();
for (const auto& cmdErr : commandErrors) {
auto command = mWriter.getCommand(cmdErr.location);
if (command == IComposerClient::Command::VALIDATE_DISPLAY ||
command == IComposerClient::Command::PRESENT_DISPLAY) {
error = cmdErr.error;
} else {
ALOGW("command 0x%x generated error %d",
command, cmdErr.error);
}
}
}
mWriter.reset();
return error;
}
CommandReader::~CommandReader()
{
resetData();
}
Error CommandReader::parse()
{
resetData();
IComposerClient::Command command;
uint16_t length = 0;
while (!isEmpty()) {
if (!beginCommand(command, length)) {
break;
}
bool parsed = false;
switch (command) {
case IComposerClient::Command::SELECT_DISPLAY:
parsed = parseSelectDisplay(length);
break;
case IComposerClient::Command::SET_ERROR:
parsed = parseSetError(length);
break;
case IComposerClient::Command::SET_CHANGED_COMPOSITION_TYPES:
parsed = parseSetChangedCompositionTypes(length);
break;
case IComposerClient::Command::SET_DISPLAY_REQUESTS:
parsed = parseSetDisplayRequests(length);
break;
case IComposerClient::Command::SET_PRESENT_FENCE:
parsed = parseSetPresentFence(length);
break;
case IComposerClient::Command::SET_RELEASE_FENCES:
parsed = parseSetReleaseFences(length);
break;
default:
parsed = false;
break;
}
endCommand();
if (!parsed) {
ALOGE("failed to parse command 0x%x length %" PRIu16,
command, length);
break;
}
}
return isEmpty() ? Error::NONE : Error::NO_RESOURCES;
}
bool CommandReader::parseSelectDisplay(uint16_t length)
{
if (length != CommandWriter::kSelectDisplayLength) {
return false;
}
mCurrentReturnData = &mReturnData[read64()];
return true;
}
bool CommandReader::parseSetError(uint16_t length)
{
if (length != CommandWriter::kSetErrorLength) {
return false;
}
auto location = read();
auto error = static_cast<Error>(readSigned());
mErrors.emplace_back(CommandError{location, error});
return true;
}
bool CommandReader::parseSetChangedCompositionTypes(uint16_t length)
{
// (layer id, composition type) pairs
if (length % 3 != 0 || !mCurrentReturnData) {
return false;
}
uint32_t count = length / 3;
mCurrentReturnData->changedLayers.reserve(count);
mCurrentReturnData->compositionTypes.reserve(count);
while (count > 0) {
auto layer = read64();
auto type = static_cast<IComposerClient::Composition>(readSigned());
mCurrentReturnData->changedLayers.push_back(layer);
mCurrentReturnData->compositionTypes.push_back(type);
count--;
}
return true;
}
bool CommandReader::parseSetDisplayRequests(uint16_t length)
{
// display requests followed by (layer id, layer requests) pairs
if (length % 3 != 1 || !mCurrentReturnData) {
return false;
}
mCurrentReturnData->displayRequests = read();
uint32_t count = (length - 1) / 3;
mCurrentReturnData->requestedLayers.reserve(count);
mCurrentReturnData->requestMasks.reserve(count);
while (count > 0) {
auto layer = read64();
auto layerRequestMask = read();
mCurrentReturnData->requestedLayers.push_back(layer);
mCurrentReturnData->requestMasks.push_back(layerRequestMask);
count--;
}
return true;
}
bool CommandReader::parseSetPresentFence(uint16_t length)
{
if (length != CommandWriter::kSetPresentFenceLength ||
!mCurrentReturnData) {
return false;
}
if (mCurrentReturnData->presentFence >= 0) {
close(mCurrentReturnData->presentFence);
}
mCurrentReturnData->presentFence = readFence();
return true;
}
bool CommandReader::parseSetReleaseFences(uint16_t length)
{
// (layer id, release fence index) pairs
if (length % 3 != 0 || !mCurrentReturnData) {
return false;
}
uint32_t count = length / 3;
mCurrentReturnData->releasedLayers.reserve(count);
mCurrentReturnData->releaseFences.reserve(count);
while (count > 0) {
auto layer = read64();
auto fence = readFence();
mCurrentReturnData->releasedLayers.push_back(layer);
mCurrentReturnData->releaseFences.push_back(fence);
count--;
}
return true;
}
void CommandReader::resetData()
{
mErrors.clear();
for (auto& data : mReturnData) {
if (data.second.presentFence >= 0) {
close(data.second.presentFence);
}
for (auto fence : data.second.releaseFences) {
if (fence >= 0) {
close(fence);
}
}
}
mReturnData.clear();
mCurrentReturnData = nullptr;
}
std::vector<CommandReader::CommandError> CommandReader::takeErrors()
{
return std::move(mErrors);
}
bool CommandReader::hasChanges(Display display,
uint32_t& numChangedCompositionTypes,
uint32_t& numLayerRequestMasks) const
{
auto found = mReturnData.find(display);
if (found == mReturnData.end()) {
numChangedCompositionTypes = 0;
numLayerRequestMasks = 0;
return false;
}
const ReturnData& data = found->second;
numChangedCompositionTypes = data.compositionTypes.size();
numLayerRequestMasks = data.requestMasks.size();
return !(data.compositionTypes.empty() && data.requestMasks.empty());
}
void CommandReader::takeChangedCompositionTypes(Display display,
std::vector<Layer>& layers,
std::vector<IComposerClient::Composition>& types)
{
auto found = mReturnData.find(display);
if (found == mReturnData.end()) {
layers.clear();
types.clear();
return;
}
ReturnData& data = found->second;
layers = std::move(data.changedLayers);
types = std::move(data.compositionTypes);
}
void CommandReader::takeDisplayRequests(Display display,
uint32_t& displayRequestMask, std::vector<Layer>& layers,
std::vector<uint32_t>& layerRequestMasks)
{
auto found = mReturnData.find(display);
if (found == mReturnData.end()) {
displayRequestMask = 0;
layers.clear();
layerRequestMasks.clear();
return;
}
ReturnData& data = found->second;
displayRequestMask = data.displayRequests;
layers = std::move(data.requestedLayers);
layerRequestMasks = std::move(data.requestMasks);
}
void CommandReader::takeReleaseFences(Display display,
std::vector<Layer>& layers, std::vector<int>& releaseFences)
{
auto found = mReturnData.find(display);
if (found == mReturnData.end()) {
layers.clear();
releaseFences.clear();
return;
}
ReturnData& data = found->second;
layers = std::move(data.releasedLayers);
releaseFences = std::move(data.releaseFences);
}
void CommandReader::takePresentFence(Display display, int& presentFence)
{
auto found = mReturnData.find(display);
if (found == mReturnData.end()) {
presentFence = -1;
return;
}
ReturnData& data = found->second;
presentFence = data.presentFence;
data.presentFence = -1;
}
} // namespace Hwc2
} // namespace android