blob: a50a214a9de525931de77360e9d35a74bc7fbe8c [file] [log] [blame]
Sean Paul98e73c82015-06-24 14:38:49 -07001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Roman Stratiienko4e994052022-02-09 17:40:35 +020017#ifndef ANDROID_DRM_ATOMIC_STATE_MANAGER_H_
18#define ANDROID_DRM_ATOMIC_STATE_MANAGER_H_
Sean Paul98e73c82015-06-24 14:38:49 -070019
Sean Paul98e73c82015-06-24 14:38:49 -070020#include <pthread.h>
Roman Stratiienkoaa3cd542020-08-29 11:26:16 +030021
Roman Stratiienkod21071f2021-03-09 21:56:50 +020022#include <functional>
Zach Reizner92f8e632015-10-12 17:47:13 -070023#include <memory>
Roman Stratiienkof81d2c82022-01-11 19:47:24 +020024#include <optional>
Sean Paul98e73c82015-06-24 14:38:49 -070025#include <sstream>
Zach Reizner92f8e632015-10-12 17:47:13 -070026#include <tuple>
Sean Paul98e73c82015-06-24 14:38:49 -070027
Roman Stratiienko4e994052022-02-09 17:40:35 +020028#include "compositor/DrmKmsPlan.h"
Roman Stratiienko9362cef2022-02-02 09:53:50 +020029#include "drm/DrmPlane.h"
Roman Stratiienko13cc3662020-08-29 21:35:39 +030030#include "drm/ResourceManager.h"
31#include "drm/VSyncWorker.h"
Roman Stratiienkoaa3cd542020-08-29 11:26:16 +030032#include "drmhwcomposer.h"
Sean Paul98e73c82015-06-24 14:38:49 -070033
34namespace android {
35
Roman Stratiienkodccc6fb2021-10-23 17:35:44 +030036struct AtomicCommitArgs {
37 /* inputs. All fields are optional, but at least one has to be specified */
38 bool test_only = false;
39 std::optional<DrmMode> display_mode;
40 std::optional<bool> active;
Roman Stratiienko9362cef2022-02-02 09:53:50 +020041 std::shared_ptr<DrmKmsPlan> composition;
Roman Stratiienkodccc6fb2021-10-23 17:35:44 +030042
43 /* out */
44 UniqueFd out_fence;
45
46 /* helpers */
47 auto HasInputs() -> bool {
Roman Stratiienkoef5348b2022-02-09 17:19:56 +020048 return display_mode || active || composition;
Roman Stratiienkodccc6fb2021-10-23 17:35:44 +030049 }
50};
51
Roman Stratiienko59bb4812022-04-09 16:13:09 +030052class PresentTrackerThread {
Sean Paul98e73c82015-06-24 14:38:49 -070053 public:
Roman Stratiienko59bb4812022-04-09 16:13:09 +030054 explicit PresentTrackerThread(DrmAtomicStateManager *st_man);
55
56 ~PresentTrackerThread();
57
58 void Stop() {
59 /* Exit thread by signalling that object is no longer valid */
60 st_man_ = nullptr;
61 Notify();
62 pt_.detach();
63 }
64
65 void Notify() {
66 cv_.notify_all();
67 }
68
69 private:
70 DrmAtomicStateManager *st_man_{};
71
72 void PresentTrackerThreadFn();
73
74 std::condition_variable cv_;
75 std::thread pt_;
76 std::mutex *mutex_;
77};
78
79class DrmAtomicStateManager {
80 friend class PresentTrackerThread;
81
82 public:
83 explicit DrmAtomicStateManager(DrmDisplayPipeline *pipe)
84 : pipe_(pipe),
85 ptt_(std::make_unique<PresentTrackerThread>(this).release()){};
86
Roman Stratiienko4e994052022-02-09 17:40:35 +020087 DrmAtomicStateManager(const DrmAtomicStateManager &) = delete;
Roman Stratiienko59bb4812022-04-09 16:13:09 +030088 ~DrmAtomicStateManager() {
89 ptt_->Stop();
90 }
Roman Kovalivskyi8fae1562020-01-30 20:20:47 +020091
Roman Stratiienkodccc6fb2021-10-23 17:35:44 +030092 auto ExecuteAtomicCommit(AtomicCommitArgs &args) -> int;
Roman Stratiienkod37b3082022-01-13 16:37:27 +020093 auto ActivateDisplayUsingDPMS() -> int;
94
Roman Stratiienkoe78235c2021-12-23 17:36:12 +020095 private:
Roman Stratiienkodccc6fb2021-10-23 17:35:44 +030096 auto CommitFrame(AtomicCommitArgs &args) -> int;
Sean Paul98e73c82015-06-24 14:38:49 -070097
Roman Stratiienko32685ba2021-12-15 13:46:05 +020098 struct KmsState {
99 /* Required to cleanup unused planes */
Roman Stratiienko9362cef2022-02-02 09:53:50 +0200100 std::vector<std::shared_ptr<BindingOwner<DrmPlane>>> used_planes;
Roman Stratiienko32685ba2021-12-15 13:46:05 +0200101 /* We have to hold a reference to framebuffer while displaying it ,
102 * otherwise picture will blink */
103 std::vector<std::shared_ptr<DrmFbIdHandle>> used_framebuffers;
104
Roman Stratiienkodccc6fb2021-10-23 17:35:44 +0300105 DrmModeUserPropertyBlobUnique mode_blob;
Roman Stratiienko32685ba2021-12-15 13:46:05 +0200106
Roman Stratiienko59bb4812022-04-09 16:13:09 +0300107 int release_fence_pt_index{};
108
Roman Stratiienko32685ba2021-12-15 13:46:05 +0200109 /* To avoid setting the inactive state twice, which will fail the commit */
110 bool crtc_active_state{};
Roman Stratiienkof815d382021-12-30 19:23:14 +0200111 } active_frame_state_;
Roman Stratiienko32685ba2021-12-15 13:46:05 +0200112
113 auto NewFrameState() -> KmsState {
Roman Stratiienko59bb4812022-04-09 16:13:09 +0300114 auto *prev_frame_state = &active_frame_state_;
Roman Stratiienko32685ba2021-12-15 13:46:05 +0200115 return (KmsState){
Roman Stratiienko59bb4812022-04-09 16:13:09 +0300116 .used_planes = prev_frame_state->used_planes,
117 .crtc_active_state = prev_frame_state->crtc_active_state,
Roman Stratiienko32685ba2021-12-15 13:46:05 +0200118 };
119 }
Sean Paul98e73c82015-06-24 14:38:49 -0700120
Roman Stratiienko19c162f2022-02-01 09:35:08 +0200121 DrmDisplayPipeline *const pipe_;
Roman Stratiienko59bb4812022-04-09 16:13:09 +0300122
123 void CleanupPriorFrameResources();
124
125 /* Present (swap) tracking */
126 PresentTrackerThread *ptt_;
127 KmsState staged_frame_state_;
128 UniqueFd last_present_fence_;
129 int frames_staged_{};
130 int frames_tracked_{};
Sean Paul98e73c82015-06-24 14:38:49 -0700131};
Roman Stratiienko59bb4812022-04-09 16:13:09 +0300132
Sean Paulf72cccd2018-08-27 13:59:08 -0400133} // namespace android
Sean Paul98e73c82015-06-24 14:38:49 -0700134
135#endif // ANDROID_DRM_DISPLAY_COMPOSITOR_H_