/* * Copyright (C) 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. */ #define LOG_TAG "drmhwc" #include "DrmHwcTwo.h" #include #include "backend/Backend.h" #include "utils/log.h" namespace android { HWC2::Error DrmHwcTwo::RegisterCallback(int32_t descriptor, hwc2_callback_data_t data, hwc2_function_pointer_t function) { switch (static_cast(descriptor)) { case HWC2::Callback::Hotplug: { hotplug_callback_ = std::make_pair(HWC2_PFN_HOTPLUG(function), data); if (function != nullptr) { GetResMan().Init(); } else { GetResMan().DeInit(); /* Headless display may still be here. Remove it! */ if (Displays().count(kPrimaryDisplay) != 0) { Displays()[kPrimaryDisplay]->Deinit(); Displays().erase(kPrimaryDisplay); } } break; } case HWC2::Callback::Refresh: { refresh_callback_ = std::make_pair(HWC2_PFN_REFRESH(function), data); break; } case HWC2::Callback::Vsync: { vsync_callback_ = std::make_pair(HWC2_PFN_VSYNC(function), data); break; } #if __ANDROID_API__ > 29 case HWC2::Callback::Vsync_2_4: { vsync_2_4_callback_ = std::make_pair(HWC2_PFN_VSYNC_2_4(function), data); break; } case HWC2::Callback::VsyncPeriodTimingChanged: { period_timing_changed_callback_ = std:: make_pair(HWC2_PFN_VSYNC_PERIOD_TIMING_CHANGED(function), data); break; } #endif default: break; } return HWC2::Error::None; } void DrmHwcTwo::SendHotplugEventToClient(hwc2_display_t displayid, DisplayStatus display_status) { auto hc = hotplug_callback_; if (hc.first != nullptr && hc.second != nullptr) { /* For some reason HWC Service will call HWC2 API in hotplug callback * handler. This is the reason we're using recursive mutex. */ hc.first(hc.second, displayid, display_status ? HWC2_CONNECTION_CONNECTED : HWC2_CONNECTION_DISCONNECTED); } } void DrmHwcTwo::SendVsyncEventToClient( hwc2_display_t displayid, int64_t timestamp, [[maybe_unused]] uint32_t vsync_period) const { /* vsync callback */ #if __ANDROID_API__ > 29 if (vsync_2_4_callback_.first != nullptr && vsync_2_4_callback_.second != nullptr) { vsync_2_4_callback_.first(vsync_2_4_callback_.second, displayid, timestamp, vsync_period); } else #endif if (vsync_callback_.first != nullptr && vsync_callback_.second != nullptr) { vsync_callback_.first(vsync_callback_.second, displayid, timestamp); } } void DrmHwcTwo::SendVsyncPeriodTimingChangedEventToClient( [[maybe_unused]] hwc2_display_t displayid, [[maybe_unused]] int64_t timestamp) const { #if __ANDROID_API__ > 29 hwc_vsync_period_change_timeline_t timeline = { .newVsyncAppliedTimeNanos = timestamp, .refreshRequired = false, .refreshTimeNanos = 0, }; if (period_timing_changed_callback_.first != nullptr && period_timing_changed_callback_.second != nullptr) { period_timing_changed_callback_ .first(period_timing_changed_callback_.second, displayid, &timeline); } #endif } void DrmHwcTwo::SendRefreshEventToClient(hwc2_display_t displayid) { if (refresh_callback_.first != nullptr && refresh_callback_.second != nullptr) { refresh_callback_.first(refresh_callback_.second, displayid); } } } // namespace android