blob: 10ed201be052d2e1454b03d858ae4da8b396a155 [file] [log] [blame]
Jamie Madillfa05f602015-05-07 13:47:11 -04001//
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//
Jamie Madill1fbc59f2016-02-24 15:25:51 -05006// Framebuffer tests:
7// Various tests related for Frambuffers.
8//
Jamie Madillfa05f602015-05-07 13:47:11 -04009
Brandon Jones9fc87332017-12-13 15:46:52 -080010#include "platform/WorkaroundsD3D.h"
Corentin Wallezd3970de2015-05-14 11:07:48 -040011#include "test_utils/ANGLETest.h"
JiangYizhou461d9a32017-01-04 16:37:26 +080012#include "test_utils/gl_raii.h"
Geoff Langb6a673a2014-06-05 14:19:16 -040013
Jamie Madillfa05f602015-05-07 13:47:11 -040014using namespace angle;
Austin Kinross18b931d2014-09-29 12:58:31 -070015
Geoff Lang857c09d2017-05-16 15:55:04 -040016namespace
17{
18
19void ExpectFramebufferCompleteOrUnsupported(GLenum binding)
20{
21 GLenum status = glCheckFramebufferStatus(binding);
22 EXPECT_TRUE(status == GL_FRAMEBUFFER_COMPLETE || status == GL_FRAMEBUFFER_UNSUPPORTED);
23}
24
25} // anonymous namespace
26
Geoff Langb6a673a2014-06-05 14:19:16 -040027class FramebufferFormatsTest : public ANGLETest
28{
Jamie Madillfa05f602015-05-07 13:47:11 -040029 protected:
Jamie Madill3215b202015-12-15 16:41:39 -050030 FramebufferFormatsTest() : mFramebuffer(0), mTexture(0), mRenderbuffer(0), mProgram(0)
Geoff Langb6a673a2014-06-05 14:19:16 -040031 {
32 setWindowWidth(128);
33 setWindowHeight(128);
34 setConfigRedBits(8);
35 setConfigGreenBits(8);
36 setConfigBlueBits(8);
37 setConfigAlphaBits(8);
38 }
39
40 void checkBitCount(GLuint fbo, GLenum channel, GLint minBits)
41 {
42 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
43
44 GLint bits = 0;
45 glGetIntegerv(channel, &bits);
46
47 if (minBits == 0)
48 {
49 EXPECT_EQ(minBits, bits);
50 }
51 else
52 {
53 EXPECT_GE(bits, minBits);
54 }
55 }
56
Jamie Madill1fbc59f2016-02-24 15:25:51 -050057 void testBitCounts(GLuint fbo,
58 GLint minRedBits,
59 GLint minGreenBits,
60 GLint minBlueBits,
61 GLint minAlphaBits,
62 GLint minDepthBits,
63 GLint minStencilBits)
Geoff Langb6a673a2014-06-05 14:19:16 -040064 {
65 checkBitCount(fbo, GL_RED_BITS, minRedBits);
66 checkBitCount(fbo, GL_GREEN_BITS, minGreenBits);
67 checkBitCount(fbo, GL_BLUE_BITS, minBlueBits);
68 checkBitCount(fbo, GL_ALPHA_BITS, minAlphaBits);
69 checkBitCount(fbo, GL_DEPTH_BITS, minDepthBits);
70 checkBitCount(fbo, GL_STENCIL_BITS, minStencilBits);
71 }
72
Jamie Madill1fbc59f2016-02-24 15:25:51 -050073 void testTextureFormat(GLenum internalFormat,
74 GLint minRedBits,
75 GLint minGreenBits,
76 GLint minBlueBits,
Geoff Langb6a673a2014-06-05 14:19:16 -040077 GLint minAlphaBits)
78 {
Jamie Madill3215b202015-12-15 16:41:39 -050079 glGenTextures(1, &mTexture);
80 glBindTexture(GL_TEXTURE_2D, mTexture);
Geoff Langc4e93662017-05-01 10:45:59 -040081
82 if (getClientMajorVersion() >= 3)
83 {
84 glTexStorage2D(GL_TEXTURE_2D, 1, internalFormat, 1, 1);
85 }
86 else
87 {
88 glTexStorage2DEXT(GL_TEXTURE_2D, 1, internalFormat, 1, 1);
89 }
Geoff Langb6a673a2014-06-05 14:19:16 -040090
Jamie Madill3215b202015-12-15 16:41:39 -050091 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, mTexture, 0);
Geoff Langb6a673a2014-06-05 14:19:16 -040092
Jamie Madill3215b202015-12-15 16:41:39 -050093 testBitCounts(mFramebuffer, minRedBits, minGreenBits, minBlueBits, minAlphaBits, 0, 0);
Geoff Langb6a673a2014-06-05 14:19:16 -040094 }
95
Jamie Madill1fbc59f2016-02-24 15:25:51 -050096 void testRenderbufferMultisampleFormat(int minESVersion,
97 GLenum attachmentType,
98 GLenum internalFormat)
Corentin Walleze0902642014-11-04 12:32:15 -080099 {
Martin Radev1be913c2016-07-11 17:59:16 +0300100 int clientVersion = getClientMajorVersion();
Jamie Madillfa05f602015-05-07 13:47:11 -0400101 if (clientVersion < minESVersion)
Corentin Walleze0902642014-11-04 12:32:15 -0800102 {
103 return;
104 }
105
106 // Check that multisample is supported with at least two samples (minimum required is 1)
107 bool supports2Samples = false;
108
Jamie Madillfa05f602015-05-07 13:47:11 -0400109 if (clientVersion == 2)
Corentin Walleze0902642014-11-04 12:32:15 -0800110 {
111 if (extensionEnabled("ANGLE_framebuffer_multisample"))
112 {
113 int maxSamples;
114 glGetIntegerv(GL_MAX_SAMPLES_ANGLE, &maxSamples);
115 supports2Samples = maxSamples >= 2;
116 }
117 }
118 else
119 {
Jamie Madillfa05f602015-05-07 13:47:11 -0400120 assert(clientVersion >= 3);
Corentin Walleze0902642014-11-04 12:32:15 -0800121 int maxSamples;
122 glGetIntegerv(GL_MAX_SAMPLES, &maxSamples);
123 supports2Samples = maxSamples >= 2;
124 }
125
126 if (!supports2Samples)
127 {
128 return;
129 }
130
Jamie Madill3215b202015-12-15 16:41:39 -0500131 glGenRenderbuffers(1, &mRenderbuffer);
132 glBindRenderbuffer(GL_RENDERBUFFER, mRenderbuffer);
Corentin Walleze0902642014-11-04 12:32:15 -0800133
134 EXPECT_GL_NO_ERROR();
135 glRenderbufferStorageMultisampleANGLE(GL_RENDERBUFFER, 2, internalFormat, 128, 128);
136 EXPECT_GL_NO_ERROR();
Jamie Madill3215b202015-12-15 16:41:39 -0500137 glFramebufferRenderbuffer(GL_FRAMEBUFFER, attachmentType, GL_RENDERBUFFER, mRenderbuffer);
Corentin Walleze0902642014-11-04 12:32:15 -0800138 EXPECT_GL_NO_ERROR();
Corentin Walleze0902642014-11-04 12:32:15 -0800139 }
140
Olli Etuahobc21e182016-02-23 16:04:57 +0200141 void testZeroHeightRenderbuffer()
142 {
143 glGenRenderbuffers(1, &mRenderbuffer);
144 glBindRenderbuffer(GL_RENDERBUFFER, mRenderbuffer);
145 glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 1, 0);
146 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER,
147 mRenderbuffer);
148 EXPECT_GL_NO_ERROR();
149 }
150
Jamie Madill3215b202015-12-15 16:41:39 -0500151 void SetUp() override
Geoff Langb6a673a2014-06-05 14:19:16 -0400152 {
153 ANGLETest::SetUp();
Jamie Madill3215b202015-12-15 16:41:39 -0500154
155 glGenFramebuffers(1, &mFramebuffer);
156 glBindFramebuffer(GL_FRAMEBUFFER, mFramebuffer);
Geoff Langb6a673a2014-06-05 14:19:16 -0400157 }
158
Jamie Madill3215b202015-12-15 16:41:39 -0500159 void TearDown() override
Geoff Langb6a673a2014-06-05 14:19:16 -0400160 {
161 ANGLETest::TearDown();
Jamie Madill3215b202015-12-15 16:41:39 -0500162
163 if (mTexture != 0)
164 {
165 glDeleteTextures(1, &mTexture);
166 mTexture = 0;
167 }
168
169 if (mRenderbuffer != 0)
170 {
171 glDeleteRenderbuffers(1, &mRenderbuffer);
172 mRenderbuffer = 0;
173 }
174
175 if (mFramebuffer != 0)
176 {
177 glDeleteFramebuffers(1, &mFramebuffer);
178 mFramebuffer = 0;
179 }
180
181 if (mProgram != 0)
182 {
183 glDeleteProgram(mProgram);
184 mProgram = 0;
185 }
Geoff Langb6a673a2014-06-05 14:19:16 -0400186 }
Jamie Madill3215b202015-12-15 16:41:39 -0500187
188 GLuint mFramebuffer;
189 GLuint mTexture;
190 GLuint mRenderbuffer;
191 GLuint mProgram;
Geoff Langb6a673a2014-06-05 14:19:16 -0400192};
193
Jamie Madillfa05f602015-05-07 13:47:11 -0400194TEST_P(FramebufferFormatsTest, RGBA4)
Geoff Langb6a673a2014-06-05 14:19:16 -0400195{
Yunchao He9550c602018-02-13 14:47:05 +0800196 ANGLE_SKIP_TEST_IF(getClientMajorVersion() < 3 && !extensionEnabled("GL_EXT_texture_storage"));
Geoff Langc4e93662017-05-01 10:45:59 -0400197
Geoff Langb6a673a2014-06-05 14:19:16 -0400198 testTextureFormat(GL_RGBA4, 4, 4, 4, 4);
199}
200
Jamie Madillfa05f602015-05-07 13:47:11 -0400201TEST_P(FramebufferFormatsTest, RGB565)
Geoff Langb6a673a2014-06-05 14:19:16 -0400202{
Yunchao He9550c602018-02-13 14:47:05 +0800203 ANGLE_SKIP_TEST_IF(getClientMajorVersion() < 3 && !extensionEnabled("GL_EXT_texture_storage"));
Geoff Langc4e93662017-05-01 10:45:59 -0400204
Geoff Langb6a673a2014-06-05 14:19:16 -0400205 testTextureFormat(GL_RGB565, 5, 6, 5, 0);
206}
207
Jamie Madillfa05f602015-05-07 13:47:11 -0400208TEST_P(FramebufferFormatsTest, RGB8)
Geoff Langb6a673a2014-06-05 14:19:16 -0400209{
Yunchao He9550c602018-02-13 14:47:05 +0800210 ANGLE_SKIP_TEST_IF(
211 getClientMajorVersion() < 3 &&
212 (!extensionEnabled("GL_OES_rgb8_rgba8") || !extensionEnabled("GL_EXT_texture_storage")));
Geoff Langf34d1db2015-05-20 14:10:46 -0400213
Geoff Langb6a673a2014-06-05 14:19:16 -0400214 testTextureFormat(GL_RGB8_OES, 8, 8, 8, 0);
215}
216
Jamie Madillfa05f602015-05-07 13:47:11 -0400217TEST_P(FramebufferFormatsTest, BGRA8)
Geoff Langb6a673a2014-06-05 14:19:16 -0400218{
Yunchao He9550c602018-02-13 14:47:05 +0800219 ANGLE_SKIP_TEST_IF(
220 !extensionEnabled("GL_EXT_texture_format_BGRA8888") ||
221 (getClientMajorVersion() < 3 && !extensionEnabled("GL_EXT_texture_storage")));
Geoff Langf34d1db2015-05-20 14:10:46 -0400222
Geoff Langb6a673a2014-06-05 14:19:16 -0400223 testTextureFormat(GL_BGRA8_EXT, 8, 8, 8, 8);
224}
225
Jamie Madillfa05f602015-05-07 13:47:11 -0400226TEST_P(FramebufferFormatsTest, RGBA8)
Geoff Langb6a673a2014-06-05 14:19:16 -0400227{
Yunchao He9550c602018-02-13 14:47:05 +0800228 ANGLE_SKIP_TEST_IF(
229 getClientMajorVersion() < 3 &&
230 (!extensionEnabled("GL_OES_rgb8_rgba8") || !extensionEnabled("GL_EXT_texture_storage")));
Geoff Langf34d1db2015-05-20 14:10:46 -0400231
Geoff Langb6a673a2014-06-05 14:19:16 -0400232 testTextureFormat(GL_RGBA8_OES, 8, 8, 8, 8);
233}
234
Jamie Madillfa05f602015-05-07 13:47:11 -0400235TEST_P(FramebufferFormatsTest, RenderbufferMultisample_DEPTH16)
Corentin Walleze0902642014-11-04 12:32:15 -0800236{
237 testRenderbufferMultisampleFormat(2, GL_DEPTH_ATTACHMENT, GL_DEPTH_COMPONENT16);
238}
239
Jamie Madillfa05f602015-05-07 13:47:11 -0400240TEST_P(FramebufferFormatsTest, RenderbufferMultisample_DEPTH24)
Corentin Walleze0902642014-11-04 12:32:15 -0800241{
242 testRenderbufferMultisampleFormat(3, GL_DEPTH_ATTACHMENT, GL_DEPTH_COMPONENT24);
243}
244
Jamie Madillfa05f602015-05-07 13:47:11 -0400245TEST_P(FramebufferFormatsTest, RenderbufferMultisample_DEPTH32F)
Corentin Walleze0902642014-11-04 12:32:15 -0800246{
Yunchao He9550c602018-02-13 14:47:05 +0800247 ANGLE_SKIP_TEST_IF(getClientMajorVersion() < 3);
Geoff Langf34d1db2015-05-20 14:10:46 -0400248
Corentin Walleze0902642014-11-04 12:32:15 -0800249 testRenderbufferMultisampleFormat(3, GL_DEPTH_ATTACHMENT, GL_DEPTH_COMPONENT32F);
250}
251
Jamie Madillfa05f602015-05-07 13:47:11 -0400252TEST_P(FramebufferFormatsTest, RenderbufferMultisample_DEPTH24_STENCIL8)
Corentin Walleze0902642014-11-04 12:32:15 -0800253{
254 testRenderbufferMultisampleFormat(3, GL_DEPTH_STENCIL_ATTACHMENT, GL_DEPTH24_STENCIL8);
255}
256
Jamie Madillfa05f602015-05-07 13:47:11 -0400257TEST_P(FramebufferFormatsTest, RenderbufferMultisample_DEPTH32F_STENCIL8)
Corentin Walleze0902642014-11-04 12:32:15 -0800258{
Yunchao He9550c602018-02-13 14:47:05 +0800259 ANGLE_SKIP_TEST_IF(getClientMajorVersion() < 3);
Geoff Langf34d1db2015-05-20 14:10:46 -0400260
Corentin Walleze0902642014-11-04 12:32:15 -0800261 testRenderbufferMultisampleFormat(3, GL_DEPTH_STENCIL_ATTACHMENT, GL_DEPTH32F_STENCIL8);
262}
263
Jamie Madillfa05f602015-05-07 13:47:11 -0400264TEST_P(FramebufferFormatsTest, RenderbufferMultisample_STENCIL_INDEX8)
Corentin Walleze0902642014-11-04 12:32:15 -0800265{
Geoff Langf34d1db2015-05-20 14:10:46 -0400266 // TODO(geofflang): Figure out how to support GLSTENCIL_INDEX8 on desktop GL
Yunchao He9550c602018-02-13 14:47:05 +0800267 ANGLE_SKIP_TEST_IF(IsDesktopOpenGL());
Geoff Langf34d1db2015-05-20 14:10:46 -0400268
Corentin Walleze0902642014-11-04 12:32:15 -0800269 testRenderbufferMultisampleFormat(2, GL_STENCIL_ATTACHMENT, GL_STENCIL_INDEX8);
270}
Jamie Madillfa05f602015-05-07 13:47:11 -0400271
Jamie Madill3215b202015-12-15 16:41:39 -0500272// Test that binding an incomplete cube map is rejected by ANGLE.
273TEST_P(FramebufferFormatsTest, IncompleteCubeMap)
274{
275 // First make a complete CubeMap.
276 glGenTextures(1, &mTexture);
277 glBindTexture(GL_TEXTURE_CUBE_MAP, mTexture);
278 glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_X, 0, GL_RGBA, 8, 8, 0, GL_RGBA, GL_UNSIGNED_BYTE,
279 nullptr);
280 glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_Y, 0, GL_RGBA, 8, 8, 0, GL_RGBA, GL_UNSIGNED_BYTE,
281 nullptr);
282 glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_Z, 0, GL_RGBA, 8, 8, 0, GL_RGBA, GL_UNSIGNED_BYTE,
283 nullptr);
284 glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_X, 0, GL_RGBA, 8, 8, 0, GL_RGBA, GL_UNSIGNED_BYTE,
285 nullptr);
286 glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_Y, 0, GL_RGBA, 8, 8, 0, GL_RGBA, GL_UNSIGNED_BYTE,
287 nullptr);
288 glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_Z, 0, GL_RGBA, 8, 8, 0, GL_RGBA, GL_UNSIGNED_BYTE,
289 nullptr);
290 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
291 glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
292
293 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_CUBE_MAP_POSITIVE_X,
294 mTexture, 0);
295
296 // Verify the framebuffer is complete.
297 ASSERT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
298
299 // Make the CubeMap cube-incomplete.
300 glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_Z, 0, GL_RGBA, 16, 16, 0, GL_RGBA, GL_UNSIGNED_BYTE,
301 nullptr);
302
303 // Verify the framebuffer is incomplete.
304 ASSERT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
305 glCheckFramebufferStatus(GL_FRAMEBUFFER));
306
Jamie Madilld84b6732018-09-06 15:54:35 -0400307 ASSERT_GL_NO_ERROR();
308
Jamie Madill3215b202015-12-15 16:41:39 -0500309 // Verify drawing with the incomplete framebuffer produces a GL error
Olli Etuaho5804dc82018-04-13 14:11:46 +0300310 mProgram = CompileProgram(essl1_shaders::vs::Simple(), essl1_shaders::fs::Red());
Jamie Madill3215b202015-12-15 16:41:39 -0500311 ASSERT_NE(0u, mProgram);
Olli Etuaho5804dc82018-04-13 14:11:46 +0300312 drawQuad(mProgram, essl1_shaders::PositionAttrib(), 0.5f);
Jamie Madill3215b202015-12-15 16:41:39 -0500313 ASSERT_GL_ERROR(GL_INVALID_FRAMEBUFFER_OPERATION);
314}
315
Olli Etuahobc21e182016-02-23 16:04:57 +0200316// Test that a renderbuffer with zero height but nonzero width is handled without crashes/asserts.
317TEST_P(FramebufferFormatsTest, ZeroHeightRenderbuffer)
318{
Yunchao He9550c602018-02-13 14:47:05 +0800319 ANGLE_SKIP_TEST_IF(getClientMajorVersion() < 3);
Olli Etuahobc21e182016-02-23 16:04:57 +0200320
321 testZeroHeightRenderbuffer();
322}
323
Geoff Lang9bf86f02018-07-26 11:46:34 -0400324// Test to cover a bug where the read framebuffer affects the completeness of the draw framebuffer.
325TEST_P(FramebufferFormatsTest, ReadDrawCompleteness)
326{
327 ANGLE_SKIP_TEST_IF(getClientMajorVersion() < 3);
328
329 GLTexture incompleteTexture;
330 glBindTexture(GL_TEXTURE_2D, incompleteTexture);
331
332 GLFramebuffer incompleteFBO;
333 glBindFramebuffer(GL_FRAMEBUFFER, incompleteFBO);
334 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, incompleteTexture,
335 0);
336 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
337 glCheckFramebufferStatus(GL_FRAMEBUFFER));
338
339 GLTexture completeTexture;
340 glBindTexture(GL_TEXTURE_2D, completeTexture);
341 glTexStorage2D(GL_TEXTURE_2D, 1, GL_RGBA8, getWindowWidth(), getWindowHeight());
342
343 GLFramebuffer completeFBO;
344 glBindFramebuffer(GL_DRAW_FRAMEBUFFER, completeFBO);
345 glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
346 completeTexture, 0);
347
348 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
349 glCheckFramebufferStatus(GL_READ_FRAMEBUFFER));
350 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_DRAW_FRAMEBUFFER));
351
352 ASSERT_GL_NO_ERROR();
353
354 // Simple draw program.
355 ANGLE_GL_PROGRAM(program, essl1_shaders::vs::Simple(), essl1_shaders::fs::Red());
356
357 drawQuad(program, essl1_shaders::PositionAttrib(), 0.5f, 1.0f, true);
358 EXPECT_GL_NO_ERROR();
359
360 glBindFramebuffer(GL_READ_FRAMEBUFFER, completeFBO);
361 EXPECT_PIXEL_COLOR_EQ(0, 0, GLColor::red);
362}
363
Jamie Madill1fbc59f2016-02-24 15:25:51 -0500364// Use this to select which configurations (e.g. which renderer, which GLES major version) these
365// tests should be run against.
Geoff Lange0cc2a42016-01-20 10:58:17 -0500366ANGLE_INSTANTIATE_TEST(FramebufferFormatsTest,
Luc Ferronfa7503c2018-05-08 11:25:06 -0400367 ES2_VULKAN(),
Geoff Lange0cc2a42016-01-20 10:58:17 -0500368 ES2_D3D9(),
369 ES2_D3D11(),
370 ES3_D3D11(),
371 ES2_OPENGL(),
372 ES3_OPENGL(),
373 ES2_OPENGLES(),
374 ES3_OPENGLES());
Jamie Madill1fbc59f2016-02-24 15:25:51 -0500375
Corentin Wallez57e6d502016-12-09 14:46:39 -0500376class FramebufferTest_ES3 : public ANGLETest
Jamie Madill1fbc59f2016-02-24 15:25:51 -0500377{
Jamie Madill1fbc59f2016-02-24 15:25:51 -0500378};
379
380// Covers invalidating an incomplete framebuffer. This should be a no-op, but should not error.
Corentin Wallez57e6d502016-12-09 14:46:39 -0500381TEST_P(FramebufferTest_ES3, InvalidateIncomplete)
Jamie Madill1fbc59f2016-02-24 15:25:51 -0500382{
Geoff Lang857c09d2017-05-16 15:55:04 -0400383 GLFramebuffer framebuffer;
384 GLRenderbuffer renderbuffer;
385
386 glBindFramebuffer(GL_FRAMEBUFFER, framebuffer);
387 glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer);
388 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, renderbuffer);
Jamie Madill1fbc59f2016-02-24 15:25:51 -0500389 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
390 glCheckFramebufferStatus(GL_FRAMEBUFFER));
391
392 std::vector<GLenum> attachments;
393 attachments.push_back(GL_COLOR_ATTACHMENT0);
394
395 glInvalidateFramebuffer(GL_FRAMEBUFFER, 1, attachments.data());
396 EXPECT_GL_NO_ERROR();
397}
398
Corentin Wallez57e6d502016-12-09 14:46:39 -0500399// Test that the framebuffer state tracking robustly handles a depth-only attachment being set
400// as a depth-stencil attachment. It is equivalent to detaching the depth-stencil attachment.
401TEST_P(FramebufferTest_ES3, DepthOnlyAsDepthStencil)
402{
Geoff Lang857c09d2017-05-16 15:55:04 -0400403 GLFramebuffer framebuffer;
404 GLRenderbuffer renderbuffer;
405
406 glBindFramebuffer(GL_FRAMEBUFFER, framebuffer);
407 glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer);
Corentin Wallez57e6d502016-12-09 14:46:39 -0500408 glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH_COMPONENT16, 4, 4);
409
410 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER,
Geoff Lang857c09d2017-05-16 15:55:04 -0400411 renderbuffer);
Corentin Wallez57e6d502016-12-09 14:46:39 -0500412 EXPECT_GLENUM_NE(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
413}
414
Geoff Lang857c09d2017-05-16 15:55:04 -0400415// Test that the framebuffer correctly returns that it is not complete if invalid texture mip levels
416// are bound
417TEST_P(FramebufferTest_ES3, TextureAttachmentMipLevels)
418{
419 GLFramebuffer framebuffer;
420 glBindFramebuffer(GL_FRAMEBUFFER, framebuffer);
421
422 GLTexture texture;
423 glBindTexture(GL_TEXTURE_2D, texture);
424
425 // Create a complete mip chain in mips 1 to 3
426 glTexImage2D(GL_TEXTURE_2D, 1, GL_RGBA8, 4, 4, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
427 glTexImage2D(GL_TEXTURE_2D, 2, GL_RGBA8, 2, 2, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
428 glTexImage2D(GL_TEXTURE_2D, 3, GL_RGBA8, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
429
430 // Create another complete mip chain in mips 4 to 5
431 glTexImage2D(GL_TEXTURE_2D, 4, GL_RGBA8, 2, 2, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
432 glTexImage2D(GL_TEXTURE_2D, 5, GL_RGBA8, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
433
434 // Create a non-complete mip chain in mip 6
435 glTexImage2D(GL_TEXTURE_2D, 6, GL_RGBA8, 2, 2, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
436
437 // Incomplete, mipLevel != baseLevel and texture is not mip complete
438 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 1);
439 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
440 glCheckFramebufferStatus(GL_FRAMEBUFFER));
441
442 // Complete, mipLevel == baseLevel
443 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_BASE_LEVEL, 1);
444 ExpectFramebufferCompleteOrUnsupported(GL_FRAMEBUFFER);
445
446 // Complete, mipLevel != baseLevel but texture is now mip complete
447 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 2);
448 ExpectFramebufferCompleteOrUnsupported(GL_FRAMEBUFFER);
449 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 3);
450 ExpectFramebufferCompleteOrUnsupported(GL_FRAMEBUFFER);
451
452 // Incomplete, attached level below the base level
453 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_BASE_LEVEL, 2);
454 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 1);
455 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
456 glCheckFramebufferStatus(GL_FRAMEBUFFER));
457
458 // Incomplete, attached level is beyond effective max level
459 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 4);
460 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT,
461 glCheckFramebufferStatus(GL_FRAMEBUFFER));
462
463 // Complete, mipLevel == baseLevel
464 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_BASE_LEVEL, 4);
465 ExpectFramebufferCompleteOrUnsupported(GL_FRAMEBUFFER);
466
467 // Complete, mipLevel != baseLevel but texture is now mip complete
468 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 5);
469 ExpectFramebufferCompleteOrUnsupported(GL_FRAMEBUFFER);
470
471 // Complete, mipLevel == baseLevel
472 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_BASE_LEVEL, 6);
473 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 6);
474 ExpectFramebufferCompleteOrUnsupported(GL_FRAMEBUFFER);
475}
476
Martin Radevd178aa42017-07-13 14:03:22 +0300477// Test that passing an attachment COLOR_ATTACHMENTm where m is equal to MAX_COLOR_ATTACHMENTS
478// generates an INVALID_OPERATION.
479// OpenGL ES Version 3.0.5 (November 3, 2016), 4.4.2.4 Attaching Texture Images to a Framebuffer, p.
480// 208
481TEST_P(FramebufferTest_ES3, ColorAttachmentIndexOutOfBounds)
482{
483 GLFramebuffer framebuffer;
484 glBindFramebuffer(GL_FRAMEBUFFER, framebuffer.get());
485
486 GLint maxColorAttachments = 0;
487 glGetIntegerv(GL_MAX_COLOR_ATTACHMENTS, &maxColorAttachments);
488 GLenum attachment = static_cast<GLenum>(maxColorAttachments + GL_COLOR_ATTACHMENT0);
489
490 GLTexture texture;
491 glBindTexture(GL_TEXTURE_2D, texture.get());
492 glTexStorage2D(GL_TEXTURE_2D, 1, GL_RGBA32F, 1, 1);
493 glFramebufferTexture2D(GL_FRAMEBUFFER, attachment, GL_TEXTURE_2D, texture.get(), 0);
494 EXPECT_GL_ERROR(GL_INVALID_OPERATION);
495}
496
Jamie Madilla0016b72017-07-14 14:30:46 -0400497// Check that depth-only attachments report the correct number of samples.
498TEST_P(FramebufferTest_ES3, MultisampleDepthOnly)
499{
500 GLRenderbuffer renderbuffer;
501 glBindRenderbuffer(GL_RENDERBUFFER, renderbuffer);
502 glRenderbufferStorageMultisample(GL_RENDERBUFFER, 2, GL_DEPTH_COMPONENT24, 32, 32);
503
504 GLFramebuffer framebuffer;
505 glBindFramebuffer(GL_FRAMEBUFFER, framebuffer);
506 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, renderbuffer);
507 ASSERT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
508 EXPECT_GL_NO_ERROR();
509
510 GLint samples = 0;
511 glGetIntegerv(GL_SAMPLES, &samples);
512 EXPECT_GL_NO_ERROR();
513 EXPECT_GE(samples, 2);
514}
515
Jeff Gilbert8f8edd62017-10-31 14:26:30 -0700516// Check that we only compare width and height of attachments, not depth.
517TEST_P(FramebufferTest_ES3, AttachmentWith3DLayers)
518{
519 GLTexture texA;
520 glBindTexture(GL_TEXTURE_2D, texA);
521 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, 4, 4, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
522
523 GLTexture texB;
524 glBindTexture(GL_TEXTURE_3D, texB);
525 glTexImage3D(GL_TEXTURE_3D, 0, GL_RGBA8, 4, 4, 2, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
526
527 GLFramebuffer framebuffer;
528 glBindFramebuffer(GL_FRAMEBUFFER, framebuffer);
529 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texA, 0);
530 glFramebufferTextureLayer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, texB, 0, 0);
531 ASSERT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
532 EXPECT_GL_NO_ERROR();
533}
534
Olli Etuahodbce1f82018-09-19 15:32:17 +0300535// Test that clearing the stencil buffer when the framebuffer only has a color attachment does not
536// crash.
537TEST_P(FramebufferTest_ES3, ClearNonexistentStencil)
538{
539 GLRenderbuffer rbo;
540 glBindRenderbuffer(GL_RENDERBUFFER, rbo);
541 glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 1, 1);
542
543 GLFramebuffer fbo;
544 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
545 glFramebufferRenderbuffer(GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rbo);
546
547 GLint clearValue = 0;
548 glClearBufferiv(GL_STENCIL, 0, &clearValue);
549
550 // There's no error specified for clearing nonexistent buffers, it's simply a no-op.
551 EXPECT_GL_NO_ERROR();
552}
553
554// Test that clearing the depth buffer when the framebuffer only has a color attachment does not
555// crash.
556TEST_P(FramebufferTest_ES3, ClearNonexistentDepth)
557{
558 GLRenderbuffer rbo;
559 glBindRenderbuffer(GL_RENDERBUFFER, rbo);
560 glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 1, 1);
561
562 GLFramebuffer fbo;
563 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
564 glFramebufferRenderbuffer(GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rbo);
565
566 GLfloat clearValue = 0.0f;
567 glClearBufferfv(GL_DEPTH, 0, &clearValue);
568
569 // There's no error specified for clearing nonexistent buffers, it's simply a no-op.
570 EXPECT_GL_NO_ERROR();
571}
572
573// Test that clearing a nonexistent color attachment does not crash.
574TEST_P(FramebufferTest_ES3, ClearNonexistentColor)
575{
576 GLRenderbuffer rbo;
577 glBindRenderbuffer(GL_RENDERBUFFER, rbo);
578 glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 1, 1);
579
580 GLFramebuffer fbo;
581 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
582 glFramebufferRenderbuffer(GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rbo);
583
584 std::vector<GLfloat> clearValue = {{0.0f, 1.0f, 0.0f, 1.0f}};
585 glClearBufferfv(GL_COLOR, 1, clearValue.data());
586
587 // There's no error specified for clearing nonexistent buffers, it's simply a no-op.
588 EXPECT_GL_NO_ERROR();
589}
590
591// Test that clearing the depth and stencil buffers when the framebuffer only has a color attachment
592// does not crash.
593TEST_P(FramebufferTest_ES3, ClearNonexistentDepthStencil)
594{
595 GLRenderbuffer rbo;
596 glBindRenderbuffer(GL_RENDERBUFFER, rbo);
597 glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8, 1, 1);
598
599 GLFramebuffer fbo;
600 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
601 glFramebufferRenderbuffer(GL_READ_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rbo);
602
603 glClearBufferfi(GL_DEPTH_STENCIL, 0, 0.0f, 0);
604
605 // There's no error specified for clearing nonexistent buffers, it's simply a no-op.
606 EXPECT_GL_NO_ERROR();
607}
608
Corentin Wallez57e6d502016-12-09 14:46:39 -0500609ANGLE_INSTANTIATE_TEST(FramebufferTest_ES3, ES3_D3D11(), ES3_OPENGL(), ES3_OPENGLES());
JiangYizhou461d9a32017-01-04 16:37:26 +0800610
611class FramebufferTest_ES31 : public ANGLETest
612{
JiangYizhou511937d2017-08-03 15:41:29 +0800613 protected:
614 void validateSamplePass(GLuint &query, GLuint &passedCount, GLint width, GLint height)
615 {
616 glUniform2i(0, width - 1, height - 1);
617 glBeginQuery(GL_ANY_SAMPLES_PASSED, query);
618 glDrawArrays(GL_TRIANGLES, 0, 6);
619 glEndQuery(GL_ANY_SAMPLES_PASSED);
620 glGetQueryObjectuiv(query, GL_QUERY_RESULT, &passedCount);
621 EXPECT_GT(static_cast<GLint>(passedCount), 0);
622
623 glUniform2i(0, width - 1, height);
624 glBeginQuery(GL_ANY_SAMPLES_PASSED, query);
625 glDrawArrays(GL_TRIANGLES, 0, 6);
626 glEndQuery(GL_ANY_SAMPLES_PASSED);
627 glGetQueryObjectuiv(query, GL_QUERY_RESULT, &passedCount);
628 EXPECT_EQ(static_cast<GLint>(passedCount), 0);
629
630 glUniform2i(0, width, height - 1);
631 glBeginQuery(GL_ANY_SAMPLES_PASSED, query);
632 glDrawArrays(GL_TRIANGLES, 0, 6);
633 glEndQuery(GL_ANY_SAMPLES_PASSED);
634 glGetQueryObjectuiv(query, GL_QUERY_RESULT, &passedCount);
635 EXPECT_EQ(static_cast<GLint>(passedCount), 0);
636 }
JiangYizhou461d9a32017-01-04 16:37:26 +0800637};
638
639// Test that without attachment, if either the value of FRAMEBUFFER_DEFAULT_WIDTH or
640// FRAMEBUFFER_DEFAULT_HEIGHT parameters is zero, the framebuffer is incomplete.
641TEST_P(FramebufferTest_ES31, IncompleteMissingAttachmentDefaultParam)
642{
643 GLFramebuffer mFramebuffer;
644 glBindFramebuffer(GL_FRAMEBUFFER, mFramebuffer.get());
645
646 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_WIDTH, 1);
647 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_HEIGHT, 1);
648 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
649
650 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_WIDTH, 0);
651 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_HEIGHT, 0);
652 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT,
653 glCheckFramebufferStatus(GL_FRAMEBUFFER));
654
655 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_WIDTH, 1);
656 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_HEIGHT, 0);
657 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT,
658 glCheckFramebufferStatus(GL_FRAMEBUFFER));
659
660 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_WIDTH, 0);
661 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_HEIGHT, 1);
662 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT,
663 glCheckFramebufferStatus(GL_FRAMEBUFFER));
664
665 ASSERT_GL_NO_ERROR();
666}
667
668// Test that the sample count of a mix of texture and renderbuffer should be same.
669TEST_P(FramebufferTest_ES31, IncompleteMultisampleSampleCountMix)
670{
671 GLFramebuffer mFramebuffer;
672 glBindFramebuffer(GL_FRAMEBUFFER, mFramebuffer.get());
673
674 GLTexture mTexture;
675 glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, mTexture.get());
676 glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, 1, GL_RGBA8, 1, 1, true);
677
678 GLRenderbuffer mRenderbuffer;
679 glBindRenderbuffer(GL_RENDERBUFFER, mRenderbuffer.get());
680 glRenderbufferStorageMultisample(GL_RENDERBUFFER, 2, GL_RGBA8, 1, 1);
681 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D_MULTISAMPLE,
682 mTexture.get(), 0);
683 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_RENDERBUFFER,
684 mRenderbuffer.get());
685 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE,
686 glCheckFramebufferStatus(GL_FRAMEBUFFER));
687
688 ASSERT_GL_NO_ERROR();
689}
690
691// Test that the sample count of texture attachments should be same.
692TEST_P(FramebufferTest_ES31, IncompleteMultisampleSampleCountTex)
693{
694 GLFramebuffer mFramebuffer;
695 glBindFramebuffer(GL_FRAMEBUFFER, mFramebuffer.get());
696
697 GLTexture mTextures[2];
698 glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, mTextures[0].get());
699 glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, 1, GL_RGBA8, 1, 1, true);
700 glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, mTextures[1].get());
701 glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, 2, GL_RGBA8, 1, 1, true);
702 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D_MULTISAMPLE,
703 mTextures[0].get(), 0);
704 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_TEXTURE_2D_MULTISAMPLE,
705 mTextures[1].get(), 0);
706 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE,
707 glCheckFramebufferStatus(GL_FRAMEBUFFER));
708
709 ASSERT_GL_NO_ERROR();
710}
711
712// Test that if the attached images are a mix of renderbuffers and textures, the value of
713// TEXTURE_FIXED_SAMPLE_LOCATIONS must be TRUE for all attached textures.
714TEST_P(FramebufferTest_ES31, IncompleteMultisampleFixedSampleLocationsMix)
715{
716 GLFramebuffer mFramebuffer;
717 glBindFramebuffer(GL_FRAMEBUFFER, mFramebuffer.get());
718
719 GLTexture mTexture;
720 glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, mTexture.get());
721 glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, 1, GL_RGBA8, 1, 1, false);
722
723 GLRenderbuffer mRenderbuffer;
724 glBindRenderbuffer(GL_RENDERBUFFER, mRenderbuffer.get());
725 glRenderbufferStorageMultisample(GL_RENDERBUFFER, 1, GL_RGBA8, 1, 1);
726 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D_MULTISAMPLE,
727 mTexture.get(), 0);
728 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_RENDERBUFFER,
729 mRenderbuffer.get());
730 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE,
731 glCheckFramebufferStatus(GL_FRAMEBUFFER));
732
733 ASSERT_GL_NO_ERROR();
734}
735
736// Test that the value of TEXTURE_FIXED_SAMPLE_LOCATIONS is the same for all attached textures.
737TEST_P(FramebufferTest_ES31, IncompleteMultisampleFixedSampleLocationsTex)
738{
739 GLFramebuffer mFramebuffer;
740 glBindFramebuffer(GL_FRAMEBUFFER, mFramebuffer.get());
741
742 GLTexture mTextures[2];
743 glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, mTextures[0].get());
744 glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, 1, GL_RGBA8, 1, 1, false);
745 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D_MULTISAMPLE,
746 mTextures[0].get(), 0);
747 glBindTexture(GL_TEXTURE_2D_MULTISAMPLE, mTextures[1].get());
748 glTexStorage2DMultisample(GL_TEXTURE_2D_MULTISAMPLE, 1, GL_RGB8, 1, 1, true);
749 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_TEXTURE_2D_MULTISAMPLE,
750 mTextures[1].get(), 0);
751 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE,
752 glCheckFramebufferStatus(GL_FRAMEBUFFER));
753
754 ASSERT_GL_NO_ERROR();
755}
756
JiangYizhou511937d2017-08-03 15:41:29 +0800757// If there are no attachments, rendering will be limited to a rectangle having a lower left of
758// (0, 0) and an upper right of(width, height), where width and height are the framebuffer
759// object's default width and height.
JiangYizhou3db40722017-08-28 17:59:13 +0800760TEST_P(FramebufferTest_ES31, RenderingLimitToDefaultFBOSizeWithNoAttachments)
JiangYizhou511937d2017-08-03 15:41:29 +0800761{
Yuly Novikov98f9f532017-11-15 19:16:19 -0500762 // anglebug.com/2253
763 ANGLE_SKIP_TEST_IF(IsLinux() && IsAMD() && IsDesktopOpenGL());
764
JiangYizhou10d41392017-12-18 18:13:36 +0800765 const std::string &vertexShader1 =
766 R"(#version 310 es
767 in layout(location = 0) highp vec2 a_position;
768 void main()
769 {
770 gl_Position = vec4(a_position, 0.0, 1.0);
771 })";
JiangYizhou511937d2017-08-03 15:41:29 +0800772
JiangYizhou10d41392017-12-18 18:13:36 +0800773 const std::string &fragShader1 =
774 R"(#version 310 es
775 uniform layout(location = 0) highp ivec2 u_expectedSize;
776 out layout(location = 5) mediump vec4 f_color;
777 void main()
778 {
779 if (ivec2(gl_FragCoord.xy) != u_expectedSize) discard;
780 f_color = vec4(1.0, 0.5, 0.25, 1.0);
781 })";
JiangYizhou511937d2017-08-03 15:41:29 +0800782
JiangYizhou10d41392017-12-18 18:13:36 +0800783 const std::string &vertexShader2 =
784 R"(#version 310 es
785 in layout(location = 0) highp vec2 a_position;
786 void main()
787 {
788 gl_Position = vec4(a_position, 0.0, 1.0);
789 })";
790
791 const std::string &fragShader2 =
792 R"(#version 310 es
793 uniform layout(location = 0) highp ivec2 u_expectedSize;
794 out layout(location = 2) mediump vec4 f_color;
795 void main()
796 {
797 if (ivec2(gl_FragCoord.xy) != u_expectedSize) discard;
798 f_color = vec4(1.0, 0.5, 0.25, 1.0);
799 })";
800
801 GLuint program1 = CompileProgram(vertexShader1, fragShader1);
802 ASSERT_NE(program1, 0u);
803
804 GLuint program2 = CompileProgram(vertexShader2, fragShader2);
805 ASSERT_NE(program2, 0u);
806
807 glUseProgram(program1);
JiangYizhou511937d2017-08-03 15:41:29 +0800808
809 GLFramebuffer mFramebuffer;
810 glBindFramebuffer(GL_DRAW_FRAMEBUFFER, mFramebuffer);
811 GLuint defaultWidth = 1;
812 GLuint defaultHeight = 1;
813
814 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_WIDTH, defaultWidth);
815 glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_HEIGHT, defaultHeight);
816 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
817
818 const float data[] = {
819 1.0f, 1.0f, 1.0f, -1.0f, -1.0f, 1.0f, -1.0f, 1.0f, 1.0f, -1.0f, -1.0f, -1.0f,
820 };
821
822 GLuint vertexArray = 0;
823 GLuint vertexBuffer = 0;
824 GLuint query = 0;
825 GLuint passedCount = 0;
826
827 glGenQueries(1, &query);
828 glGenVertexArrays(1, &vertexArray);
829 glBindVertexArray(vertexArray);
830
831 glGenBuffers(1, &vertexBuffer);
832 glBindBuffer(GL_ARRAY_BUFFER, vertexBuffer);
833 glBufferData(GL_ARRAY_BUFFER, sizeof(data), data, GL_STATIC_DRAW);
834
835 glEnableVertexAttribArray(0);
836 glVertexAttribPointer(0, 2, GL_FLOAT, false, 0, 0);
JiangYizhou38d92b52017-09-13 13:47:52 +0800837 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
JiangYizhou511937d2017-08-03 15:41:29 +0800838
839 validateSamplePass(query, passedCount, defaultWidth, defaultHeight);
840
JiangYizhou10d41392017-12-18 18:13:36 +0800841 glUseProgram(program2);
842 validateSamplePass(query, passedCount, defaultWidth, defaultHeight);
843
844 glUseProgram(program1);
JiangYizhou511937d2017-08-03 15:41:29 +0800845 // If fbo has attachments, the rendering size should be the same as its attachment.
846 GLTexture mTexture;
847 GLuint width = 2;
848 GLuint height = 2;
849 glBindTexture(GL_TEXTURE_2D, mTexture.get());
850 glTexStorage2D(GL_TEXTURE_2D, 1, GL_RGBA8, width, height);
JiangYizhou38d92b52017-09-13 13:47:52 +0800851
852 const GLenum bufs[] = {GL_NONE, GL_NONE, GL_NONE, GL_NONE, GL_NONE, GL_COLOR_ATTACHMENT5};
853
854 glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT5, GL_TEXTURE_2D, mTexture.get(),
JiangYizhou511937d2017-08-03 15:41:29 +0800855 0);
856 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
JiangYizhou38d92b52017-09-13 13:47:52 +0800857 glDrawBuffers(6, bufs);
JiangYizhou511937d2017-08-03 15:41:29 +0800858
859 validateSamplePass(query, passedCount, width, height);
860
861 // If fbo's attachment has been removed, the rendering size should be the same as framebuffer
862 // default size.
JiangYizhou38d92b52017-09-13 13:47:52 +0800863 glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT5, 0, 0, 0);
JiangYizhou511937d2017-08-03 15:41:29 +0800864 EXPECT_GLENUM_EQ(GL_FRAMEBUFFER_COMPLETE, glCheckFramebufferStatus(GL_FRAMEBUFFER));
865
866 validateSamplePass(query, passedCount, defaultWidth, defaultHeight);
867
868 glDisableVertexAttribArray(0);
869 glBindBuffer(GL_ARRAY_BUFFER, 0);
870 glBindVertexArray(0);
871 glDeleteBuffers(1, &vertexBuffer);
872 glDeleteVertexArrays(1, &vertexArray);
873
874 ASSERT_GL_NO_ERROR();
875}
876
877ANGLE_INSTANTIATE_TEST(FramebufferTest_ES31, ES31_D3D11(), ES31_OPENGL(), ES31_OPENGLES());
Brandon Jones9fc87332017-12-13 15:46:52 -0800878
879class AddDummyTextureNoRenderTargetTest : public ANGLETest
880{
881 public:
882 AddDummyTextureNoRenderTargetTest()
883 {
884 setWindowWidth(512);
885 setWindowHeight(512);
886 setConfigRedBits(8);
887 setConfigGreenBits(8);
888 setConfigBlueBits(8);
889 setConfigAlphaBits(8);
890 }
891
892 void overrideWorkaroundsD3D(WorkaroundsD3D *workarounds) override
893 {
894 workarounds->addDummyTextureNoRenderTarget = true;
895 }
896};
897
898// Test to verify workaround succeeds when no program outputs exist http://anglebug.com/2283
899TEST_P(AddDummyTextureNoRenderTargetTest, NoProgramOutputWorkaround)
900{
901 const std::string &vShader = "void main() {}";
902 const std::string &fShader = "void main() {}";
903
904 ANGLE_GL_PROGRAM(drawProgram, vShader, fShader);
905
906 glUseProgram(drawProgram);
907
908 glDrawArrays(GL_TRIANGLES, 0, 6);
909
910 ASSERT_GL_NO_ERROR();
911}
912
913ANGLE_INSTANTIATE_TEST(AddDummyTextureNoRenderTargetTest, ES2_D3D11());