1 //
2 // Copyright 2015 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6
7 // BindUniformLocationTest.cpp : Tests of the GL_CHROMIUM_bind_uniform_location extension.
8
9 #include "test_utils/ANGLETest.h"
10 #include "test_utils/gl_raii.h"
11
12 #include <cmath>
13
14 using namespace angle;
15
16 namespace
17 {
18
19 class BindUniformLocationTest : public ANGLETest
20 {
21 protected:
BindUniformLocationTest()22 BindUniformLocationTest()
23 {
24 setWindowWidth(128);
25 setWindowHeight(128);
26 setConfigRedBits(8);
27 setConfigGreenBits(8);
28 setConfigBlueBits(8);
29 setConfigAlphaBits(8);
30 }
31
testTearDown()32 void testTearDown() override
33 {
34 if (mProgram != 0)
35 {
36 glDeleteProgram(mProgram);
37 }
38 }
39
40 GLuint mProgram = 0;
41 };
42
43 // Test basic functionality of GL_CHROMIUM_bind_uniform_location
TEST_P(BindUniformLocationTest,Basic)44 TEST_P(BindUniformLocationTest, Basic)
45 {
46 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
47
48 constexpr char kFS[] = R"(precision mediump float;
49 uniform vec4 u_colorC;
50 uniform vec4 u_colorB[2];
51 uniform vec4 u_colorA;
52 void main()
53 {
54 gl_FragColor = u_colorA + u_colorB[0] + u_colorB[1] + u_colorC;
55 })";
56
57 GLint colorALocation = 3;
58 GLint colorBLocation = 10;
59 GLint colorCLocation = 5;
60
61 mProgram = CompileProgram(essl1_shaders::vs::Simple(), kFS, [&](GLuint program) {
62 glBindUniformLocationCHROMIUM(program, colorALocation, "u_colorA");
63 glBindUniformLocationCHROMIUM(program, colorBLocation, "u_colorB[0]");
64 glBindUniformLocationCHROMIUM(program, colorCLocation, "u_colorC");
65 });
66 ASSERT_NE(0u, mProgram);
67
68 glUseProgram(mProgram);
69
70 static const float colorB[] = {
71 0.0f, 0.50f, 0.0f, 0.0f, 0.0f, 0.0f, 0.75f, 0.0f,
72 };
73
74 glUniform4f(colorALocation, 0.25f, 0.0f, 0.0f, 0.0f);
75 glUniform4fv(colorBLocation, 2, colorB);
76 glUniform4f(colorCLocation, 0.0f, 0.0f, 0.0f, 1.0f);
77
78 drawQuad(mProgram, essl1_shaders::PositionAttrib(), 0.5f);
79
80 EXPECT_GL_NO_ERROR();
81 EXPECT_PIXEL_NEAR(0, 0, 64, 128, 192, 255, 1.0);
82 }
83
84 // Force a sampler location and make sure it samples the correct texture
TEST_P(BindUniformLocationTest,SamplerLocation)85 TEST_P(BindUniformLocationTest, SamplerLocation)
86 {
87 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
88
89 constexpr char kFS[] = R"(precision mediump float;
90 uniform vec4 u_colorA;
91 uniform vec4 u_colorB[2];
92 uniform sampler2D u_sampler;
93 void main()
94 {
95 gl_FragColor = u_colorA + u_colorB[0] + u_colorB[1] + texture2D(u_sampler, vec2(0, 0));
96 })";
97
98 GLint colorALocation = 3;
99 GLint colorBLocation = 10;
100 GLint samplerLocation = 1;
101
102 mProgram = CompileProgram(essl1_shaders::vs::Simple(), kFS, [&](GLuint program) {
103 glBindUniformLocationCHROMIUM(program, colorALocation, "u_colorA");
104 glBindUniformLocationCHROMIUM(program, colorBLocation, "u_colorB[0]");
105 glBindUniformLocationCHROMIUM(program, samplerLocation, "u_sampler");
106 });
107 ASSERT_NE(0u, mProgram);
108
109 glUseProgram(mProgram);
110
111 static const float colorB[] = {
112 0.0f, 0.50f, 0.0f, 0.0f, 0.0f, 0.0f, 0.75f, 0.0f,
113 };
114
115 glUniform4f(colorALocation, 0.25f, 0.0f, 0.0f, 0.0f);
116 glUniform4fv(colorBLocation, 2, colorB);
117
118 // Point the texture at texture unit 2
119 glUniform1i(samplerLocation, 2);
120
121 GLTexture texture;
122 glActiveTexture(GL_TEXTURE2);
123 glBindTexture(GL_TEXTURE_2D, texture);
124 constexpr GLubyte kTextureData[] = {32, 32, 32, 255};
125 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE, kTextureData);
126
127 drawQuad(mProgram, essl1_shaders::PositionAttrib(), 0.5f);
128
129 EXPECT_GL_NO_ERROR();
130 EXPECT_PIXEL_NEAR(0, 0, 96, 160, 224, 255, 1.0);
131 }
132
133 // Test that conflicts are detected when two uniforms are bound to the same location
TEST_P(BindUniformLocationTest,ConflictsDetection)134 TEST_P(BindUniformLocationTest, ConflictsDetection)
135 {
136 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
137
138 constexpr char kFS[] =
139 R"(precision mediump float;
140 uniform vec4 u_colorA;
141 uniform vec4 u_colorB;
142 void main()
143 {
144 gl_FragColor = u_colorA + u_colorB;
145 })";
146
147 GLint colorALocation = 3;
148 GLint colorBLocation = 4;
149
150 GLuint vs = CompileShader(GL_VERTEX_SHADER, essl1_shaders::vs::Simple());
151 GLuint fs = CompileShader(GL_FRAGMENT_SHADER, kFS);
152
153 mProgram = glCreateProgram();
154 glAttachShader(mProgram, vs);
155 glDeleteShader(vs);
156 glAttachShader(mProgram, fs);
157 glDeleteShader(fs);
158
159 glBindUniformLocationCHROMIUM(mProgram, colorALocation, "u_colorA");
160 // Bind u_colorB to location a, causing conflicts, link should fail.
161 glBindUniformLocationCHROMIUM(mProgram, colorALocation, "u_colorB");
162 glLinkProgram(mProgram);
163 GLint linked = 0;
164 glGetProgramiv(mProgram, GL_LINK_STATUS, &linked);
165 ASSERT_EQ(0, linked);
166
167 // Bind u_colorB to location b, no conflicts, link should succeed.
168 glBindUniformLocationCHROMIUM(mProgram, colorBLocation, "u_colorB");
169 glLinkProgram(mProgram);
170 linked = 0;
171 glGetProgramiv(mProgram, GL_LINK_STATUS, &linked);
172 EXPECT_EQ(1, linked);
173 }
174
175 // Test a use case of the chromium compositor
TEST_P(BindUniformLocationTest,Compositor)176 TEST_P(BindUniformLocationTest, Compositor)
177 {
178 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
179
180 constexpr char kVS[] =
181 R"(attribute vec4 a_position;
182 attribute vec2 a_texCoord;
183 uniform mat4 matrix;
184 uniform vec2 color_a[4];
185 uniform vec4 color_b;
186 varying vec4 v_color;
187 void main()
188 {
189 v_color.xy = color_a[0] + color_a[1];
190 v_color.zw = color_a[2] + color_a[3];
191 v_color += color_b;
192 gl_Position = matrix * a_position;
193 })";
194
195 constexpr char kFS[] =
196 R"(precision mediump float;
197 varying vec4 v_color;
198 uniform float alpha;
199 uniform vec4 multiplier;
200 uniform vec3 color_c[8];
201 void main()
202 {
203 vec4 color_c_sum = vec4(0.0);
204 color_c_sum.xyz += color_c[0];
205 color_c_sum.xyz += color_c[1];
206 color_c_sum.xyz += color_c[2];
207 color_c_sum.xyz += color_c[3];
208 color_c_sum.xyz += color_c[4];
209 color_c_sum.xyz += color_c[5];
210 color_c_sum.xyz += color_c[6];
211 color_c_sum.xyz += color_c[7];
212 color_c_sum.w = alpha;
213 color_c_sum *= multiplier;
214 gl_FragColor = v_color + color_c_sum;
215 })";
216
217 int counter = 6;
218 int matrixLocation = counter++;
219 int colorALocation = counter++;
220 int colorBLocation = counter++;
221 int alphaLocation = counter++;
222 int multiplierLocation = counter++;
223 int colorCLocation = counter++;
224
225 mProgram = CompileProgram(kVS, kFS, [&](GLuint program) {
226 glBindUniformLocationCHROMIUM(program, matrixLocation, "matrix");
227 glBindUniformLocationCHROMIUM(program, colorALocation, "color_a");
228 glBindUniformLocationCHROMIUM(program, colorBLocation, "color_b");
229 glBindUniformLocationCHROMIUM(program, alphaLocation, "alpha");
230 glBindUniformLocationCHROMIUM(program, multiplierLocation, "multiplier");
231 glBindUniformLocationCHROMIUM(program, colorCLocation, "color_c");
232 });
233 ASSERT_NE(0u, mProgram);
234
235 glUseProgram(mProgram);
236
237 static const float colorA[] = {
238 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f,
239 };
240
241 static const float colorC[] = {
242 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f,
243 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f, 0.1f,
244 };
245
246 static const float identity[] = {
247 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1,
248 };
249
250 glUniformMatrix4fv(matrixLocation, 1, false, identity);
251 glUniform2fv(colorALocation, 4, colorA);
252 glUniform4f(colorBLocation, 0.2f, 0.2f, 0.2f, 0.2f);
253 glUniform1f(alphaLocation, 0.8f);
254 glUniform4f(multiplierLocation, 0.5f, 0.5f, 0.5f, 0.5f);
255 glUniform3fv(colorCLocation, 8, colorC);
256
257 glDrawArrays(GL_TRIANGLES, 0, 6);
258
259 drawQuad(mProgram, "a_position", 0.5f);
260
261 EXPECT_PIXEL_EQ(0, 0, 204, 204, 204, 204);
262 }
263
264 // Test that unused uniforms don't conflict when bound to the same location
TEST_P(BindUniformLocationTest,UnusedUniformUpdate)265 TEST_P(BindUniformLocationTest, UnusedUniformUpdate)
266 {
267 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
268
269 ASSERT_NE(nullptr, glBindUniformLocationCHROMIUM);
270
271 constexpr char kFS[] = R"(precision mediump float;
272 uniform vec4 u_colorA;
273 uniform float u_colorU;
274 uniform vec4 u_colorC;
275 void main()
276 {
277 gl_FragColor = u_colorA + u_colorC;
278 })";
279
280 const GLint colorULocation = 1;
281 const GLint nonexistingLocation = 5;
282 const GLint unboundLocation = 6;
283
284 mProgram = CompileProgram(essl1_shaders::vs::Simple(), kFS, [&](GLuint program) {
285 glBindUniformLocationCHROMIUM(program, colorULocation, "u_colorU");
286 // The non-existing uniform should behave like existing, but optimized away
287 // uniform.
288 glBindUniformLocationCHROMIUM(program, nonexistingLocation, "nonexisting");
289 // Let A and C be assigned automatic locations.
290 });
291 ASSERT_NE(0u, mProgram);
292
293 glUseProgram(mProgram);
294
295 // No errors on bound locations, since caller does not know
296 // if the driver optimizes them away or not.
297 glUniform1f(colorULocation, 0.25f);
298 EXPECT_GL_NO_ERROR();
299
300 // No errors on bound locations of names that do not exist
301 // in the shader. Otherwise it would be inconsistent wrt the
302 // optimization case.
303 glUniform1f(nonexistingLocation, 0.25f);
304 EXPECT_GL_NO_ERROR();
305
306 // The above are equal to updating -1.
307 glUniform1f(-1, 0.25f);
308 EXPECT_GL_NO_ERROR();
309
310 // No errors when updating with other type either.
311 // The type can not be known with the non-existing case.
312 glUniform2f(colorULocation, 0.25f, 0.25f);
313 EXPECT_GL_NO_ERROR();
314 glUniform2f(nonexistingLocation, 0.25f, 0.25f);
315 EXPECT_GL_NO_ERROR();
316 glUniform2f(-1, 0.25f, 0.25f);
317 EXPECT_GL_NO_ERROR();
318
319 // Ensure that driver or ANGLE has optimized the variable
320 // away and the test tests what it is supposed to.
321 EXPECT_EQ(-1, glGetUniformLocation(mProgram, "u_colorU"));
322
323 // The bound location gets marked as used and the driver
324 // does not allocate other variables to that location.
325 EXPECT_NE(colorULocation, glGetUniformLocation(mProgram, "u_colorA"));
326 EXPECT_NE(colorULocation, glGetUniformLocation(mProgram, "u_colorC"));
327 EXPECT_NE(nonexistingLocation, glGetUniformLocation(mProgram, "u_colorA"));
328 EXPECT_NE(nonexistingLocation, glGetUniformLocation(mProgram, "u_colorC"));
329
330 // Unintuitive: while specifying value works, getting the value does not.
331 GLfloat getResult = 0.0f;
332 glGetUniformfv(mProgram, colorULocation, &getResult);
333 EXPECT_GL_ERROR(GL_INVALID_OPERATION);
334 glGetUniformfv(mProgram, nonexistingLocation, &getResult);
335 EXPECT_GL_ERROR(GL_INVALID_OPERATION);
336 glGetUniformfv(mProgram, -1, &getResult);
337 EXPECT_GL_ERROR(GL_INVALID_OPERATION);
338
339 // Updating an unbound, non-existing location still causes
340 // an error.
341 glUniform1f(unboundLocation, 0.25f);
342 EXPECT_GL_ERROR(GL_INVALID_OPERATION);
343 }
344
345 // GL backend optimizes away a uniform in the vertex shader if it's only used to
346 // compute a varying that is never referenced in the fragment shader.
TEST_P(BindUniformLocationTest,UnusedUniformUpdateComplex)347 TEST_P(BindUniformLocationTest, UnusedUniformUpdateComplex)
348 {
349 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
350
351 ASSERT_NE(nullptr, glBindUniformLocationCHROMIUM);
352
353 constexpr char kVS[] = R"(precision highp float;
354 attribute vec4 a_position;
355 varying vec4 v_unused;
356 uniform vec4 u_unused;
357 void main()
358 {
359 gl_Position = a_position;
360 v_unused = u_unused;
361 }
362 )";
363
364 constexpr char kFS[] = R"(precision mediump float;
365 varying vec4 v_unused;
366 void main()
367 {
368 gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);
369 })";
370
371 const GLint unusedLocation = 1;
372
373 mProgram = CompileProgram(kVS, kFS, [&](GLuint program) {
374 glBindUniformLocationCHROMIUM(program, unusedLocation, "u_unused");
375 });
376 ASSERT_NE(0u, mProgram);
377
378 glUseProgram(mProgram);
379
380 // No errors on bound locations of names that do not exist
381 // in the shader. Otherwise it would be inconsistent wrt the
382 // optimization case.
383 glUniform4f(unusedLocation, 0.25f, 0.25f, 0.25f, 0.25f);
384 EXPECT_GL_NO_ERROR();
385 }
386
387 // Test for a bug where using a sampler caused GL error if the mProgram had
388 // uniforms that were optimized away by the driver. This was only a problem with
389 // glBindUniformLocationCHROMIUM implementation. This could be reproed by
390 // binding the sampler to a location higher than the amount of active uniforms.
TEST_P(BindUniformLocationTest,UseSamplerWhenUnusedUniforms)391 TEST_P(BindUniformLocationTest, UseSamplerWhenUnusedUniforms)
392 {
393 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
394
395 constexpr char kFS[] =
396 R"(uniform sampler2D tex;
397 void main()
398 {
399 gl_FragColor = texture2D(tex, vec2(1));
400 })";
401
402 const GLuint texLocation = 54;
403
404 mProgram = CompileProgram(essl1_shaders::vs::Simple(), kFS, [&](GLuint program) {
405 glBindUniformLocationCHROMIUM(program, texLocation, "tex");
406 });
407 ASSERT_NE(0u, mProgram);
408
409 glUseProgram(mProgram);
410 glUniform1i(texLocation, 0);
411 EXPECT_GL_NO_ERROR();
412 }
413
414 // Test for binding a statically used uniform to the same location as a non-statically used uniform.
415 // This is valid according to the extension spec.
TEST_P(BindUniformLocationTest,SameLocationForUsedAndUnusedUniform)416 TEST_P(BindUniformLocationTest, SameLocationForUsedAndUnusedUniform)
417 {
418 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
419
420 constexpr char kFS[] =
421 R"(precision mediump float;
422 uniform vec4 a;
423 uniform vec4 b;
424 void main()
425 {
426 gl_FragColor = a;
427 })";
428
429 const GLuint location = 54;
430
431 mProgram = CompileProgram(essl1_shaders::vs::Zero(), kFS, [&](GLuint program) {
432 glBindUniformLocationCHROMIUM(program, location, "a");
433 glBindUniformLocationCHROMIUM(program, location, "b");
434 });
435 ASSERT_NE(0u, mProgram);
436
437 glUseProgram(mProgram);
438 glUniform4f(location, 0.0, 1.0, 0.0, 1.0);
439 EXPECT_GL_NO_ERROR();
440 }
441
442 class BindUniformLocationES31Test : public BindUniformLocationTest
443 {
444 protected:
BindUniformLocationES31Test()445 BindUniformLocationES31Test() : BindUniformLocationTest() {}
446
linkProgramWithUniformLocation(const char * vs,const char * fs,const char * uniformName,GLint uniformLocation)447 void linkProgramWithUniformLocation(const char *vs,
448 const char *fs,
449 const char *uniformName,
450 GLint uniformLocation)
451 {
452 mProgram = CompileProgram(vs, fs, [&](GLuint program) {
453 glBindUniformLocationCHROMIUM(program, uniformLocation, uniformName);
454 });
455 }
456 };
457
458 // Test for when the shader specifies an explicit uniform location with a layout qualifier and the
459 // bindUniformLocation API sets a consistent location.
TEST_P(BindUniformLocationES31Test,ConsistentWithLocationLayoutQualifier)460 TEST_P(BindUniformLocationES31Test, ConsistentWithLocationLayoutQualifier)
461 {
462 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
463
464 constexpr char kFS[] =
465 "#version 310 es\n"
466 "uniform layout(location=2) highp sampler2D tex;\n"
467 "out highp vec4 my_FragColor;\n"
468 "void main()\n"
469 "{\n"
470 " my_FragColor = texture(tex, vec2(1));\n"
471 "}\n";
472
473 const GLuint texLocation = 2;
474
475 linkProgramWithUniformLocation(essl31_shaders::vs::Zero(), kFS, "tex", texLocation);
476
477 GLint linked = GL_FALSE;
478 glGetProgramiv(mProgram, GL_LINK_STATUS, &linked);
479 ASSERT_GL_TRUE(linked);
480
481 EXPECT_EQ(static_cast<GLint>(texLocation), glGetUniformLocation(mProgram, "tex"));
482 glUseProgram(mProgram);
483 glUniform1i(texLocation, 0);
484 EXPECT_GL_NO_ERROR();
485 }
486
487 // Test for when the shader specifies an explicit uniform location with a layout qualifier and the
488 // bindUniformLocation API sets a conflicting location for the same variable. The shader-set
489 // location should prevail.
TEST_P(BindUniformLocationES31Test,LocationLayoutQualifierOverridesAPIBinding)490 TEST_P(BindUniformLocationES31Test, LocationLayoutQualifierOverridesAPIBinding)
491 {
492 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
493
494 constexpr char kFS[] =
495 "#version 310 es\n"
496 "uniform layout(location=2) highp sampler2D tex;\n"
497 "out highp vec4 my_FragColor;\n"
498 "void main()\n"
499 "{\n"
500 " my_FragColor = texture(tex, vec2(1));\n"
501 "}\n";
502
503 const GLuint shaderTexLocation = 2;
504 const GLuint texLocation = 3;
505
506 linkProgramWithUniformLocation(essl31_shaders::vs::Zero(), kFS, "tex", texLocation);
507
508 GLint linked = GL_FALSE;
509 glGetProgramiv(mProgram, GL_LINK_STATUS, &linked);
510 ASSERT_GL_TRUE(linked);
511
512 EXPECT_EQ(static_cast<GLint>(shaderTexLocation), glGetUniformLocation(mProgram, "tex"));
513 glUseProgram(mProgram);
514 glUniform1i(shaderTexLocation, 1);
515 EXPECT_GL_NO_ERROR();
516 glUniform1i(texLocation, 2);
517 EXPECT_GL_NO_ERROR();
518 }
519
520 // Test for when the shader specifies an explicit uniform location with a layout qualifier and the
521 // bindUniformLocation API sets a conflicting location for a different variable. Linking should
522 // fail.
TEST_P(BindUniformLocationES31Test,LocationLayoutQualifierConflictsWithAPIBinding)523 TEST_P(BindUniformLocationES31Test, LocationLayoutQualifierConflictsWithAPIBinding)
524 {
525 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
526
527 constexpr char kFS[] =
528 "#version 310 es\n"
529 "uniform layout(location=2) highp sampler2D tex;\n"
530 "uniform highp sampler2D tex2;\n"
531 "out highp vec4 my_FragColor;\n"
532 "void main()\n"
533 "{\n"
534 " my_FragColor = texture(tex2, vec2(1));\n"
535 "}\n";
536
537 const GLuint tex2Location = 2;
538
539 linkProgramWithUniformLocation(essl31_shaders::vs::Zero(), kFS, "tex2", tex2Location);
540
541 GLint linked = GL_FALSE;
542 glGetProgramiv(mProgram, GL_LINK_STATUS, &linked);
543 ASSERT_GL_FALSE(linked);
544 }
545
546 // Test for binding a location for an array of arrays uniform.
TEST_P(BindUniformLocationES31Test,ArrayOfArrays)547 TEST_P(BindUniformLocationES31Test, ArrayOfArrays)
548 {
549 ANGLE_SKIP_TEST_IF(!IsGLExtensionEnabled("GL_CHROMIUM_bind_uniform_location"));
550
551 constexpr char kFS[] =
552 R"(#version 310 es
553 precision highp float;
554 uniform vec4 sourceColor[2][1];
555 out highp vec4 my_FragColor;
556 void main()
557 {
558 my_FragColor = sourceColor[1][0];
559 })";
560
561 const GLuint location = 8;
562
563 linkProgramWithUniformLocation(essl31_shaders::vs::Simple(), kFS, "sourceColor[1]", location);
564
565 GLint linked = GL_FALSE;
566 glGetProgramiv(mProgram, GL_LINK_STATUS, &linked);
567 ASSERT_GL_TRUE(linked);
568
569 glUseProgram(mProgram);
570 glUniform4f(location, 0.0f, 1.0f, 0.0f, 1.0f);
571
572 drawQuad(mProgram, essl31_shaders::PositionAttrib(), 0.5f);
573 EXPECT_GL_NO_ERROR();
574
575 EXPECT_PIXEL_COLOR_EQ(0, 0, GLColor::green);
576 }
577
578 // Use this to select which configurations (e.g. which renderer, which GLES major version) these
579 // tests should be run against.
580 ANGLE_INSTANTIATE_TEST_ES2(BindUniformLocationTest);
581
582 ANGLE_INSTANTIATE_TEST_ES31(BindUniformLocationES31Test);
583
584 } // namespace
585