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 		throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
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 // TextureCube
287 
TextureCube(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)288 TextureCube::TextureCube (const RenderContext& context, const ContextInfo& contextInfo, int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
289 	: m_context			(context)
290 	, m_isCompressed	(true)
291 	, m_format			(getGLFormat(levels[0].getFormat()))
292 	, m_refTexture		(getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth())
293 	, m_glTexture		(0)
294 {
295 	const glw::Functions& gl = m_context.getFunctions();
296 
297 	TCU_CHECK_INTERNAL(levels[0].getWidth() == levels[0].getHeight());
298 
299 	if (!contextInfo.isCompressedTextureFormatSupported(m_format))
300 		throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
301 
302 	gl.genTextures(1, &m_glTexture);
303 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
304 
305 	try
306 	{
307 		loadCompressed(numLevels, levels, decompressionParams);
308 	}
309 	catch (const std::exception&)
310 	{
311 		gl.deleteTextures(1, &m_glTexture);
312 		throw;
313 	}
314 }
315 
TextureCube(const RenderContext & context,deUint32 format,deUint32 dataType,int size)316 TextureCube::TextureCube (const RenderContext& context, deUint32 format, deUint32 dataType, int size)
317 	: m_context			(context)
318 	, m_isCompressed	(false)
319 	, m_format			(format)
320 	, m_refTexture		(mapGLTransferFormat(format, dataType), size)
321 	, m_glTexture		(0)
322 {
323 	const glw::Functions& gl = m_context.getFunctions();
324 	gl.genTextures(1, &m_glTexture);
325 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
326 }
327 
TextureCube(const RenderContext & context,deUint32 internalFormat,int size)328 TextureCube::TextureCube (const RenderContext& context, deUint32 internalFormat, int size)
329 	: m_context			(context)
330 	, m_isCompressed	(false)
331 	, m_format			(internalFormat)
332 	, m_refTexture		(mapGLInternalFormat(internalFormat), size)
333 	, m_glTexture		(0)
334 {
335 	const glw::Functions& gl = m_context.getFunctions();
336 	gl.genTextures(1, &m_glTexture);
337 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
338 }
339 
~TextureCube(void)340 TextureCube::~TextureCube (void)
341 {
342 	if (m_glTexture)
343 		m_context.getFunctions().deleteTextures(1, &m_glTexture);
344 }
345 
upload(void)346 void TextureCube::upload (void)
347 {
348 	const glw::Functions& gl = m_context.getFunctions();
349 
350 	DE_ASSERT(!m_isCompressed);
351 
352 	TCU_CHECK(m_glTexture);
353 	gl.bindTexture(GL_TEXTURE_CUBE_MAP, m_glTexture);
354 	gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
355 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
356 
357 	TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
358 
359 	for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
360 	{
361 		for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
362 		{
363 			if (m_refTexture.isLevelEmpty((tcu::CubeFace)face, levelNdx))
364 				continue; // Don't upload.
365 
366 			tcu::ConstPixelBufferAccess access = m_refTexture.getLevelFace(levelNdx, (tcu::CubeFace)face);
367 			DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
368 			gl.texImage2D(getGLCubeFace((tcu::CubeFace)face), levelNdx, m_format, access.getWidth(), access.getHeight(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
369 		}
370 	}
371 
372 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
373 }
374 
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)375 void TextureCube::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
376 {
377 	const glw::Functions&	gl					= m_context.getFunctions();
378 	deUint32				compressedFormat	= getGLFormat(levels[0].getFormat());
379 
380 	TCU_CHECK(m_glTexture);
381 	gl.bindTexture(GL_TEXTURE_CUBE_MAP, m_glTexture);
382 
383 	for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
384 	{
385 		for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
386 		{
387 			const tcu::CompressedTexture& level = levels[levelNdx*tcu::CUBEFACE_LAST + face];
388 
389 			// Decompress to reference texture.
390 			m_refTexture.allocLevel((tcu::CubeFace)face, levelNdx);
391 			tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevelFace(levelNdx, (tcu::CubeFace)face);
392 			TCU_CHECK(level.getWidth()	== refLevelAccess.getWidth() &&
393 					  level.getHeight()	== refLevelAccess.getHeight());
394 			level.decompress(refLevelAccess, decompressionParams);
395 
396 			// Upload to GL texture in compressed form.
397 			gl.compressedTexImage2D(getGLCubeFace((tcu::CubeFace)face), levelNdx, compressedFormat,
398 									level.getWidth(), level.getHeight(), 0 /* border */, level.getDataSize(), level.getData());
399 		}
400 	}
401 
402 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
403 }
404 
create(const RenderContext & context,const ContextInfo & contextInfo,const tcu::Archive & archive,int numLevels,const char * const * filenames)405 TextureCube* TextureCube::create (const RenderContext& context, const ContextInfo& contextInfo, const tcu::Archive& archive, int numLevels, const char* const* filenames)
406 {
407 	DE_ASSERT(numLevels > 0);
408 
409 	std::string ext = de::FilePath(filenames[0]).getFileExtension();
410 
411 	// \todo [2011-11-21 pyry] Support PNG images.
412 	if (ext == "pkm")
413 	{
414 		// Compressed texture.
415 		int								numImages	= numLevels*tcu::CUBEFACE_LAST;
416 		vector<tcu::CompressedTexture>	levels		(numImages);
417 
418 		for (int ndx = 0; ndx < numImages; ndx++)
419 			tcu::ImageIO::loadPKM(levels[ndx], archive, filenames[ndx]);
420 
421 		return new TextureCube(context, contextInfo, numLevels, &levels[0]);
422 	}
423 	else
424 		TCU_FAIL("Unsupported file format");
425 }
426 
create(const RenderContext & context,const ContextInfo & contextInfo,const tcu::Archive & archive,int numLevels,const std::vector<std::string> & filenames)427 TextureCube* TextureCube::create (const RenderContext& context, const ContextInfo& contextInfo, const tcu::Archive& archive, int numLevels, const std::vector<std::string>& filenames)
428 {
429 	DE_STATIC_ASSERT(tcu::CUBEFACE_LAST == 6);
430 	TCU_CHECK(numLevels*tcu::CUBEFACE_LAST == (int)filenames.size());
431 
432 	std::vector<const char*> charPtrs(filenames.size());
433 	for (int ndx = 0; ndx < (int)filenames.size(); ndx++)
434 		charPtrs[ndx] = filenames[ndx].c_str();
435 
436 	return TextureCube::create(context, contextInfo, archive, numLevels, &charPtrs[0]);
437 }
438 
439 // Texture1DArray
440 
Texture1DArray(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int numLevels)441 Texture1DArray::Texture1DArray (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int numLevels)
442 	: m_context			(context)
443 	, m_format			(format)
444 	, m_refTexture		(mapGLTransferFormat(format, dataType), width, numLevels)
445 	, m_glTexture		(0)
446 {
447 	const glw::Functions& gl = m_context.getFunctions();
448 	gl.genTextures(1, &m_glTexture);
449 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
450 }
451 
Texture1DArray(const RenderContext & context,deUint32 sizedFormat,int width,int numLevels)452 Texture1DArray::Texture1DArray (const RenderContext& context, deUint32 sizedFormat, int width, int numLevels)
453 	: m_context			(context)
454 	, m_format			(sizedFormat)
455 	, m_refTexture		(mapGLInternalFormat(sizedFormat), width, numLevels)
456 	, m_glTexture		(0)
457 {
458 	const glw::Functions& gl = m_context.getFunctions();
459 	gl.genTextures(1, &m_glTexture);
460 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
461 }
462 
~Texture1DArray(void)463 Texture1DArray::~Texture1DArray (void)
464 {
465 	if (m_glTexture)
466 		m_context.getFunctions().deleteTextures(1, &m_glTexture);
467 }
468 
upload(void)469 void Texture1DArray::upload (void)
470 {
471 	const glw::Functions& gl = m_context.getFunctions();
472 
473 	TCU_CHECK(m_glTexture);
474 	gl.bindTexture(GL_TEXTURE_1D_ARRAY, m_glTexture);
475 	gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
476 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
477 
478 	TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
479 
480 	for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
481 	{
482 		if (m_refTexture.isLevelEmpty(levelNdx))
483 			continue; // Don't upload.
484 
485 		tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
486 		DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
487 		gl.texImage2D(GL_TEXTURE_1D_ARRAY, levelNdx, m_format, access.getWidth(), access.getHeight(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
488 	}
489 
490 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
491 }
492 
493 // Texture2DArray
494 
Texture2DArray(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int height,int numLevels)495 Texture2DArray::Texture2DArray (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int height, int numLevels)
496 	: m_context			(context)
497 	, m_isCompressed	(false)
498 	, m_format			(format)
499 	, m_refTexture		(mapGLTransferFormat(format, dataType), width, height, numLevels)
500 	, m_glTexture		(0)
501 {
502 	// \todo [2013-04-08 pyry] Check support here.
503 	const glw::Functions& gl = m_context.getFunctions();
504 	gl.genTextures(1, &m_glTexture);
505 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
506 }
507 
Texture2DArray(const RenderContext & context,deUint32 sizedFormat,int width,int height,int numLevels)508 Texture2DArray::Texture2DArray (const RenderContext& context, deUint32 sizedFormat, int width, int height, int numLevels)
509 	: m_context			(context)
510 	, m_isCompressed	(false)
511 	, m_format			(sizedFormat)
512 	, m_refTexture		(mapGLInternalFormat(sizedFormat), width, height, numLevels)
513 	, m_glTexture		(0)
514 {
515 	// \todo [2013-04-08 pyry] Check support here.
516 	const glw::Functions& gl = m_context.getFunctions();
517 	gl.genTextures(1, &m_glTexture);
518 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
519 }
520 
Texture2DArray(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)521 Texture2DArray::Texture2DArray (const RenderContext& context, const ContextInfo& contextInfo, int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
522 	: m_context			(context)
523 	, m_isCompressed	(true)
524 	, m_format			(getGLFormat(levels[0].getFormat()))
525 	, m_refTexture		(getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth(), levels[0].getHeight(), levels[0].getDepth())
526 	, m_glTexture		(0)
527 {
528 	const glw::Functions& gl = context.getFunctions();
529 
530 	if (!contextInfo.isCompressedTextureFormatSupported(m_format))
531 		throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
532 
533 	gl.genTextures(1, &m_glTexture);
534 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
535 
536 	try
537 	{
538 		loadCompressed(numLevels, levels, decompressionParams);
539 	}
540 	catch (const std::exception&)
541 	{
542 		gl.deleteTextures(1, &m_glTexture);
543 		throw;
544 	}
545 }
546 
~Texture2DArray(void)547 Texture2DArray::~Texture2DArray (void)
548 {
549 	if (m_glTexture)
550 		m_context.getFunctions().deleteTextures(1, &m_glTexture);
551 }
552 
upload(void)553 void Texture2DArray::upload (void)
554 {
555 	const glw::Functions& gl = m_context.getFunctions();
556 
557 	if (!gl.texImage3D)
558 		throw tcu::NotSupportedError("glTexImage3D() is not supported");
559 
560 	TCU_CHECK(m_glTexture);
561 	gl.bindTexture(GL_TEXTURE_2D_ARRAY, m_glTexture);
562 	gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
563 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
564 
565 	TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
566 
567 	for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
568 	{
569 		if (m_refTexture.isLevelEmpty(levelNdx))
570 			continue; // Don't upload.
571 
572 		tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
573 		DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
574 		DE_ASSERT(access.getSlicePitch() == access.getFormat().getPixelSize()*access.getWidth()*access.getHeight());
575 		gl.texImage3D(GL_TEXTURE_2D_ARRAY, levelNdx, m_format, access.getWidth(), access.getHeight(), access.getDepth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
576 	}
577 
578 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
579 }
580 
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)581 void Texture2DArray::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
582 {
583 	const glw::Functions&	gl					= m_context.getFunctions();
584 	deUint32				compressedFormat	= getGLFormat(levels[0].getFormat());
585 
586 	TCU_CHECK(m_glTexture);
587 	gl.bindTexture(GL_TEXTURE_2D_ARRAY, m_glTexture);
588 
589 	for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
590 	{
591 		const tcu::CompressedTexture& level = levels[levelNdx];
592 
593 		// Decompress to reference texture.
594 		m_refTexture.allocLevel(levelNdx);
595 		tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevel(levelNdx);
596 		TCU_CHECK(level.getWidth()	== refLevelAccess.getWidth() &&
597 				  level.getHeight()	== refLevelAccess.getHeight() &&
598 				  level.getDepth()	== refLevelAccess.getDepth());
599 		level.decompress(refLevelAccess, decompressionParams);
600 
601 		// Upload to GL texture in compressed form.
602 		gl.compressedTexImage3D(GL_TEXTURE_2D_ARRAY, levelNdx, compressedFormat,
603 								level.getWidth(), level.getHeight(), m_refTexture.getLevel(levelNdx).getDepth(), 0 /* border */, level.getDataSize(), level.getData());
604 	}
605 
606 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
607 }
608 
609 // Texture3D
610 
Texture3D(const RenderContext & context,deUint32 format,deUint32 dataType,int width,int height,int depth)611 Texture3D::Texture3D (const RenderContext& context, deUint32 format, deUint32 dataType, int width, int height, int depth)
612 	: m_context			(context)
613 	, m_isCompressed	(false)
614 	, m_format			(format)
615 	, m_refTexture		(mapGLTransferFormat(format, dataType), width, height, depth)
616 	, m_glTexture		(0)
617 {
618 	// \todo [2013-04-08 pyry] Check support here.
619 	const glw::Functions& gl = m_context.getFunctions();
620 	gl.genTextures(1, &m_glTexture);
621 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
622 }
623 
Texture3D(const RenderContext & context,deUint32 sizedFormat,int width,int height,int depth)624 Texture3D::Texture3D (const RenderContext& context, deUint32 sizedFormat, int width, int height, int depth)
625 	: m_context			(context)
626 	, m_isCompressed	(false)
627 	, m_format			(sizedFormat)
628 	, m_refTexture		(mapGLInternalFormat(sizedFormat), width, height, depth)
629 	, m_glTexture		(0)
630 {
631 	// \todo [2013-04-08 pyry] Check support here.
632 	const glw::Functions& gl = m_context.getFunctions();
633 	gl.genTextures(1, &m_glTexture);
634 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
635 }
636 
Texture3D(const RenderContext & context,const ContextInfo & contextInfo,int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)637 Texture3D::Texture3D (const RenderContext&					context,
638 					  const ContextInfo&					contextInfo,
639 					  int									numLevels,
640 					  const tcu::CompressedTexture*			levels,
641 					  const tcu::TexDecompressionParams&	decompressionParams)
642 	: m_context			(context)
643 	, m_isCompressed	(true)
644 	, m_format			(getGLFormat(levels[0].getFormat()))
645 	, m_refTexture		(getUncompressedFormat(levels[0].getFormat()), levels[0].getWidth(), levels[0].getHeight(), levels[0].getDepth())
646 	, m_glTexture		(0)
647 {
648 	const glw::Functions& gl = context.getFunctions();
649 
650 	if (!contextInfo.isCompressedTextureFormatSupported(m_format))
651 		throw tcu::NotSupportedError("Compressed texture format not supported", "", __FILE__, __LINE__);
652 
653 	gl.genTextures(1, &m_glTexture);
654 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
655 
656 	try
657 	{
658 		loadCompressed(numLevels, levels, decompressionParams);
659 	}
660 	catch (const std::exception&)
661 	{
662 		gl.deleteTextures(1, &m_glTexture);
663 		throw;
664 	}
665 }
666 
~Texture3D(void)667 Texture3D::~Texture3D (void)
668 {
669 	if (m_glTexture)
670 		m_context.getFunctions().deleteTextures(1, &m_glTexture);
671 }
672 
upload(void)673 void Texture3D::upload (void)
674 {
675 	const glw::Functions& gl = m_context.getFunctions();
676 
677 	DE_ASSERT(!m_isCompressed);
678 
679 	if (!gl.texImage3D)
680 		throw tcu::NotSupportedError("glTexImage3D() is not supported");
681 
682 	TCU_CHECK(m_glTexture);
683 	gl.bindTexture(GL_TEXTURE_3D, m_glTexture);
684 	gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
685 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
686 
687 	TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
688 
689 	for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
690 	{
691 		if (m_refTexture.isLevelEmpty(levelNdx))
692 			continue; // Don't upload.
693 
694 		tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
695 		DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
696 		DE_ASSERT(access.getSlicePitch() == access.getFormat().getPixelSize()*access.getWidth()*access.getHeight());
697 		gl.texImage3D(GL_TEXTURE_3D, levelNdx, m_format, access.getWidth(), access.getHeight(), access.getDepth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
698 	}
699 
700 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
701 }
702 
loadCompressed(int numLevels,const tcu::CompressedTexture * levels,const tcu::TexDecompressionParams & decompressionParams)703 void Texture3D::loadCompressed (int numLevels, const tcu::CompressedTexture* levels, const tcu::TexDecompressionParams& decompressionParams)
704 {
705 	const glw::Functions&	gl					= m_context.getFunctions();
706 	deUint32				compressedFormat	= getGLFormat(levels[0].getFormat());
707 
708 	if (!gl.compressedTexImage3D)
709 		throw tcu::NotSupportedError("glCompressedTexImage3D() is not supported");
710 
711 	TCU_CHECK(m_glTexture);
712 	gl.bindTexture(GL_TEXTURE_3D, m_glTexture);
713 
714 	for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
715 	{
716 		const tcu::CompressedTexture& level = levels[levelNdx];
717 
718 		// Decompress to reference texture.
719 		m_refTexture.allocLevel(levelNdx);
720 		tcu::PixelBufferAccess refLevelAccess = m_refTexture.getLevel(levelNdx);
721 		TCU_CHECK(level.getWidth()	== refLevelAccess.getWidth() &&
722 				  level.getHeight()	== refLevelAccess.getHeight() &&
723 				  level.getDepth()	== refLevelAccess.getDepth());
724 		level.decompress(refLevelAccess, decompressionParams);
725 
726 		// Upload to GL texture in compressed form.
727 		gl.compressedTexImage3D(GL_TEXTURE_3D, levelNdx, compressedFormat,
728 								level.getWidth(), level.getHeight(), level.getDepth(), 0 /* border */, level.getDataSize(), level.getData());
729 	}
730 
731 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
732 }
733 
734 // TextureCubeArray
735 
TextureCubeArray(const RenderContext & context,deUint32 format,deUint32 dataType,int size,int numLayers)736 TextureCubeArray::TextureCubeArray (const RenderContext& context, deUint32 format, deUint32 dataType, int size, int numLayers)
737 	: m_context			(context)
738 	, m_format			(format)
739 	, m_refTexture		(mapGLTransferFormat(format, dataType), size, numLayers)
740 	, m_glTexture		(0)
741 {
742 	// \todo [2013-04-08 pyry] Check support here.
743 	const glw::Functions& gl = m_context.getFunctions();
744 	gl.genTextures(1, &m_glTexture);
745 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
746 }
747 
TextureCubeArray(const RenderContext & context,deUint32 sizedFormat,int size,int numLayers)748 TextureCubeArray::TextureCubeArray (const RenderContext& context, deUint32 sizedFormat, int size, int numLayers)
749 	: m_context			(context)
750 	, m_format			(sizedFormat)
751 	, m_refTexture		(mapGLInternalFormat(sizedFormat), size, numLayers)
752 	, m_glTexture		(0)
753 {
754 	// \todo [2013-04-08 pyry] Check support here.
755 	const glw::Functions& gl = m_context.getFunctions();
756 	gl.genTextures(1, &m_glTexture);
757 	GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
758 }
759 
~TextureCubeArray(void)760 TextureCubeArray::~TextureCubeArray (void)
761 {
762 	if (m_glTexture)
763 		m_context.getFunctions().deleteTextures(1, &m_glTexture);
764 }
765 
upload(void)766 void TextureCubeArray::upload (void)
767 {
768 	const glw::Functions& gl = m_context.getFunctions();
769 
770 	if (!gl.texImage3D)
771 		throw tcu::NotSupportedError("glTexImage3D() is not supported");
772 
773 	TCU_CHECK(m_glTexture);
774 	gl.bindTexture(GL_TEXTURE_CUBE_MAP_ARRAY, m_glTexture);
775 	gl.pixelStorei(GL_UNPACK_ALIGNMENT, computePixelStore(m_refTexture.getFormat()));
776 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
777 
778 	TransferFormat transferFormat = getTransferFormat(m_refTexture.getFormat());
779 
780 	for (int levelNdx = 0; levelNdx < m_refTexture.getNumLevels(); levelNdx++)
781 	{
782 		if (m_refTexture.isLevelEmpty(levelNdx))
783 			continue; // Don't upload.
784 
785 		tcu::ConstPixelBufferAccess access = m_refTexture.getLevel(levelNdx);
786 		DE_ASSERT(access.getRowPitch() == access.getFormat().getPixelSize()*access.getWidth());
787 		DE_ASSERT(access.getSlicePitch() == access.getFormat().getPixelSize()*access.getWidth()*access.getHeight());
788 		gl.texImage3D(GL_TEXTURE_CUBE_MAP_ARRAY, levelNdx, m_format, access.getWidth(), access.getHeight(), access.getDepth(), 0 /* border */, transferFormat.format, transferFormat.dataType, access.getDataPtr());
789 	}
790 
791 	GLU_EXPECT_NO_ERROR(gl.getError(), "Texture upload failed");
792 }
793 
794 // TextureBuffer
795 
TextureBuffer(const RenderContext & context,deUint32 internalFormat,size_t bufferSize)796 TextureBuffer::TextureBuffer (const RenderContext& context, deUint32 internalFormat, size_t bufferSize)
797 	: m_context			(context)
798 	, m_format			(0)
799 	, m_offset			(0)
800 	, m_size			(0)
801 	, m_glTexture		(0)
802 	, m_glBuffer		(0)
803 {
804 	init(internalFormat, bufferSize, 0, 0, DE_NULL);
805 }
806 
TextureBuffer(const RenderContext & context,deUint32 internalFormat,size_t bufferSize,size_t offset,size_t size,const void * data)807 TextureBuffer::TextureBuffer (const RenderContext& context, deUint32 internalFormat, size_t bufferSize, size_t offset, size_t size, const void* data)
808 	: m_context			(context)
809 	, m_format			(0)
810 	, m_offset			(0)
811 	, m_size			(0)
812 	, m_glTexture		(0)
813 	, m_glBuffer		(0)
814 {
815 	init(internalFormat, bufferSize, offset, size, data);
816 }
817 
init(deUint32 internalFormat,size_t bufferSize,size_t offset,size_t size,const void * data)818 void TextureBuffer::init (deUint32 internalFormat, size_t bufferSize, size_t offset, size_t size, const void* data)
819 {
820 	const glw::Functions&		gl		= m_context.getFunctions();
821 	de::UniquePtr<ContextInfo>	info 	(ContextInfo::create(m_context));
822 
823 	if (offset != 0 || size != 0)
824 	{
825 		if (!(contextSupports(m_context.getType(), glu::ApiType(3, 3, glu::PROFILE_CORE)) && info->isExtensionSupported("GL_ARB_texture_buffer_range"))
826 			&& !(contextSupports(m_context.getType(), glu::ApiType(3, 1, glu::PROFILE_ES))
827 				&& info->isExtensionSupported("GL_EXT_texture_buffer")))
828 		{
829 			throw tcu::NotSupportedError("Ranged texture buffers not supported", "", __FILE__, __LINE__);
830 		}
831 	}
832 	else
833 	{
834 		if (!contextSupports(m_context.getType(), glu::ApiType(3, 3, glu::PROFILE_CORE))
835 			&& !(contextSupports(m_context.getType(), glu::ApiType(3, 1, glu::PROFILE_ES))
836 				&& info->isExtensionSupported("GL_EXT_texture_buffer")))
837 		{
838 			throw tcu::NotSupportedError("Texture buffers not supported", "", __FILE__, __LINE__);
839 		}
840 	}
841 
842 	m_refBuffer.resize(bufferSize, 0);
843 
844 	if (data)
845 		deMemcpy(&m_refBuffer[0], data, (int)bufferSize);
846 
847 	m_format	= internalFormat;
848 	m_offset	= offset;
849 	m_size		= size;
850 
851 	DE_ASSERT(size != 0 || offset == 0);
852 
853 	{
854 		const tcu::TextureFormat	format			= mapGLInternalFormat(internalFormat);
855 		deInt32						maxTextureSize	= 0;
856 
857 		gl.getIntegerv(GL_MAX_TEXTURE_BUFFER_SIZE, &maxTextureSize);
858 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv(GL_MAX_TEXTURE_BUFFER_SIZE) failed");
859 
860 		m_refTexture = tcu::PixelBufferAccess(format, de::min<int>((int)maxTextureSize, (int)((m_size != 0 ? m_size : bufferSize) / format.getPixelSize())), 1, 1, &(m_refBuffer[m_offset]));
861 	}
862 
863 	{
864 		gl.genTextures(1, &m_glTexture);
865 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGenTextures() failed");
866 
867 		gl.genBuffers(1, &m_glBuffer);
868 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGenBuffers() failed");
869 
870 		gl.bindBuffer(GL_TEXTURE_BUFFER, m_glBuffer);
871 		gl.bufferData(GL_TEXTURE_BUFFER, (glw::GLsizei)m_refBuffer.size(), &(m_refBuffer[0]), GL_STATIC_DRAW);
872 		gl.bindBuffer(GL_TEXTURE_BUFFER, 0);
873 		GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to create buffer");
874 
875 		gl.bindTexture(GL_TEXTURE_BUFFER, m_glTexture);
876 
877 		if (offset != 0 || size != 0)
878 			gl.texBufferRange(GL_TEXTURE_BUFFER, m_format, m_glBuffer, (glw::GLintptr)m_offset, (glw::GLsizeiptr)m_size);
879 		else
880 			gl.texBuffer(GL_TEXTURE_BUFFER, m_format, m_glBuffer);
881 
882 		gl.bindTexture(GL_TEXTURE_BUFFER, 0);
883 		GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to bind buffer to texture");
884 	}
885 }
886 
~TextureBuffer(void)887 TextureBuffer::~TextureBuffer (void)
888 {
889 	if (m_glTexture)
890 		m_context.getFunctions().deleteTextures(1, &m_glTexture);
891 
892 	if (m_glBuffer)
893 		m_context.getFunctions().deleteBuffers(1, &m_glBuffer);
894 }
895 
upload(void)896 void TextureBuffer::upload (void)
897 {
898 	const glw::Functions& gl = m_context.getFunctions();
899 
900 	gl.bindBuffer(GL_TEXTURE_BUFFER, m_glBuffer);
901 	gl.bufferData(GL_TEXTURE_BUFFER, (glw::GLsizei)m_refBuffer.size(), &(m_refBuffer[0]), GL_STATIC_DRAW);
902 	gl.bindBuffer(GL_TEXTURE_BUFFER, 0);
903 	GLU_EXPECT_NO_ERROR(gl.getError(), "Failed to upload buffer");
904 }
905 
bufferData(const deUint8 * data,size_t size)906 void TextureBuffer::bufferData (const deUint8* data, size_t size)
907 {
908 	const glw::Functions& gl = m_context.getFunctions();
909 
910 	m_refBuffer = vector<deUint8>(data, data+size);
911 
912 	{
913 		const tcu::TextureFormat	format			= mapGLInternalFormat(m_format);
914 		deInt32						maxTextureSize	= 0;
915 
916 		gl.getIntegerv(GL_MAX_TEXTURE_BUFFER_SIZE, &maxTextureSize);
917 		GLU_EXPECT_NO_ERROR(gl.getError(), "glGetIntegerv(GL_MAX_TEXTURE_BUFFER_SIZE) failed");
918 
919 		m_refTexture = tcu::PixelBufferAccess(format, de::min<int>((int)maxTextureSize, (int)((m_size != 0 ? m_size : m_refBuffer.size())  / format.getPixelSize())), 1, 1, &(m_refBuffer[m_offset]));
920 	}
921 }
922 
923 } // glu
924