1 /*
2 * Copyright 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "GLTest.h"
18
19 #include <gui/Surface.h>
20
21 #include <GLES2/gl2.h>
22
23 namespace android {
24
25 using Transaction = SurfaceComposerClient::Transaction;
26
abs(int value)27 static int abs(int value) {
28 return value > 0 ? value : -value;
29 }
30
SetUp()31 void GLTest::SetUp() {
32 const ::testing::TestInfo* const testInfo =
33 ::testing::UnitTest::GetInstance()->current_test_info();
34 ALOGV("Begin test: %s.%s", testInfo->test_case_name(), testInfo->name());
35
36 mEglDisplay = eglGetDisplay(EGL_DEFAULT_DISPLAY);
37 ASSERT_EQ(EGL_SUCCESS, eglGetError());
38 ASSERT_NE(EGL_NO_DISPLAY, mEglDisplay);
39
40 EGLint majorVersion;
41 EGLint minorVersion;
42 EXPECT_TRUE(eglInitialize(mEglDisplay, &majorVersion, &minorVersion));
43 ASSERT_EQ(EGL_SUCCESS, eglGetError());
44 RecordProperty("EglVersionMajor", majorVersion);
45 RecordProperty("EglVersionMinor", minorVersion);
46
47 EGLint numConfigs = 0;
48 EXPECT_TRUE(eglChooseConfig(mEglDisplay, getConfigAttribs(), &mGlConfig, 1,
49 &numConfigs));
50 ASSERT_EQ(EGL_SUCCESS, eglGetError());
51
52 char* displaySecsEnv = getenv("GLTEST_DISPLAY_SECS");
53 if (displaySecsEnv != NULL) {
54 mDisplaySecs = atoi(displaySecsEnv);
55 if (mDisplaySecs < 0) {
56 mDisplaySecs = 0;
57 }
58 } else {
59 mDisplaySecs = 0;
60 }
61
62 if (mDisplaySecs > 0) {
63 mComposerClient = new SurfaceComposerClient;
64 ASSERT_EQ(NO_ERROR, mComposerClient->initCheck());
65
66 mSurfaceControl = mComposerClient->createSurface(
67 String8("Test Surface"), getSurfaceWidth(), getSurfaceHeight(),
68 PIXEL_FORMAT_RGB_888, 0);
69
70 ASSERT_TRUE(mSurfaceControl != NULL);
71 ASSERT_TRUE(mSurfaceControl->isValid());
72
73 Transaction t;
74 ASSERT_EQ(NO_ERROR, t.setLayer(mSurfaceControl, 0x7FFFFFFF)
75 .show(mSurfaceControl)
76 .apply());
77
78 sp<ANativeWindow> window = mSurfaceControl->getSurface();
79 mEglSurface = createWindowSurface(mEglDisplay, mGlConfig, window);
80 } else {
81 EGLint pbufferAttribs[] = {
82 EGL_WIDTH, getSurfaceWidth(),
83 EGL_HEIGHT, getSurfaceHeight(),
84 EGL_NONE };
85
86 mEglSurface = eglCreatePbufferSurface(mEglDisplay, mGlConfig,
87 pbufferAttribs);
88 }
89 ASSERT_EQ(EGL_SUCCESS, eglGetError());
90 ASSERT_NE(EGL_NO_SURFACE, mEglSurface);
91
92 mEglContext = eglCreateContext(mEglDisplay, mGlConfig, EGL_NO_CONTEXT,
93 getContextAttribs());
94 ASSERT_EQ(EGL_SUCCESS, eglGetError());
95 ASSERT_NE(EGL_NO_CONTEXT, mEglContext);
96
97 EXPECT_TRUE(eglMakeCurrent(mEglDisplay, mEglSurface, mEglSurface,
98 mEglContext));
99 ASSERT_EQ(EGL_SUCCESS, eglGetError());
100
101 EGLint w, h;
102 EXPECT_TRUE(eglQuerySurface(mEglDisplay, mEglSurface, EGL_WIDTH, &w));
103 ASSERT_EQ(EGL_SUCCESS, eglGetError());
104 EXPECT_TRUE(eglQuerySurface(mEglDisplay, mEglSurface, EGL_HEIGHT, &h));
105 ASSERT_EQ(EGL_SUCCESS, eglGetError());
106 RecordProperty("EglSurfaceWidth", w);
107 RecordProperty("EglSurfaceHeight", h);
108
109 glViewport(0, 0, w, h);
110 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
111 }
112
TearDown()113 void GLTest::TearDown() {
114 // Display the result
115 if (mDisplaySecs > 0 && mEglSurface != EGL_NO_SURFACE) {
116 eglSwapBuffers(mEglDisplay, mEglSurface);
117 sleep(mDisplaySecs);
118 }
119
120 if (mComposerClient != NULL) {
121 mComposerClient->dispose();
122 }
123 if (mEglContext != EGL_NO_CONTEXT) {
124 eglDestroyContext(mEglDisplay, mEglContext);
125 }
126 if (mEglSurface != EGL_NO_SURFACE) {
127 eglDestroySurface(mEglDisplay, mEglSurface);
128 }
129 if (mEglDisplay != EGL_NO_DISPLAY) {
130 eglMakeCurrent(mEglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE,
131 EGL_NO_CONTEXT);
132 eglTerminate(mEglDisplay);
133 }
134 ASSERT_EQ(EGL_SUCCESS, eglGetError());
135
136 const ::testing::TestInfo* const testInfo =
137 ::testing::UnitTest::GetInstance()->current_test_info();
138 ALOGV("End test: %s.%s", testInfo->test_case_name(), testInfo->name());
139 }
140
getConfigAttribs()141 EGLint const* GLTest::getConfigAttribs() {
142 static const EGLint sDefaultConfigAttribs[] = {
143 EGL_SURFACE_TYPE, EGL_PBUFFER_BIT,
144 EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
145 EGL_RED_SIZE, 8,
146 EGL_GREEN_SIZE, 8,
147 EGL_BLUE_SIZE, 8,
148 EGL_ALPHA_SIZE, 8,
149 EGL_DEPTH_SIZE, 16,
150 EGL_STENCIL_SIZE, 8,
151 EGL_NONE };
152
153 return sDefaultConfigAttribs;
154 }
155
getContextAttribs()156 EGLint const* GLTest::getContextAttribs() {
157 static const EGLint sDefaultContextAttribs[] = {
158 EGL_CONTEXT_CLIENT_VERSION, 2,
159 EGL_NONE };
160
161 return sDefaultContextAttribs;
162 }
163
getSurfaceWidth()164 EGLint GLTest::getSurfaceWidth() {
165 return 512;
166 }
167
getSurfaceHeight()168 EGLint GLTest::getSurfaceHeight() {
169 return 512;
170 }
171
createWindowSurface(EGLDisplay display,EGLConfig config,sp<ANativeWindow> & window) const172 EGLSurface GLTest::createWindowSurface(EGLDisplay display, EGLConfig config,
173 sp<ANativeWindow>& window) const {
174 return eglCreateWindowSurface(display, config, window.get(), NULL);
175 }
176
checkPixel(int x,int y,int r,int g,int b,int a,int tolerance)177 ::testing::AssertionResult GLTest::checkPixel(int x, int y,
178 int r, int g, int b, int a, int tolerance) {
179 GLubyte pixel[4];
180 String8 msg;
181 glReadPixels(x, y, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, pixel);
182 GLenum err = glGetError();
183 if (err != GL_NO_ERROR) {
184 msg += String8::format("error reading pixel: %#x", err);
185 while ((err = glGetError()) != GL_NO_ERROR) {
186 msg += String8::format(", %#x", err);
187 }
188 return ::testing::AssertionFailure(::testing::Message(msg.string()));
189 }
190 if (r >= 0 && abs(r - int(pixel[0])) > tolerance) {
191 msg += String8::format("r(%d isn't %d)", pixel[0], r);
192 }
193 if (g >= 0 && abs(g - int(pixel[1])) > tolerance) {
194 if (!msg.isEmpty()) {
195 msg += " ";
196 }
197 msg += String8::format("g(%d isn't %d)", pixel[1], g);
198 }
199 if (b >= 0 && abs(b - int(pixel[2])) > tolerance) {
200 if (!msg.isEmpty()) {
201 msg += " ";
202 }
203 msg += String8::format("b(%d isn't %d)", pixel[2], b);
204 }
205 if (a >= 0 && abs(a - int(pixel[3])) > tolerance) {
206 if (!msg.isEmpty()) {
207 msg += " ";
208 }
209 msg += String8::format("a(%d isn't %d)", pixel[3], a);
210 }
211 if (!msg.isEmpty()) {
212 return ::testing::AssertionFailure(::testing::Message(msg.string()));
213 } else {
214 return ::testing::AssertionSuccess();
215 }
216 }
217
assertRectEq(const Rect & r1,const Rect & r2,int tolerance)218 ::testing::AssertionResult GLTest::assertRectEq(const Rect &r1, const Rect &r2,
219 int tolerance) {
220 String8 msg;
221
222 if (abs(r1.left - r2.left) > tolerance) {
223 msg += String8::format("left(%d isn't %d)", r1.left, r2.left);
224 }
225 if (abs(r1.top - r2.top) > tolerance) {
226 if (!msg.isEmpty()) {
227 msg += " ";
228 }
229 msg += String8::format("top(%d isn't %d)", r1.top, r2.top);
230 }
231 if (abs(r1.right - r2.right) > tolerance) {
232 if (!msg.isEmpty()) {
233 msg += " ";
234 }
235 msg += String8::format("right(%d isn't %d)", r1.right, r2.right);
236 }
237 if (abs(r1.bottom - r2.bottom) > tolerance) {
238 if (!msg.isEmpty()) {
239 msg += " ";
240 }
241 msg += String8::format("bottom(%d isn't %d)", r1.bottom, r2.bottom);
242 }
243 if (!msg.isEmpty()) {
244 msg += String8::format(" R1: [%d %d %d %d] R2: [%d %d %d %d]",
245 r1.left, r1.top, r1.right, r1.bottom,
246 r2.left, r2.top, r2.right, r2.bottom);
247 fprintf(stderr, "assertRectEq: %s\n", msg.string());
248 return ::testing::AssertionFailure(::testing::Message(msg.string()));
249 } else {
250 return ::testing::AssertionSuccess();
251 }
252 }
253
loadShader(GLenum shaderType,const char * pSource,GLuint * outShader)254 void GLTest::loadShader(GLenum shaderType, const char* pSource,
255 GLuint* outShader) {
256 GLuint shader = glCreateShader(shaderType);
257 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
258 if (shader) {
259 glShaderSource(shader, 1, &pSource, NULL);
260 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
261 glCompileShader(shader);
262 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
263 GLint compiled = 0;
264 glGetShaderiv(shader, GL_COMPILE_STATUS, &compiled);
265 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
266 if (!compiled) {
267 GLint infoLen = 0;
268 glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &infoLen);
269 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
270 if (infoLen) {
271 char* buf = (char*) malloc(infoLen);
272 if (buf) {
273 glGetShaderInfoLog(shader, infoLen, NULL, buf);
274 printf("Shader compile log:\n%s\n", buf);
275 free(buf);
276 FAIL();
277 }
278 } else {
279 char* buf = (char*) malloc(0x1000);
280 if (buf) {
281 glGetShaderInfoLog(shader, 0x1000, NULL, buf);
282 printf("Shader compile log:\n%s\n", buf);
283 free(buf);
284 FAIL();
285 }
286 }
287 glDeleteShader(shader);
288 shader = 0;
289 }
290 }
291 ASSERT_TRUE(shader != 0);
292 *outShader = shader;
293 }
294
createProgram(const char * pVertexSource,const char * pFragmentSource,GLuint * outPgm)295 void GLTest::createProgram(const char* pVertexSource,
296 const char* pFragmentSource, GLuint* outPgm) {
297 GLuint vertexShader, fragmentShader;
298 {
299 SCOPED_TRACE("compiling vertex shader");
300 ASSERT_NO_FATAL_FAILURE(loadShader(GL_VERTEX_SHADER, pVertexSource,
301 &vertexShader));
302 }
303 {
304 SCOPED_TRACE("compiling fragment shader");
305 ASSERT_NO_FATAL_FAILURE(loadShader(GL_FRAGMENT_SHADER, pFragmentSource,
306 &fragmentShader));
307 }
308
309 GLuint program = glCreateProgram();
310 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
311 if (program) {
312 glAttachShader(program, vertexShader);
313 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
314 glAttachShader(program, fragmentShader);
315 ASSERT_EQ(GLenum(GL_NO_ERROR), glGetError());
316 glLinkProgram(program);
317 GLint linkStatus = GL_FALSE;
318 glGetProgramiv(program, GL_LINK_STATUS, &linkStatus);
319 if (linkStatus != GL_TRUE) {
320 GLint bufLength = 0;
321 glGetProgramiv(program, GL_INFO_LOG_LENGTH, &bufLength);
322 if (bufLength) {
323 char* buf = (char*) malloc(bufLength);
324 if (buf) {
325 glGetProgramInfoLog(program, bufLength, NULL, buf);
326 printf("Program link log:\n%s\n", buf);
327 free(buf);
328 FAIL();
329 }
330 }
331 glDeleteProgram(program);
332 program = 0;
333 }
334 }
335 glDeleteShader(vertexShader);
336 glDeleteShader(fragmentShader);
337 ASSERT_TRUE(program != 0);
338 *outPgm = program;
339 }
340
341 } // namespace android
342