blob: e756fde47854126bf077de26ae5ea535f47803a2 [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
17#ifndef ANDROID_DRM_DISPLAY_COMPOSITOR_H_
18#define ANDROID_DRM_DISPLAY_COMPOSITOR_H_
19
20#include "drm_hwcomposer.h"
21#include "drmcomposition.h"
22#include "drmcompositorworker.h"
Zach Reizner713a6782015-07-31 15:12:44 -070023#include "drmframebuffer.h"
Zach Reizner92f8e632015-10-12 17:47:13 -070024#include "seperate_rects.h"
Sean Paul98e73c82015-06-24 14:38:49 -070025
26#include <pthread.h>
Zach Reizner92f8e632015-10-12 17:47:13 -070027#include <memory>
Sean Paul98e73c82015-06-24 14:38:49 -070028#include <queue>
29#include <sstream>
Zach Reizner92f8e632015-10-12 17:47:13 -070030#include <tuple>
Sean Paul98e73c82015-06-24 14:38:49 -070031
32#include <hardware/hardware.h>
33#include <hardware/hwcomposer.h>
34
Zach Reizner713a6782015-07-31 15:12:44 -070035#define DRM_DISPLAY_BUFFERS 2
36
Sean Paul98e73c82015-06-24 14:38:49 -070037namespace android {
38
Zach Reizner713a6782015-07-31 15:12:44 -070039class GLWorkerCompositor;
40
Zach Reizner92f8e632015-10-12 17:47:13 -070041class SquashState {
42 public:
Haixia Shidda2fab2015-10-22 18:12:49 -070043 static const unsigned kHistoryLength = 6; // TODO: make this number not magic
Zach Reizner92f8e632015-10-12 17:47:13 -070044 static const unsigned kMaxLayers = 64;
45
46 struct Region {
47 DrmHwcRect<int> rect;
48 std::bitset<kMaxLayers> layer_refs;
49 std::bitset<kHistoryLength> change_history;
50 bool squashed = false;
51 };
52
53 bool is_stable(int region_index) const {
54 return valid_history_ >= kHistoryLength &&
55 regions_[region_index].change_history.none();
56 }
57
58 const std::vector<Region> &regions() const {
59 return regions_;
60 }
61
62 void Init(DrmHwcLayer *layers, size_t num_layers);
Zach Reizner5757e822015-10-16 19:06:31 -070063 void GenerateHistory(DrmHwcLayer *layers, size_t num_layers,
Zach Reizner92f8e632015-10-12 17:47:13 -070064 std::vector<bool> &changed_regions) const;
65 void StableRegionsWithMarginalHistory(
66 const std::vector<bool> &changed_regions,
67 std::vector<bool> &stable_regions) const;
Zach Reizner5757e822015-10-16 19:06:31 -070068 void RecordHistory(DrmHwcLayer *layers, size_t num_layers,
Zach Reizner92f8e632015-10-12 17:47:13 -070069 const std::vector<bool> &changed_regions);
Zach Reizner5757e822015-10-16 19:06:31 -070070 bool RecordAndCompareSquashed(const std::vector<bool> &squashed_regions);
Zach Reizner92f8e632015-10-12 17:47:13 -070071
Zach Reiznerfd6dc332015-10-13 21:12:48 -070072 void Dump(std::ostringstream *out) const;
73
Zach Reizner92f8e632015-10-12 17:47:13 -070074 private:
75 size_t generation_number_ = 0;
76 unsigned valid_history_ = 0;
77 std::vector<buffer_handle_t> last_handles_;
78
79 std::vector<Region> regions_;
80};
81
Sean Paul98e73c82015-06-24 14:38:49 -070082class DrmDisplayCompositor {
83 public:
84 DrmDisplayCompositor();
85 ~DrmDisplayCompositor();
86
87 int Init(DrmResources *drm, int display);
88
Zach Reizner92f8e632015-10-12 17:47:13 -070089 std::unique_ptr<DrmDisplayComposition> CreateComposition() const;
Sean Paul98e73c82015-06-24 14:38:49 -070090 int QueueComposition(std::unique_ptr<DrmDisplayComposition> composition);
91 int Composite();
92 void Dump(std::ostringstream *out) const;
93
Zach Reizner92f8e632015-10-12 17:47:13 -070094 std::tuple<uint32_t, uint32_t, int> GetActiveModeResolution();
95
Sean Paul98e73c82015-06-24 14:38:49 -070096 bool HaveQueuedComposites() const;
97
Zach Reizner92f8e632015-10-12 17:47:13 -070098 SquashState *squash_state() {
Zach Reizner5757e822015-10-16 19:06:31 -070099 return &squash_state_;
Zach Reizner92f8e632015-10-12 17:47:13 -0700100 }
101
Sean Paul98e73c82015-06-24 14:38:49 -0700102 private:
Haixia Shidda2fab2015-10-22 18:12:49 -0700103 struct FrameState {
104 std::unique_ptr<DrmDisplayComposition> composition;
105 int status = 0;
106 };
107
108 class FrameWorker : public Worker {
109 public:
110 FrameWorker(DrmDisplayCompositor *compositor);
Haixia Shi479412c2015-10-27 10:40:48 -0700111 ~FrameWorker() override;
Haixia Shidda2fab2015-10-22 18:12:49 -0700112
113 int Init();
114 void QueueFrame(std::unique_ptr<DrmDisplayComposition> composition,
115 int status);
116
117 protected:
Haixia Shi479412c2015-10-27 10:40:48 -0700118 void Routine() override;
Haixia Shidda2fab2015-10-22 18:12:49 -0700119
120 private:
121 DrmDisplayCompositor *compositor_;
122 std::queue<FrameState> frame_queue_;
123 };
124
Sean Paul98e73c82015-06-24 14:38:49 -0700125 DrmDisplayCompositor(const DrmDisplayCompositor &) = delete;
126
Sean Paul971be152015-10-13 15:44:45 -0400127 // We'll wait for acquire fences to fire for kAcquireWaitTimeoutMs,
128 // kAcquireWaitTries times, logging a warning in between.
129 static const int kAcquireWaitTries = 5;
130 static const int kAcquireWaitTimeoutMs = 100;
Sean Pauld106b912015-09-29 00:56:00 -0400131
Zach Reizner92f8e632015-10-12 17:47:13 -0700132 int PrepareFramebuffer(DrmFramebuffer &fb,
133 DrmDisplayComposition *display_comp);
Zach Reizner5757e822015-10-16 19:06:31 -0700134 int ApplySquash(DrmDisplayComposition *display_comp);
Zach Reizner713a6782015-07-31 15:12:44 -0700135 int ApplyPreComposite(DrmDisplayComposition *display_comp);
Haixia Shidda2fab2015-10-22 18:12:49 -0700136 int PrepareFrame(DrmDisplayComposition *display_comp);
137 int CommitFrame(DrmDisplayComposition *display_comp);
Sean Pauldb7a17d2015-06-24 18:46:05 -0700138 int ApplyDpms(DrmDisplayComposition *display_comp);
Sean Paul7b1e4bc2015-10-13 15:47:22 -0400139 int DisablePlanes(DrmDisplayComposition *display_comp);
Sean Paul98e73c82015-06-24 14:38:49 -0700140
Haixia Shidda2fab2015-10-22 18:12:49 -0700141 void ApplyFrame(std::unique_ptr<DrmDisplayComposition> composition,
142 int status);
143
Sean Paul98e73c82015-06-24 14:38:49 -0700144 DrmResources *drm_;
145 int display_;
146
147 DrmCompositorWorker worker_;
Haixia Shidda2fab2015-10-22 18:12:49 -0700148 FrameWorker frame_worker_;
Sean Paul98e73c82015-06-24 14:38:49 -0700149
150 std::queue<std::unique_ptr<DrmDisplayComposition>> composite_queue_;
151 std::unique_ptr<DrmDisplayComposition> active_composition_;
152
Sean Paul98e73c82015-06-24 14:38:49 -0700153 bool initialized_;
Sean Pauldb7a17d2015-06-24 18:46:05 -0700154 bool active_;
Sean Paul98e73c82015-06-24 14:38:49 -0700155
Sean Paul57355412015-09-19 09:14:34 -0400156 DrmMode next_mode_;
157 bool needs_modeset_;
158
Zach Reizner713a6782015-07-31 15:12:44 -0700159 int framebuffer_index_;
160 DrmFramebuffer framebuffers_[DRM_DISPLAY_BUFFERS];
161 std::unique_ptr<GLWorkerCompositor> pre_compositor_;
162
Zach Reizner5757e822015-10-16 19:06:31 -0700163 SquashState squash_state_;
164 int squash_framebuffer_index_;
165 DrmFramebuffer squash_framebuffers_[2];
166
Sean Paul98e73c82015-06-24 14:38:49 -0700167 // mutable since we need to acquire in HaveQueuedComposites
168 mutable pthread_mutex_t lock_;
169
170 // State tracking progress since our last Dump(). These are mutable since
171 // we need to reset them on every Dump() call.
172 mutable uint64_t dump_frames_composited_;
173 mutable uint64_t dump_last_timestamp_ns_;
174};
175}
176
177#endif // ANDROID_DRM_DISPLAY_COMPOSITOR_H_