blob: 62305565f17b674709dd982a53a62bc15ae7d8cf [file] [log] [blame]
Geoff Lang051dbc72015-01-05 15:48:58 -05001//
Jamie Madilld2b50a02016-06-09 00:13:35 -07002// Copyright 2016 The ANGLE Project Authors. All rights reserved.
Geoff Lang051dbc72015-01-05 15:48:58 -05003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
Jamie Madilld2b50a02016-06-09 00:13:35 -07006// renderer_utils:
7// Helper methods pertaining to most or all back-ends.
8//
Geoff Lang051dbc72015-01-05 15:48:58 -05009
Jamie Madilld2b50a02016-06-09 00:13:35 -070010#include "libANGLE/renderer/renderer_utils.h"
Geoff Lang051dbc72015-01-05 15:48:58 -050011
Geoff Lang6e4cfce2016-06-13 15:06:31 -040012#include "image_util/copyimage.h"
13#include "image_util/imageformats.h"
14
Jamie Madill222c5172017-07-19 16:15:42 -040015#include "libANGLE/AttributeMap.h"
Jamie Madill42975642017-10-12 12:31:51 -040016#include "libANGLE/Context.h"
Jamie Madilld2b50a02016-06-09 00:13:35 -070017#include "libANGLE/formatutils.h"
Jamie Madill42975642017-10-12 12:31:51 -040018#include "libANGLE/renderer/ContextImpl.h"
Jamie Madill86e0b7f2016-08-09 11:10:29 -040019#include "libANGLE/renderer/Format.h"
Geoff Lang6e4cfce2016-06-13 15:06:31 -040020
21#include <string.h>
Luc Ferron48cdc2e2018-05-31 09:58:34 -040022#include "common/utilities.h"
Geoff Lang051dbc72015-01-05 15:48:58 -050023
24namespace rx
25{
26
Jamie Madilld2b50a02016-06-09 00:13:35 -070027namespace
28{
Geoff Langd93cd6c2017-08-11 16:32:41 -040029void CopyColor(gl::ColorF *color)
30{
31 // No-op
32}
33
34void PremultiplyAlpha(gl::ColorF *color)
35{
36 color->red *= color->alpha;
37 color->green *= color->alpha;
38 color->blue *= color->alpha;
39}
40
41void UnmultiplyAlpha(gl::ColorF *color)
42{
43 if (color->alpha != 0.0f)
44 {
45 float invAlpha = 1.0f / color->alpha;
46 color->red *= invAlpha;
47 color->green *= invAlpha;
48 color->blue *= invAlpha;
49 }
50}
51
52void ClipChannelsR(gl::ColorF *color)
53{
54 color->green = 0.0f;
55 color->blue = 0.0f;
56 color->alpha = 1.0f;
57}
58
59void ClipChannelsRG(gl::ColorF *color)
60{
61 color->blue = 0.0f;
62 color->alpha = 1.0f;
63}
64
65void ClipChannelsRGB(gl::ColorF *color)
66{
67 color->alpha = 1.0f;
68}
69
70void ClipChannelsLuminance(gl::ColorF *color)
71{
72 color->alpha = 1.0f;
73}
74
75void ClipChannelsAlpha(gl::ColorF *color)
76{
77 color->red = 0.0f;
78 color->green = 0.0f;
79 color->blue = 0.0f;
80}
81
82void ClipChannelsNoOp(gl::ColorF *color)
83{
84}
85
86void WriteUintColor(const gl::ColorF &color,
Jamie Madill522095f2018-07-23 14:59:41 -040087 PixelWriteFunction colorWriteFunction,
Geoff Langd93cd6c2017-08-11 16:32:41 -040088 uint8_t *destPixelData)
89{
90 gl::ColorUI destColor(
91 static_cast<unsigned int>(color.red * 255), static_cast<unsigned int>(color.green * 255),
92 static_cast<unsigned int>(color.blue * 255), static_cast<unsigned int>(color.alpha * 255));
93 colorWriteFunction(reinterpret_cast<const uint8_t *>(&destColor), destPixelData);
94}
95
96void WriteFloatColor(const gl::ColorF &color,
Jamie Madill522095f2018-07-23 14:59:41 -040097 PixelWriteFunction colorWriteFunction,
Geoff Langd93cd6c2017-08-11 16:32:41 -040098 uint8_t *destPixelData)
99{
100 colorWriteFunction(reinterpret_cast<const uint8_t *>(&color), destPixelData);
101}
102
Luc Ferron46bcea52018-05-31 09:48:36 -0400103template <typename T, int cols, int rows>
104bool TransposeExpandMatrix(T *target, const GLfloat *value)
105{
106 constexpr int targetWidth = 4;
107 constexpr int targetHeight = rows;
108 constexpr int srcWidth = rows;
109 constexpr int srcHeight = cols;
110
111 constexpr int copyWidth = std::min(targetHeight, srcWidth);
112 constexpr int copyHeight = std::min(targetWidth, srcHeight);
113
114 T staging[targetWidth * targetHeight] = {0};
115
116 for (int x = 0; x < copyWidth; x++)
117 {
118 for (int y = 0; y < copyHeight; y++)
119 {
120 staging[x * targetWidth + y] = static_cast<T>(value[y * srcWidth + x]);
121 }
122 }
123
124 if (memcmp(target, staging, targetWidth * targetHeight * sizeof(T)) == 0)
125 {
126 return false;
127 }
128
129 memcpy(target, staging, targetWidth * targetHeight * sizeof(T));
130 return true;
131}
132
133template <typename T, int cols, int rows>
134bool ExpandMatrix(T *target, const GLfloat *value)
135{
136 constexpr int kTargetWidth = 4;
137 constexpr int kTargetHeight = rows;
138 constexpr int kSrcWidth = cols;
139 constexpr int kSrcHeight = rows;
140
141 constexpr int kCopyWidth = std::min(kTargetWidth, kSrcWidth);
142 constexpr int kCopyHeight = std::min(kTargetHeight, kSrcHeight);
143
144 T staging[kTargetWidth * kTargetHeight] = {0};
145
146 for (int y = 0; y < kCopyHeight; y++)
147 {
148 for (int x = 0; x < kCopyWidth; x++)
149 {
150 staging[y * kTargetWidth + x] = static_cast<T>(value[y * kSrcWidth + x]);
151 }
152 }
153
154 if (memcmp(target, staging, kTargetWidth * kTargetHeight * sizeof(T)) == 0)
155 {
156 return false;
157 }
158
159 memcpy(target, staging, kTargetWidth * kTargetHeight * sizeof(T));
160 return true;
161}
Jamie Madilld2b50a02016-06-09 00:13:35 -0700162} // anonymous namespace
Geoff Lang051dbc72015-01-05 15:48:58 -0500163
Jamie Madilld2b50a02016-06-09 00:13:35 -0700164PackPixelsParams::PackPixelsParams()
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400165 : destFormat(nullptr), outputPitch(0), packBuffer(nullptr), offset(0)
Jamie Madilld2b50a02016-06-09 00:13:35 -0700166{
167}
168
169PackPixelsParams::PackPixelsParams(const gl::Rectangle &areaIn,
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400170 const angle::Format &destFormat,
Jamie Madilld2b50a02016-06-09 00:13:35 -0700171 GLuint outputPitchIn,
172 const gl::PixelPackState &packIn,
Corentin Wallezcda6af12017-10-30 19:20:37 -0400173 gl::Buffer *packBufferIn,
Jamie Madilld2b50a02016-06-09 00:13:35 -0700174 ptrdiff_t offsetIn)
175 : area(areaIn),
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400176 destFormat(&destFormat),
Jamie Madilld2b50a02016-06-09 00:13:35 -0700177 outputPitch(outputPitchIn),
Corentin Wallezcda6af12017-10-30 19:20:37 -0400178 packBuffer(packBufferIn),
179 pack(),
Jamie Madilld2b50a02016-06-09 00:13:35 -0700180 offset(offsetIn)
181{
Corentin Wallezcda6af12017-10-30 19:20:37 -0400182 pack.alignment = packIn.alignment;
183 pack.reverseRowOrder = packIn.reverseRowOrder;
Jamie Madill4928b7c2017-06-20 12:57:39 -0400184}
185
Jamie Madilld2b50a02016-06-09 00:13:35 -0700186void PackPixels(const PackPixelsParams &params,
Jamie Madill86e0b7f2016-08-09 11:10:29 -0400187 const angle::Format &sourceFormat,
Jamie Madilld2b50a02016-06-09 00:13:35 -0700188 int inputPitchIn,
189 const uint8_t *sourceIn,
190 uint8_t *destWithoutOffset)
191{
192 uint8_t *destWithOffset = destWithoutOffset + params.offset;
193
194 const uint8_t *source = sourceIn;
195 int inputPitch = inputPitchIn;
196
197 if (params.pack.reverseRowOrder)
198 {
199 source += inputPitch * (params.area.height - 1);
200 inputPitch = -inputPitch;
201 }
202
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400203 if (sourceFormat == *params.destFormat)
Jamie Madilld2b50a02016-06-09 00:13:35 -0700204 {
205 // Direct copy possible
206 for (int y = 0; y < params.area.height; ++y)
207 {
208 memcpy(destWithOffset + y * params.outputPitch, source + y * inputPitch,
Luc Ferron339b95e2018-06-12 10:21:52 -0400209 params.area.width * sourceFormat.pixelBytes);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700210 }
211 return;
212 }
213
Jamie Madill522095f2018-07-23 14:59:41 -0400214 PixelCopyFunction fastCopyFunc = sourceFormat.fastCopyFunctions.get(params.destFormat->id);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700215
216 if (fastCopyFunc)
217 {
218 // Fast copy is possible through some special function
219 for (int y = 0; y < params.area.height; ++y)
220 {
221 for (int x = 0; x < params.area.width; ++x)
222 {
223 uint8_t *dest =
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400224 destWithOffset + y * params.outputPitch + x * params.destFormat->pixelBytes;
Luc Ferron339b95e2018-06-12 10:21:52 -0400225 const uint8_t *src = source + y * inputPitch + x * sourceFormat.pixelBytes;
Jamie Madilld2b50a02016-06-09 00:13:35 -0700226
227 fastCopyFunc(src, dest);
228 }
229 }
230 return;
231 }
232
Jamie Madill522095f2018-07-23 14:59:41 -0400233 PixelWriteFunction pixelWriteFunction = params.destFormat->pixelWriteFunction;
234 ASSERT(pixelWriteFunction != nullptr);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700235
236 // Maximum size of any Color<T> type used.
237 uint8_t temp[16];
238 static_assert(sizeof(temp) >= sizeof(gl::ColorF) && sizeof(temp) >= sizeof(gl::ColorUI) &&
Jamie Madillb436aac2018-07-18 17:23:48 -0400239 sizeof(temp) >= sizeof(gl::ColorI) &&
240 sizeof(temp) >= sizeof(angle::DepthStencil),
241 "Unexpected size of pixel struct.");
Jamie Madilld2b50a02016-06-09 00:13:35 -0700242
Jamie Madill522095f2018-07-23 14:59:41 -0400243 PixelReadFunction pixelReadFunction = sourceFormat.pixelReadFunction;
244 ASSERT(pixelReadFunction != nullptr);
Jamie Madill86e0b7f2016-08-09 11:10:29 -0400245
Jamie Madilld2b50a02016-06-09 00:13:35 -0700246 for (int y = 0; y < params.area.height; ++y)
247 {
248 for (int x = 0; x < params.area.width; ++x)
249 {
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400250 uint8_t *dest =
251 destWithOffset + y * params.outputPitch + x * params.destFormat->pixelBytes;
Luc Ferron339b95e2018-06-12 10:21:52 -0400252 const uint8_t *src = source + y * inputPitch + x * sourceFormat.pixelBytes;
Jamie Madilld2b50a02016-06-09 00:13:35 -0700253
254 // readFunc and writeFunc will be using the same type of color, CopyTexImage
255 // will not allow the copy otherwise.
Jamie Madill522095f2018-07-23 14:59:41 -0400256 pixelReadFunction(src, temp);
257 pixelWriteFunction(temp, dest);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700258 }
259 }
260}
261
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400262bool FastCopyFunctionMap::has(angle::FormatID formatID) const
Geoff Lang051dbc72015-01-05 15:48:58 -0500263{
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400264 return (get(formatID) != nullptr);
Geoff Lang051dbc72015-01-05 15:48:58 -0500265}
266
Jamie Madill522095f2018-07-23 14:59:41 -0400267PixelCopyFunction FastCopyFunctionMap::get(angle::FormatID formatID) const
Jamie Madill4f57e5f2016-10-27 17:36:53 -0400268{
269 for (size_t index = 0; index < mSize; ++index)
270 {
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400271 if (mData[index].formatID == formatID)
Jamie Madill4f57e5f2016-10-27 17:36:53 -0400272 {
273 return mData[index].func;
274 }
275 }
276
277 return nullptr;
Geoff Lang051dbc72015-01-05 15:48:58 -0500278}
Jamie Madilld2b50a02016-06-09 00:13:35 -0700279
Jamie Madill222c5172017-07-19 16:15:42 -0400280bool ShouldUseDebugLayers(const egl::AttributeMap &attribs)
281{
282 EGLAttrib debugSetting =
283 attribs.get(EGL_PLATFORM_ANGLE_DEBUG_LAYERS_ENABLED_ANGLE, EGL_DONT_CARE);
284
285// Prefer to enable debug layers if compiling in Debug, and disabled in Release.
Geoff Lang58ba6bf2017-11-28 16:40:58 -0500286#if defined(ANGLE_ENABLE_ASSERTS)
Jamie Madill222c5172017-07-19 16:15:42 -0400287 return (debugSetting != EGL_FALSE);
288#else
289 return (debugSetting == EGL_TRUE);
Geoff Lang58ba6bf2017-11-28 16:40:58 -0500290#endif // defined(ANGLE_ENABLE_ASSERTS)
Jamie Madill222c5172017-07-19 16:15:42 -0400291}
292
Geoff Lang24ddc7a2018-06-11 14:56:34 -0400293bool ShouldUseVirtualizedContexts(const egl::AttributeMap &attribs, bool defaultValue)
294{
295 EGLAttrib virtualizedContextRequest =
296 attribs.get(EGL_PLATFORM_ANGLE_CONTEXT_VIRTUALIZATION_ANGLE, EGL_DONT_CARE);
297 if (defaultValue)
298 {
299 return (virtualizedContextRequest != EGL_FALSE);
300 }
301 else
302 {
303 return (virtualizedContextRequest == EGL_TRUE);
304 }
305}
306
Geoff Langd93cd6c2017-08-11 16:32:41 -0400307void CopyImageCHROMIUM(const uint8_t *sourceData,
308 size_t sourceRowPitch,
309 size_t sourcePixelBytes,
Jamie Madill522095f2018-07-23 14:59:41 -0400310 PixelReadFunction pixelReadFunction,
Geoff Langd93cd6c2017-08-11 16:32:41 -0400311 uint8_t *destData,
312 size_t destRowPitch,
313 size_t destPixelBytes,
Jamie Madill522095f2018-07-23 14:59:41 -0400314 PixelWriteFunction pixelWriteFunction,
Geoff Langd93cd6c2017-08-11 16:32:41 -0400315 GLenum destUnsizedFormat,
316 GLenum destComponentType,
317 size_t width,
318 size_t height,
319 bool unpackFlipY,
320 bool unpackPremultiplyAlpha,
321 bool unpackUnmultiplyAlpha)
322{
323 using ConversionFunction = void (*)(gl::ColorF *);
324 ConversionFunction conversionFunction = CopyColor;
325 if (unpackPremultiplyAlpha != unpackUnmultiplyAlpha)
326 {
327 if (unpackPremultiplyAlpha)
328 {
329 conversionFunction = PremultiplyAlpha;
330 }
331 else
332 {
333 conversionFunction = UnmultiplyAlpha;
334 }
335 }
336
337 auto clipChannelsFunction = ClipChannelsNoOp;
338 switch (destUnsizedFormat)
339 {
340 case GL_RED:
341 clipChannelsFunction = ClipChannelsR;
342 break;
343 case GL_RG:
344 clipChannelsFunction = ClipChannelsRG;
345 break;
346 case GL_RGB:
347 clipChannelsFunction = ClipChannelsRGB;
348 break;
349 case GL_LUMINANCE:
350 clipChannelsFunction = ClipChannelsLuminance;
351 break;
352 case GL_ALPHA:
353 clipChannelsFunction = ClipChannelsAlpha;
354 break;
355 }
356
357 auto writeFunction = (destComponentType == GL_UNSIGNED_INT) ? WriteUintColor : WriteFloatColor;
358
359 for (size_t y = 0; y < height; y++)
360 {
361 for (size_t x = 0; x < width; x++)
362 {
363 const uint8_t *sourcePixelData = sourceData + y * sourceRowPitch + x * sourcePixelBytes;
364
365 gl::ColorF sourceColor;
Jamie Madill522095f2018-07-23 14:59:41 -0400366 pixelReadFunction(sourcePixelData, reinterpret_cast<uint8_t *>(&sourceColor));
Geoff Langd93cd6c2017-08-11 16:32:41 -0400367
368 conversionFunction(&sourceColor);
369 clipChannelsFunction(&sourceColor);
370
371 size_t destY = 0;
372 if (unpackFlipY)
373 {
374 destY += (height - 1);
375 destY -= y;
376 }
377 else
378 {
379 destY += y;
380 }
381
382 uint8_t *destPixelData = destData + destY * destRowPitch + x * destPixelBytes;
Jamie Madill522095f2018-07-23 14:59:41 -0400383 writeFunction(sourceColor, pixelWriteFunction, destPixelData);
Geoff Langd93cd6c2017-08-11 16:32:41 -0400384 }
385 }
386}
387
Jamie Madill42975642017-10-12 12:31:51 -0400388// IncompleteTextureSet implementation.
389IncompleteTextureSet::IncompleteTextureSet()
390{
391}
392
393IncompleteTextureSet::~IncompleteTextureSet()
394{
395}
396
397void IncompleteTextureSet::onDestroy(const gl::Context *context)
398{
399 // Clear incomplete textures.
400 for (auto &incompleteTexture : mIncompleteTextures)
401 {
Corentin Wallez99d492c2018-02-27 15:17:10 -0500402 if (incompleteTexture.get() != nullptr)
403 {
404 ANGLE_SWALLOW_ERR(incompleteTexture->onDestroy(context));
405 incompleteTexture.set(context, nullptr);
406 }
Jamie Madill42975642017-10-12 12:31:51 -0400407 }
Jamie Madill42975642017-10-12 12:31:51 -0400408}
409
410gl::Error IncompleteTextureSet::getIncompleteTexture(
411 const gl::Context *context,
Corentin Wallez99d492c2018-02-27 15:17:10 -0500412 gl::TextureType type,
Jamie Madill42975642017-10-12 12:31:51 -0400413 MultisampleTextureInitializer *multisampleInitializer,
414 gl::Texture **textureOut)
415{
Corentin Wallez99d492c2018-02-27 15:17:10 -0500416 *textureOut = mIncompleteTextures[type].get();
417 if (*textureOut != nullptr)
Jamie Madill42975642017-10-12 12:31:51 -0400418 {
Jamie Madill42975642017-10-12 12:31:51 -0400419 return gl::NoError();
420 }
421
422 ContextImpl *implFactory = context->getImplementation();
423
424 const GLubyte color[] = {0, 0, 0, 255};
425 const gl::Extents colorSize(1, 1, 1);
Corentin Wallezcda6af12017-10-30 19:20:37 -0400426 gl::PixelUnpackState unpack;
427 unpack.alignment = 1;
Jamie Madill42975642017-10-12 12:31:51 -0400428 const gl::Box area(0, 0, 0, 1, 1, 1);
429
430 // If a texture is external use a 2D texture for the incomplete texture
Corentin Wallez99d492c2018-02-27 15:17:10 -0500431 gl::TextureType createType = (type == gl::TextureType::External) ? gl::TextureType::_2D : type;
Jamie Madill42975642017-10-12 12:31:51 -0400432
433 gl::Texture *tex = new gl::Texture(implFactory, std::numeric_limits<GLuint>::max(), createType);
434 angle::UniqueObjectPointer<gl::Texture, gl::Context> t(tex, context);
435
Corentin Wallez99d492c2018-02-27 15:17:10 -0500436 if (createType == gl::TextureType::_2DMultisample)
Jamie Madill42975642017-10-12 12:31:51 -0400437 {
438 ANGLE_TRY(t->setStorageMultisample(context, createType, 1, GL_RGBA8, colorSize, true));
439 }
440 else
441 {
442 ANGLE_TRY(t->setStorage(context, createType, 1, GL_RGBA8, colorSize));
443 }
444
Corentin Wallez99d492c2018-02-27 15:17:10 -0500445 if (type == gl::TextureType::CubeMap)
Jamie Madill42975642017-10-12 12:31:51 -0400446 {
Corentin Wallez99d492c2018-02-27 15:17:10 -0500447 for (gl::TextureTarget face : gl::AllCubeFaceTextureTargets())
Jamie Madill42975642017-10-12 12:31:51 -0400448 {
449 ANGLE_TRY(
450 t->setSubImage(context, unpack, face, 0, area, GL_RGBA, GL_UNSIGNED_BYTE, color));
451 }
452 }
Corentin Wallez99d492c2018-02-27 15:17:10 -0500453 else if (type == gl::TextureType::_2DMultisample)
Jamie Madill42975642017-10-12 12:31:51 -0400454 {
455 // Call a specialized clear function to init a multisample texture.
456 ANGLE_TRY(multisampleInitializer->initializeMultisampleTextureToBlack(context, t.get()));
457 }
458 else
459 {
Corentin Wallez99d492c2018-02-27 15:17:10 -0500460 ANGLE_TRY(t->setSubImage(context, unpack, gl::NonCubeTextureTypeToTarget(createType), 0,
461 area, GL_RGBA, GL_UNSIGNED_BYTE, color));
Jamie Madill42975642017-10-12 12:31:51 -0400462 }
463
Luc Ferron4bba74f2018-04-19 14:40:45 -0400464 ANGLE_TRY(t->syncState(context));
Jamie Madill42975642017-10-12 12:31:51 -0400465
466 mIncompleteTextures[type].set(context, t.release());
467 *textureOut = mIncompleteTextures[type].get();
468 return gl::NoError();
469}
470
Luc Ferron46bcea52018-05-31 09:48:36 -0400471#define ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(cols, rows) \
472 template bool SetFloatUniformMatrix<cols, rows>(unsigned int, unsigned int, GLsizei, \
473 GLboolean, const GLfloat *, uint8_t *)
474
475ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(2, 2);
476ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(3, 3);
477ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(4, 4);
478ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(2, 3);
479ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(3, 2);
480ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(2, 4);
481ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(4, 2);
482ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(3, 4);
483ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(4, 3);
484
485#undef ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC
486
487template <int cols, int rows>
488bool SetFloatUniformMatrix(unsigned int arrayElementOffset,
489 unsigned int elementCount,
490 GLsizei countIn,
491 GLboolean transpose,
492 const GLfloat *value,
493 uint8_t *targetData)
494{
495 unsigned int count =
496 std::min(elementCount - arrayElementOffset, static_cast<unsigned int>(countIn));
497
498 const unsigned int targetMatrixStride = (4 * rows);
499 GLfloat *target = reinterpret_cast<GLfloat *>(
500 targetData + arrayElementOffset * sizeof(GLfloat) * targetMatrixStride);
501
502 bool dirty = false;
503
504 for (unsigned int i = 0; i < count; i++)
505 {
506 if (transpose == GL_FALSE)
507 {
508 dirty = ExpandMatrix<GLfloat, cols, rows>(target, value) || dirty;
509 }
510 else
511 {
512 dirty = TransposeExpandMatrix<GLfloat, cols, rows>(target, value) || dirty;
513 }
514 target += targetMatrixStride;
515 value += cols * rows;
516 }
517
518 return dirty;
519}
520
Luc Ferron48cdc2e2018-05-31 09:58:34 -0400521template void GetMatrixUniform<GLint>(GLenum, GLint *, const GLint *, bool);
522template void GetMatrixUniform<GLuint>(GLenum, GLuint *, const GLuint *, bool);
523
524void GetMatrixUniform(GLenum type, GLfloat *dataOut, const GLfloat *source, bool transpose)
525{
526 int columns = gl::VariableColumnCount(type);
527 int rows = gl::VariableRowCount(type);
528 for (GLint col = 0; col < columns; ++col)
529 {
530 for (GLint row = 0; row < rows; ++row)
531 {
532 GLfloat *outptr = dataOut + ((col * rows) + row);
533 const GLfloat *inptr =
534 transpose ? source + ((row * 4) + col) : source + ((col * 4) + row);
535 *outptr = *inptr;
536 }
537 }
538}
539
540template <typename NonFloatT>
541void GetMatrixUniform(GLenum type, NonFloatT *dataOut, const NonFloatT *source, bool transpose)
542{
543 UNREACHABLE();
544}
545
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400546const angle::Format &GetFormatFromFormatType(GLenum format, GLenum type)
547{
548 GLenum sizedInternalFormat = gl::GetInternalFormatInfo(format, type).sizedInternalFormat;
549 angle::FormatID angleFormatID = angle::Format::InternalFormatToID(sizedInternalFormat);
550 return angle::Format::Get(angleFormatID);
551}
Jamie Madilld2b50a02016-06-09 00:13:35 -0700552} // namespace rx