| /* |
| * Copyright (c) 2016-2017, 2019-2020 The Linux Foundation. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are |
| * met: |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following |
| * disclaimer in the documentation and/or other materials provided |
| * with the distribution. |
| * * Neither the name of The Linux Foundation. nor the names of its |
| * contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED |
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT |
| * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR |
| * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
| * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE |
| * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN |
| * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| #include <utils/constants.h> |
| #include <utils/debug.h> |
| #include <utils/locker.h> |
| #include "hwc_callbacks.h" |
| |
| #define __CLASS__ "HWCCallbacks" |
| |
| namespace sdm { |
| |
| HWC2::Error HWCCallbacks::Hotplug(hwc2_display_t display, HWC2::Connection state) { |
| SCOPE_LOCK(hotplug_lock_); |
| DTRACE_SCOPED(); |
| |
| // If client has not registered hotplug, wait for it finitely: |
| // 1. spurious wake up (!hotplug_), wait again |
| // 2. error = ETIMEDOUT, return with warning 1 |
| // 3. error != ETIMEDOUT, return with warning 2 |
| // 4. error == NONE and no spurious wake up, then !hotplug_ is false, exit loop |
| while (!hotplug_) { |
| DLOGW("Attempting to send client a hotplug too early Display = %" PRIu64 " state = %d", display, |
| state); |
| int ret = hotplug_lock_.WaitFinite(5000); |
| if (ret == ETIMEDOUT) { |
| DLOGW("Client didn't connect on time, dropping hotplug!"); |
| return HWC2::Error::None; |
| } else if (ret != 0) { |
| DLOGW("Failed client connection wait. Error %s, dropping hotplug!", strerror(ret)); |
| return HWC2::Error::None; |
| } |
| } |
| hotplug_(hotplug_data_, display, INT32(state)); |
| return HWC2::Error::None; |
| } |
| |
| HWC2::Error HWCCallbacks::Refresh(hwc2_display_t display) { |
| SCOPE_LOCK(refresh_lock_); |
| // Do not lock, will cause hotplug deadlock |
| DTRACE_SCOPED(); |
| // If client has not registered refresh, drop it |
| if (!refresh_) { |
| return HWC2::Error::NoResources; |
| } |
| refresh_(refresh_data_, display); |
| pending_refresh_.set(UINT32(display)); |
| return HWC2::Error::None; |
| } |
| |
| HWC2::Error HWCCallbacks::Vsync(hwc2_display_t display, int64_t timestamp) { |
| SCOPE_LOCK(vsync_lock_); |
| // Do not lock, may cause hotplug deadlock |
| DTRACE_SCOPED(); |
| // If client has not registered vsync, drop it |
| if (!vsync_) { |
| return HWC2::Error::NoResources; |
| } |
| vsync_(vsync_data_, display, timestamp); |
| return HWC2::Error::None; |
| } |
| |
| HWC2::Error HWCCallbacks::Vsync_2_4(hwc2_display_t display, int64_t timestamp, uint32_t period) { |
| SCOPE_LOCK(vsync_2_4_lock_); |
| DTRACE_SCOPED(); |
| if (!vsync_2_4_) { |
| return HWC2::Error::NoResources; |
| } |
| |
| vsync_2_4_(vsync_2_4_data_, display, timestamp, period); |
| return HWC2::Error::None; |
| } |
| |
| HWC2::Error HWCCallbacks::VsyncPeriodTimingChanged( |
| hwc2_display_t display, hwc_vsync_period_change_timeline_t *updated_timeline) { |
| SCOPE_LOCK(vsync_period_timing_changed_lock_); |
| DTRACE_SCOPED(); |
| if (!vsync_period_timing_changed_) { |
| return HWC2::Error::NoResources; |
| } |
| |
| vsync_period_timing_changed_(vsync_period_timing_changed_data_, display, updated_timeline); |
| return HWC2::Error::None; |
| } |
| |
| HWC2::Error HWCCallbacks::SeamlessPossible(hwc2_display_t display) { |
| SCOPE_LOCK(seamless_possible_lock_); |
| DTRACE_SCOPED(); |
| if (!seamless_possible_) { |
| return HWC2::Error::NoResources; |
| } |
| |
| seamless_possible_(seamless_possible_data_, display); |
| return HWC2::Error::None; |
| } |
| |
| HWC2::Error HWCCallbacks::Register(HWC2::Callback descriptor, hwc2_callback_data_t callback_data, |
| hwc2_function_pointer_t pointer) { |
| switch (descriptor) { |
| case HWC2::Callback::Hotplug: { |
| SCOPE_LOCK(hotplug_lock_); |
| hotplug_data_ = callback_data; |
| hotplug_ = reinterpret_cast<HWC2_PFN_HOTPLUG>(pointer); |
| client_connected_ = true; |
| hotplug_lock_.Broadcast(); |
| } break; |
| case HWC2::Callback::Refresh: { |
| SCOPE_LOCK(refresh_lock_); |
| refresh_data_ = callback_data; |
| refresh_ = reinterpret_cast<HWC2_PFN_REFRESH>(pointer); |
| } break; |
| case HWC2::Callback::Vsync: { |
| SCOPE_LOCK(vsync_lock_); |
| vsync_data_ = callback_data; |
| vsync_ = reinterpret_cast<HWC2_PFN_VSYNC>(pointer); |
| } break; |
| case HWC2::Callback::Vsync_2_4: { |
| SCOPE_LOCK(vsync_2_4_lock_); |
| vsync_2_4_data_ = callback_data; |
| vsync_2_4_ = reinterpret_cast<HWC2_PFN_VSYNC_2_4>(pointer); |
| } break; |
| case HWC2::Callback::VsyncPeriodTimingChanged: { |
| SCOPE_LOCK(vsync_period_timing_changed_lock_); |
| vsync_period_timing_changed_data_ = callback_data; |
| vsync_period_timing_changed_ = |
| reinterpret_cast<HWC2_PFN_VSYNC_PERIOD_TIMING_CHANGED>(pointer); |
| } break; |
| case HWC2::Callback::SeamlessPossible: { |
| SCOPE_LOCK(seamless_possible_lock_); |
| seamless_possible_data_ = callback_data; |
| seamless_possible_ = reinterpret_cast<HWC2_PFN_SEAMLESS_POSSIBLE>(pointer); |
| } break; |
| default: |
| return HWC2::Error::BadParameter; |
| } |
| |
| return HWC2::Error::None; |
| } |
| |
| } // namespace sdm |