blob: a2a80d9f37f19ceb2ce3de8aa3f87a1874dbf303 [file] [log] [blame]
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00001//
Geoff Lang1b6edcb2014-02-03 14:27:56 -05002// Copyright (c) 2002-2014 The ANGLE Project Authors. All rights reserved.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
7// Program.h: Defines the gl::Program class. Implements GL program objects
8// and related functionality. [OpenGL ES 2.0.24] section 2.10.3 page 28.
9
Geoff Lang0a73dd82014-11-19 16:18:08 -050010#ifndef LIBANGLE_PROGRAM_H_
11#define LIBANGLE_PROGRAM_H_
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000012
Geoff Lang0b7eef72014-06-12 14:10:47 -040013#include <GLES2/gl2.h>
Jamie Madill861ebb32016-11-17 16:40:22 -050014#include <GLSLANG/ShaderVars.h>
Geoff Lang0b7eef72014-06-12 14:10:47 -040015
Jamie Madilla7d12dc2016-12-13 15:08:19 -050016#include <array>
Jamie Madill861ebb32016-11-17 16:40:22 -050017#include <map>
Jamie Madill63805b42015-08-25 13:17:39 -040018#include <set>
Jamie Madill71c3b2c2015-05-07 11:49:20 -040019#include <sstream>
Geoff Lang0b7eef72014-06-12 14:10:47 -040020#include <string>
Jamie Madill63805b42015-08-25 13:17:39 -040021#include <vector>
Geoff Lang0b7eef72014-06-12 14:10:47 -040022
Jamie Madill3d3d2f22015-09-23 16:47:51 -040023#include "common/angleutils.h"
24#include "common/mathutil.h"
25#include "common/Optional.h"
26
Jamie Madill3d3d2f22015-09-23 16:47:51 -040027#include "libANGLE/Constants.h"
Geoff Lang70d0f492015-12-10 17:45:46 -050028#include "libANGLE/Debug.h"
Jamie Madill3d3d2f22015-09-23 16:47:51 -040029#include "libANGLE/Error.h"
30#include "libANGLE/RefCountObject.h"
jchen107a20b972017-06-13 14:25:26 +080031#include "libANGLE/Uniform.h"
32#include "libANGLE/angletypes.h"
Jamie Madill3d3d2f22015-09-23 16:47:51 -040033
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000034namespace rx
35{
Jamie Madill7aea7e02016-05-10 10:39:45 -040036class GLImplFactory;
Geoff Lang7dd2e102014-11-10 15:19:26 -050037class ProgramImpl;
Jamie Madill5c6b7bf2015-08-17 12:53:35 -040038struct TranslatedAttribute;
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000039}
daniel@transgaming.come6842292010-04-20 18:52:50 +000040
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000041namespace gl
42{
Brandon Jones43a53e22014-08-28 16:23:22 -070043struct Caps;
Jamie Madilla2c74982016-12-12 11:20:42 -050044class Context;
Jamie Madilldfde6ab2016-06-09 07:07:18 -070045class ContextState;
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000046class Shader;
Geoff Lang4ddf5af2016-12-01 14:30:44 -050047class ShaderProgramManager;
Jamie Madillf4f8db82017-02-15 09:31:39 -050048class State;
Geoff Lang7dd2e102014-11-10 15:19:26 -050049class InfoLog;
Geoff Lang7dd2e102014-11-10 15:19:26 -050050class Buffer;
51class Framebuffer;
Jamie Madill192745a2016-12-22 15:58:21 -050052struct PackedVarying;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000053
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000054extern const char * const g_fakepath;
55
Jamie Madillf0d10f82015-03-31 12:56:52 -040056class InfoLog : angle::NonCopyable
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000057{
58 public:
59 InfoLog();
60 ~InfoLog();
61
Jamie Madill71c3b2c2015-05-07 11:49:20 -040062 size_t getLength() const;
Geoff Lange1a27752015-10-05 13:16:04 -040063 void getLog(GLsizei bufSize, GLsizei *length, char *infoLog) const;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000064
65 void appendSanitized(const char *message);
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000066 void reset();
Jamie Madillf6113162015-05-07 11:49:21 -040067
68 // This helper class ensures we append a newline after writing a line.
69 class StreamHelper : angle::NonCopyable
70 {
71 public:
72 StreamHelper(StreamHelper &&rhs)
73 : mStream(rhs.mStream)
74 {
75 rhs.mStream = nullptr;
76 }
77
78 StreamHelper &operator=(StreamHelper &&rhs)
79 {
80 std::swap(mStream, rhs.mStream);
81 return *this;
82 }
83
84 ~StreamHelper()
85 {
86 // Write newline when destroyed on the stack
87 if (mStream)
88 {
89 (*mStream) << std::endl;
90 }
91 }
92
93 template <typename T>
94 StreamHelper &operator<<(const T &value)
95 {
96 (*mStream) << value;
97 return *this;
98 }
99
100 private:
101 friend class InfoLog;
102
103 StreamHelper(std::stringstream *stream)
104 : mStream(stream)
105 {
106 ASSERT(stream);
107 }
108
109 std::stringstream *mStream;
110 };
111
112 template <typename T>
113 StreamHelper operator<<(const T &value)
114 {
Jamie Madill23176ce2017-07-31 14:14:33 -0400115 ensureInitialized();
116 StreamHelper helper(mLazyStream.get());
Jamie Madillf6113162015-05-07 11:49:21 -0400117 helper << value;
118 return helper;
119 }
120
Jamie Madill23176ce2017-07-31 14:14:33 -0400121 std::string str() const { return mLazyStream ? mLazyStream->str() : ""; }
Jamie Madillf6113162015-05-07 11:49:21 -0400122
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000123 private:
Jamie Madill23176ce2017-07-31 14:14:33 -0400124 void ensureInitialized()
125 {
126 if (!mLazyStream)
127 {
128 mLazyStream.reset(new std::stringstream());
129 }
130 }
131
132 std::unique_ptr<std::stringstream> mLazyStream;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000133};
134
Geoff Lang7dd2e102014-11-10 15:19:26 -0500135// Struct used for correlating uniforms/elements of uniform arrays to handles
136struct VariableLocation
137{
Jamie Madillfb997ec2017-09-20 15:44:27 -0400138 static constexpr unsigned int kUnused = GL_INVALID_INDEX;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500139
Jamie Madillfb997ec2017-09-20 15:44:27 -0400140 VariableLocation();
Olli Etuahoc8538042017-09-27 11:20:15 +0300141 VariableLocation(unsigned int arrayIndex, unsigned int index);
Geoff Langd8605522016-04-13 10:19:12 -0400142
Jamie Madillfe8b5982017-09-07 17:00:18 -0400143 // If used is false, it means this location is only used to fill an empty space in an array,
144 // and there is no corresponding uniform variable for this location. It can also mean the
145 // uniform was optimized out by the implementation.
Jamie Madillfb997ec2017-09-20 15:44:27 -0400146 bool used() const { return (index != kUnused); }
147 void markUnused() { index = kUnused; }
148 void markIgnored() { ignored = true; }
149
Olli Etuahoc8538042017-09-27 11:20:15 +0300150 bool areAllArrayIndicesZero() const;
151
152 // The "arrayIndices" vector stores indices for the GLSL array. "index" is an index of the
153 // location.
154 std::vector<unsigned int> arrayIndices; // Outermost array indices are in the back.
Jamie Madillfb997ec2017-09-20 15:44:27 -0400155 unsigned int index;
Geoff Langd8605522016-04-13 10:19:12 -0400156
Olli Etuahoc8538042017-09-27 11:20:15 +0300157 unsigned int flattenedArrayOffset; // For non-nested arrays this is the same as the array
158 // index. For arrays of arrays, the indices are converted to
159 // a single offset inside a one-dimensional array made up of
160 // the elements of the innermost arrays.
161
Geoff Langd8605522016-04-13 10:19:12 -0400162 // If this location was bound to an unreferenced uniform. Setting data on this uniform is a
163 // no-op.
164 bool ignored;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500165};
166
Sami Väisänen46eaa942016-06-29 10:26:37 +0300167// Information about a variable binding.
168// Currently used by CHROMIUM_path_rendering
169struct BindingInfo
170{
171 // The type of binding, for example GL_FLOAT_VEC3.
172 // This can be GL_NONE if the variable is optimized away.
173 GLenum type;
174
175 // This is the name of the variable in
176 // the translated shader program. Note that
177 // this can be empty in the case where the
178 // variable has been optimized away.
179 std::string name;
180
181 // True if the binding is valid, otherwise false.
182 bool valid;
183};
184
Jamie Madille7d84322017-01-10 18:21:59 -0500185// This small structure encapsulates binding sampler uniforms to active GL textures.
186struct SamplerBinding
187{
Jamie Madill54164b02017-08-28 15:17:37 -0400188 SamplerBinding(GLenum textureTypeIn, size_t elementCount, bool unreferenced)
189 : textureType(textureTypeIn), boundTextureUnits(elementCount, 0), unreferenced(unreferenced)
Jamie Madille7d84322017-01-10 18:21:59 -0500190 {
191 }
192
193 // Necessary for retrieving active textures from the GL state.
194 GLenum textureType;
195
196 // List of all textures bound to this sampler, of type textureType.
197 std::vector<GLuint> boundTextureUnits;
Jamie Madill54164b02017-08-28 15:17:37 -0400198
199 // A note if this sampler is an unreferenced uniform.
200 bool unreferenced;
Jamie Madille7d84322017-01-10 18:21:59 -0500201};
202
jchen10a9042d32017-03-17 08:50:45 +0800203// A varying with tranform feedback enabled. If it's an array, either the whole array or one of its
204// elements specified by 'arrayIndex' can set to be enabled.
205struct TransformFeedbackVarying : public sh::Varying
206{
207 TransformFeedbackVarying(const sh::Varying &varyingIn, GLuint index)
208 : sh::Varying(varyingIn), arrayIndex(index)
209 {
210 }
211 std::string nameWithArrayIndex() const
212 {
213 std::stringstream fullNameStr;
214 fullNameStr << name;
215 if (arrayIndex != GL_INVALID_INDEX)
216 {
217 fullNameStr << "[" << arrayIndex << "]";
218 }
219 return fullNameStr.str();
220 }
221 GLsizei size() const { return (arrayIndex == GL_INVALID_INDEX ? elementCount() : 1); }
222
223 GLuint arrayIndex;
224};
225
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800226struct ImageBinding
227{
Xinghua Cao0328b572017-06-26 15:51:36 +0800228 ImageBinding(size_t count) : boundImageUnits(count, 0) {}
229 ImageBinding(GLuint imageUnit, size_t count)
230 {
231 for (size_t index = 0; index < count; ++index)
232 {
233 boundImageUnits.push_back(imageUnit + static_cast<GLuint>(index));
234 }
235 }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800236
Xinghua Cao0328b572017-06-26 15:51:36 +0800237 std::vector<GLuint> boundImageUnits;
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800238};
239
Jamie Madill48ef11b2016-04-27 15:21:52 -0400240class ProgramState final : angle::NonCopyable
241{
242 public:
243 ProgramState();
244 ~ProgramState();
245
246 const std::string &getLabel();
247
Jamie Madillbd044ed2017-06-05 12:59:21 -0400248 Shader *getAttachedVertexShader() const { return mAttachedVertexShader; }
249 Shader *getAttachedFragmentShader() const { return mAttachedFragmentShader; }
250 Shader *getAttachedComputeShader() const { return mAttachedComputeShader; }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400251 const std::vector<std::string> &getTransformFeedbackVaryingNames() const
252 {
253 return mTransformFeedbackVaryingNames;
254 }
255 GLint getTransformFeedbackBufferMode() const { return mTransformFeedbackBufferMode; }
256 GLuint getUniformBlockBinding(GLuint uniformBlockIndex) const
257 {
jchen107a20b972017-06-13 14:25:26 +0800258 ASSERT(uniformBlockIndex < mUniformBlocks.size());
259 return mUniformBlocks[uniformBlockIndex].binding;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400260 }
Jiajia Qin729b2c62017-08-14 09:36:11 +0800261 GLuint getShaderStorageBlockBinding(GLuint blockIndex) const
262 {
263 ASSERT(blockIndex < mShaderStorageBlocks.size());
264 return mShaderStorageBlocks[blockIndex].binding;
265 }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400266 const UniformBlockBindingMask &getActiveUniformBlockBindingsMask() const
267 {
268 return mActiveUniformBlockBindings;
269 }
270 const std::vector<sh::Attribute> &getAttributes() const { return mAttributes; }
271 const AttributesMask &getActiveAttribLocationsMask() const
272 {
273 return mActiveAttribLocationsMask;
274 }
Corentin Walleze7557742017-06-01 13:09:57 -0400275 DrawBufferMask getActiveOutputVariables() const { return mActiveOutputVariables; }
Jamie Madill4f86d052017-06-05 12:59:26 -0400276 const std::vector<sh::OutputVariable> &getOutputVariables() const { return mOutputVariables; }
jchen1015015f72017-03-16 13:54:21 +0800277 const std::map<int, VariableLocation> &getOutputLocations() const { return mOutputLocations; }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400278 const std::vector<LinkedUniform> &getUniforms() const { return mUniforms; }
279 const std::vector<VariableLocation> &getUniformLocations() const { return mUniformLocations; }
Jiajia Qin729b2c62017-08-14 09:36:11 +0800280 const std::vector<InterfaceBlock> &getUniformBlocks() const { return mUniformBlocks; }
281 const std::vector<InterfaceBlock> &getShaderStorageBlocks() const
282 {
283 return mShaderStorageBlocks;
284 }
Jamie Madille7d84322017-01-10 18:21:59 -0500285 const std::vector<SamplerBinding> &getSamplerBindings() const { return mSamplerBindings; }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800286 const std::vector<ImageBinding> &getImageBindings() const { return mImageBindings; }
Jamie Madill4f86d052017-06-05 12:59:26 -0400287 const sh::WorkGroupSize &getComputeShaderLocalSize() const { return mComputeShaderLocalSize; }
288 const RangeUI &getSamplerUniformRange() const { return mSamplerUniformRange; }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800289 const RangeUI &getImageUniformRange() const { return mImageUniformRange; }
jchen10eaef1e52017-06-13 10:44:11 +0800290 const RangeUI &getAtomicCounterUniformRange() const { return mAtomicCounterUniformRange; }
Jamie Madill4f86d052017-06-05 12:59:26 -0400291
Jamie Madill4f86d052017-06-05 12:59:26 -0400292 const std::vector<TransformFeedbackVarying> &getLinkedTransformFeedbackVaryings() const
293 {
294 return mLinkedTransformFeedbackVaryings;
295 }
jchen10eaef1e52017-06-13 10:44:11 +0800296 const std::vector<AtomicCounterBuffer> &getAtomicCounterBuffers() const
297 {
298 return mAtomicCounterBuffers;
299 }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400300
Jamie Madill48ef11b2016-04-27 15:21:52 -0400301 GLint getUniformLocation(const std::string &name) const;
Jamie Madille7d84322017-01-10 18:21:59 -0500302 GLuint getUniformIndexFromName(const std::string &name) const;
303 GLuint getUniformIndexFromLocation(GLint location) const;
304 Optional<GLuint> getSamplerIndex(GLint location) const;
305 bool isSamplerUniformIndex(GLuint index) const;
306 GLuint getSamplerIndexFromUniformIndex(GLuint uniformIndex) const;
Jamie Madill34ca4f52017-06-13 11:49:39 -0400307 GLuint getAttributeLocation(const std::string &name) const;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400308
Martin Radev4e619f52017-08-09 11:50:06 +0300309 int getNumViews() const { return mNumViews; }
310 bool usesMultiview() const { return mNumViews != -1; }
311
Jamie Madill48ef11b2016-04-27 15:21:52 -0400312 private:
Jamie Madill4f86d052017-06-05 12:59:26 -0400313 friend class MemoryProgramCache;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400314 friend class Program;
315
316 std::string mLabel;
317
Martin Radev4c4c8e72016-08-04 12:25:34 +0300318 sh::WorkGroupSize mComputeShaderLocalSize;
319
Jamie Madill48ef11b2016-04-27 15:21:52 -0400320 Shader *mAttachedFragmentShader;
321 Shader *mAttachedVertexShader;
Martin Radev4c4c8e72016-08-04 12:25:34 +0300322 Shader *mAttachedComputeShader;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400323
324 std::vector<std::string> mTransformFeedbackVaryingNames;
jchen10a9042d32017-03-17 08:50:45 +0800325 std::vector<TransformFeedbackVarying> mLinkedTransformFeedbackVaryings;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400326 GLenum mTransformFeedbackBufferMode;
327
jchen107a20b972017-06-13 14:25:26 +0800328 // For faster iteration on the blocks currently being bound.
Jamie Madill48ef11b2016-04-27 15:21:52 -0400329 UniformBlockBindingMask mActiveUniformBlockBindings;
330
331 std::vector<sh::Attribute> mAttributes;
Jamie Madill6de51852017-04-12 09:53:01 -0400332 angle::BitSet<MAX_VERTEX_ATTRIBS> mActiveAttribLocationsMask;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400333
334 // Uniforms are sorted in order:
jchen10eaef1e52017-06-13 10:44:11 +0800335 // 1. Non-opaque uniforms
Jamie Madill48ef11b2016-04-27 15:21:52 -0400336 // 2. Sampler uniforms
jchen10eaef1e52017-06-13 10:44:11 +0800337 // 3. Image uniforms
338 // 4. Atomic counter uniforms
339 // 5. Uniform block uniforms
340 // This makes opaque uniform validation easier, since we don't need a separate list.
Jamie Madill48ef11b2016-04-27 15:21:52 -0400341 std::vector<LinkedUniform> mUniforms;
342 std::vector<VariableLocation> mUniformLocations;
Jiajia Qin729b2c62017-08-14 09:36:11 +0800343 std::vector<InterfaceBlock> mUniformBlocks;
344 std::vector<InterfaceBlock> mShaderStorageBlocks;
jchen10eaef1e52017-06-13 10:44:11 +0800345 std::vector<AtomicCounterBuffer> mAtomicCounterBuffers;
Jamie Madille7d84322017-01-10 18:21:59 -0500346 RangeUI mSamplerUniformRange;
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800347 RangeUI mImageUniformRange;
jchen10eaef1e52017-06-13 10:44:11 +0800348 RangeUI mAtomicCounterUniformRange;
Jamie Madille7d84322017-01-10 18:21:59 -0500349
350 // An array of the samplers that are used by the program
351 std::vector<gl::SamplerBinding> mSamplerBindings;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400352
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800353 // An array of the images that are used by the program
354 std::vector<gl::ImageBinding> mImageBindings;
355
jchen1015015f72017-03-16 13:54:21 +0800356 std::vector<sh::OutputVariable> mOutputVariables;
jchen1015015f72017-03-16 13:54:21 +0800357 std::map<int, VariableLocation> mOutputLocations;
Corentin Walleze7557742017-06-01 13:09:57 -0400358 DrawBufferMask mActiveOutputVariables;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400359
Geoff Lange0cff192017-05-30 13:04:56 -0400360 // Fragment output variable base types: FLOAT, INT, or UINT. Ordered by location.
361 std::vector<GLenum> mOutputVariableTypes;
362
Jamie Madill48ef11b2016-04-27 15:21:52 -0400363 bool mBinaryRetrieveableHint;
Yunchao He61afff12017-03-14 15:34:03 +0800364 bool mSeparable;
Martin Radev7cf61662017-07-26 17:10:53 +0300365
366 // ANGLE_multiview.
367 int mNumViews;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400368};
369
Geoff Lang70d0f492015-12-10 17:45:46 -0500370class Program final : angle::NonCopyable, public LabeledObject
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000371{
372 public:
Geoff Lang4ddf5af2016-12-01 14:30:44 -0500373 Program(rx::GLImplFactory *factory, ShaderProgramManager *manager, GLuint handle);
Jamie Madill4928b7c2017-06-20 12:57:39 -0400374 void onDestroy(const Context *context);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000375
Geoff Lang7dd2e102014-11-10 15:19:26 -0500376 GLuint id() const { return mHandle; }
377
Geoff Lang70d0f492015-12-10 17:45:46 -0500378 void setLabel(const std::string &label) override;
379 const std::string &getLabel() const override;
380
Geoff Lang47110bf2016-04-20 11:13:22 -0700381 rx::ProgramImpl *getImplementation() const { return mProgram; }
Geoff Lang7dd2e102014-11-10 15:19:26 -0500382
Jamie Madillef300b12016-10-07 15:12:09 -0400383 void attachShader(Shader *shader);
Jamie Madillc1d770e2017-04-13 17:31:24 -0400384 void detachShader(const Context *context, Shader *shader);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000385 int getAttachedShadersCount() const;
386
Jamie Madillef300b12016-10-07 15:12:09 -0400387 const Shader *getAttachedVertexShader() const { return mState.mAttachedVertexShader; }
388 const Shader *getAttachedFragmentShader() const { return mState.mAttachedFragmentShader; }
389 const Shader *getAttachedComputeShader() const { return mState.mAttachedComputeShader; }
390
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000391 void bindAttributeLocation(GLuint index, const char *name);
Geoff Langd8605522016-04-13 10:19:12 -0400392 void bindUniformLocation(GLuint index, const char *name);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000393
Sami Väisänen46eaa942016-06-29 10:26:37 +0300394 // CHROMIUM_path_rendering
Jamie Madillbd044ed2017-06-05 12:59:21 -0400395 BindingInfo getFragmentInputBindingInfo(const Context *context, GLint index) const;
Sami Väisänen46eaa942016-06-29 10:26:37 +0300396 void bindFragmentInputLocation(GLint index, const char *name);
Jamie Madillbd044ed2017-06-05 12:59:21 -0400397 void pathFragmentInputGen(const Context *context,
398 GLint index,
Sami Väisänen46eaa942016-06-29 10:26:37 +0300399 GLenum genMode,
400 GLint components,
401 const GLfloat *coeffs);
402
Jamie Madill8ecf7f92017-01-13 17:29:52 -0500403 Error link(const gl::Context *context);
Geoff Lange1a27752015-10-05 13:16:04 -0400404 bool isLinked() const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500405
Jamie Madilla2c74982016-12-12 11:20:42 -0500406 Error loadBinary(const Context *context,
407 GLenum binaryFormat,
408 const void *binary,
409 GLsizei length);
410 Error saveBinary(const Context *context,
411 GLenum *binaryFormat,
412 void *binary,
413 GLsizei bufSize,
414 GLsizei *length) const;
Jamie Madillffe00c02017-06-27 16:26:55 -0400415 GLint getBinaryLength(const Context *context) const;
Geoff Langc5629752015-12-07 16:29:04 -0500416 void setBinaryRetrievableHint(bool retrievable);
417 bool getBinaryRetrievableHint() const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000418
Yunchao He61afff12017-03-14 15:34:03 +0800419 void setSeparable(bool separable);
420 bool isSeparable() const;
421
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000422 int getInfoLogLength() const;
Geoff Lange1a27752015-10-05 13:16:04 -0400423 void getInfoLog(GLsizei bufSize, GLsizei *length, char *infoLog) const;
424 void getAttachedShaders(GLsizei maxCount, GLsizei *count, GLuint *shaders) const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000425
Geoff Lange1a27752015-10-05 13:16:04 -0400426 GLuint getAttributeLocation(const std::string &name) const;
Jamie Madill63805b42015-08-25 13:17:39 -0400427 bool isAttribLocationActive(size_t attribLocation) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500428
jchen10fd7c3b52017-03-21 15:36:03 +0800429 void getActiveAttribute(GLuint index,
430 GLsizei bufsize,
431 GLsizei *length,
432 GLint *size,
433 GLenum *type,
434 GLchar *name) const;
Geoff Lange1a27752015-10-05 13:16:04 -0400435 GLint getActiveAttributeCount() const;
436 GLint getActiveAttributeMaxLength() const;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400437 const std::vector<sh::Attribute> &getAttributes() const { return mState.mAttributes; }
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000438
Geoff Lang7dd2e102014-11-10 15:19:26 -0500439 GLint getFragDataLocation(const std::string &name) const;
jchen10fd7c3b52017-03-21 15:36:03 +0800440 size_t getOutputResourceCount() const;
Geoff Lange0cff192017-05-30 13:04:56 -0400441 const std::vector<GLenum> &getOutputVariableTypes() const
442 {
443 return mState.mOutputVariableTypes;
444 }
Corentin Wallezdb9e5d32017-06-12 12:05:45 -0700445 DrawBufferMask getActiveOutputVariables() const { return mState.mActiveOutputVariables; }
Geoff Lang7dd2e102014-11-10 15:19:26 -0500446
Geoff Lange1a27752015-10-05 13:16:04 -0400447 void getActiveUniform(GLuint index,
448 GLsizei bufsize,
449 GLsizei *length,
450 GLint *size,
451 GLenum *type,
452 GLchar *name) const;
453 GLint getActiveUniformCount() const;
454 GLint getActiveUniformMaxLength() const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500455 bool isValidUniformLocation(GLint location) const;
Jamie Madill62d31cb2015-09-11 13:25:51 -0400456 const LinkedUniform &getUniformByLocation(GLint location) const;
Jamie Madillac4e9c32017-01-13 14:07:12 -0500457 const VariableLocation &getUniformLocation(GLint location) const;
458 const std::vector<VariableLocation> &getUniformLocations() const;
459 const LinkedUniform &getUniformByIndex(GLuint index) const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000460
Jamie Madill81c2e252017-09-09 23:32:46 -0400461 enum SetUniformResult
462 {
463 SamplerChanged,
464 NoSamplerChange,
465 };
466
Jamie Madill62d31cb2015-09-11 13:25:51 -0400467 GLint getUniformLocation(const std::string &name) const;
468 GLuint getUniformIndex(const std::string &name) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500469 void setUniform1fv(GLint location, GLsizei count, const GLfloat *v);
470 void setUniform2fv(GLint location, GLsizei count, const GLfloat *v);
471 void setUniform3fv(GLint location, GLsizei count, const GLfloat *v);
472 void setUniform4fv(GLint location, GLsizei count, const GLfloat *v);
Jamie Madill81c2e252017-09-09 23:32:46 -0400473 SetUniformResult setUniform1iv(GLint location, GLsizei count, const GLint *v);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500474 void setUniform2iv(GLint location, GLsizei count, const GLint *v);
475 void setUniform3iv(GLint location, GLsizei count, const GLint *v);
476 void setUniform4iv(GLint location, GLsizei count, const GLint *v);
477 void setUniform1uiv(GLint location, GLsizei count, const GLuint *v);
478 void setUniform2uiv(GLint location, GLsizei count, const GLuint *v);
479 void setUniform3uiv(GLint location, GLsizei count, const GLuint *v);
480 void setUniform4uiv(GLint location, GLsizei count, const GLuint *v);
481 void setUniformMatrix2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
482 void setUniformMatrix3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
483 void setUniformMatrix4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
484 void setUniformMatrix2x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
485 void setUniformMatrix3x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
486 void setUniformMatrix2x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
487 void setUniformMatrix4x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
488 void setUniformMatrix3x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
489 void setUniformMatrix4x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
490
Jamie Madill54164b02017-08-28 15:17:37 -0400491 void getUniformfv(const Context *context, GLint location, GLfloat *params) const;
492 void getUniformiv(const Context *context, GLint location, GLint *params) const;
493 void getUniformuiv(const Context *context, GLint location, GLuint *params) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500494
Geoff Lang7dd2e102014-11-10 15:19:26 -0500495 void getActiveUniformBlockName(GLuint uniformBlockIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformBlockName) const;
Geoff Lange1a27752015-10-05 13:16:04 -0400496 GLuint getActiveUniformBlockCount() const;
Jiajia Qin729b2c62017-08-14 09:36:11 +0800497 GLuint getActiveShaderStorageBlockCount() const;
Geoff Lange1a27752015-10-05 13:16:04 -0400498 GLint getActiveUniformBlockMaxLength() const;
shannonwoods@chromium.orge684b582013-05-30 00:07:42 +0000499
Geoff Lange1a27752015-10-05 13:16:04 -0400500 GLuint getUniformBlockIndex(const std::string &name) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500501
shannonwoods@chromium.org70eb1ea2013-05-30 00:07:20 +0000502 void bindUniformBlock(GLuint uniformBlockIndex, GLuint uniformBlockBinding);
503 GLuint getUniformBlockBinding(GLuint uniformBlockIndex) const;
Jiajia Qin729b2c62017-08-14 09:36:11 +0800504 GLuint getShaderStorageBlockBinding(GLuint shaderStorageBlockIndex) const;
shannonwoods@chromium.org70eb1ea2013-05-30 00:07:20 +0000505
Jiajia Qin729b2c62017-08-14 09:36:11 +0800506 const InterfaceBlock &getUniformBlockByIndex(GLuint index) const;
Gregoire Payen de La Garanderie68694e92015-03-24 14:03:37 +0000507
Geoff Lang48dcae72014-02-05 16:28:24 -0500508 void setTransformFeedbackVaryings(GLsizei count, const GLchar *const *varyings, GLenum bufferMode);
509 void getTransformFeedbackVarying(GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name) const;
Geoff Lang1b6edcb2014-02-03 14:27:56 -0500510 GLsizei getTransformFeedbackVaryingCount() const;
511 GLsizei getTransformFeedbackVaryingMaxLength() const;
512 GLenum getTransformFeedbackBufferMode() const;
513
Frank Henigmanfccbac22017-05-28 17:29:26 -0400514 static bool linkValidateInterfaceBlockFields(InfoLog &infoLog,
515 const std::string &uniformName,
516 const sh::InterfaceBlockField &vertexUniform,
517 const sh::InterfaceBlockField &fragmentUniform,
518 bool webglCompatibility);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500519
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000520 void addRef();
Jamie Madill6c1f6712017-02-14 19:08:04 -0500521 void release(const Context *context);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000522 unsigned int getRefCount() const;
523 void flagForDeletion();
524 bool isFlaggedForDeletion() const;
525
Brandon Jones43a53e22014-08-28 16:23:22 -0700526 void validate(const Caps &caps);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500527 bool validateSamplers(InfoLog *infoLog, const Caps &caps);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000528 bool isValidated() const;
Jamie Madilla4595b82017-01-11 17:36:34 -0500529 bool samplesFromTexture(const gl::State &state, GLuint textureID) const;
apatrick@chromium.org90080e32012-07-09 22:15:33 +0000530
Jamie Madill63805b42015-08-25 13:17:39 -0400531 const AttributesMask &getActiveAttribLocationsMask() const
532 {
Jamie Madill48ef11b2016-04-27 15:21:52 -0400533 return mState.mActiveAttribLocationsMask;
Jamie Madill63805b42015-08-25 13:17:39 -0400534 }
535
Jamie Madille7d84322017-01-10 18:21:59 -0500536 const std::vector<SamplerBinding> &getSamplerBindings() const
537 {
538 return mState.mSamplerBindings;
539 }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800540
541 const std::vector<ImageBinding> &getImageBindings() const { return mState.mImageBindings; }
542
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500543 const ProgramState &getState() const { return mState; }
Jamie Madille7d84322017-01-10 18:21:59 -0500544
Olli Etuahob78707c2017-03-09 15:03:11 +0000545 static bool linkValidateVariablesBase(InfoLog &infoLog,
546 const std::string &variableName,
547 const sh::ShaderVariable &vertexVariable,
548 const sh::ShaderVariable &fragmentVariable,
549 bool validatePrecision);
550
jchen1015015f72017-03-16 13:54:21 +0800551 GLuint getInputResourceIndex(const GLchar *name) const;
552 GLuint getOutputResourceIndex(const GLchar *name) const;
jchen10fd7c3b52017-03-21 15:36:03 +0800553 void getInputResourceName(GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name) const;
554 void getOutputResourceName(GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name) const;
jchen10baf5d942017-08-28 20:45:48 +0800555 void getUniformResourceName(GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name) const;
jchen10880683b2017-04-12 16:21:55 +0800556 const sh::Attribute &getInputResource(GLuint index) const;
557 const sh::OutputVariable &getOutputResource(GLuint index) const;
jchen1015015f72017-03-16 13:54:21 +0800558
Geoff Langd8605522016-04-13 10:19:12 -0400559 class Bindings final : angle::NonCopyable
560 {
561 public:
562 void bindLocation(GLuint index, const std::string &name);
563 int getBinding(const std::string &name) const;
564
565 typedef std::unordered_map<std::string, GLuint>::const_iterator const_iterator;
566 const_iterator begin() const;
567 const_iterator end() const;
568
569 private:
570 std::unordered_map<std::string, GLuint> mBindings;
571 };
572
Jamie Madill32447362017-06-28 14:53:52 -0400573 const Bindings &getAttributeBindings() const { return mAttributeBindings; }
574 const Bindings &getUniformLocationBindings() const { return mUniformLocationBindings; }
575 const Bindings &getFragmentInputBindings() const { return mFragmentInputBindings; }
576
Martin Radev4e619f52017-08-09 11:50:06 +0300577 int getNumViews() const { return mState.getNumViews(); }
578 bool usesMultiview() const { return mState.usesMultiview(); }
Martin Radev7cf61662017-07-26 17:10:53 +0300579
Olli Etuahob78707c2017-03-09 15:03:11 +0000580 private:
Jamie Madill4928b7c2017-06-20 12:57:39 -0400581 ~Program();
582
Jamie Madill192745a2016-12-22 15:58:21 -0500583 struct VaryingRef
584 {
585 const sh::Varying *get() const { return vertex ? vertex : fragment; }
586
587 const sh::Varying *vertex = nullptr;
588 const sh::Varying *fragment = nullptr;
589 };
590
591 using MergedVaryings = std::map<std::string, VaryingRef>;
592
Jamie Madill6c1f6712017-02-14 19:08:04 -0500593 void unlink();
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000594
Jamie Madillbd044ed2017-06-05 12:59:21 -0400595 bool linkAttributes(const Context *context, InfoLog &infoLog);
Martin Radev4c4c8e72016-08-04 12:25:34 +0300596 bool validateVertexAndFragmentInterfaceBlocks(
597 const std::vector<sh::InterfaceBlock> &vertexInterfaceBlocks,
598 const std::vector<sh::InterfaceBlock> &fragmentInterfaceBlocks,
Frank Henigmanfccbac22017-05-28 17:29:26 -0400599 InfoLog &infoLog,
600 bool webglCompatibility) const;
Jiajia Qin729b2c62017-08-14 09:36:11 +0800601 bool linkInterfaceBlocks(const Context *context, InfoLog &infoLog);
Jamie Madillbd044ed2017-06-05 12:59:21 -0400602 bool linkVaryings(const Context *context, InfoLog &infoLog) const;
Olli Etuaho6ca2b652017-02-19 18:05:10 +0000603
Jamie Madillbd044ed2017-06-05 12:59:21 -0400604 bool linkUniforms(const Context *context,
605 InfoLog &infoLog,
606 const Bindings &uniformLocationBindings);
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800607 void linkSamplerAndImageBindings();
jchen10eaef1e52017-06-13 10:44:11 +0800608 bool linkAtomicCounterBuffers();
Olli Etuaho6ca2b652017-02-19 18:05:10 +0000609
Jamie Madilla2c74982016-12-12 11:20:42 -0500610 bool areMatchingInterfaceBlocks(InfoLog &infoLog,
Martin Radev4c4c8e72016-08-04 12:25:34 +0300611 const sh::InterfaceBlock &vertexInterfaceBlock,
Frank Henigmanfccbac22017-05-28 17:29:26 -0400612 const sh::InterfaceBlock &fragmentInterfaceBlock,
613 bool webglCompatibility) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500614
Yuly Novikova1f6dc92016-06-15 23:27:04 -0400615 static bool linkValidateVaryings(InfoLog &infoLog,
616 const std::string &varyingName,
617 const sh::Varying &vertexVarying,
618 const sh::Varying &fragmentVarying,
619 int shaderVersion);
Jamie Madillbd044ed2017-06-05 12:59:21 -0400620 bool linkValidateBuiltInVaryings(const Context *context, InfoLog &infoLog) const;
jchen10a9042d32017-03-17 08:50:45 +0800621 bool linkValidateTransformFeedback(const gl::Context *context,
622 InfoLog &infoLog,
Jamie Madill192745a2016-12-22 15:58:21 -0500623 const MergedVaryings &linkedVaryings,
Jamie Madillccdf74b2015-08-18 10:46:12 -0400624 const Caps &caps) const;
Yuly Novikovcaa5cda2017-06-15 21:14:03 -0400625 bool linkValidateGlobalNames(const Context *context, InfoLog &infoLog) const;
Jamie Madillccdf74b2015-08-18 10:46:12 -0400626
Jamie Madill192745a2016-12-22 15:58:21 -0500627 void gatherTransformFeedbackVaryings(const MergedVaryings &varyings);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500628
Jamie Madillbd044ed2017-06-05 12:59:21 -0400629 MergedVaryings getMergedVaryings(const Context *context) const;
Jamie Madill192745a2016-12-22 15:58:21 -0500630 std::vector<PackedVarying> getPackedVaryings(const MergedVaryings &mergedVaryings) const;
Jamie Madillbd044ed2017-06-05 12:59:21 -0400631 void linkOutputVariables(const Context *context);
Jamie Madillccdf74b2015-08-18 10:46:12 -0400632
Olli Etuaho48fed632017-03-16 12:05:30 +0000633 void setUniformValuesFromBindingQualifiers();
634
jchen10eaef1e52017-06-13 10:44:11 +0800635 void gatherAtomicCounterBuffers();
Jiajia Qin729b2c62017-08-14 09:36:11 +0800636 void gatherComputeBlockInfo(const std::vector<sh::InterfaceBlock> &computeBlocks);
637 void gatherVertexAndFragmentBlockInfo(
638 const std::vector<sh::InterfaceBlock> &vertexInterfaceBlocks,
639 const std::vector<sh::InterfaceBlock> &fragmentInterfaceBlocks);
Jamie Madillbd044ed2017-06-05 12:59:21 -0400640 void gatherInterfaceBlockInfo(const Context *context);
Jamie Madill4a3c2342015-10-08 12:58:45 -0400641 template <typename VarT>
642 void defineUniformBlockMembers(const std::vector<VarT> &fields,
643 const std::string &prefix,
Olli Etuaho855d9642017-05-17 14:05:06 +0300644 const std::string &mappedPrefix,
Jamie Madill4a3c2342015-10-08 12:58:45 -0400645 int blockIndex);
646
Jiajia Qin729b2c62017-08-14 09:36:11 +0800647 void defineInterfaceBlock(const sh::InterfaceBlock &interfaceBlock, GLenum shaderType);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400648
Corentin Wallez8b7d8142016-11-15 13:40:37 -0500649 // Both these function update the cached uniform values and return a modified "count"
650 // so that the uniform update doesn't overflow the uniform.
Jamie Madill62d31cb2015-09-11 13:25:51 -0400651 template <typename T>
Jamie Madillbe5e2ec2017-08-31 13:28:28 -0400652 GLsizei clampUniformCount(const VariableLocation &locationInfo,
653 GLsizei count,
654 int vectorSize,
Jamie Madille7d84322017-01-10 18:21:59 -0500655 const T *v);
Jamie Madillbe5e2ec2017-08-31 13:28:28 -0400656 template <size_t cols, size_t rows, typename T>
657 GLsizei clampMatrixUniformCount(GLint location, GLsizei count, GLboolean transpose, const T *v);
658
659 void updateSamplerUniform(const VariableLocation &locationInfo,
660 GLsizei clampedCount,
661 const GLint *v);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400662
663 template <typename DestT>
Jamie Madill54164b02017-08-28 15:17:37 -0400664 void getUniformInternal(const Context *context,
665 DestT *dataOut,
666 GLint location,
667 GLenum nativeType,
668 int components) const;
Jamie Madill62d31cb2015-09-11 13:25:51 -0400669
jchen10baf5d942017-08-28 20:45:48 +0800670 template <typename T>
671 void getResourceName(GLuint index,
672 const std::vector<T> &resources,
673 GLsizei bufSize,
674 GLsizei *length,
675 GLchar *name) const;
676
Jamie Madill48ef11b2016-04-27 15:21:52 -0400677 ProgramState mState;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500678 rx::ProgramImpl *mProgram;
679
Geoff Lang7dd2e102014-11-10 15:19:26 -0500680 bool mValidated;
681
Geoff Langd8605522016-04-13 10:19:12 -0400682 Bindings mAttributeBindings;
Olli Etuaho4a92ceb2017-02-19 17:51:24 +0000683
684 // Note that this has nothing to do with binding layout qualifiers that can be set for some
685 // uniforms in GLES3.1+. It is used to pre-set the location of uniforms.
686 Bindings mUniformLocationBindings;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000687
Sami Väisänen46eaa942016-06-29 10:26:37 +0300688 // CHROMIUM_path_rendering
689 Bindings mFragmentInputBindings;
690
daniel@transgaming.com716056c2012-07-24 18:38:59 +0000691 bool mLinked;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000692 bool mDeleteStatus; // Flag to indicate that the program can be deleted when no longer in use
daniel@transgaming.com4fa08332010-05-11 02:29:27 +0000693
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +0000694 unsigned int mRefCount;
695
Geoff Lang4ddf5af2016-12-01 14:30:44 -0500696 ShaderProgramManager *mResourceManager;
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +0000697 const GLuint mHandle;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000698
699 InfoLog mInfoLog;
Jamie Madill3d3d2f22015-09-23 16:47:51 -0400700
701 // Cache for sampler validation
702 Optional<bool> mCachedValidateSamplersResult;
703 std::vector<GLenum> mTextureUnitTypesCache;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000704};
Jamie Madilla2c74982016-12-12 11:20:42 -0500705} // namespace gl
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000706
Geoff Lang0a73dd82014-11-19 16:18:08 -0500707#endif // LIBANGLE_PROGRAM_H_