blob: fb14bc9f9fdbaee3a6c85bc5ba2375920ea31efb [file] [log] [blame] [edit]
/*
* Copyright (C) 2024 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 "drmhwc"
#include "DrmHwcThree.h"
#include <cinttypes>
#include "Utils.h"
#include "aidl/android/hardware/graphics/common/Dataspace.h"
#if __ANDROID_API__ >= 35
#include "aidl/android/hardware/graphics/common/DisplayHotplugEvent.h"
#endif
namespace aidl::android::hardware::graphics::composer3::impl {
using ::android::HwcDisplay;
DrmHwcThree::~DrmHwcThree() {
/* Display deinit routine is handled by resource manager */
GetResMan().DeInit();
}
void DrmHwcThree::Init(std::shared_ptr<IComposerCallback> callback) {
composer_callback_ = std::move(callback);
GetResMan().Init();
}
void DrmHwcThree::SendVsyncPeriodTimingChangedEventToClient(
uint64_t display_id, int64_t timestamp) const {
VsyncPeriodChangeTimeline timeline;
timeline.newVsyncAppliedTimeNanos = timestamp;
timeline.refreshRequired = false;
timeline.refreshTimeNanos = 0;
composer_callback_->onVsyncPeriodTimingChanged(static_cast<int64_t>(
display_id),
timeline);
}
void DrmHwcThree::SendRefreshEventToClient(uint64_t display_id) {
composer_resources_->SetDisplayMustValidateState(display_id, true);
composer_callback_->onRefresh(static_cast<int64_t>(display_id));
}
void DrmHwcThree::SendVsyncEventToClient(uint64_t display_id, int64_t timestamp,
uint32_t vsync_period) const {
composer_callback_->onVsync(static_cast<int64_t>(display_id), timestamp,
static_cast<int32_t>(vsync_period));
}
#if __ANDROID_API__ >= 35
void DrmHwcThree::SendHotplugEventToClient(
hwc2_display_t display_id, DrmHwc::DisplayStatus display_status) {
common::DisplayHotplugEvent event = common::DisplayHotplugEvent::DISCONNECTED;
switch (display_status) {
case DrmHwc::kDisconnected:
event = common::DisplayHotplugEvent::DISCONNECTED;
HandleDisplayHotplugEvent(static_cast<uint64_t>(display_id), false);
break;
case DrmHwc::kConnected:
event = common::DisplayHotplugEvent::CONNECTED;
HandleDisplayHotplugEvent(static_cast<uint64_t>(display_id), true);
break;
case DrmHwc::kLinkTrainingFailed:
event = common::DisplayHotplugEvent::ERROR_INCOMPATIBLE_CABLE;
break;
}
composer_callback_->onHotplugEvent(static_cast<int64_t>(display_id), event);
}
#else
void DrmHwcThree::SendHotplugEventToClient(
hwc2_display_t display_id, DrmHwc::DisplayStatus display_status) {
bool connected = display_status != DrmHwc::kDisconnected;
HandleDisplayHotplugEvent(static_cast<uint64_t>(display_id), connected);
composer_callback_->onHotplug(static_cast<int64_t>(display_id), connected);
}
#endif
void DrmHwcThree::CleanDisplayResources(uint64_t display_id) {
DEBUG_FUNC();
HwcDisplay* display = GetDisplay(display_id);
if (display == nullptr) {
return;
}
display->SetPowerMode(static_cast<int32_t>(PowerMode::OFF));
size_t cache_size = 0;
auto err = composer_resources_->GetDisplayClientTargetCacheSize(display_id,
&cache_size);
if (err != hwc3::Error::kNone) {
ALOGE("%s: Could not clear target buffer cache for display: %" PRIu64,
__func__, display_id);
return;
}
for (size_t slot = 0; slot < cache_size; slot++) {
buffer_handle_t buffer_handle = nullptr;
auto buf_releaser = ComposerResources::CreateResourceReleaser(true);
Buffer buf{};
buf.slot = static_cast<int32_t>(slot);
err = composer_resources_->GetDisplayClientTarget(display_id, buf,
&buffer_handle,
buf_releaser.get());
if (err != hwc3::Error::kNone) {
continue;
}
err = Hwc2toHwc3Error(
display->SetClientTarget(buffer_handle, -1,
static_cast<int32_t>(
common::Dataspace::UNKNOWN),
{}));
if (err != hwc3::Error::kNone) {
ALOGE(
"%s: Could not clear slot %zu of the target buffer cache for "
"display %" PRIu64,
__func__, slot, display_id);
}
}
}
void DrmHwcThree::HandleDisplayHotplugEvent(uint64_t display_id,
bool connected) {
DEBUG_FUNC();
if (!connected) {
composer_resources_->RemoveDisplay(display_id);
Displays().erase(display_id);
return;
}
if (composer_resources_->HasDisplay(display_id)) {
/* Cleanup existing display resources */
CleanDisplayResources(display_id);
composer_resources_->RemoveDisplay(display_id);
Displays().erase(display_id);
}
composer_resources_->AddPhysicalDisplay(display_id);
}
} // namespace aidl::android::hardware::graphics::composer3::impl