1 /*-------------------------------------------------------------------------
2 * drawElements Quality Program OpenGL ES Utilities
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 Texture classes.
22 *//*--------------------------------------------------------------------*/
23
24 #include "gluTexture.hpp"
25 #include "gluTextureUtil.hpp"
26 #include "deFilePath.hpp"
27 #include "tcuImageIO.hpp"
28 #include "tcuSurface.hpp"
29 #include "tcuTextureUtil.hpp"
30
31 #include "glwFunctions.hpp"
32 #include "glwEnums.hpp"
33
34 #include "deUniquePtr.hpp"
35
36 using std::vector;
37
38 namespace glu
39 {
40
computePixelStore(const tcu::TextureFormat & format)41 static inline int computePixelStore (const tcu::TextureFormat& format)
42 {
43 int pixelSize = format.getPixelSize();
44 if (deIsPowerOfTwo32(pixelSize))
45 return de::min(pixelSize, 8);
46 else
47 return 1;
48 }
49
50 // Texture1D
51
Texture1D(const RenderContext & context,deUint32 format,deUint32 dataType,int width)52 Texture1D::Texture1D (const RenderContext& context, deUint32 format, deUint32 dataType, int width)
53 : m_context (context)
54 , m_format (format)
55 , m_refTexture (mapGLTransferFormat(format, dataType), width)
56 , m_glTexture (0)
57 {
58 const glw::Functions& gl = context.getFunctions();
59 gl.genTextures(1, &m_glTexture);
60 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
61 }
62
Texture1D(const RenderContext & context,deUint32 sizedFormat,int width)63 Texture1D::Texture1D (const RenderContext& context, deUint32 sizedFormat, int width)
64 : m_context (context)
65 , m_format (sizedFormat)
66 , m_refTexture (mapGLInternalFormat(sizedFormat), width)
67 , m_glTexture (0)
68 {
69 const glw::Functions& gl = context.getFunctions();
70 gl.genTextures(1, &m_glTexture);
71 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
72 }
73
~Texture1D(void)74 Texture1D::~Texture1D (void)
75 {
76 if (m_glTexture)
77 m_context.getFunctions().deleteTextures(1, &m_glTexture);
78 }
79
upload(void)80 void Texture1D::upload (void)
81 {
82 const glw::Functions& gl = m_context.getFunctions();
83
84 TCU_CHECK(m_glTexture);
85 gl.bindTexture(GL_TEXTURE_1D, m_glTexture);
86 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
87 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
88
89 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
90
91 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
92 {
93 if (m_refTexture.isLevelEmpty(levelNdx))
94 continue; // Don't upload.
95
96 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
97 gl.texImage1D(GL_TEXTURE_1D, levelNdx, m_format, access.getWidth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
98 }
99
100 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
101 }
102
103 // Texture2D
104
Texture2D(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int height)105 Texture2D::Texture2D (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int height)
106 : m_context (context)
107 , m_isCompressed (false)
108 , m_format (format)
109 , m_refTexture (mapGLTransferFormat(format, dataType), width, height)
110 , m_glTexture (0)
111 {
112 const glw::Functions& gl = context.getFunctions();
113 gl.genTextures(1, &m_glTexture);
114 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
115 }
116
Texture2D(const RenderContext & context,deUint32 sizedFormat,int width,int height)117 Texture2D::Texture2D (const RenderContext& context, deUint32 sizedFormat, int width, int height)
118 : m_context (context)
119 , m_isCompressed (false)
120 , m_format (sizedFormat)
121 , m_refTexture (mapGLInternalFormat(sizedFormat), width, height)
122 , m_glTexture (0)
123 {
124 const glw::Functions& gl = context.getFunctions();
125 gl.genTextures(1, &m_glTexture);
126 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
127 }
128
Texture2D(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)129 Texture2D::Texture2D (const RenderContext& context, const ContextInfo& contextInfo, int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
130 : m_context (context)
131 , m_isCompressed (true)
132 , m_format (getGLFormat(levels[0].getFormat()))
133 , m_refTexture (getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth(), levels[0].getHeight())
134 , m_glTexture (0)
135 {
136 const glw::Functions& gl = context.getFunctions();
137
138 if (!contextInfo.isCompressedTextureFormatSupported(m_format))
139 TCU_THROW(NotSupportedError, "Compressed texture format not supported");
140
141 gl.genTextures(1, &m_glTexture);
142 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
143
144 try
145 {
146 loadCompressed(numLevels, levels, decompressionParams);
147 }
148 catch (const std::exception&)
149 {
150 gl.deleteTextures(1, &m_glTexture);
151 throw;
152 }
153 }
154
~Texture2D(void)155 Texture2D::~Texture2D (void)
156 {
157 if (m_glTexture)
158 m_context.getFunctions().deleteTextures(1, &m_glTexture);
159 }
160
upload(void)161 void Texture2D::upload (void)
162 {
163 const glw::Functions& gl = m_context.getFunctions();
164
165 DE_ASSERT(!m_isCompressed);
166
167 TCU_CHECK(m_glTexture);
168 gl.bindTexture(GL_TEXTURE_2D, m_glTexture);
169 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
170 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
171
172 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
173
174 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
175 {
176 if (m_refTexture.isLevelEmpty(levelNdx))
177 continue; // Don't upload.
178
179 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
180 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
181 gl.texImage2D(GL_TEXTURE_2D, levelNdx, m_format, access.getWidth(), access.getHeight(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
182 }
183
184 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
185 }
186
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)187 void Texture2D::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
188 {
189 const glw::Functions& gl = m_context.getFunctions();
190 deUint32 compressedFormat = getGLFormat(levels[0].getFormat());
191
192 TCU_CHECK(m_glTexture);
193 gl.bindTexture(GL_TEXTURE_2D, m_glTexture);
194
195 for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
196 {
197 const tcu::CompressedTexture& level = levels[levelNdx];
198
199 // Decompress to reference texture.
200 m_refTexture.allocLevel(levelNdx);
201 tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevel(levelNdx);
202 TCU_CHECK(level.getWidth() == refLevelAccess.getWidth() &&
203 level.getHeight() == refLevelAccess.getHeight());
204 level.decompress(refLevelAccess, decompressionParams);
205
206 // Upload to GL texture in compressed form.
207 gl.compressedTexImage2D(GL_TEXTURE_2D, levelNdx, compressedFormat,
208 level.getWidth(), level.getHeight(), 0 /* border */, level.getDataSize(), level.getData());
209 }
210
211 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
212 }
213
create(const RenderContext & context,const ContextInfo & contextInfo,const tcu::Archive & archive,int numLevels,const char * const * levelFileNames)214 Texture2D* Texture2D::create (const RenderContext& context, const ContextInfo& contextInfo, const tcu::Archive& archive, int numLevels, const char* const* levelFileNames)
215 {
216 DE_ASSERT(numLevels > 0);
217
218 std::string ext = de::FilePath(levelFileNames[0]).getFileExtension();
219
220 if (ext == "png")
221 {
222 // Uncompressed texture.
223
224 tcu::TextureLevel level;
225
226 // Load level 0.
227 tcu::ImageIO::loadPNG(level, archive, levelFileNames[0]);
228
229 TCU_CHECK_INTERNAL(level.getFormat() == tcu::TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8) ||
230 level.getFormat() == tcu::TextureFormat(tcu::TextureFormat::RGB, tcu::TextureFormat::UNORM_INT8));
231
232 bool isRGBA = level.getFormat() == tcu::TextureFormat(tcu::TextureFormat::RGBA, tcu::TextureFormat::UNORM_INT8);
233 Texture2D* texture = new Texture2D(context, isRGBA ? GL_RGBA : GL_RGB, GL_UNSIGNED_BYTE, level.getWidth(), level.getHeight());
234
235 try
236 {
237 // Fill level 0.
238 texture->getRefTexture().allocLevel(0);
239 tcu::copy(texture->getRefTexture().getLevel(0), level.getAccess());
240
241 // Fill remaining levels.
242 for (int levelNdx = 1; levelNdx < numLevels; levelNdx++)
243 {
244 tcu::ImageIO::loadPNG(level, archive, levelFileNames[levelNdx]);
245
246 texture->getRefTexture().allocLevel(levelNdx);
247 tcu::copy(texture->getRefTexture().getLevel(levelNdx), level.getAccess());
248 }
249
250 // Upload data.
251 texture->upload();
252 }
253 catch (const std::exception&)
254 {
255 delete texture;
256 throw;
257 }
258
259 return texture;
260 }
261 else if (ext == "pkm")
262 {
263 // Compressed texture.
264 vector<tcu::CompressedTexture> levels(numLevels);
265
266 for (int ndx = 0; ndx < numLevels; ndx++)
267 tcu::ImageIO::loadPKM(levels[ndx], archive, levelFileNames[ndx]);
268
269 return new Texture2D(context, contextInfo, numLevels, &levels[0]);
270 }
271 else
272 TCU_FAIL("Unsupported file format");
273 }
274
create(const RenderContext & context,const ContextInfo & contextInfo,const tcu::Archive & archive,int numLevels,const std::vector<std::string> & filenames)275 Texture2D* Texture2D::create (const RenderContext& context, const ContextInfo& contextInfo, const tcu::Archive& archive, int numLevels, const std::vector<std::string>& filenames)
276 {
277 TCU_CHECK(numLevels == (int)filenames.size());
278
279 std::vector<const char*> charPtrs(filenames.size());
280 for (int ndx = 0; ndx < (int)filenames.size(); ndx++)
281 charPtrs[ndx] = filenames[ndx].c_str();
282
283 return Texture2D::create(context, contextInfo, archive, numLevels, &charPtrs[0]);
284 }
285
286 // ImmutableTexture2D
287
ImmutableTexture2D(const RenderContext & context,deUint32 sizedFormat,int width,int height)288 ImmutableTexture2D::ImmutableTexture2D (const RenderContext& context, deUint32 sizedFormat, int width, int height)
289 : Texture2D(context, sizedFormat, width, height)
290 {
291 }
292
upload(void)293 void ImmutableTexture2D::upload (void)
294 {
295 const glw::Functions& gl = m_context.getFunctions();
296
297 DE_ASSERT(!m_isCompressed);
298
299 TCU_CHECK(m_glTexture);
300 gl.bindTexture(GL_TEXTURE_2D, m_glTexture);
301 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
302 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
303
304 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
305
306 gl.texStorage2D(GL_TEXTURE_2D, m_refTexture.getNumLevels(), m_format, m_refTexture.getWidth(), m_refTexture.getHeight());
307 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
308 {
309 if (m_refTexture.isLevelEmpty(levelNdx))
310 continue; // Don't upload.
311
312 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
313 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
314 gl.texSubImage2D(GL_TEXTURE_2D, levelNdx, 0, 0, access.getWidth(), access.getHeight(), transferFormat.format, transferFormat.dataType, access.getDataPtr());
315 }
316
317 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
318 }
319
320 // TextureCube
321
TextureCube(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)322 TextureCube::TextureCube (const RenderContext& context, const ContextInfo& contextInfo, int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
323 : m_context (context)
324 , m_isCompressed (true)
325 , m_format (getGLFormat(levels[0].getFormat()))
326 , m_refTexture (getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth())
327 , m_glTexture (0)
328 {
329 const glw::Functions& gl = m_context.getFunctions();
330
331 TCU_CHECK_INTERNAL(levels[0].getWidth() == levels[0].getHeight());
332
333 if (!contextInfo.isCompressedTextureFormatSupported(m_format))
334 throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
335
336 gl.genTextures(1, &m_glTexture);
337 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
338
339 try
340 {
341 loadCompressed(numLevels, levels, decompressionParams);
342 }
343 catch (const std::exception&)
344 {
345 gl.deleteTextures(1, &m_glTexture);
346 throw;
347 }
348 }
349
TextureCube(const RenderContext & context,deUint32 format,deUint32 dataType,int size)350 TextureCube::TextureCube (const RenderContext& context, deUint32 format, deUint32 dataType, int size)
351 : m_context (context)
352 , m_isCompressed (false)
353 , m_format (format)
354 , m_refTexture (mapGLTransferFormat(format, dataType), size)
355 , m_glTexture (0)
356 {
357 const glw::Functions& gl = m_context.getFunctions();
358 gl.genTextures(1, &m_glTexture);
359 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
360 }
361
TextureCube(const RenderContext & context,deUint32 internalFormat,int size)362 TextureCube::TextureCube (const RenderContext& context, deUint32 internalFormat, int size)
363 : m_context (context)
364 , m_isCompressed (false)
365 , m_format (internalFormat)
366 , m_refTexture (mapGLInternalFormat(internalFormat), size)
367 , m_glTexture (0)
368 {
369 const glw::Functions& gl = m_context.getFunctions();
370 gl.genTextures(1, &m_glTexture);
371 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
372 }
373
~TextureCube(void)374 TextureCube::~TextureCube (void)
375 {
376 if (m_glTexture)
377 m_context.getFunctions().deleteTextures(1, &m_glTexture);
378 }
379
upload(void)380 void TextureCube::upload (void)
381 {
382 const glw::Functions& gl = m_context.getFunctions();
383
384 DE_ASSERT(!m_isCompressed);
385
386 TCU_CHECK(m_glTexture);
387 gl.bindTexture(GL_TEXTURE_CUBE_MAP, m_glTexture);
388 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
389 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
390
391 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
392
393 for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
394 {
395 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
396 {
397 if (m_refTexture.isLevelEmpty((tcu::CubeFace)face, levelNdx))
398 continue; // Don't upload.
399
400 tcu::ConstPixelBufferAccess access = m_refTexture.getLevelFace(levelNdx, (tcu::CubeFace)face);
401 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
402 gl.texImage2D(getGLCubeFace((tcu::CubeFace)face), levelNdx, m_format, access.getWidth(), access.getHeight(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
403 }
404 }
405
406 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
407 }
408
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)409 void TextureCube::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
410 {
411 const glw::Functions& gl = m_context.getFunctions();
412 deUint32 compressedFormat = getGLFormat(levels[0].getFormat());
413
414 TCU_CHECK(m_glTexture);
415 gl.bindTexture(GL_TEXTURE_CUBE_MAP, m_glTexture);
416
417 for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
418 {
419 for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
420 {
421 const tcu::CompressedTexture& level = levels[levelNdx*tcu::CUBEFACE_LAST + face];
422
423 // Decompress to reference texture.
424 m_refTexture.allocLevel((tcu::CubeFace)face, levelNdx);
425 tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevelFace(levelNdx, (tcu::CubeFace)face);
426 TCU_CHECK(level.getWidth() == refLevelAccess.getWidth() &&
427 level.getHeight() == refLevelAccess.getHeight());
428 level.decompress(refLevelAccess, decompressionParams);
429
430 // Upload to GL texture in compressed form.
431 gl.compressedTexImage2D(getGLCubeFace((tcu::CubeFace)face), levelNdx, compressedFormat,
432 level.getWidth(), level.getHeight(), 0 /* border */, level.getDataSize(), level.getData());
433 }
434 }
435
436 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
437 }
438
create(const RenderContext & context,const ContextInfo & contextInfo,const tcu::Archive & archive,int numLevels,const char * const * filenames)439 TextureCube* TextureCube::create (const RenderContext& context, const ContextInfo& contextInfo, const tcu::Archive& archive, int numLevels, const char* const* filenames)
440 {
441 DE_ASSERT(numLevels > 0);
442
443 std::string ext = de::FilePath(filenames[0]).getFileExtension();
444
445 // \todo [2011-11-21 pyry] Support PNG images.
446 if (ext == "pkm")
447 {
448 // Compressed texture.
449 int numImages = numLevels*tcu::CUBEFACE_LAST;
450 vector<tcu::CompressedTexture> levels (numImages);
451
452 for (int ndx = 0; ndx < numImages; ndx++)
453 tcu::ImageIO::loadPKM(levels[ndx], archive, filenames[ndx]);
454
455 return new TextureCube(context, contextInfo, numLevels, &levels[0]);
456 }
457 else
458 TCU_FAIL("Unsupported file format");
459 }
460
create(const RenderContext & context,const ContextInfo & contextInfo,const tcu::Archive & archive,int numLevels,const std::vector<std::string> & filenames)461 TextureCube* TextureCube::create (const RenderContext& context, const ContextInfo& contextInfo, const tcu::Archive& archive, int numLevels, const std::vector<std::string>& filenames)
462 {
463 DE_STATIC_ASSERT(tcu::CUBEFACE_LAST == 6);
464 TCU_CHECK(numLevels*tcu::CUBEFACE_LAST == (int)filenames.size());
465
466 std::vector<const char*> charPtrs(filenames.size());
467 for (int ndx = 0; ndx < (int)filenames.size(); ndx++)
468 charPtrs[ndx] = filenames[ndx].c_str();
469
470 return TextureCube::create(context, contextInfo, archive, numLevels, &charPtrs[0]);
471 }
472
473 // Texture1DArray
474
Texture1DArray(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int numLevels)475 Texture1DArray::Texture1DArray (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int numLevels)
476 : m_context (context)
477 , m_format (format)
478 , m_refTexture (mapGLTransferFormat(format, dataType), width, numLevels)
479 , m_glTexture (0)
480 {
481 const glw::Functions& gl = m_context.getFunctions();
482 gl.genTextures(1, &m_glTexture);
483 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
484 }
485
Texture1DArray(const RenderContext & context,deUint32 sizedFormat,int width,int numLevels)486 Texture1DArray::Texture1DArray (const RenderContext& context, deUint32 sizedFormat, int width, int numLevels)
487 : m_context (context)
488 , m_format (sizedFormat)
489 , m_refTexture (mapGLInternalFormat(sizedFormat), width, numLevels)
490 , m_glTexture (0)
491 {
492 const glw::Functions& gl = m_context.getFunctions();
493 gl.genTextures(1, &m_glTexture);
494 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
495 }
496
~Texture1DArray(void)497 Texture1DArray::~Texture1DArray (void)
498 {
499 if (m_glTexture)
500 m_context.getFunctions().deleteTextures(1, &m_glTexture);
501 }
502
upload(void)503 void Texture1DArray::upload (void)
504 {
505 const glw::Functions& gl = m_context.getFunctions();
506
507 TCU_CHECK(m_glTexture);
508 gl.bindTexture(GL_TEXTURE_1D_ARRAY, m_glTexture);
509 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
510 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
511
512 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
513
514 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
515 {
516 if (m_refTexture.isLevelEmpty(levelNdx))
517 continue; // Don't upload.
518
519 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
520 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
521 gl.texImage2D(GL_TEXTURE_1D_ARRAY, levelNdx, m_format, access.getWidth(), access.getHeight(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
522 }
523
524 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
525 }
526
527 // Texture2DArray
528
Texture2DArray(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int height,int numLevels)529 Texture2DArray::Texture2DArray (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int height, int numLevels)
530 : m_context (context)
531 , m_isCompressed (false)
532 , m_format (format)
533 , m_refTexture (mapGLTransferFormat(format, dataType), width, height, numLevels)
534 , m_glTexture (0)
535 {
536 // \todo [2013-04-08 pyry] Check support here.
537 const glw::Functions& gl = m_context.getFunctions();
538 gl.genTextures(1, &m_glTexture);
539 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
540 }
541
Texture2DArray(const RenderContext & context,deUint32 sizedFormat,int width,int height,int numLevels)542 Texture2DArray::Texture2DArray (const RenderContext& context, deUint32 sizedFormat, int width, int height, int numLevels)
543 : m_context (context)
544 , m_isCompressed (false)
545 , m_format (sizedFormat)
546 , m_refTexture (mapGLInternalFormat(sizedFormat), width, height, numLevels)
547 , m_glTexture (0)
548 {
549 // \todo [2013-04-08 pyry] Check support here.
550 const glw::Functions& gl = m_context.getFunctions();
551 gl.genTextures(1, &m_glTexture);
552 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
553 }
554
Texture2DArray(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)555 Texture2DArray::Texture2DArray (const RenderContext& context, const ContextInfo& contextInfo, int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
556 : m_context (context)
557 , m_isCompressed (true)
558 , m_format (getGLFormat(levels[0].getFormat()))
559 , m_refTexture (getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth(), levels[0].getHeight(), levels[0].getDepth())
560 , m_glTexture (0)
561 {
562 const glw::Functions& gl = context.getFunctions();
563
564 if (!contextInfo.isCompressedTextureFormatSupported(m_format))
565 throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
566
567 gl.genTextures(1, &m_glTexture);
568 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
569
570 try
571 {
572 loadCompressed(numLevels, levels, decompressionParams);
573 }
574 catch (const std::exception&)
575 {
576 gl.deleteTextures(1, &m_glTexture);
577 throw;
578 }
579 }
580
~Texture2DArray(void)581 Texture2DArray::~Texture2DArray (void)
582 {
583 if (m_glTexture)
584 m_context.getFunctions().deleteTextures(1, &m_glTexture);
585 }
586
upload(void)587 void Texture2DArray::upload (void)
588 {
589 const glw::Functions& gl = m_context.getFunctions();
590
591 if (!gl.texImage3D)
592 throw tcu::NotSupportedError("glTexImage3D() is not supported");
593
594 TCU_CHECK(m_glTexture);
595 gl.bindTexture(GL_TEXTURE_2D_ARRAY, m_glTexture);
596 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
597 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
598
599 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
600
601 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
602 {
603 if (m_refTexture.isLevelEmpty(levelNdx))
604 continue; // Don't upload.
605
606 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
607 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
608 DE_ASSERT(access.getSlicePitch() == access.getFormat().getPixelSize()*access.getWidth()*access.getHeight());
609 gl.texImage3D(GL_TEXTURE_2D_ARRAY, levelNdx, m_format, access.getWidth(), access.getHeight(), access.getDepth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
610 }
611
612 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
613 }
614
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)615 void Texture2DArray::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
616 {
617 const glw::Functions& gl = m_context.getFunctions();
618 deUint32 compressedFormat = getGLFormat(levels[0].getFormat());
619
620 TCU_CHECK(m_glTexture);
621 gl.bindTexture(GL_TEXTURE_2D_ARRAY, m_glTexture);
622
623 for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
624 {
625 const tcu::CompressedTexture& level = levels[levelNdx];
626
627 // Decompress to reference texture.
628 m_refTexture.allocLevel(levelNdx);
629 tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevel(levelNdx);
630 TCU_CHECK(level.getWidth() == refLevelAccess.getWidth() &&
631 level.getHeight() == refLevelAccess.getHeight() &&
632 level.getDepth() == refLevelAccess.getDepth());
633 level.decompress(refLevelAccess, decompressionParams);
634
635 // Upload to GL texture in compressed form.
636 gl.compressedTexImage3D(GL_TEXTURE_2D_ARRAY, levelNdx, compressedFormat,
637 level.getWidth(), level.getHeight(), m_refTexture.getLevel(levelNdx).getDepth(), 0 /* border */, level.getDataSize(), level.getData());
638 }
639
640 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
641 }
642
643 // Texture3D
644
Texture3D(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int height,int depth)645 Texture3D::Texture3D (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int height, int depth)
646 : m_context (context)
647 , m_isCompressed (false)
648 , m_format (format)
649 , m_refTexture (mapGLTransferFormat(format, dataType), width, height, depth)
650 , m_glTexture (0)
651 {
652 // \todo [2013-04-08 pyry] Check support here.
653 const glw::Functions& gl = m_context.getFunctions();
654 gl.genTextures(1, &m_glTexture);
655 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
656 }
657
Texture3D(const RenderContext & context,deUint32 sizedFormat,int width,int height,int depth)658 Texture3D::Texture3D (const RenderContext& context, deUint32 sizedFormat, int width, int height, int depth)
659 : m_context (context)
660 , m_isCompressed (false)
661 , m_format (sizedFormat)
662 , m_refTexture (mapGLInternalFormat(sizedFormat), width, height, depth)
663 , m_glTexture (0)
664 {
665 // \todo [2013-04-08 pyry] Check support here.
666 const glw::Functions& gl = m_context.getFunctions();
667 gl.genTextures(1, &m_glTexture);
668 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
669 }
670
Texture3D(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)671 Texture3D::Texture3D (const RenderContext& context,
672 const ContextInfo& contextInfo,
673 int numLevels,
674 const tcu::CompressedTexture* levels,
675 const tcu::TexDecompressionParams& decompressionParams)
676 : m_context (context)
677 , m_isCompressed (true)
678 , m_format (getGLFormat(levels[0].getFormat()))
679 , m_refTexture (getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth(), levels[0].getHeight(), levels[0].getDepth())
680 , m_glTexture (0)
681 {
682 const glw::Functions& gl = context.getFunctions();
683
684 if (!contextInfo.isCompressedTextureFormatSupported(m_format))
685 throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
686
687 gl.genTextures(1, &m_glTexture);
688 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
689
690 try
691 {
692 loadCompressed(numLevels, levels, decompressionParams);
693 }
694 catch (const std::exception&)
695 {
696 gl.deleteTextures(1, &m_glTexture);
697 throw;
698 }
699 }
700
~Texture3D(void)701 Texture3D::~Texture3D (void)
702 {
703 if (m_glTexture)
704 m_context.getFunctions().deleteTextures(1, &m_glTexture);
705 }
706
upload(void)707 void Texture3D::upload (void)
708 {
709 const glw::Functions& gl = m_context.getFunctions();
710
711 DE_ASSERT(!m_isCompressed);
712
713 if (!gl.texImage3D)
714 throw tcu::NotSupportedError("glTexImage3D() is not supported");
715
716 TCU_CHECK(m_glTexture);
717 gl.bindTexture(GL_TEXTURE_3D, m_glTexture);
718 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
719 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
720
721 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
722
723 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
724 {
725 if (m_refTexture.isLevelEmpty(levelNdx))
726 continue; // Don't upload.
727
728 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
729 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
730 DE_ASSERT(access.getSlicePitch() == access.getFormat().getPixelSize()*access.getWidth()*access.getHeight());
731 gl.texImage3D(GL_TEXTURE_3D, levelNdx, m_format, access.getWidth(), access.getHeight(), access.getDepth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
732 }
733
734 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
735 }
736
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)737 void Texture3D::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
738 {
739 const glw::Functions& gl = m_context.getFunctions();
740 deUint32 compressedFormat = getGLFormat(levels[0].getFormat());
741
742 if (!gl.compressedTexImage3D)
743 throw tcu::NotSupportedError("glCompressedTexImage3D() is not supported");
744
745 TCU_CHECK(m_glTexture);
746 gl.bindTexture(GL_TEXTURE_3D, m_glTexture);
747
748 for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
749 {
750 const tcu::CompressedTexture& level = levels[levelNdx];
751
752 // Decompress to reference texture.
753 m_refTexture.allocLevel(levelNdx);
754 tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevel(levelNdx);
755 TCU_CHECK(level.getWidth() == refLevelAccess.getWidth() &&
756 level.getHeight() == refLevelAccess.getHeight() &&
757 level.getDepth() == refLevelAccess.getDepth());
758 level.decompress(refLevelAccess, decompressionParams);
759
760 // Upload to GL texture in compressed form.
761 gl.compressedTexImage3D(GL_TEXTURE_3D, levelNdx, compressedFormat,
762 level.getWidth(), level.getHeight(), level.getDepth(), 0 /* border */, level.getDataSize(), level.getData());
763 }
764
765 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
766 }
767
768 // TextureCubeArray
769
TextureCubeArray(const RenderContext & context,deUint32 format,deUint32 dataType,int size,int numLayers)770 TextureCubeArray::TextureCubeArray (const RenderContext& context, deUint32 format, deUint32 dataType, int size, int numLayers)
771 : m_context (context)
772 , m_format (format)
773 , m_refTexture (mapGLTransferFormat(format, dataType), size, numLayers)
774 , m_glTexture (0)
775 {
776 // \todo [2013-04-08 pyry] Check support here.
777 const glw::Functions& gl = m_context.getFunctions();
778 gl.genTextures(1, &m_glTexture);
779 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
780 }
781
TextureCubeArray(const RenderContext & context,deUint32 sizedFormat,int size,int numLayers)782 TextureCubeArray::TextureCubeArray (const RenderContext& context, deUint32 sizedFormat, int size, int numLayers)
783 : m_context (context)
784 , m_format (sizedFormat)
785 , m_refTexture (mapGLInternalFormat(sizedFormat), size, numLayers)
786 , m_glTexture (0)
787 {
788 // \todo [2013-04-08 pyry] Check support here.
789 const glw::Functions& gl = m_context.getFunctions();
790 gl.genTextures(1, &m_glTexture);
791 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
792 }
793
~TextureCubeArray(void)794 TextureCubeArray::~TextureCubeArray (void)
795 {
796 if (m_glTexture)
797 m_context.getFunctions().deleteTextures(1, &m_glTexture);
798 }
799
upload(void)800 void TextureCubeArray::upload (void)
801 {
802 const glw::Functions& gl = m_context.getFunctions();
803
804 if (!gl.texImage3D)
805 throw tcu::NotSupportedError("glTexImage3D() is not supported");
806
807 TCU_CHECK(m_glTexture);
808 gl.bindTexture(GL_TEXTURE_CUBE_MAP_ARRAY, m_glTexture);
809 gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
810 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
811
812 TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
813
814 for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
815 {
816 if (m_refTexture.isLevelEmpty(levelNdx))
817 continue; // Don't upload.
818
819 tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
820 DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
821 DE_ASSERT(access.getSlicePitch() == access.getFormat().getPixelSize()*access.getWidth()*access.getHeight());
822 gl.texImage3D(GL_TEXTURE_CUBE_MAP_ARRAY, levelNdx, m_format, access.getWidth(), access.getHeight(), access.getDepth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
823 }
824
825 GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
826 }
827
828 // TextureBuffer
829
TextureBuffer(const RenderContext & context,deUint32 internalFormat,size_t bufferSize)830 TextureBuffer::TextureBuffer (const RenderContext& context, deUint32 internalFormat, size_t bufferSize)
831 : m_context (context)
832 , m_format (0)
833 , m_offset (0)
834 , m_size (0)
835 , m_glTexture (0)
836 , m_glBuffer (0)
837 {
838 init(internalFormat, bufferSize, 0, 0, DE_NULL);
839 }
840
TextureBuffer(const RenderContext & context,deUint32 internalFormat,size_t bufferSize,size_t offset,size_t size,const void * data)841 TextureBuffer::TextureBuffer (const RenderContext& context, deUint32 internalFormat, size_t bufferSize, size_t offset, size_t size, const void* data)
842 : m_context (context)
843 , m_format (0)
844 , m_offset (0)
845 , m_size (0)
846 , m_glTexture (0)
847 , m_glBuffer (0)
848 {
849 init(internalFormat, bufferSize, offset, size, data);
850 }
851
init(deUint32 internalFormat,size_t bufferSize,size_t offset,size_t size,const void * data)852 void TextureBuffer::init (deUint32 internalFormat, size_t bufferSize, size_t offset, size_t size, const void* data)
853 {
854 const glw::Functions& gl = m_context.getFunctions();
855 de::UniquePtr<ContextInfo> info (ContextInfo::create(m_context));
856
857 if (offset != 0 || size != 0)
858 {
859 if (!(contextSupports(m_context.getType(), glu::ApiType(3, 3, glu::PROFILE_CORE)) && info->isExtensionSupported("GL_ARB_texture_buffer_range"))
860 && !(contextSupports(m_context.getType(), glu::ApiType(3, 1, glu::PROFILE_ES))
861 && info->isExtensionSupported("GL_EXT_texture_buffer")))
862 {
863 throw tcu::NotSupportedError("Ranged texture buffers not supported", "", __FILE__, __LINE__);
864 }
865 }
866 else
867 {
868 if (!contextSupports(m_context.getType(), glu::ApiType(3, 3, glu::PROFILE_CORE))
869 && !(contextSupports(m_context.getType(), glu::ApiType(3, 1, glu::PROFILE_ES))
870 && info->isExtensionSupported("GL_EXT_texture_buffer")))
871 {
872 throw tcu::NotSupportedError("Texture buffers not supported", "", __FILE__, __LINE__);
873 }
874 }
875
876 m_refBuffer.setStorage(bufferSize);
877 if (data)
878 deMemcpy(m_refBuffer.getPtr(), data, (int)bufferSize);
879
880 m_format = internalFormat;
881 m_offset = offset;
882 m_size = size;
883
884 DE_ASSERT(size != 0 || offset == 0);
885
886 {
887 gl.genTextures(1, &m_glTexture);
888 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
889
890 gl.genBuffers(1, &m_glBuffer);
891 GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers() failed");
892
893 gl.bindBuffer(GL_TEXTURE_BUFFER, m_glBuffer);
894 gl.bufferData(GL_TEXTURE_BUFFER, (glw::GLsizei)m_refBuffer.size(), data, GL_STATIC_DRAW);
895 gl.bindBuffer(GL_TEXTURE_BUFFER, 0);
896 GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to create buffer");
897
898 gl.bindTexture(GL_TEXTURE_BUFFER, m_glTexture);
899
900 if (offset != 0 || size != 0)
901 gl.texBufferRange(GL_TEXTURE_BUFFER, m_format, m_glBuffer, (glw::GLintptr)m_offset, (glw::GLsizeiptr)m_size);
902 else
903 gl.texBuffer(GL_TEXTURE_BUFFER, m_format, m_glBuffer);
904
905 gl.bindTexture(GL_TEXTURE_BUFFER, 0);
906 GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to bind buffer to texture");
907 }
908 }
909
~TextureBuffer(void)910 TextureBuffer::~TextureBuffer (void)
911 {
912 if (m_glTexture)
913 m_context.getFunctions().deleteTextures(1, &m_glTexture);
914
915 if (m_glBuffer)
916 m_context.getFunctions().deleteBuffers(1, &m_glBuffer);
917 }
918
919
getFullRefTexture(void)920 const tcu::PixelBufferAccess TextureBuffer::getFullRefTexture (void)
921 {
922 const tcu::TextureFormat format = mapGLInternalFormat(m_format);
923 const size_t bufferLengthBytes = (m_size != 0) ? (m_size) : (m_refBuffer.size());
924 const int bufferLengthPixels = (int)bufferLengthBytes / format.getPixelSize();
925
926 return tcu::PixelBufferAccess(format,
927 tcu::IVec3(bufferLengthPixels, 1, 1),
928 (deUint8*)m_refBuffer.getPtr() + m_offset);
929 }
930
getFullRefTexture(void) const931 const tcu::ConstPixelBufferAccess TextureBuffer::getFullRefTexture (void) const
932 {
933 return const_cast<TextureBuffer*>(this)->getFullRefTexture();
934 }
935
upload(void)936 void TextureBuffer::upload (void)
937 {
938 const glw::Functions& gl = m_context.getFunctions();
939
940 gl.bindBuffer(GL_TEXTURE_BUFFER, m_glBuffer);
941 gl.bufferData(GL_TEXTURE_BUFFER, (glw::GLsizei)m_refBuffer.size(), m_refBuffer.getPtr(), GL_STATIC_DRAW);
942 gl.bindBuffer(GL_TEXTURE_BUFFER, 0);
943 GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to upload buffer");
944 }
945
946 } // glu
947