Jamie Madill | fa05f60 | 2015-05-07 13:47:11 -0400 | [diff] [blame] | 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 | |
Corentin Wallez | d3970de | 2015-05-14 11:07:48 -0400 | [diff] [blame] | 7 | #include "test_utils/ANGLETest.h" |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 8 | |
Jamie Madill | fa05f60 | 2015-05-07 13:47:11 -0400 | [diff] [blame] | 9 | using namespace angle; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 10 | |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 11 | class FramebufferRenderMipmapTest : public ANGLETest |
| 12 | { |
Jamie Madill | fa05f60 | 2015-05-07 13:47:11 -0400 | [diff] [blame] | 13 | protected: |
| 14 | FramebufferRenderMipmapTest() |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 15 | { |
| 16 | setWindowWidth(256); |
| 17 | setWindowHeight(256); |
| 18 | setConfigRedBits(8); |
| 19 | setConfigGreenBits(8); |
| 20 | setConfigBlueBits(8); |
| 21 | setConfigAlphaBits(8); |
| 22 | } |
| 23 | |
| 24 | virtual void SetUp() |
| 25 | { |
| 26 | ANGLETest::SetUp(); |
| 27 | |
Olli Etuaho | a20af6d | 2017-09-18 13:32:29 +0300 | [diff] [blame^] | 28 | const std::string vsSource = |
| 29 | R"(attribute highp vec4 position; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 30 | void main(void) |
| 31 | { |
| 32 | gl_Position = position; |
Olli Etuaho | a20af6d | 2017-09-18 13:32:29 +0300 | [diff] [blame^] | 33 | })"; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 34 | |
Olli Etuaho | a20af6d | 2017-09-18 13:32:29 +0300 | [diff] [blame^] | 35 | const std::string fsSource = |
| 36 | R"(uniform highp vec4 color; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 37 | void main(void) |
| 38 | { |
| 39 | gl_FragColor = color; |
Olli Etuaho | a20af6d | 2017-09-18 13:32:29 +0300 | [diff] [blame^] | 40 | })"; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 41 | |
| 42 | mProgram = CompileProgram(vsSource, fsSource); |
| 43 | if (mProgram == 0) |
| 44 | { |
| 45 | FAIL() << "shader compilation failed."; |
| 46 | } |
| 47 | |
| 48 | mColorLocation = glGetUniformLocation(mProgram, "color"); |
| 49 | |
| 50 | glUseProgram(mProgram); |
| 51 | |
| 52 | glClearColor(0, 0, 0, 0); |
| 53 | glClearDepthf(0.0); |
| 54 | glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); |
| 55 | |
| 56 | glEnable(GL_BLEND); |
| 57 | glDisable(GL_DEPTH_TEST); |
| 58 | |
| 59 | ASSERT_GL_NO_ERROR(); |
| 60 | } |
| 61 | |
| 62 | virtual void TearDown() |
| 63 | { |
| 64 | glDeleteProgram(mProgram); |
| 65 | |
| 66 | ANGLETest::TearDown(); |
| 67 | } |
| 68 | |
| 69 | GLuint mProgram; |
| 70 | GLint mColorLocation; |
| 71 | }; |
| 72 | |
| 73 | // Validate that if we are in ES3 or GL_OES_fbo_render_mipmap exists, there are no validation errors |
| 74 | // when using a non-zero level in glFramebufferTexture2D. |
Jamie Madill | fa05f60 | 2015-05-07 13:47:11 -0400 | [diff] [blame] | 75 | TEST_P(FramebufferRenderMipmapTest, Validation) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 76 | { |
Martin Radev | 1be913c | 2016-07-11 17:59:16 +0300 | [diff] [blame] | 77 | bool renderToMipmapSupported = |
| 78 | extensionEnabled("GL_OES_fbo_render_mipmap") || getClientMajorVersion() > 2; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 79 | |
| 80 | GLuint tex = 0; |
| 81 | glGenTextures(1, &tex); |
| 82 | glBindTexture(GL_TEXTURE_2D, tex); |
| 83 | |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 84 | const GLint levels = 5; |
| 85 | for (GLint i = 0; i < levels; i++) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 86 | { |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 87 | GLsizei size = 1 << ((levels - 1) - i); |
Yunchao He | f81ce4a | 2017-04-24 10:49:17 +0800 | [diff] [blame] | 88 | glTexImage2D(GL_TEXTURE_2D, i, GL_RGBA, size, size, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr); |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 89 | } |
| 90 | |
| 91 | EXPECT_GL_NO_ERROR(); |
| 92 | |
| 93 | GLuint fbo = 0; |
| 94 | glGenFramebuffers(1, &fbo); |
| 95 | glBindFramebuffer(GL_FRAMEBUFFER, fbo); |
| 96 | EXPECT_GL_NO_ERROR(); |
| 97 | |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 98 | for (GLint i = 0; i < levels; i++) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 99 | { |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 100 | glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, tex, i); |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 101 | |
| 102 | if (i > 0 && !renderToMipmapSupported) |
| 103 | { |
| 104 | EXPECT_GL_ERROR(GL_INVALID_VALUE); |
| 105 | } |
| 106 | else |
| 107 | { |
| 108 | EXPECT_GL_NO_ERROR(); |
| 109 | EXPECT_EQ(glCheckFramebufferStatus(GL_FRAMEBUFFER), GLenum(GL_FRAMEBUFFER_COMPLETE)); |
| 110 | } |
| 111 | } |
| 112 | |
| 113 | glDeleteFramebuffers(1, &fbo); |
| 114 | glDeleteTextures(1, &tex); |
| 115 | } |
| 116 | |
| 117 | // Render to various levels of a texture and check that they have the correct color data via ReadPixels |
Jamie Madill | fa05f60 | 2015-05-07 13:47:11 -0400 | [diff] [blame] | 118 | TEST_P(FramebufferRenderMipmapTest, RenderToMipmap) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 119 | { |
Geoff Lang | 1a0847c | 2015-04-08 13:49:31 -0400 | [diff] [blame] | 120 | // TODO(geofflang): Figure out why this is broken on Intel OpenGL |
Jamie Madill | 518b9fa | 2016-03-02 11:26:02 -0500 | [diff] [blame] | 121 | if (IsIntel() && getPlatformRenderer() == EGL_PLATFORM_ANGLE_TYPE_OPENGL_ANGLE) |
Geoff Lang | 1a0847c | 2015-04-08 13:49:31 -0400 | [diff] [blame] | 122 | { |
| 123 | std::cout << "Test skipped on Intel OpenGL." << std::endl; |
| 124 | return; |
| 125 | } |
| 126 | |
Martin Radev | 1be913c | 2016-07-11 17:59:16 +0300 | [diff] [blame] | 127 | bool renderToMipmapSupported = |
| 128 | extensionEnabled("GL_OES_fbo_render_mipmap") || getClientMajorVersion() > 2; |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 129 | if (!renderToMipmapSupported) |
| 130 | { |
| 131 | std::cout << "Test skipped because GL_OES_fbo_render_mipmap or ES3 is not available." << std::endl; |
| 132 | return; |
| 133 | } |
| 134 | |
| 135 | const GLfloat levelColors[] = |
| 136 | { |
| 137 | 1.0f, 0.0f, 0.0f, 1.0f, |
| 138 | 0.0f, 1.0f, 0.0f, 1.0f, |
| 139 | 0.0f, 0.0f, 1.0f, 1.0f, |
| 140 | 1.0f, 1.0f, 0.0f, 1.0f, |
| 141 | 1.0f, 0.0f, 1.0f, 1.0f, |
| 142 | 0.0f, 1.0f, 1.0f, 1.0f, |
| 143 | }; |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 144 | const GLint testLevels = static_cast<GLint>(ArraySize(levelColors) / 4); |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 145 | |
| 146 | GLuint tex = 0; |
| 147 | glGenTextures(1, &tex); |
| 148 | glBindTexture(GL_TEXTURE_2D, tex); |
| 149 | |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 150 | for (GLint i = 0; i < testLevels; i++) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 151 | { |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 152 | GLsizei size = 1 << ((testLevels - 1) - i); |
Yunchao He | f81ce4a | 2017-04-24 10:49:17 +0800 | [diff] [blame] | 153 | glTexImage2D(GL_TEXTURE_2D, i, GL_RGBA, size, size, 0, GL_RGBA, GL_UNSIGNED_BYTE, nullptr); |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 154 | } |
| 155 | |
| 156 | EXPECT_GL_NO_ERROR(); |
| 157 | |
| 158 | GLuint fbo = 0; |
| 159 | glGenFramebuffers(1, &fbo); |
| 160 | glBindFramebuffer(GL_FRAMEBUFFER, fbo); |
| 161 | EXPECT_GL_NO_ERROR(); |
| 162 | |
| 163 | // Render to the levels of the texture with different colors |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 164 | for (GLint i = 0; i < testLevels; i++) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 165 | { |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 166 | glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, tex, i); |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 167 | EXPECT_GL_NO_ERROR(); |
| 168 | |
| 169 | glUseProgram(mProgram); |
| 170 | glUniform4fv(mColorLocation, 1, levelColors + (i * 4)); |
| 171 | |
| 172 | drawQuad(mProgram, "position", 0.5f); |
| 173 | EXPECT_GL_NO_ERROR(); |
| 174 | } |
| 175 | |
| 176 | // Test that the levels of the texture are correct |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 177 | for (GLint i = 0; i < testLevels; i++) |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 178 | { |
Bruce Dawson | e2fcf5c | 2016-03-30 11:27:12 -0700 | [diff] [blame] | 179 | glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, tex, i); |
Geoff Lang | 9566391 | 2015-04-02 15:54:45 -0400 | [diff] [blame] | 180 | EXPECT_GL_NO_ERROR(); |
| 181 | |
| 182 | const GLfloat *color = levelColors + (i * 4); |
| 183 | EXPECT_PIXEL_EQ(0, 0, color[0] * 255, color[1] * 255, color[2] * 255, color[3] * 255); |
| 184 | } |
| 185 | |
| 186 | glDeleteFramebuffers(1, &fbo); |
| 187 | glDeleteTextures(1, &tex); |
| 188 | |
| 189 | EXPECT_GL_NO_ERROR(); |
| 190 | } |
Jamie Madill | fa05f60 | 2015-05-07 13:47:11 -0400 | [diff] [blame] | 191 | |
| 192 | // Use this to select which configurations (e.g. which renderer, which GLES major version) these tests should be run against. |
Geoff Lang | e0cc2a4 | 2016-01-20 10:58:17 -0500 | [diff] [blame] | 193 | ANGLE_INSTANTIATE_TEST(FramebufferRenderMipmapTest, |
| 194 | ES2_D3D9(), |
| 195 | ES2_D3D11(), |
| 196 | ES3_D3D11(), |
| 197 | ES2_OPENGL(), |
| 198 | ES3_OPENGL(), |
| 199 | ES2_OPENGLES(), |
| 200 | ES3_OPENGLES()); |