1 /*-------------------------------------------------------------------------
2 * drawElements Quality Program EGL Module
3 * ---------------------------------------
4 *
5 * Copyright 2014 The Android Open Source Project
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *
19 *//*!
20 * \file
21 * \brief Simple surface construction test.
22 *//*--------------------------------------------------------------------*/
23
24 #include "teglCreateSurfaceTests.hpp"
25
26 #include "egluNativeDisplay.hpp"
27 #include "egluNativeWindow.hpp"
28 #include "egluNativePixmap.hpp"
29 #include "egluUtil.hpp"
30 #include "egluUnique.hpp"
31
32 #include "eglwLibrary.hpp"
33 #include "eglwEnums.hpp"
34
35 #include "teglSimpleConfigCase.hpp"
36 #include "tcuTestContext.hpp"
37 #include "tcuCommandLine.hpp"
38 #include "tcuTestLog.hpp"
39
40 #include "deSTLUtil.hpp"
41 #include "deUniquePtr.hpp"
42
43 #include <memory>
44
45 namespace deqp
46 {
47 namespace egl
48 {
49
50 using std::vector;
51 using tcu::TestLog;
52 using namespace eglw;
53
54 namespace
55 {
56
checkEGLPlatformSupport(const Library & egl,const char * platformExt)57 void checkEGLPlatformSupport (const Library& egl, const char* platformExt)
58 {
59 std::vector<std::string> extensions = eglu::getPlatformExtensions(egl);
60
61 if (!de::contains(extensions.begin(), extensions.end(), platformExt))
62 throw tcu::NotSupportedError((std::string("Platform extension '") + platformExt + "' not supported").c_str(), "", __FILE__, __LINE__);
63 }
64
createWindowSurface(EGLDisplay display,EGLConfig config,eglu::NativeDisplay & nativeDisplay,eglu::NativeWindow & window,bool useLegacyCreate)65 EGLSurface createWindowSurface (EGLDisplay display, EGLConfig config, eglu::NativeDisplay& nativeDisplay, eglu::NativeWindow& window, bool useLegacyCreate)
66 {
67 const Library& egl = nativeDisplay.getLibrary();
68 EGLSurface surface = EGL_NO_SURFACE;
69
70 if (useLegacyCreate)
71 {
72 surface = egl.createWindowSurface(display, config, window.getLegacyNative(), DE_NULL);
73 EGLU_CHECK_MSG(egl, "eglCreateWindowSurface() failed");
74 }
75 else
76 {
77 checkEGLPlatformSupport(egl, nativeDisplay.getPlatformExtensionName());
78
79 surface = egl.createPlatformWindowSurfaceEXT(display, config, window.getPlatformNative(), DE_NULL);
80 EGLU_CHECK_MSG(egl, "eglCreatePlatformWindowSurfaceEXT() failed");
81 }
82
83 return surface;
84 }
85
createPixmapSurface(EGLDisplay display,EGLConfig config,eglu::NativeDisplay & nativeDisplay,eglu::NativePixmap & pixmap,bool useLegacyCreate)86 EGLSurface createPixmapSurface (EGLDisplay display, EGLConfig config, eglu::NativeDisplay& nativeDisplay, eglu::NativePixmap& pixmap, bool useLegacyCreate)
87 {
88 const Library& egl = nativeDisplay.getLibrary();
89 EGLSurface surface = EGL_NO_SURFACE;
90
91 if (useLegacyCreate)
92 {
93 surface = egl.createPixmapSurface(display, config, pixmap.getLegacyNative(), DE_NULL);
94 EGLU_CHECK_MSG(egl, "eglCreatePixmapSurface() failed");
95 }
96 else
97 {
98 checkEGLPlatformSupport(egl, nativeDisplay.getPlatformExtensionName());
99
100 surface = egl.createPlatformPixmapSurfaceEXT(display, config, pixmap.getPlatformNative(), DE_NULL);
101 EGLU_CHECK_MSG(egl, "eglCreatePlatformPixmapSurfaceEXT() failed");
102 }
103
104 return surface;
105 }
106
107 class CreateWindowSurfaceCase : public SimpleConfigCase
108 {
109 public:
CreateWindowSurfaceCase(EglTestContext & eglTestCtx,const char * name,const char * description,bool useLegacyCreate,const eglu::FilterList & filters)110 CreateWindowSurfaceCase (EglTestContext& eglTestCtx, const char* name, const char* description, bool useLegacyCreate, const eglu::FilterList& filters)
111 : SimpleConfigCase (eglTestCtx, name, description, filters)
112 , m_useLegacyCreate (useLegacyCreate)
113 {
114 }
115
executeForConfig(EGLDisplay display,EGLConfig config)116 void executeForConfig (EGLDisplay display, EGLConfig config)
117 {
118 const Library& egl = m_eglTestCtx.getLibrary();
119 TestLog& log = m_testCtx.getLog();
120 EGLint id = eglu::getConfigID(egl, display, config);
121 const eglu::NativeWindowFactory& windowFactory = eglu::selectNativeWindowFactory(m_eglTestCtx.getNativeDisplayFactory(), m_testCtx.getCommandLine());
122
123 // \todo [2011-03-23 pyry] Iterate thru all possible combinations of EGL_RENDER_BUFFER, EGL_VG_COLORSPACE and EGL_VG_ALPHA_FORMAT
124
125 if (m_useLegacyCreate)
126 {
127 if ((windowFactory.getCapabilities() & eglu::NativeWindow::CAPABILITY_CREATE_SURFACE_LEGACY) == 0)
128 TCU_THROW(NotSupportedError, "Native window doesn't support legacy eglCreateWindowSurface()");
129 }
130 else
131 {
132 if ((windowFactory.getCapabilities() & eglu::NativeWindow::CAPABILITY_CREATE_SURFACE_PLATFORM) == 0)
133 TCU_THROW(NotSupportedError, "Native window doesn't support eglCreatePlatformWindowSurfaceEXT()");
134 }
135
136 log << TestLog::Message << "Creating window surface with config ID " << id << TestLog::EndMessage;
137 EGLU_CHECK_MSG(egl, "init");
138
139 {
140 const int width = 64;
141 const int height = 64;
142 de::UniquePtr<eglu::NativeWindow> window (windowFactory.createWindow(&m_eglTestCtx.getNativeDisplay(), display, config, DE_NULL, eglu::WindowParams(width, height, eglu::parseWindowVisibility(m_testCtx.getCommandLine()))));
143 eglu::UniqueSurface surface (egl, display, createWindowSurface(display, config, m_eglTestCtx.getNativeDisplay(), *window, m_useLegacyCreate));
144
145 EGLint windowWidth = 0;
146 EGLint windowHeight = 0;
147
148 EGLU_CHECK_CALL(egl, querySurface(display, *surface, EGL_WIDTH, &windowWidth));
149 EGLU_CHECK_CALL(egl, querySurface(display, *surface, EGL_HEIGHT, &windowHeight));
150
151 if (windowWidth <= 0 || windowHeight <= 0)
152 {
153 log << TestLog::Message << " Fail, invalid surface size " << windowWidth << "x" << windowHeight << TestLog::EndMessage;
154 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Invalid surface size");
155 }
156 else
157 log << TestLog::Message << " Pass" << TestLog::EndMessage;
158 }
159 }
160
161 private:
162 bool m_useLegacyCreate;
163 };
164
165 class CreatePixmapSurfaceCase : public SimpleConfigCase
166 {
167 public:
CreatePixmapSurfaceCase(EglTestContext & eglTestCtx,const char * name,const char * description,bool useLegacyCreate,const eglu::FilterList & filters)168 CreatePixmapSurfaceCase (EglTestContext& eglTestCtx, const char* name, const char* description, bool useLegacyCreate, const eglu::FilterList& filters)
169 : SimpleConfigCase(eglTestCtx, name, description, filters)
170 , m_useLegacyCreate (useLegacyCreate)
171 {
172 }
173
executeForConfig(EGLDisplay display,EGLConfig config)174 void executeForConfig (EGLDisplay display, EGLConfig config)
175 {
176 const Library& egl = m_eglTestCtx.getLibrary();
177 TestLog& log = m_testCtx.getLog();
178 EGLint id = eglu::getConfigID(egl, display, config);
179 const eglu::NativePixmapFactory& pixmapFactory = eglu::selectNativePixmapFactory(m_eglTestCtx.getNativeDisplayFactory(), m_testCtx.getCommandLine());
180
181 // \todo [2011-03-23 pyry] Iterate thru all possible combinations of EGL_RENDER_BUFFER, EGL_VG_COLORSPACE and EGL_VG_ALPHA_FORMAT
182
183 if (m_useLegacyCreate)
184 {
185 if ((pixmapFactory.getCapabilities() & eglu::NativePixmap::CAPABILITY_CREATE_SURFACE_LEGACY) == 0)
186 TCU_THROW(NotSupportedError, "Native pixmap doesn't support legacy eglCreatePixmapSurface()");
187 }
188 else
189 {
190 if ((pixmapFactory.getCapabilities() & eglu::NativePixmap::CAPABILITY_CREATE_SURFACE_PLATFORM) == 0)
191 TCU_THROW(NotSupportedError, "Native pixmap doesn't support eglCreatePlatformPixmapSurfaceEXT()");
192 }
193
194 log << TestLog::Message << "Creating pixmap surface with config ID " << id << TestLog::EndMessage;
195 EGLU_CHECK_MSG(egl, "init");
196
197 {
198 const int width = 64;
199 const int height = 64;
200 de::UniquePtr<eglu::NativePixmap> pixmap (pixmapFactory.createPixmap(&m_eglTestCtx.getNativeDisplay(), display, config, DE_NULL, width, height));
201 eglu::UniqueSurface surface (egl, display, createPixmapSurface(display, config, m_eglTestCtx.getNativeDisplay(), *pixmap, m_useLegacyCreate));
202 EGLint pixmapWidth = 0;
203 EGLint pixmapHeight = 0;
204
205 EGLU_CHECK_CALL(egl, querySurface(display, *surface, EGL_WIDTH, &pixmapWidth));
206 EGLU_CHECK_CALL(egl, querySurface(display, *surface, EGL_HEIGHT, &pixmapHeight));
207
208 if (pixmapWidth <= 0 || pixmapHeight <= 0)
209 {
210 log << TestLog::Message << " Fail, invalid surface size " << pixmapWidth << "x" << pixmapHeight << TestLog::EndMessage;
211 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Invalid surface size");
212 }
213 else
214 log << TestLog::Message << " Pass" << TestLog::EndMessage;
215 }
216 }
217
218 private:
219 bool m_useLegacyCreate;
220 };
221
222 class CreatePbufferSurfaceCase : public SimpleConfigCase
223 {
224 public:
CreatePbufferSurfaceCase(EglTestContext & eglTestCtx,const char * name,const char * description,const eglu::FilterList & filters)225 CreatePbufferSurfaceCase (EglTestContext& eglTestCtx, const char* name, const char* description, const eglu::FilterList& filters)
226 : SimpleConfigCase(eglTestCtx, name, description, filters)
227 {
228 }
229
executeForConfig(EGLDisplay display,EGLConfig config)230 void executeForConfig (EGLDisplay display, EGLConfig config)
231 {
232 const Library& egl = m_eglTestCtx.getLibrary();
233 TestLog& log = m_testCtx.getLog();
234 EGLint id = eglu::getConfigID(egl, display, config);
235 int width = 64;
236 int height = 64;
237
238 // \todo [2011-03-23 pyry] Iterate thru all possible combinations of EGL_RENDER_BUFFER, EGL_VG_COLORSPACE and EGL_VG_ALPHA_FORMAT
239
240 log << TestLog::Message << "Creating pbuffer surface with config ID " << id << TestLog::EndMessage;
241 EGLU_CHECK_MSG(egl, "init");
242
243 // Clamp to maximums reported by implementation
244 width = deMin32(width, eglu::getConfigAttribInt(egl, display, config, EGL_MAX_PBUFFER_WIDTH));
245 height = deMin32(height, eglu::getConfigAttribInt(egl, display, config, EGL_MAX_PBUFFER_HEIGHT));
246
247 if (width == 0 || height == 0)
248 {
249 log << TestLog::Message << " Fail, maximum pbuffer size of " << width << "x" << height << " reported" << TestLog::EndMessage;
250 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Invalid maximum pbuffer size");
251 return;
252 }
253
254 // \todo [2011-03-23 pyry] Texture-backed variants!
255
256 const EGLint attribs[] =
257 {
258 EGL_WIDTH, width,
259 EGL_HEIGHT, height,
260 EGL_TEXTURE_FORMAT, EGL_NO_TEXTURE,
261 EGL_NONE
262 };
263
264 EGLSurface surface = egl.createPbufferSurface(display, config, attribs);
265 EGLU_CHECK_MSG(egl, "Failed to create pbuffer");
266 TCU_CHECK(surface != EGL_NO_SURFACE);
267 egl.destroySurface(display, surface);
268
269 log << TestLog::Message << " Pass" << TestLog::EndMessage;
270 }
271 };
272
273 } // anonymous
274
CreateSurfaceTests(EglTestContext & eglTestCtx)275 CreateSurfaceTests::CreateSurfaceTests (EglTestContext& eglTestCtx)
276 : TestCaseGroup(eglTestCtx, "create_surface", "Basic surface construction tests")
277 {
278 }
279
~CreateSurfaceTests(void)280 CreateSurfaceTests::~CreateSurfaceTests (void)
281 {
282 }
283
284 template <deUint32 Type>
surfaceType(const eglu::CandidateConfig & c)285 static bool surfaceType (const eglu::CandidateConfig& c)
286 {
287 return (c.surfaceType() & Type) == Type;
288 }
289
init(void)290 void CreateSurfaceTests::init (void)
291 {
292 // Window surfaces
293 {
294 tcu::TestCaseGroup* windowGroup = new tcu::TestCaseGroup(m_testCtx, "window", "Window surfaces");
295 addChild(windowGroup);
296
297 eglu::FilterList baseFilters;
298 baseFilters << surfaceType<EGL_WINDOW_BIT>;
299
300 vector<NamedFilterList> filterLists;
301 getDefaultFilterLists(filterLists, baseFilters);
302
303 for (vector<NamedFilterList>::iterator i = filterLists.begin(); i != filterLists.end(); i++)
304 windowGroup->addChild(new CreateWindowSurfaceCase(m_eglTestCtx, i->getName(), i->getDescription(), true, *i));
305 }
306
307 // Pixmap surfaces
308 {
309 tcu::TestCaseGroup* pixmapGroup = new tcu::TestCaseGroup(m_testCtx, "pixmap", "Pixmap surfaces");
310 addChild(pixmapGroup);
311
312 eglu::FilterList baseFilters;
313 baseFilters << surfaceType<EGL_PIXMAP_BIT>;
314
315 vector<NamedFilterList> filterLists;
316 getDefaultFilterLists(filterLists, baseFilters);
317
318 for (vector<NamedFilterList>::iterator i = filterLists.begin(); i != filterLists.end(); i++)
319 pixmapGroup->addChild(new CreatePixmapSurfaceCase(m_eglTestCtx, i->getName(), i->getDescription(), true, *i));
320 }
321
322 // Pbuffer surfaces
323 {
324 tcu::TestCaseGroup* pbufferGroup = new tcu::TestCaseGroup(m_testCtx, "pbuffer", "Pbuffer surfaces");
325 addChild(pbufferGroup);
326
327 eglu::FilterList baseFilters;
328 baseFilters << surfaceType<EGL_PBUFFER_BIT>;
329
330 vector<NamedFilterList> filterLists;
331 getDefaultFilterLists(filterLists, baseFilters);
332
333 for (vector<NamedFilterList>::iterator i = filterLists.begin(); i != filterLists.end(); i++)
334 pbufferGroup->addChild(new CreatePbufferSurfaceCase(m_eglTestCtx, i->getName(), i->getDescription(), *i));
335 }
336
337 // Window surfaces with new platform extension
338 {
339 tcu::TestCaseGroup* windowGroup = new tcu::TestCaseGroup(m_testCtx, "platform_window", "Window surfaces with platform extension");
340 addChild(windowGroup);
341
342 eglu::FilterList baseFilters;
343 baseFilters << surfaceType<EGL_WINDOW_BIT>;
344
345 vector<NamedFilterList> filterLists;
346 getDefaultFilterLists(filterLists, baseFilters);
347
348 for (vector<NamedFilterList>::iterator i = filterLists.begin(); i != filterLists.end(); i++)
349 windowGroup->addChild(new CreateWindowSurfaceCase(m_eglTestCtx, i->getName(), i->getDescription(), false, *i));
350 }
351
352 // Pixmap surfaces with new platform extension
353 {
354 tcu::TestCaseGroup* pixmapGroup = new tcu::TestCaseGroup(m_testCtx, "platform_pixmap", "Pixmap surfaces with platform extension");
355 addChild(pixmapGroup);
356
357 eglu::FilterList baseFilters;
358 baseFilters << surfaceType<EGL_PIXMAP_BIT>;
359
360 vector<NamedFilterList> filterLists;
361 getDefaultFilterLists(filterLists, baseFilters);
362
363 for (vector<NamedFilterList>::iterator i = filterLists.begin(); i != filterLists.end(); i++)
364 pixmapGroup->addChild(new CreatePixmapSurfaceCase(m_eglTestCtx, i->getName(), i->getDescription(), false, *i));
365 }
366 }
367
368 } // egl
369 } // deqp
370