blob: da67f92e78466f1df8f2e4313b564552ce62546a [file] [log] [blame]
Mathias Agopian3f844832013-08-07 21:24:32 -07001/*Gluint
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
Peiyong Lin833074a2018-08-28 11:53:54 -070017#include "Program.h"
Peiyong Lincbc184f2018-08-22 13:24:10 -070018
Mathias Agopian3f844832013-08-07 21:24:32 -070019#include <stdint.h>
20
Mark Salyzyn7823e122016-09-29 08:08:05 -070021#include <log/log.h>
Chia-I Wub027f802017-11-29 14:00:52 -080022#include <math/mat4.h>
Peiyong Lincbc184f2018-08-22 13:24:10 -070023#include <utils/String8.h>
Peiyong Lin833074a2018-08-28 11:53:54 -070024#include "ProgramCache.h"
Mathias Agopian3f844832013-08-07 21:24:32 -070025
26namespace android {
Peiyong Lin833074a2018-08-28 11:53:54 -070027namespace renderengine {
28namespace gl {
Mathias Agopian3f844832013-08-07 21:24:32 -070029
Mark Salyzyn92dc3fc2014-03-12 13:12:44 -070030Program::Program(const ProgramCache::Key& /*needs*/, const char* vertex, const char* fragment)
Chia-I Wub027f802017-11-29 14:00:52 -080031 : mInitialized(false) {
Mathias Agopian3f844832013-08-07 21:24:32 -070032 GLuint vertexId = buildShader(vertex, GL_VERTEX_SHADER);
33 GLuint fragmentId = buildShader(fragment, GL_FRAGMENT_SHADER);
34 GLuint programId = glCreateProgram();
35 glAttachShader(programId, vertexId);
36 glAttachShader(programId, fragmentId);
37 glBindAttribLocation(programId, position, "position");
38 glBindAttribLocation(programId, texCoords, "texCoords");
39 glLinkProgram(programId);
40
41 GLint status;
42 glGetProgramiv(programId, GL_LINK_STATUS, &status);
43 if (status != GL_TRUE) {
44 ALOGE("Error while linking shaders:");
45 GLint infoLen = 0;
46 glGetProgramiv(programId, GL_INFO_LOG_LENGTH, &infoLen);
47 if (infoLen > 1) {
48 GLchar log[infoLen];
49 glGetProgramInfoLog(programId, infoLen, 0, &log[0]);
50 ALOGE("%s", log);
51 }
52 glDetachShader(programId, vertexId);
53 glDetachShader(programId, fragmentId);
54 glDeleteShader(vertexId);
55 glDeleteShader(fragmentId);
56 glDeleteProgram(programId);
57 } else {
58 mProgram = programId;
59 mVertexShader = vertexId;
60 mFragmentShader = fragmentId;
61 mInitialized = true;
Mathias Agopian3f844832013-08-07 21:24:32 -070062 mProjectionMatrixLoc = glGetUniformLocation(programId, "projection");
63 mTextureMatrixLoc = glGetUniformLocation(programId, "texture");
64 mSamplerLoc = glGetUniformLocation(programId, "sampler");
65 mColorLoc = glGetUniformLocation(programId, "color");
Peiyong Linfb069302018-04-25 14:34:31 -070066 mDisplayMaxLuminanceLoc = glGetUniformLocation(programId, "displayMaxLuminance");
Peiyong Lina296b0c2018-04-30 16:55:29 -070067 mInputTransformMatrixLoc = glGetUniformLocation(programId, "inputTransformMatrix");
68 mOutputTransformMatrixLoc = glGetUniformLocation(programId, "outputTransformMatrix");
Mathias Agopian3f844832013-08-07 21:24:32 -070069
70 // set-up the default values for our uniforms
71 glUseProgram(programId);
Chia-I Wub027f802017-11-29 14:00:52 -080072 glUniformMatrix4fv(mProjectionMatrixLoc, 1, GL_FALSE, mat4().asArray());
Mathias Agopian3f844832013-08-07 21:24:32 -070073 glEnableVertexAttribArray(0);
74 }
75}
76
Chia-I Wub027f802017-11-29 14:00:52 -080077Program::~Program() {}
Mathias Agopian3f844832013-08-07 21:24:32 -070078
79bool Program::isValid() const {
80 return mInitialized;
81}
82
83void Program::use() {
84 glUseProgram(mProgram);
85}
86
87GLuint Program::getAttrib(const char* name) const {
88 // TODO: maybe use a local cache
89 return glGetAttribLocation(mProgram, name);
90}
91
92GLint Program::getUniform(const char* name) const {
93 // TODO: maybe use a local cache
94 return glGetUniformLocation(mProgram, name);
95}
96
97GLuint Program::buildShader(const char* source, GLenum type) {
98 GLuint shader = glCreateShader(type);
99 glShaderSource(shader, 1, &source, 0);
100 glCompileShader(shader);
101 GLint status;
102 glGetShaderiv(shader, GL_COMPILE_STATUS, &status);
103 if (status != GL_TRUE) {
104 // Some drivers return wrong values for GL_INFO_LOG_LENGTH
105 // use a fixed size instead
106 GLchar log[512];
107 glGetShaderInfoLog(shader, sizeof(log), 0, log);
108 ALOGE("Error while compiling shader: \n%s\n%s", source, log);
109 glDeleteShader(shader);
110 return 0;
111 }
112 return shader;
113}
114
Mark Salyzyn92dc3fc2014-03-12 13:12:44 -0700115String8& Program::dumpShader(String8& result, GLenum /*type*/) {
Mathias Agopian3f844832013-08-07 21:24:32 -0700116 GLuint shader = GL_FRAGMENT_SHADER ? mFragmentShader : mVertexShader;
117 GLint l;
118 glGetShaderiv(shader, GL_SHADER_SOURCE_LENGTH, &l);
119 char* src = new char[l];
Peiyong Lin566a3b42018-01-09 18:22:43 -0800120 glGetShaderSource(shader, l, nullptr, src);
Mathias Agopian3f844832013-08-07 21:24:32 -0700121 result.append(src);
Chia-I Wub027f802017-11-29 14:00:52 -0800122 delete[] src;
Mathias Agopian3f844832013-08-07 21:24:32 -0700123 return result;
124}
125
126void Program::setUniforms(const Description& desc) {
Mathias Agopian3f844832013-08-07 21:24:32 -0700127 // TODO: we should have a mechanism here to not always reset uniforms that
128 // didn't change for this program.
129
130 if (mSamplerLoc >= 0) {
131 glUniform1i(mSamplerLoc, 0);
Peiyong Lin70b26ce2018-09-18 19:02:39 -0700132 glUniformMatrix4fv(mTextureMatrixLoc, 1, GL_FALSE, desc.texture.getMatrix().asArray());
Mathias Agopian3f844832013-08-07 21:24:32 -0700133 }
Mathias Agopian3f844832013-08-07 21:24:32 -0700134 if (mColorLoc >= 0) {
Peiyong Lin70b26ce2018-09-18 19:02:39 -0700135 const float color[4] = {desc.color.r, desc.color.g, desc.color.b, desc.color.a};
chaviw13fdc492017-06-27 12:40:18 -0700136 glUniform4fv(mColorLoc, 1, color);
Mathias Agopian3f844832013-08-07 21:24:32 -0700137 }
Peiyong Lina296b0c2018-04-30 16:55:29 -0700138 if (mInputTransformMatrixLoc >= 0) {
Peiyong Lin70b26ce2018-09-18 19:02:39 -0700139 mat4 inputTransformMatrix = desc.inputTransformMatrix;
Peiyong Lin00f9c022018-05-09 15:35:33 -0700140 glUniformMatrix4fv(mInputTransformMatrixLoc, 1, GL_FALSE, inputTransformMatrix.asArray());
Peiyong Lina296b0c2018-04-30 16:55:29 -0700141 }
142 if (mOutputTransformMatrixLoc >= 0) {
143 // The output transform matrix and color matrix can be combined as one matrix
144 // that is applied right before applying OETF.
Peiyong Lin70b26ce2018-09-18 19:02:39 -0700145 mat4 outputTransformMatrix = desc.colorMatrix * desc.outputTransformMatrix;
Peiyong Lina296b0c2018-04-30 16:55:29 -0700146 glUniformMatrix4fv(mOutputTransformMatrixLoc, 1, GL_FALSE,
147 outputTransformMatrix.asArray());
Mathias Agopianff2ed702013-09-01 21:36:12 -0700148 }
Peiyong Linfb069302018-04-25 14:34:31 -0700149 if (mDisplayMaxLuminanceLoc >= 0) {
Peiyong Lin70b26ce2018-09-18 19:02:39 -0700150 glUniform1f(mDisplayMaxLuminanceLoc, desc.displayMaxLuminance) ;
Peiyong Linfb069302018-04-25 14:34:31 -0700151 }
Mathias Agopian3f844832013-08-07 21:24:32 -0700152 // these uniforms are always present
Peiyong Lin70b26ce2018-09-18 19:02:39 -0700153 glUniformMatrix4fv(mProjectionMatrixLoc, 1, GL_FALSE, desc.projectionMatrix.asArray());
Mathias Agopian3f844832013-08-07 21:24:32 -0700154}
155
Peiyong Lin833074a2018-08-28 11:53:54 -0700156} // namespace gl
157} // namespace renderengine
158} // namespace android