blob: 280b3d9624701fb46da8d00c59ee07caa84ceb2b [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,
Frank Henigman1ffad842018-09-24 23:40:45 -0400172 bool reverseRowOrderIn,
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),
Frank Henigman1ffad842018-09-24 23:40:45 -0400179 reverseRowOrder(reverseRowOrderIn),
Jamie Madilld2b50a02016-06-09 00:13:35 -0700180 offset(offsetIn)
181{
Jamie Madill4928b7c2017-06-20 12:57:39 -0400182}
183
Jamie Madilld2b50a02016-06-09 00:13:35 -0700184void PackPixels(const PackPixelsParams &params,
Jamie Madill86e0b7f2016-08-09 11:10:29 -0400185 const angle::Format &sourceFormat,
Jamie Madilld2b50a02016-06-09 00:13:35 -0700186 int inputPitchIn,
187 const uint8_t *sourceIn,
188 uint8_t *destWithoutOffset)
189{
190 uint8_t *destWithOffset = destWithoutOffset + params.offset;
191
192 const uint8_t *source = sourceIn;
193 int inputPitch = inputPitchIn;
194
Frank Henigman1ffad842018-09-24 23:40:45 -0400195 if (params.reverseRowOrder)
Jamie Madilld2b50a02016-06-09 00:13:35 -0700196 {
197 source += inputPitch * (params.area.height - 1);
198 inputPitch = -inputPitch;
199 }
200
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400201 if (sourceFormat == *params.destFormat)
Jamie Madilld2b50a02016-06-09 00:13:35 -0700202 {
203 // Direct copy possible
204 for (int y = 0; y < params.area.height; ++y)
205 {
206 memcpy(destWithOffset + y * params.outputPitch, source + y * inputPitch,
Luc Ferron339b95e2018-06-12 10:21:52 -0400207 params.area.width * sourceFormat.pixelBytes);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700208 }
209 return;
210 }
211
Jamie Madill522095f2018-07-23 14:59:41 -0400212 PixelCopyFunction fastCopyFunc = sourceFormat.fastCopyFunctions.get(params.destFormat->id);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700213
214 if (fastCopyFunc)
215 {
216 // Fast copy is possible through some special function
217 for (int y = 0; y < params.area.height; ++y)
218 {
219 for (int x = 0; x < params.area.width; ++x)
220 {
221 uint8_t *dest =
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400222 destWithOffset + y * params.outputPitch + x * params.destFormat->pixelBytes;
Luc Ferron339b95e2018-06-12 10:21:52 -0400223 const uint8_t *src = source + y * inputPitch + x * sourceFormat.pixelBytes;
Jamie Madilld2b50a02016-06-09 00:13:35 -0700224
225 fastCopyFunc(src, dest);
226 }
227 }
228 return;
229 }
230
Jamie Madill522095f2018-07-23 14:59:41 -0400231 PixelWriteFunction pixelWriteFunction = params.destFormat->pixelWriteFunction;
232 ASSERT(pixelWriteFunction != nullptr);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700233
234 // Maximum size of any Color<T> type used.
235 uint8_t temp[16];
236 static_assert(sizeof(temp) >= sizeof(gl::ColorF) && sizeof(temp) >= sizeof(gl::ColorUI) &&
Jamie Madillb436aac2018-07-18 17:23:48 -0400237 sizeof(temp) >= sizeof(gl::ColorI) &&
238 sizeof(temp) >= sizeof(angle::DepthStencil),
239 "Unexpected size of pixel struct.");
Jamie Madilld2b50a02016-06-09 00:13:35 -0700240
Jamie Madill522095f2018-07-23 14:59:41 -0400241 PixelReadFunction pixelReadFunction = sourceFormat.pixelReadFunction;
242 ASSERT(pixelReadFunction != nullptr);
Jamie Madill86e0b7f2016-08-09 11:10:29 -0400243
Jamie Madilld2b50a02016-06-09 00:13:35 -0700244 for (int y = 0; y < params.area.height; ++y)
245 {
246 for (int x = 0; x < params.area.width; ++x)
247 {
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400248 uint8_t *dest =
249 destWithOffset + y * params.outputPitch + x * params.destFormat->pixelBytes;
Luc Ferron339b95e2018-06-12 10:21:52 -0400250 const uint8_t *src = source + y * inputPitch + x * sourceFormat.pixelBytes;
Jamie Madilld2b50a02016-06-09 00:13:35 -0700251
252 // readFunc and writeFunc will be using the same type of color, CopyTexImage
253 // will not allow the copy otherwise.
Jamie Madill522095f2018-07-23 14:59:41 -0400254 pixelReadFunction(src, temp);
255 pixelWriteFunction(temp, dest);
Jamie Madilld2b50a02016-06-09 00:13:35 -0700256 }
257 }
258}
259
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400260bool FastCopyFunctionMap::has(angle::FormatID formatID) const
Geoff Lang051dbc72015-01-05 15:48:58 -0500261{
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400262 return (get(formatID) != nullptr);
Geoff Lang051dbc72015-01-05 15:48:58 -0500263}
264
Jamie Madill522095f2018-07-23 14:59:41 -0400265PixelCopyFunction FastCopyFunctionMap::get(angle::FormatID formatID) const
Jamie Madill4f57e5f2016-10-27 17:36:53 -0400266{
267 for (size_t index = 0; index < mSize; ++index)
268 {
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400269 if (mData[index].formatID == formatID)
Jamie Madill4f57e5f2016-10-27 17:36:53 -0400270 {
271 return mData[index].func;
272 }
273 }
274
275 return nullptr;
Geoff Lang051dbc72015-01-05 15:48:58 -0500276}
Jamie Madilld2b50a02016-06-09 00:13:35 -0700277
Jamie Madill222c5172017-07-19 16:15:42 -0400278bool ShouldUseDebugLayers(const egl::AttributeMap &attribs)
279{
280 EGLAttrib debugSetting =
281 attribs.get(EGL_PLATFORM_ANGLE_DEBUG_LAYERS_ENABLED_ANGLE, EGL_DONT_CARE);
282
283// Prefer to enable debug layers if compiling in Debug, and disabled in Release.
Geoff Lang58ba6bf2017-11-28 16:40:58 -0500284#if defined(ANGLE_ENABLE_ASSERTS)
Jamie Madill222c5172017-07-19 16:15:42 -0400285 return (debugSetting != EGL_FALSE);
286#else
287 return (debugSetting == EGL_TRUE);
Geoff Lang58ba6bf2017-11-28 16:40:58 -0500288#endif // defined(ANGLE_ENABLE_ASSERTS)
Jamie Madill222c5172017-07-19 16:15:42 -0400289}
290
Geoff Lang24ddc7a2018-06-11 14:56:34 -0400291bool ShouldUseVirtualizedContexts(const egl::AttributeMap &attribs, bool defaultValue)
292{
293 EGLAttrib virtualizedContextRequest =
294 attribs.get(EGL_PLATFORM_ANGLE_CONTEXT_VIRTUALIZATION_ANGLE, EGL_DONT_CARE);
295 if (defaultValue)
296 {
297 return (virtualizedContextRequest != EGL_FALSE);
298 }
299 else
300 {
301 return (virtualizedContextRequest == EGL_TRUE);
302 }
303}
304
Geoff Langd93cd6c2017-08-11 16:32:41 -0400305void CopyImageCHROMIUM(const uint8_t *sourceData,
306 size_t sourceRowPitch,
307 size_t sourcePixelBytes,
Jamie Madill522095f2018-07-23 14:59:41 -0400308 PixelReadFunction pixelReadFunction,
Geoff Langd93cd6c2017-08-11 16:32:41 -0400309 uint8_t *destData,
310 size_t destRowPitch,
311 size_t destPixelBytes,
Jamie Madill522095f2018-07-23 14:59:41 -0400312 PixelWriteFunction pixelWriteFunction,
Geoff Langd93cd6c2017-08-11 16:32:41 -0400313 GLenum destUnsizedFormat,
314 GLenum destComponentType,
315 size_t width,
316 size_t height,
317 bool unpackFlipY,
318 bool unpackPremultiplyAlpha,
319 bool unpackUnmultiplyAlpha)
320{
321 using ConversionFunction = void (*)(gl::ColorF *);
322 ConversionFunction conversionFunction = CopyColor;
323 if (unpackPremultiplyAlpha != unpackUnmultiplyAlpha)
324 {
325 if (unpackPremultiplyAlpha)
326 {
327 conversionFunction = PremultiplyAlpha;
328 }
329 else
330 {
331 conversionFunction = UnmultiplyAlpha;
332 }
333 }
334
335 auto clipChannelsFunction = ClipChannelsNoOp;
336 switch (destUnsizedFormat)
337 {
338 case GL_RED:
339 clipChannelsFunction = ClipChannelsR;
340 break;
341 case GL_RG:
342 clipChannelsFunction = ClipChannelsRG;
343 break;
344 case GL_RGB:
345 clipChannelsFunction = ClipChannelsRGB;
346 break;
347 case GL_LUMINANCE:
348 clipChannelsFunction = ClipChannelsLuminance;
349 break;
350 case GL_ALPHA:
351 clipChannelsFunction = ClipChannelsAlpha;
352 break;
353 }
354
355 auto writeFunction = (destComponentType == GL_UNSIGNED_INT) ? WriteUintColor : WriteFloatColor;
356
357 for (size_t y = 0; y < height; y++)
358 {
359 for (size_t x = 0; x < width; x++)
360 {
361 const uint8_t *sourcePixelData = sourceData + y * sourceRowPitch + x * sourcePixelBytes;
362
363 gl::ColorF sourceColor;
Jamie Madill522095f2018-07-23 14:59:41 -0400364 pixelReadFunction(sourcePixelData, reinterpret_cast<uint8_t *>(&sourceColor));
Geoff Langd93cd6c2017-08-11 16:32:41 -0400365
366 conversionFunction(&sourceColor);
367 clipChannelsFunction(&sourceColor);
368
369 size_t destY = 0;
370 if (unpackFlipY)
371 {
372 destY += (height - 1);
373 destY -= y;
374 }
375 else
376 {
377 destY += y;
378 }
379
380 uint8_t *destPixelData = destData + destY * destRowPitch + x * destPixelBytes;
Jamie Madill522095f2018-07-23 14:59:41 -0400381 writeFunction(sourceColor, pixelWriteFunction, destPixelData);
Geoff Langd93cd6c2017-08-11 16:32:41 -0400382 }
383 }
384}
385
Jamie Madill42975642017-10-12 12:31:51 -0400386// IncompleteTextureSet implementation.
387IncompleteTextureSet::IncompleteTextureSet()
388{
389}
390
391IncompleteTextureSet::~IncompleteTextureSet()
392{
393}
394
395void IncompleteTextureSet::onDestroy(const gl::Context *context)
396{
397 // Clear incomplete textures.
398 for (auto &incompleteTexture : mIncompleteTextures)
399 {
Corentin Wallez99d492c2018-02-27 15:17:10 -0500400 if (incompleteTexture.get() != nullptr)
401 {
402 ANGLE_SWALLOW_ERR(incompleteTexture->onDestroy(context));
403 incompleteTexture.set(context, nullptr);
404 }
Jamie Madill42975642017-10-12 12:31:51 -0400405 }
Jamie Madill42975642017-10-12 12:31:51 -0400406}
407
408gl::Error IncompleteTextureSet::getIncompleteTexture(
409 const gl::Context *context,
Corentin Wallez99d492c2018-02-27 15:17:10 -0500410 gl::TextureType type,
Jamie Madill42975642017-10-12 12:31:51 -0400411 MultisampleTextureInitializer *multisampleInitializer,
412 gl::Texture **textureOut)
413{
Corentin Wallez99d492c2018-02-27 15:17:10 -0500414 *textureOut = mIncompleteTextures[type].get();
415 if (*textureOut != nullptr)
Jamie Madill42975642017-10-12 12:31:51 -0400416 {
Jamie Madill42975642017-10-12 12:31:51 -0400417 return gl::NoError();
418 }
419
420 ContextImpl *implFactory = context->getImplementation();
421
422 const GLubyte color[] = {0, 0, 0, 255};
423 const gl::Extents colorSize(1, 1, 1);
Corentin Wallezcda6af12017-10-30 19:20:37 -0400424 gl::PixelUnpackState unpack;
425 unpack.alignment = 1;
Jamie Madill42975642017-10-12 12:31:51 -0400426 const gl::Box area(0, 0, 0, 1, 1, 1);
427
428 // If a texture is external use a 2D texture for the incomplete texture
Corentin Wallez99d492c2018-02-27 15:17:10 -0500429 gl::TextureType createType = (type == gl::TextureType::External) ? gl::TextureType::_2D : type;
Jamie Madill42975642017-10-12 12:31:51 -0400430
431 gl::Texture *tex = new gl::Texture(implFactory, std::numeric_limits<GLuint>::max(), createType);
432 angle::UniqueObjectPointer<gl::Texture, gl::Context> t(tex, context);
433
Corentin Wallez99d492c2018-02-27 15:17:10 -0500434 if (createType == gl::TextureType::_2DMultisample)
Jamie Madill42975642017-10-12 12:31:51 -0400435 {
436 ANGLE_TRY(t->setStorageMultisample(context, createType, 1, GL_RGBA8, colorSize, true));
437 }
438 else
439 {
440 ANGLE_TRY(t->setStorage(context, createType, 1, GL_RGBA8, colorSize));
441 }
442
Corentin Wallez99d492c2018-02-27 15:17:10 -0500443 if (type == gl::TextureType::CubeMap)
Jamie Madill42975642017-10-12 12:31:51 -0400444 {
Corentin Wallez99d492c2018-02-27 15:17:10 -0500445 for (gl::TextureTarget face : gl::AllCubeFaceTextureTargets())
Jamie Madill42975642017-10-12 12:31:51 -0400446 {
Jamie Madill0d0fb432018-09-07 17:43:32 -0400447 ANGLE_TRY(t->setSubImage(context, unpack, nullptr, face, 0, area, GL_RGBA,
448 GL_UNSIGNED_BYTE, color));
Jamie Madill42975642017-10-12 12:31:51 -0400449 }
450 }
Corentin Wallez99d492c2018-02-27 15:17:10 -0500451 else if (type == gl::TextureType::_2DMultisample)
Jamie Madill42975642017-10-12 12:31:51 -0400452 {
453 // Call a specialized clear function to init a multisample texture.
454 ANGLE_TRY(multisampleInitializer->initializeMultisampleTextureToBlack(context, t.get()));
455 }
456 else
457 {
Jamie Madill0d0fb432018-09-07 17:43:32 -0400458 ANGLE_TRY(t->setSubImage(context, unpack, nullptr,
459 gl::NonCubeTextureTypeToTarget(createType), 0, area, GL_RGBA,
460 GL_UNSIGNED_BYTE, color));
Jamie Madill42975642017-10-12 12:31:51 -0400461 }
462
Luc Ferron4bba74f2018-04-19 14:40:45 -0400463 ANGLE_TRY(t->syncState(context));
Jamie Madill42975642017-10-12 12:31:51 -0400464
465 mIncompleteTextures[type].set(context, t.release());
466 *textureOut = mIncompleteTextures[type].get();
467 return gl::NoError();
468}
469
Luc Ferron46bcea52018-05-31 09:48:36 -0400470#define ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(cols, rows) \
471 template bool SetFloatUniformMatrix<cols, rows>(unsigned int, unsigned int, GLsizei, \
472 GLboolean, const GLfloat *, uint8_t *)
473
474ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(2, 2);
475ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(3, 3);
476ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(4, 4);
477ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(2, 3);
478ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(3, 2);
479ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(2, 4);
480ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(4, 2);
481ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(3, 4);
482ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC(4, 3);
483
484#undef ANGLE_INSTANTIATE_SET_UNIFORM_MATRIX_FUNC
485
486template <int cols, int rows>
487bool SetFloatUniformMatrix(unsigned int arrayElementOffset,
488 unsigned int elementCount,
489 GLsizei countIn,
490 GLboolean transpose,
491 const GLfloat *value,
492 uint8_t *targetData)
493{
494 unsigned int count =
495 std::min(elementCount - arrayElementOffset, static_cast<unsigned int>(countIn));
496
497 const unsigned int targetMatrixStride = (4 * rows);
498 GLfloat *target = reinterpret_cast<GLfloat *>(
499 targetData + arrayElementOffset * sizeof(GLfloat) * targetMatrixStride);
500
501 bool dirty = false;
502
503 for (unsigned int i = 0; i < count; i++)
504 {
505 if (transpose == GL_FALSE)
506 {
507 dirty = ExpandMatrix<GLfloat, cols, rows>(target, value) || dirty;
508 }
509 else
510 {
511 dirty = TransposeExpandMatrix<GLfloat, cols, rows>(target, value) || dirty;
512 }
513 target += targetMatrixStride;
514 value += cols * rows;
515 }
516
517 return dirty;
518}
519
Luc Ferron48cdc2e2018-05-31 09:58:34 -0400520template void GetMatrixUniform<GLint>(GLenum, GLint *, const GLint *, bool);
521template void GetMatrixUniform<GLuint>(GLenum, GLuint *, const GLuint *, bool);
522
523void GetMatrixUniform(GLenum type, GLfloat *dataOut, const GLfloat *source, bool transpose)
524{
525 int columns = gl::VariableColumnCount(type);
526 int rows = gl::VariableRowCount(type);
527 for (GLint col = 0; col < columns; ++col)
528 {
529 for (GLint row = 0; row < rows; ++row)
530 {
531 GLfloat *outptr = dataOut + ((col * rows) + row);
532 const GLfloat *inptr =
533 transpose ? source + ((row * 4) + col) : source + ((col * 4) + row);
534 *outptr = *inptr;
535 }
536 }
537}
538
539template <typename NonFloatT>
540void GetMatrixUniform(GLenum type, NonFloatT *dataOut, const NonFloatT *source, bool transpose)
541{
542 UNREACHABLE();
543}
544
Jamie Madilldb9c69e2018-07-18 17:23:47 -0400545const angle::Format &GetFormatFromFormatType(GLenum format, GLenum type)
546{
547 GLenum sizedInternalFormat = gl::GetInternalFormatInfo(format, type).sizedInternalFormat;
548 angle::FormatID angleFormatID = angle::Format::InternalFormatToID(sizedInternalFormat);
549 return angle::Format::Get(angleFormatID);
550}
Jamie Madilld2b50a02016-06-09 00:13:35 -0700551} // namespace rx