blob: 864bc3f618301f10aedfbb00ef4b2d59aebb51e3 [file] [log] [blame]
Mathias Agopian3f844832013-08-07 21:24:32 -07001/*
2 * Copyright 2013 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 SF_RENDER_ENGINE_PROGRAMCACHE_H
18#define SF_RENDER_ENGINE_PROGRAMCACHE_H
19
20#include <GLES2/gl2.h>
21
Mathias Agopian3f844832013-08-07 21:24:32 -070022#include <utils/KeyedVector.h>
Chia-I Wub027f802017-11-29 14:00:52 -080023#include <utils/Singleton.h>
Mathias Agopian3f844832013-08-07 21:24:32 -070024#include <utils/TypeHelpers.h>
25
26#include "Description.h"
27
28namespace android {
29
30class Description;
Peiyong Lin2542cb02018-04-30 17:29:53 -070031class Formatter;
Mathias Agopian3f844832013-08-07 21:24:32 -070032class Program;
33class String8;
34
35/*
36 * This class generates GLSL programs suitable to handle a given
37 * Description. It's responsible for figuring out what to
38 * generate from a Description.
39 * It also maintains a cache of these Programs.
40 */
41class ProgramCache : public Singleton<ProgramCache> {
42public:
43 /*
44 * Key is used to retrieve a Program in the cache.
45 * A Key is generated from a Description.
46 */
47 class Key {
48 friend class ProgramCache;
49 typedef uint32_t key_t;
50 key_t mKey;
Chia-I Wub027f802017-11-29 14:00:52 -080051
Mathias Agopian3f844832013-08-07 21:24:32 -070052 public:
53 enum {
Chia-I Wu7e65bc02018-01-11 14:31:38 -080054 BLEND_SHIFT = 0,
55 BLEND_MASK = 1 << BLEND_SHIFT,
56 BLEND_PREMULT = 1 << BLEND_SHIFT,
57 BLEND_NORMAL = 0 << BLEND_SHIFT,
Mathias Agopian3f844832013-08-07 21:24:32 -070058
Chia-I Wu7e65bc02018-01-11 14:31:38 -080059 OPACITY_SHIFT = 1,
60 OPACITY_MASK = 1 << OPACITY_SHIFT,
61 OPACITY_OPAQUE = 1 << OPACITY_SHIFT,
62 OPACITY_TRANSLUCENT = 0 << OPACITY_SHIFT,
Mathias Agopian3f844832013-08-07 21:24:32 -070063
Chia-I Wu7e65bc02018-01-11 14:31:38 -080064 ALPHA_SHIFT = 2,
65 ALPHA_MASK = 1 << ALPHA_SHIFT,
66 ALPHA_LT_ONE = 1 << ALPHA_SHIFT,
67 ALPHA_EQ_ONE = 0 << ALPHA_SHIFT,
Mathias Agopian3f844832013-08-07 21:24:32 -070068
Chia-I Wu7e65bc02018-01-11 14:31:38 -080069 TEXTURE_SHIFT = 3,
70 TEXTURE_MASK = 3 << TEXTURE_SHIFT,
71 TEXTURE_OFF = 0 << TEXTURE_SHIFT,
72 TEXTURE_EXT = 1 << TEXTURE_SHIFT,
73 TEXTURE_2D = 2 << TEXTURE_SHIFT,
Mathias Agopianff2ed702013-09-01 21:36:12 -070074
Peiyong Lina296b0c2018-04-30 16:55:29 -070075 INPUT_TRANSFORM_MATRIX_SHIFT = 5,
76 INPUT_TRANSFORM_MATRIX_MASK = 1 << INPUT_TRANSFORM_MATRIX_SHIFT,
77 INPUT_TRANSFORM_MATRIX_OFF = 0 << INPUT_TRANSFORM_MATRIX_SHIFT,
78 INPUT_TRANSFORM_MATRIX_ON = 1 << INPUT_TRANSFORM_MATRIX_SHIFT,
Romain Guy88d37dd2017-05-26 17:57:05 -070079
Peiyong Lina296b0c2018-04-30 16:55:29 -070080 OUTPUT_TRANSFORM_MATRIX_SHIFT = 6,
81 OUTPUT_TRANSFORM_MATRIX_MASK = 1 << OUTPUT_TRANSFORM_MATRIX_SHIFT,
82 OUTPUT_TRANSFORM_MATRIX_OFF = 0 << OUTPUT_TRANSFORM_MATRIX_SHIFT,
83 OUTPUT_TRANSFORM_MATRIX_ON = 1 << OUTPUT_TRANSFORM_MATRIX_SHIFT,
84
85 INPUT_TF_SHIFT = 7,
Chia-I Wu7e65bc02018-01-11 14:31:38 -080086 INPUT_TF_MASK = 3 << INPUT_TF_SHIFT,
87 INPUT_TF_LINEAR = 0 << INPUT_TF_SHIFT,
88 INPUT_TF_SRGB = 1 << INPUT_TF_SHIFT,
Chia-I Wu131d3762018-01-11 14:35:27 -080089 INPUT_TF_ST2084 = 2 << INPUT_TF_SHIFT,
Peiyong Lina3fb7d62018-04-11 17:41:47 -070090 INPUT_TF_HLG = 3 << INPUT_TF_SHIFT,
Chia-I Wu7e65bc02018-01-11 14:31:38 -080091
Peiyong Lina296b0c2018-04-30 16:55:29 -070092 OUTPUT_TF_SHIFT = 9,
Chia-I Wu7e65bc02018-01-11 14:31:38 -080093 OUTPUT_TF_MASK = 3 << OUTPUT_TF_SHIFT,
94 OUTPUT_TF_LINEAR = 0 << OUTPUT_TF_SHIFT,
95 OUTPUT_TF_SRGB = 1 << OUTPUT_TF_SHIFT,
Chia-I Wu131d3762018-01-11 14:35:27 -080096 OUTPUT_TF_ST2084 = 2 << OUTPUT_TF_SHIFT,
Peiyong Lina3fb7d62018-04-11 17:41:47 -070097 OUTPUT_TF_HLG = 3 << OUTPUT_TF_SHIFT,
Chia-I Wu131d3762018-01-11 14:35:27 -080098
Peiyong Lina296b0c2018-04-30 16:55:29 -070099 Y410_BT2020_SHIFT = 11,
Chia-I Wu131d3762018-01-11 14:35:27 -0800100 Y410_BT2020_MASK = 1 << Y410_BT2020_SHIFT,
101 Y410_BT2020_OFF = 0 << Y410_BT2020_SHIFT,
102 Y410_BT2020_ON = 1 << Y410_BT2020_SHIFT,
Mathias Agopian3f844832013-08-07 21:24:32 -0700103 };
104
Chia-I Wub027f802017-11-29 14:00:52 -0800105 inline Key() : mKey(0) {}
106 inline Key(const Key& rhs) : mKey(rhs.mKey) {}
Mathias Agopian3f844832013-08-07 21:24:32 -0700107
108 inline Key& set(key_t mask, key_t value) {
109 mKey = (mKey & ~mask) | value;
110 return *this;
111 }
112
Chia-I Wub027f802017-11-29 14:00:52 -0800113 inline bool isTexturing() const { return (mKey & TEXTURE_MASK) != TEXTURE_OFF; }
114 inline int getTextureTarget() const { return (mKey & TEXTURE_MASK); }
115 inline bool isPremultiplied() const { return (mKey & BLEND_MASK) == BLEND_PREMULT; }
116 inline bool isOpaque() const { return (mKey & OPACITY_MASK) == OPACITY_OPAQUE; }
117 inline bool hasAlpha() const { return (mKey & ALPHA_MASK) == ALPHA_LT_ONE; }
Peiyong Lina296b0c2018-04-30 16:55:29 -0700118 inline bool hasInputTransformMatrix() const {
119 return (mKey & INPUT_TRANSFORM_MATRIX_MASK) == INPUT_TRANSFORM_MATRIX_ON;
120 }
121 inline bool hasOutputTransformMatrix() const {
122 return (mKey & OUTPUT_TRANSFORM_MATRIX_MASK) == OUTPUT_TRANSFORM_MATRIX_ON;
123 }
124 inline bool hasTransformMatrix() const {
125 return hasInputTransformMatrix() || hasOutputTransformMatrix();
126 }
Chia-I Wu7e65bc02018-01-11 14:31:38 -0800127 inline int getInputTF() const { return (mKey & INPUT_TF_MASK); }
128 inline int getOutputTF() const { return (mKey & OUTPUT_TF_MASK); }
Peiyong Lin53f320e2018-04-23 17:31:06 -0700129
130 // When HDR and non-HDR contents are mixed, or different types of HDR contents are
131 // mixed, we will do a tone mapping process to tone map the input content to output
132 // content. Currently, the following conversions handled, they are:
133 // * SDR -> HLG
134 // * SDR -> PQ
135 // * HLG -> PQ
136 inline bool needsToneMapping() const {
137 int inputTF = getInputTF();
138 int outputTF = getOutputTF();
139
140 // Return false when converting from SDR to SDR.
141 if (inputTF == Key::INPUT_TF_SRGB && outputTF == Key::OUTPUT_TF_LINEAR) {
142 return false;
143 }
144 if (inputTF == Key::INPUT_TF_LINEAR && outputTF == Key::OUTPUT_TF_SRGB) {
145 return false;
146 }
147
148 inputTF >>= Key::INPUT_TF_SHIFT;
149 outputTF >>= Key::OUTPUT_TF_SHIFT;
150 return inputTF != outputTF;
151 }
Chia-I Wu131d3762018-01-11 14:35:27 -0800152 inline bool isY410BT2020() const { return (mKey & Y410_BT2020_MASK) == Y410_BT2020_ON; }
Mathias Agopian3f844832013-08-07 21:24:32 -0700153
154 // this is the definition of a friend function -- not a method of class Needs
155 friend inline int strictly_order_type(const Key& lhs, const Key& rhs) {
Romain Guy88d37dd2017-05-26 17:57:05 -0700156 return (lhs.mKey < rhs.mKey) ? 1 : 0;
Mathias Agopian3f844832013-08-07 21:24:32 -0700157 }
158 };
159
160 ProgramCache();
161 ~ProgramCache();
162
163 // useProgram lookup a suitable program in the cache or generates one
164 // if none can be found.
165 void useProgram(const Description& description);
166
167private:
Riley Andrewsa51fafc2014-09-29 13:29:40 -0700168 // Generate shaders to populate the cache
169 void primeCache();
Mathias Agopian3f844832013-08-07 21:24:32 -0700170 // compute a cache Key from a Description
171 static Key computeKey(const Description& description);
Peiyong Lin2542cb02018-04-30 17:29:53 -0700172 // Generate EOTF based from Key.
173 static void generateEOTF(Formatter& fs, const Key& needs);
Peiyong Lin53f320e2018-04-23 17:31:06 -0700174 // Generate necessary tone mapping methods for OOTF.
175 static void generateToneMappingProcess(Formatter& fs, const Key& needs);
Peiyong Lin2542cb02018-04-30 17:29:53 -0700176 // Generate OOTF based from Key.
177 static void generateOOTF(Formatter& fs, const Key& needs);
178 // Generate OETF based from Key.
179 static void generateOETF(Formatter& fs, const Key& needs);
Mathias Agopian3f844832013-08-07 21:24:32 -0700180 // generates a program from the Key
181 static Program* generateProgram(const Key& needs);
182 // generates the vertex shader from the Key
183 static String8 generateVertexShader(const Key& needs);
184 // generates the fragment shader from the Key
185 static String8 generateFragmentShader(const Key& needs);
186
187 // Key/Value map used for caching Programs. Currently the cache
188 // is never shrunk.
189 DefaultKeyedVector<Key, Program*> mCache;
190};
191
Mathias Agopian3f844832013-08-07 21:24:32 -0700192ANDROID_BASIC_TYPES_TRAITS(ProgramCache::Key)
193
194} /* namespace android */
195
196#endif /* SF_RENDER_ENGINE_PROGRAMCACHE_H */