• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2010 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 #define LOG_TAG "OpenGLRenderer"
18 
19 #include <utils/String8.h>
20 
21 #include "Caches.h"
22 #include "Dither.h"
23 #include "ProgramCache.h"
24 
25 namespace android {
26 namespace uirenderer {
27 
28 ///////////////////////////////////////////////////////////////////////////////
29 // Defines
30 ///////////////////////////////////////////////////////////////////////////////
31 
32 #define MODULATE_OP_NO_MODULATE 0
33 #define MODULATE_OP_MODULATE 1
34 #define MODULATE_OP_MODULATE_A8 2
35 
36 #define STR(x) STR1(x)
37 #define STR1(x) #x
38 
39 ///////////////////////////////////////////////////////////////////////////////
40 // Vertex shaders snippets
41 ///////////////////////////////////////////////////////////////////////////////
42 
43 const char* gVS_Header_Attributes =
44         "attribute vec4 position;\n";
45 const char* gVS_Header_Attributes_TexCoords =
46         "attribute vec2 texCoords;\n";
47 const char* gVS_Header_Attributes_Colors =
48         "attribute vec4 colors;\n";
49 const char* gVS_Header_Attributes_VertexAlphaParameters =
50         "attribute float vtxAlpha;\n";
51 const char* gVS_Header_Uniforms_TextureTransform =
52         "uniform mat4 mainTextureTransform;\n";
53 const char* gVS_Header_Uniforms =
54         "uniform mat4 projection;\n" \
55         "uniform mat4 transform;\n";
56 const char* gVS_Header_Uniforms_HasGradient =
57         "uniform mat4 screenSpace;\n";
58 const char* gVS_Header_Uniforms_HasBitmap =
59         "uniform mat4 textureTransform;\n"
60         "uniform mediump vec2 textureDimension;\n";
61 const char* gVS_Header_Uniforms_HasRoundRectClip =
62         "uniform mat4 roundRectInvTransform;\n";
63 const char* gVS_Header_Varyings_HasTexture =
64         "varying vec2 outTexCoords;\n";
65 const char* gVS_Header_Varyings_HasColors =
66         "varying vec4 outColors;\n";
67 const char* gVS_Header_Varyings_HasVertexAlpha =
68         "varying float alpha;\n";
69 const char* gVS_Header_Varyings_HasBitmap =
70         "varying highp vec2 outBitmapTexCoords;\n";
71 const char* gVS_Header_Varyings_HasGradient[6] = {
72         // Linear
73         "varying highp vec2 linear;\n"
74         "varying vec2 ditherTexCoords;\n",
75         "varying float linear;\n"
76         "varying vec2 ditherTexCoords;\n",
77 
78         // Circular
79         "varying highp vec2 circular;\n"
80         "varying vec2 ditherTexCoords;\n",
81         "varying highp vec2 circular;\n"
82         "varying vec2 ditherTexCoords;\n",
83 
84         // Sweep
85         "varying highp vec2 sweep;\n"
86         "varying vec2 ditherTexCoords;\n",
87         "varying highp vec2 sweep;\n"
88         "varying vec2 ditherTexCoords;\n",
89 };
90 const char* gVS_Header_Varyings_HasRoundRectClip =
91         "varying highp vec2 roundRectPos;\n";
92 const char* gVS_Main =
93         "\nvoid main(void) {\n";
94 const char* gVS_Main_OutTexCoords =
95         "    outTexCoords = texCoords;\n";
96 const char* gVS_Main_OutColors =
97         "    outColors = colors;\n";
98 const char* gVS_Main_OutTransformedTexCoords =
99         "    outTexCoords = (mainTextureTransform * vec4(texCoords, 0.0, 1.0)).xy;\n";
100 const char* gVS_Main_OutGradient[6] = {
101         // Linear
102         "    linear = vec2((screenSpace * position).x, 0.5);\n"
103         "    ditherTexCoords = (transform * position).xy * " STR(DITHER_KERNEL_SIZE_INV) ";\n",
104         "    linear = (screenSpace * position).x;\n"
105         "    ditherTexCoords = (transform * position).xy * " STR(DITHER_KERNEL_SIZE_INV) ";\n",
106 
107         // Circular
108         "    circular = (screenSpace * position).xy;\n"
109         "    ditherTexCoords = (transform * position).xy * " STR(DITHER_KERNEL_SIZE_INV) ";\n",
110         "    circular = (screenSpace * position).xy;\n"
111         "    ditherTexCoords = (transform * position).xy * " STR(DITHER_KERNEL_SIZE_INV) ";\n",
112 
113         // Sweep
114         "    sweep = (screenSpace * position).xy;\n"
115         "    ditherTexCoords = (transform * position).xy * " STR(DITHER_KERNEL_SIZE_INV) ";\n",
116         "    sweep = (screenSpace * position).xy;\n"
117         "    ditherTexCoords = (transform * position).xy * " STR(DITHER_KERNEL_SIZE_INV) ";\n",
118 };
119 const char* gVS_Main_OutBitmapTexCoords =
120         "    outBitmapTexCoords = (textureTransform * position).xy * textureDimension;\n";
121 const char* gVS_Main_Position =
122         "    vec4 transformedPosition = projection * transform * position;\n"
123         "    gl_Position = transformedPosition;\n";
124 
125 const char* gVS_Main_VertexAlpha =
126         "    alpha = vtxAlpha;\n";
127 
128 const char* gVS_Main_HasRoundRectClip =
129         "    roundRectPos = (roundRectInvTransform * transformedPosition).xy;\n";
130 const char* gVS_Footer =
131         "}\n\n";
132 
133 ///////////////////////////////////////////////////////////////////////////////
134 // Fragment shaders snippets
135 ///////////////////////////////////////////////////////////////////////////////
136 
137 const char* gFS_Header_Extension_FramebufferFetch =
138         "#extension GL_NV_shader_framebuffer_fetch : enable\n\n";
139 const char* gFS_Header_Extension_ExternalTexture =
140         "#extension GL_OES_EGL_image_external : require\n\n";
141 const char* gFS_Header =
142         "precision mediump float;\n\n";
143 const char* gFS_Uniforms_Color =
144         "uniform vec4 color;\n";
145 const char* gFS_Uniforms_TextureSampler =
146         "uniform sampler2D baseSampler;\n";
147 const char* gFS_Uniforms_ExternalTextureSampler =
148         "uniform samplerExternalOES baseSampler;\n";
149 const char* gFS_Uniforms_Dither =
150         "uniform sampler2D ditherSampler;";
151 const char* gFS_Uniforms_GradientSampler[2] = {
152         "%s\n"
153         "uniform sampler2D gradientSampler;\n",
154         "%s\n"
155         "uniform vec4 startColor;\n"
156         "uniform vec4 endColor;\n"
157 };
158 const char* gFS_Uniforms_BitmapSampler =
159         "uniform sampler2D bitmapSampler;\n";
160 const char* gFS_Uniforms_ColorOp[3] = {
161         // None
162         "",
163         // Matrix
164         "uniform mat4 colorMatrix;\n"
165         "uniform vec4 colorMatrixVector;\n",
166         // PorterDuff
167         "uniform vec4 colorBlend;\n"
168 };
169 const char* gFS_Uniforms_Gamma =
170         "uniform float gamma;\n";
171 
172 const char* gFS_Uniforms_HasRoundRectClip =
173         "uniform vec4 roundRectInnerRectLTRB;\n"
174         "uniform float roundRectRadius;\n";
175 
176 const char* gFS_Main =
177         "\nvoid main(void) {\n"
178         "    lowp vec4 fragColor;\n";
179 
180 const char* gFS_Main_Dither[2] = {
181         // ES 2.0
182         "texture2D(ditherSampler, ditherTexCoords).a * " STR(DITHER_KERNEL_SIZE_INV_SQUARE),
183         // ES 3.0
184         "texture2D(ditherSampler, ditherTexCoords).a"
185 };
186 const char* gFS_Main_AddDitherToGradient =
187         "    gradientColor += %s;\n";
188 
189 // Fast cases
190 const char* gFS_Fast_SingleColor =
191         "\nvoid main(void) {\n"
192         "    gl_FragColor = color;\n"
193         "}\n\n";
194 const char* gFS_Fast_SingleTexture =
195         "\nvoid main(void) {\n"
196         "    gl_FragColor = texture2D(baseSampler, outTexCoords);\n"
197         "}\n\n";
198 const char* gFS_Fast_SingleModulateTexture =
199         "\nvoid main(void) {\n"
200         "    gl_FragColor = color.a * texture2D(baseSampler, outTexCoords);\n"
201         "}\n\n";
202 const char* gFS_Fast_SingleA8Texture =
203         "\nvoid main(void) {\n"
204         "    gl_FragColor = texture2D(baseSampler, outTexCoords);\n"
205         "}\n\n";
206 const char* gFS_Fast_SingleA8Texture_ApplyGamma =
207         "\nvoid main(void) {\n"
208         "    gl_FragColor = vec4(0.0, 0.0, 0.0, pow(texture2D(baseSampler, outTexCoords).a, gamma));\n"
209         "}\n\n";
210 const char* gFS_Fast_SingleModulateA8Texture =
211         "\nvoid main(void) {\n"
212         "    gl_FragColor = color * texture2D(baseSampler, outTexCoords).a;\n"
213         "}\n\n";
214 const char* gFS_Fast_SingleModulateA8Texture_ApplyGamma =
215         "\nvoid main(void) {\n"
216         "    gl_FragColor = color * pow(texture2D(baseSampler, outTexCoords).a, gamma);\n"
217         "}\n\n";
218 const char* gFS_Fast_SingleGradient[2] = {
219         "\nvoid main(void) {\n"
220         "    gl_FragColor = %s + texture2D(gradientSampler, linear);\n"
221         "}\n\n",
222         "\nvoid main(void) {\n"
223         "    gl_FragColor = %s + mix(startColor, endColor, clamp(linear, 0.0, 1.0));\n"
224         "}\n\n",
225 };
226 const char* gFS_Fast_SingleModulateGradient[2] = {
227         "\nvoid main(void) {\n"
228         "    gl_FragColor = %s + color.a * texture2D(gradientSampler, linear);\n"
229         "}\n\n",
230         "\nvoid main(void) {\n"
231         "    gl_FragColor = %s + color.a * mix(startColor, endColor, clamp(linear, 0.0, 1.0));\n"
232         "}\n\n"
233 };
234 
235 // General case
236 const char* gFS_Main_FetchColor =
237         "    fragColor = color;\n";
238 const char* gFS_Main_ModulateColor =
239         "    fragColor *= color.a;\n";
240 const char* gFS_Main_ApplyVertexAlphaLinearInterp =
241         "    fragColor *= alpha;\n";
242 const char* gFS_Main_ApplyVertexAlphaShadowInterp =
243         "    fragColor *= (1.0 - cos(alpha)) / 2.0;\n";
244 
245 const char* gFS_Main_FetchTexture[2] = {
246         // Don't modulate
247         "    fragColor = texture2D(baseSampler, outTexCoords);\n",
248         // Modulate
249         "    fragColor = color * texture2D(baseSampler, outTexCoords);\n"
250 };
251 const char* gFS_Main_FetchA8Texture[4] = {
252         // Don't modulate
253         "    fragColor = texture2D(baseSampler, outTexCoords);\n",
254         "    fragColor = texture2D(baseSampler, outTexCoords);\n",
255         // Modulate
256         "    fragColor = color * texture2D(baseSampler, outTexCoords).a;\n",
257         "    fragColor = color * pow(texture2D(baseSampler, outTexCoords).a, gamma);\n"
258 };
259 const char* gFS_Main_FetchGradient[6] = {
260         // Linear
261         "    vec4 gradientColor = texture2D(gradientSampler, linear);\n",
262 
263         "    vec4 gradientColor = mix(startColor, endColor, clamp(linear, 0.0, 1.0));\n",
264 
265         // Circular
266         "    vec4 gradientColor = texture2D(gradientSampler, vec2(length(circular), 0.5));\n",
267 
268         "    vec4 gradientColor = mix(startColor, endColor, clamp(length(circular), 0.0, 1.0));\n",
269 
270         // Sweep
271         "    highp float index = atan(sweep.y, sweep.x) * 0.15915494309; // inv(2 * PI)\n"
272         "    vec4 gradientColor = texture2D(gradientSampler, vec2(index - floor(index), 0.5));\n",
273 
274         "    highp float index = atan(sweep.y, sweep.x) * 0.15915494309; // inv(2 * PI)\n"
275         "    vec4 gradientColor = mix(startColor, endColor, clamp(index - floor(index), 0.0, 1.0));\n"
276 };
277 const char* gFS_Main_FetchBitmap =
278         "    vec4 bitmapColor = texture2D(bitmapSampler, outBitmapTexCoords);\n";
279 const char* gFS_Main_FetchBitmapNpot =
280         "    vec4 bitmapColor = texture2D(bitmapSampler, wrap(outBitmapTexCoords));\n";
281 const char* gFS_Main_BlendShadersBG =
282         "    fragColor = blendShaders(gradientColor, bitmapColor)";
283 const char* gFS_Main_BlendShadersGB =
284         "    fragColor = blendShaders(bitmapColor, gradientColor)";
285 const char* gFS_Main_BlendShaders_Modulate[6] = {
286         // Don't modulate
287         ";\n",
288         ";\n",
289         // Modulate
290         " * color.a;\n",
291         " * color.a;\n",
292         // Modulate with alpha 8 texture
293         " * texture2D(baseSampler, outTexCoords).a;\n",
294         " * pow(texture2D(baseSampler, outTexCoords).a, gamma);\n"
295 };
296 const char* gFS_Main_GradientShader_Modulate[6] = {
297         // Don't modulate
298         "    fragColor = gradientColor;\n",
299         "    fragColor = gradientColor;\n",
300         // Modulate
301         "    fragColor = gradientColor * color.a;\n",
302         "    fragColor = gradientColor * color.a;\n",
303         // Modulate with alpha 8 texture
304         "    fragColor = gradientColor * texture2D(baseSampler, outTexCoords).a;\n",
305         "    fragColor = gradientColor * pow(texture2D(baseSampler, outTexCoords).a, gamma);\n"
306     };
307 const char* gFS_Main_BitmapShader_Modulate[6] = {
308         // Don't modulate
309         "    fragColor = bitmapColor;\n",
310         "    fragColor = bitmapColor;\n",
311         // Modulate
312         "    fragColor = bitmapColor * color.a;\n",
313         "    fragColor = bitmapColor * color.a;\n",
314         // Modulate with alpha 8 texture
315         "    fragColor = bitmapColor * texture2D(baseSampler, outTexCoords).a;\n",
316         "    fragColor = bitmapColor * pow(texture2D(baseSampler, outTexCoords).a, gamma);\n"
317     };
318 const char* gFS_Main_FragColor =
319         "    gl_FragColor = fragColor;\n";
320 const char* gFS_Main_FragColor_HasColors =
321         "    gl_FragColor *= outColors;\n";
322 const char* gFS_Main_FragColor_Blend =
323         "    gl_FragColor = blendFramebuffer(fragColor, gl_LastFragColor);\n";
324 const char* gFS_Main_FragColor_Blend_Swap =
325         "    gl_FragColor = blendFramebuffer(gl_LastFragColor, fragColor);\n";
326 const char* gFS_Main_ApplyColorOp[3] = {
327         // None
328         "",
329         // Matrix
330         "    fragColor.rgb /= (fragColor.a + 0.0019);\n" // un-premultiply
331         "    fragColor *= colorMatrix;\n"
332         "    fragColor += colorMatrixVector;\n"
333         "    fragColor.rgb *= (fragColor.a + 0.0019);\n", // re-premultiply
334         // PorterDuff
335         "    fragColor = blendColors(colorBlend, fragColor);\n"
336 };
337 
338 // Note: LTRB -> xyzw
339 const char* gFS_Main_FragColor_HasRoundRectClip =
340         "    mediump vec2 fragToLT = roundRectInnerRectLTRB.xy - roundRectPos;\n"
341         "    mediump vec2 fragFromRB = roundRectPos - roundRectInnerRectLTRB.zw;\n"
342 
343         // divide + multiply by 128 to avoid falling out of range in length() function
344         "    mediump vec2 dist = max(max(fragToLT, fragFromRB), vec2(0.0, 0.0)) / 128.0;\n"
345         "    mediump float linearDist = roundRectRadius - (length(dist) * 128.0);\n"
346         "    gl_FragColor *= clamp(linearDist, 0.0, 1.0);\n";
347 
348 const char* gFS_Main_DebugHighlight =
349         "    gl_FragColor.rgb = vec3(0.0, gl_FragColor.a, 0.0);\n";
350 const char* gFS_Footer =
351         "}\n\n";
352 
353 ///////////////////////////////////////////////////////////////////////////////
354 // PorterDuff snippets
355 ///////////////////////////////////////////////////////////////////////////////
356 
357 const char* gBlendOps[18] = {
358         // Clear
359         "return vec4(0.0, 0.0, 0.0, 0.0);\n",
360         // Src
361         "return src;\n",
362         // Dst
363         "return dst;\n",
364         // SrcOver
365         "return src + dst * (1.0 - src.a);\n",
366         // DstOver
367         "return dst + src * (1.0 - dst.a);\n",
368         // SrcIn
369         "return src * dst.a;\n",
370         // DstIn
371         "return dst * src.a;\n",
372         // SrcOut
373         "return src * (1.0 - dst.a);\n",
374         // DstOut
375         "return dst * (1.0 - src.a);\n",
376         // SrcAtop
377         "return vec4(src.rgb * dst.a + (1.0 - src.a) * dst.rgb, dst.a);\n",
378         // DstAtop
379         "return vec4(dst.rgb * src.a + (1.0 - dst.a) * src.rgb, src.a);\n",
380         // Xor
381         "return vec4(src.rgb * (1.0 - dst.a) + (1.0 - src.a) * dst.rgb, "
382                 "src.a + dst.a - 2.0 * src.a * dst.a);\n",
383         // Add
384         "return min(src + dst, 1.0);\n",
385         // Multiply
386         "return src * dst;\n",
387         // Screen
388         "return src + dst - src * dst;\n",
389         // Overlay
390         "return clamp(vec4(mix("
391                 "2.0 * src.rgb * dst.rgb + src.rgb * (1.0 - dst.a) + dst.rgb * (1.0 - src.a), "
392                 "src.a * dst.a - 2.0 * (dst.a - dst.rgb) * (src.a - src.rgb) + src.rgb * (1.0 - dst.a) + dst.rgb * (1.0 - src.a), "
393                 "step(dst.a, 2.0 * dst.rgb)), "
394                 "src.a + dst.a - src.a * dst.a), 0.0, 1.0);\n",
395         // Darken
396         "return vec4(src.rgb * (1.0 - dst.a) + (1.0 - src.a) * dst.rgb + "
397                 "min(src.rgb * dst.a, dst.rgb * src.a), src.a + dst.a - src.a * dst.a);\n",
398         // Lighten
399         "return vec4(src.rgb * (1.0 - dst.a) + (1.0 - src.a) * dst.rgb + "
400                 "max(src.rgb * dst.a, dst.rgb * src.a), src.a + dst.a - src.a * dst.a);\n",
401 };
402 
403 ///////////////////////////////////////////////////////////////////////////////
404 // Constructors/destructors
405 ///////////////////////////////////////////////////////////////////////////////
406 
ProgramCache()407 ProgramCache::ProgramCache(): mHasES3(Extensions::getInstance().getMajorGlVersion() >= 3) {
408 }
409 
~ProgramCache()410 ProgramCache::~ProgramCache() {
411     clear();
412 }
413 
414 ///////////////////////////////////////////////////////////////////////////////
415 // Cache management
416 ///////////////////////////////////////////////////////////////////////////////
417 
clear()418 void ProgramCache::clear() {
419     PROGRAM_LOGD("Clearing program cache");
420 
421     size_t count = mCache.size();
422     for (size_t i = 0; i < count; i++) {
423         delete mCache.valueAt(i);
424     }
425     mCache.clear();
426 }
427 
get(const ProgramDescription & description)428 Program* ProgramCache::get(const ProgramDescription& description) {
429     programid key = description.key();
430     if (key == (PROGRAM_KEY_TEXTURE | PROGRAM_KEY_A8_TEXTURE)) {
431         // program for A8, unmodulated, texture w/o shader (black text/path textures) is equivalent
432         // to standard texture program (bitmaps, patches). Consider them equivalent.
433         key = PROGRAM_KEY_TEXTURE;
434     }
435 
436     ssize_t index = mCache.indexOfKey(key);
437     Program* program = NULL;
438     if (index < 0) {
439         description.log("Could not find program");
440         program = generateProgram(description, key);
441         mCache.add(key, program);
442     } else {
443         program = mCache.valueAt(index);
444     }
445     return program;
446 }
447 
448 ///////////////////////////////////////////////////////////////////////////////
449 // Program generation
450 ///////////////////////////////////////////////////////////////////////////////
451 
generateProgram(const ProgramDescription & description,programid key)452 Program* ProgramCache::generateProgram(const ProgramDescription& description, programid key) {
453     String8 vertexShader = generateVertexShader(description);
454     String8 fragmentShader = generateFragmentShader(description);
455 
456     return new Program(description, vertexShader.string(), fragmentShader.string());
457 }
458 
gradientIndex(const ProgramDescription & description)459 static inline size_t gradientIndex(const ProgramDescription& description) {
460     return description.gradientType * 2 + description.isSimpleGradient;
461 }
462 
generateVertexShader(const ProgramDescription & description)463 String8 ProgramCache::generateVertexShader(const ProgramDescription& description) {
464     // Add attributes
465     String8 shader(gVS_Header_Attributes);
466     if (description.hasTexture || description.hasExternalTexture) {
467         shader.append(gVS_Header_Attributes_TexCoords);
468     }
469     if (description.hasVertexAlpha) {
470         shader.append(gVS_Header_Attributes_VertexAlphaParameters);
471     }
472     if (description.hasColors) {
473         shader.append(gVS_Header_Attributes_Colors);
474     }
475     // Uniforms
476     shader.append(gVS_Header_Uniforms);
477     if (description.hasTextureTransform) {
478         shader.append(gVS_Header_Uniforms_TextureTransform);
479     }
480     if (description.hasGradient) {
481         shader.append(gVS_Header_Uniforms_HasGradient);
482     }
483     if (description.hasBitmap) {
484         shader.append(gVS_Header_Uniforms_HasBitmap);
485     }
486     if (description.hasRoundRectClip) {
487         shader.append(gVS_Header_Uniforms_HasRoundRectClip);
488     }
489     // Varyings
490     if (description.hasTexture || description.hasExternalTexture) {
491         shader.append(gVS_Header_Varyings_HasTexture);
492     }
493     if (description.hasVertexAlpha) {
494         shader.append(gVS_Header_Varyings_HasVertexAlpha);
495     }
496     if (description.hasColors) {
497         shader.append(gVS_Header_Varyings_HasColors);
498     }
499     if (description.hasGradient) {
500         shader.append(gVS_Header_Varyings_HasGradient[gradientIndex(description)]);
501     }
502     if (description.hasBitmap) {
503         shader.append(gVS_Header_Varyings_HasBitmap);
504     }
505     if (description.hasRoundRectClip) {
506         shader.append(gVS_Header_Varyings_HasRoundRectClip);
507     }
508 
509     // Begin the shader
510     shader.append(gVS_Main); {
511         if (description.hasTextureTransform) {
512             shader.append(gVS_Main_OutTransformedTexCoords);
513         } else if (description.hasTexture || description.hasExternalTexture) {
514             shader.append(gVS_Main_OutTexCoords);
515         }
516         if (description.hasVertexAlpha) {
517             shader.append(gVS_Main_VertexAlpha);
518         }
519         if (description.hasColors) {
520             shader.append(gVS_Main_OutColors);
521         }
522         if (description.hasBitmap) {
523             shader.append(gVS_Main_OutBitmapTexCoords);
524         }
525         // Output transformed position
526         shader.append(gVS_Main_Position);
527         if (description.hasGradient) {
528             shader.append(gVS_Main_OutGradient[gradientIndex(description)]);
529         }
530         if (description.hasRoundRectClip) {
531             shader.append(gVS_Main_HasRoundRectClip);
532         }
533     }
534     // End the shader
535     shader.append(gVS_Footer);
536 
537     PROGRAM_LOGD("*** Generated vertex shader:\n\n%s", shader.string());
538 
539     return shader;
540 }
541 
shaderOp(const ProgramDescription & description,String8 & shader,const int modulateOp,const char ** snippets)542 static bool shaderOp(const ProgramDescription& description, String8& shader,
543         const int modulateOp, const char** snippets) {
544     int op = description.hasAlpha8Texture ? MODULATE_OP_MODULATE_A8 : modulateOp;
545     op = op * 2 + description.hasGammaCorrection;
546     shader.append(snippets[op]);
547     return description.hasAlpha8Texture;
548 }
549 
generateFragmentShader(const ProgramDescription & description)550 String8 ProgramCache::generateFragmentShader(const ProgramDescription& description) {
551     String8 shader;
552 
553     const bool blendFramebuffer = description.framebufferMode >= SkXfermode::kPlus_Mode;
554     if (blendFramebuffer) {
555         shader.append(gFS_Header_Extension_FramebufferFetch);
556     }
557     if (description.hasExternalTexture) {
558         shader.append(gFS_Header_Extension_ExternalTexture);
559     }
560 
561     shader.append(gFS_Header);
562 
563     // Varyings
564     if (description.hasTexture || description.hasExternalTexture) {
565         shader.append(gVS_Header_Varyings_HasTexture);
566     }
567     if (description.hasVertexAlpha) {
568         shader.append(gVS_Header_Varyings_HasVertexAlpha);
569     }
570     if (description.hasColors) {
571         shader.append(gVS_Header_Varyings_HasColors);
572     }
573     if (description.hasGradient) {
574         shader.append(gVS_Header_Varyings_HasGradient[gradientIndex(description)]);
575     }
576     if (description.hasBitmap) {
577         shader.append(gVS_Header_Varyings_HasBitmap);
578     }
579     if (description.hasRoundRectClip) {
580         shader.append(gVS_Header_Varyings_HasRoundRectClip);
581     }
582 
583     // Uniforms
584     int modulateOp = MODULATE_OP_NO_MODULATE;
585     const bool singleColor = !description.hasTexture && !description.hasExternalTexture &&
586             !description.hasGradient && !description.hasBitmap;
587 
588     if (description.modulate || singleColor) {
589         shader.append(gFS_Uniforms_Color);
590         if (!singleColor) modulateOp = MODULATE_OP_MODULATE;
591     }
592     if (description.hasTexture) {
593         shader.append(gFS_Uniforms_TextureSampler);
594     } else if (description.hasExternalTexture) {
595         shader.append(gFS_Uniforms_ExternalTextureSampler);
596     }
597     if (description.hasGradient) {
598         shader.appendFormat(gFS_Uniforms_GradientSampler[description.isSimpleGradient],
599                 gFS_Uniforms_Dither);
600     }
601     if (description.hasGammaCorrection) {
602         shader.append(gFS_Uniforms_Gamma);
603     }
604     if (description.hasRoundRectClip) {
605         shader.append(gFS_Uniforms_HasRoundRectClip);
606     }
607 
608     // Optimization for common cases
609     if (!description.hasVertexAlpha
610             && !blendFramebuffer
611             && !description.hasColors
612             && description.colorOp == ProgramDescription::kColorNone
613             && !description.hasDebugHighlight
614             && !description.hasRoundRectClip) {
615         bool fast = false;
616 
617         const bool noShader = !description.hasGradient && !description.hasBitmap;
618         const bool singleTexture = (description.hasTexture || description.hasExternalTexture) &&
619                 !description.hasAlpha8Texture && noShader;
620         const bool singleA8Texture = description.hasTexture &&
621                 description.hasAlpha8Texture && noShader;
622         const bool singleGradient = !description.hasTexture && !description.hasExternalTexture &&
623                 description.hasGradient && !description.hasBitmap &&
624                 description.gradientType == ProgramDescription::kGradientLinear;
625 
626         if (singleColor) {
627             shader.append(gFS_Fast_SingleColor);
628             fast = true;
629         } else if (singleTexture) {
630             if (!description.modulate) {
631                 shader.append(gFS_Fast_SingleTexture);
632             } else {
633                 shader.append(gFS_Fast_SingleModulateTexture);
634             }
635             fast = true;
636         } else if (singleA8Texture) {
637             if (!description.modulate) {
638                 if (description.hasGammaCorrection) {
639                     shader.append(gFS_Fast_SingleA8Texture_ApplyGamma);
640                 } else {
641                     shader.append(gFS_Fast_SingleA8Texture);
642                 }
643             } else {
644                 if (description.hasGammaCorrection) {
645                     shader.append(gFS_Fast_SingleModulateA8Texture_ApplyGamma);
646                 } else {
647                     shader.append(gFS_Fast_SingleModulateA8Texture);
648                 }
649             }
650             fast = true;
651         } else if (singleGradient) {
652             if (!description.modulate) {
653                 shader.appendFormat(gFS_Fast_SingleGradient[description.isSimpleGradient],
654                         gFS_Main_Dither[mHasES3]);
655             } else {
656                 shader.appendFormat(gFS_Fast_SingleModulateGradient[description.isSimpleGradient],
657                         gFS_Main_Dither[mHasES3]);
658             }
659             fast = true;
660         }
661 
662         if (fast) {
663 #if DEBUG_PROGRAMS
664                 PROGRAM_LOGD("*** Fast case:\n");
665                 PROGRAM_LOGD("*** Generated fragment shader:\n\n");
666                 printLongString(shader);
667 #endif
668 
669             return shader;
670         }
671     }
672 
673     if (description.hasBitmap) {
674         shader.append(gFS_Uniforms_BitmapSampler);
675     }
676     shader.append(gFS_Uniforms_ColorOp[description.colorOp]);
677 
678     // Generate required functions
679     if (description.hasGradient && description.hasBitmap) {
680         generateBlend(shader, "blendShaders", description.shadersMode);
681     }
682     if (description.colorOp == ProgramDescription::kColorBlend) {
683         generateBlend(shader, "blendColors", description.colorMode);
684     }
685     if (blendFramebuffer) {
686         generateBlend(shader, "blendFramebuffer", description.framebufferMode);
687     }
688     if (description.isBitmapNpot) {
689         generateTextureWrap(shader, description.bitmapWrapS, description.bitmapWrapT);
690     }
691 
692     // Begin the shader
693     shader.append(gFS_Main); {
694         // Stores the result in fragColor directly
695         if (description.hasTexture || description.hasExternalTexture) {
696             if (description.hasAlpha8Texture) {
697                 if (!description.hasGradient && !description.hasBitmap) {
698                     shader.append(gFS_Main_FetchA8Texture[modulateOp * 2 +
699                                                           description.hasGammaCorrection]);
700                 }
701             } else {
702                 shader.append(gFS_Main_FetchTexture[modulateOp]);
703             }
704         } else {
705             if (!description.hasGradient && !description.hasBitmap) {
706                 shader.append(gFS_Main_FetchColor);
707             }
708         }
709         if (description.hasGradient) {
710             shader.append(gFS_Main_FetchGradient[gradientIndex(description)]);
711             shader.appendFormat(gFS_Main_AddDitherToGradient, gFS_Main_Dither[mHasES3]);
712         }
713         if (description.hasBitmap) {
714             if (!description.isBitmapNpot) {
715                 shader.append(gFS_Main_FetchBitmap);
716             } else {
717                 shader.append(gFS_Main_FetchBitmapNpot);
718             }
719         }
720         bool applyModulate = false;
721         // Case when we have two shaders set
722         if (description.hasGradient && description.hasBitmap) {
723             if (description.isBitmapFirst) {
724                 shader.append(gFS_Main_BlendShadersBG);
725             } else {
726                 shader.append(gFS_Main_BlendShadersGB);
727             }
728             applyModulate = shaderOp(description, shader, modulateOp,
729                     gFS_Main_BlendShaders_Modulate);
730         } else {
731             if (description.hasGradient) {
732                 applyModulate = shaderOp(description, shader, modulateOp,
733                         gFS_Main_GradientShader_Modulate);
734             } else if (description.hasBitmap) {
735                 applyModulate = shaderOp(description, shader, modulateOp,
736                         gFS_Main_BitmapShader_Modulate);
737             }
738         }
739 
740         if (description.modulate && applyModulate) {
741             shader.append(gFS_Main_ModulateColor);
742         }
743 
744         // Apply the color op if needed
745         shader.append(gFS_Main_ApplyColorOp[description.colorOp]);
746 
747         if (description.hasVertexAlpha) {
748             if (description.useShadowAlphaInterp) {
749                 shader.append(gFS_Main_ApplyVertexAlphaShadowInterp);
750             } else {
751                 shader.append(gFS_Main_ApplyVertexAlphaLinearInterp);
752             }
753         }
754 
755         // Output the fragment
756         if (!blendFramebuffer) {
757             shader.append(gFS_Main_FragColor);
758         } else {
759             shader.append(!description.swapSrcDst ?
760                     gFS_Main_FragColor_Blend : gFS_Main_FragColor_Blend_Swap);
761         }
762         if (description.hasColors) {
763             shader.append(gFS_Main_FragColor_HasColors);
764         }
765         if (description.hasRoundRectClip) {
766             shader.append(gFS_Main_FragColor_HasRoundRectClip);
767         }
768         if (description.hasDebugHighlight) {
769             shader.append(gFS_Main_DebugHighlight);
770         }
771     }
772     // End the shader
773     shader.append(gFS_Footer);
774 
775 #if DEBUG_PROGRAMS
776         PROGRAM_LOGD("*** Generated fragment shader:\n\n");
777         printLongString(shader);
778 #endif
779 
780     return shader;
781 }
782 
generateBlend(String8 & shader,const char * name,SkXfermode::Mode mode)783 void ProgramCache::generateBlend(String8& shader, const char* name, SkXfermode::Mode mode) {
784     shader.append("\nvec4 ");
785     shader.append(name);
786     shader.append("(vec4 src, vec4 dst) {\n");
787     shader.append("    ");
788     shader.append(gBlendOps[mode]);
789     shader.append("}\n");
790 }
791 
generateTextureWrap(String8 & shader,GLenum wrapS,GLenum wrapT)792 void ProgramCache::generateTextureWrap(String8& shader, GLenum wrapS, GLenum wrapT) {
793     shader.append("\nhighp vec2 wrap(highp vec2 texCoords) {\n");
794     if (wrapS == GL_MIRRORED_REPEAT) {
795         shader.append("    highp float xMod2 = mod(texCoords.x, 2.0);\n");
796         shader.append("    if (xMod2 > 1.0) xMod2 = 2.0 - xMod2;\n");
797     }
798     if (wrapT == GL_MIRRORED_REPEAT) {
799         shader.append("    highp float yMod2 = mod(texCoords.y, 2.0);\n");
800         shader.append("    if (yMod2 > 1.0) yMod2 = 2.0 - yMod2;\n");
801     }
802     shader.append("    return vec2(");
803     switch (wrapS) {
804         case GL_CLAMP_TO_EDGE:
805             shader.append("texCoords.x");
806             break;
807         case GL_REPEAT:
808             shader.append("mod(texCoords.x, 1.0)");
809             break;
810         case GL_MIRRORED_REPEAT:
811             shader.append("xMod2");
812             break;
813     }
814     shader.append(", ");
815     switch (wrapT) {
816         case GL_CLAMP_TO_EDGE:
817             shader.append("texCoords.y");
818             break;
819         case GL_REPEAT:
820             shader.append("mod(texCoords.y, 1.0)");
821             break;
822         case GL_MIRRORED_REPEAT:
823             shader.append("yMod2");
824             break;
825     }
826     shader.append(");\n");
827     shader.append("}\n");
828 }
829 
printLongString(const String8 & shader) const830 void ProgramCache::printLongString(const String8& shader) const {
831     ssize_t index = 0;
832     ssize_t lastIndex = 0;
833     const char* str = shader.string();
834     while ((index = shader.find("\n", index)) > -1) {
835         String8 line(str, index - lastIndex);
836         if (line.length() == 0) line.append("\n");
837         PROGRAM_LOGD("%s", line.string());
838         index++;
839         str += (index - lastIndex);
840         lastIndex = index;
841     }
842 }
843 
844 }; // namespace uirenderer
845 }; // namespace android
846