1 /*
2 * Copyright 2010, The Android Open Source Project
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * * Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * * Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 */
25
26 #define LOG_TAG "GLUtils"
27 #define LOG_NDEBUG 1
28
29 #include "config.h"
30 #include "GLUtils.h"
31
32 #if USE(ACCELERATED_COMPOSITING)
33
34 #include "AndroidLog.h"
35 #include "BaseRenderer.h"
36 #include "TextureInfo.h"
37 #include "Tile.h"
38 #include "TilesManager.h"
39 #include "TransferQueue.h"
40
41 #include <android/native_window.h>
42 #include <gui/SurfaceTexture.h>
43 #include <wtf/CurrentTime.h>
44
45 // We will limit GL error logging for LOG_VOLUME_PER_CYCLE times every
46 // LOG_VOLUME_PER_CYCLE seconds.
47 #define LOG_CYCLE 30.0
48 #define LOG_VOLUME_PER_CYCLE 20
49
50 struct ANativeWindowBuffer;
51
52 namespace WebCore {
53
54 using namespace android;
55
56 /////////////////////////////////////////////////////////////////////////////////////////
57 // Matrix utilities
58 /////////////////////////////////////////////////////////////////////////////////////////
59
toGLMatrix(GLfloat * flattened,const TransformationMatrix & m)60 void GLUtils::toGLMatrix(GLfloat* flattened, const TransformationMatrix& m)
61 {
62 flattened[0] = m.m11(); // scaleX
63 flattened[1] = m.m12(); // skewY
64 flattened[2] = m.m13();
65 flattened[3] = m.m14(); // persp0
66 flattened[4] = m.m21(); // skewX
67 flattened[5] = m.m22(); // scaleY
68 flattened[6] = m.m23();
69 flattened[7] = m.m24(); // persp1
70 flattened[8] = m.m31();
71 flattened[9] = m.m32();
72 flattened[10] = m.m33();
73 flattened[11] = m.m34();
74 flattened[12] = m.m41(); // transX
75 flattened[13] = m.m42(); // transY
76 flattened[14] = m.m43();
77 flattened[15] = m.m44(); // persp2
78 }
79
toSkMatrix(SkMatrix & matrix,const TransformationMatrix & m)80 void GLUtils::toSkMatrix(SkMatrix& matrix, const TransformationMatrix& m)
81 {
82 matrix[0] = m.m11(); // scaleX
83 matrix[1] = m.m21(); // skewX
84 matrix[2] = m.m41(); // transX
85 matrix[3] = m.m12(); // skewY
86 matrix[4] = m.m22(); // scaleY
87 matrix[5] = m.m42(); // transY
88 matrix[6] = m.m14(); // persp0
89 matrix[7] = m.m24(); // persp1
90 matrix[8] = m.m44(); // persp2
91 }
92
setOrthographicMatrix(TransformationMatrix & ortho,float left,float top,float right,float bottom,float nearZ,float farZ)93 void GLUtils::setOrthographicMatrix(TransformationMatrix& ortho, float left, float top,
94 float right, float bottom, float nearZ, float farZ)
95 {
96 float deltaX = right - left;
97 float deltaY = top - bottom;
98 float deltaZ = farZ - nearZ;
99 if (!deltaX || !deltaY || !deltaZ)
100 return;
101
102 ortho.setM11(2.0f / deltaX);
103 ortho.setM41(-(right + left) / deltaX);
104 ortho.setM22(2.0f / deltaY);
105 ortho.setM42(-(top + bottom) / deltaY);
106 ortho.setM33(-2.0f / deltaZ);
107 ortho.setM43(-(nearZ + farZ) / deltaZ);
108 }
109
has3dTransform(const TransformationMatrix & matrix)110 bool GLUtils::has3dTransform(const TransformationMatrix& matrix)
111 {
112 return matrix.m13() != 0 || matrix.m23() != 0
113 || matrix.m31() != 0 || matrix.m32() != 0
114 || matrix.m33() != 1 || matrix.m34() != 0
115 || matrix.m43() != 0;
116 }
117
118 /////////////////////////////////////////////////////////////////////////////////////////
119 // GL & EGL error checks
120 /////////////////////////////////////////////////////////////////////////////////////////
121
122 double GLUtils::m_previousLogTime = 0;
123 int GLUtils::m_currentLogCounter = 0;
124
allowGLLog()125 bool GLUtils::allowGLLog()
126 {
127 if (m_currentLogCounter < LOG_VOLUME_PER_CYCLE) {
128 m_currentLogCounter++;
129 return true;
130 }
131
132 // when we are in Log cycle and over the log limit, just return false
133 double currentTime = WTF::currentTime();
134 double delta = currentTime - m_previousLogTime;
135 bool inLogCycle = (delta <= LOG_CYCLE) && (delta > 0);
136 if (inLogCycle)
137 return false;
138
139 // When we are out of Log Cycle and over the log limit, we need to reset
140 // the counter and timer.
141 m_previousLogTime = currentTime;
142 m_currentLogCounter = 0;
143 return false;
144 }
145
crashIfOOM(GLint errorCode)146 static void crashIfOOM(GLint errorCode)
147 {
148 const GLint OOM_ERROR_CODE = 0x505;
149 if (errorCode == OOM_ERROR_CODE) {
150 ALOGE("ERROR: Fatal OOM detected.");
151 CRASH();
152 }
153 }
154
checkEglError(const char * op,EGLBoolean returnVal)155 void GLUtils::checkEglError(const char* op, EGLBoolean returnVal)
156 {
157 if (returnVal != EGL_TRUE) {
158 #ifndef DEBUG
159 if (allowGLLog())
160 #endif
161 ALOGE("EGL ERROR - %s() returned %d\n", op, returnVal);
162 }
163
164 for (EGLint error = eglGetError(); error != EGL_SUCCESS; error = eglGetError()) {
165 #ifndef DEBUG
166 if (allowGLLog())
167 #endif
168 ALOGE("after %s() eglError (0x%x)\n", op, error);
169 crashIfOOM(error);
170 }
171 }
172
checkGlError(const char * op)173 bool GLUtils::checkGlError(const char* op)
174 {
175 bool ret = false;
176 for (GLint error = glGetError(); error; error = glGetError()) {
177 #ifndef DEBUG
178 if (allowGLLog())
179 #endif
180 ALOGE("GL ERROR - after %s() glError (0x%x)\n", op, error);
181 crashIfOOM(error);
182 ret = true;
183 }
184 return ret;
185 }
186
checkGlErrorOn(void * p,const char * op)187 bool GLUtils::checkGlErrorOn(void* p, const char* op)
188 {
189 bool ret = false;
190 for (GLint error = glGetError(); error; error = glGetError()) {
191 #ifndef DEBUG
192 if (allowGLLog())
193 #endif
194 ALOGE("GL ERROR on %x - after %s() glError (0x%x)\n", p, op, error);
195 crashIfOOM(error);
196 ret = true;
197 }
198 return ret;
199 }
200
checkSurfaceTextureError(const char * functionName,int status)201 void GLUtils::checkSurfaceTextureError(const char* functionName, int status)
202 {
203 if (status != NO_ERROR) {
204 #ifndef DEBUG
205 if (allowGLLog())
206 #endif
207 ALOGE("ERROR at calling %s status is (%d)", functionName, status);
208 }
209 }
210 /////////////////////////////////////////////////////////////////////////////////////////
211 // GL & EGL extension checks
212 /////////////////////////////////////////////////////////////////////////////////////////
213
isEGLImageSupported()214 bool GLUtils::isEGLImageSupported()
215 {
216 const char* eglExtensions = eglQueryString(eglGetCurrentDisplay(), EGL_EXTENSIONS);
217 const char* glExtensions = reinterpret_cast<const char*>(glGetString(GL_EXTENSIONS));
218
219 return eglExtensions && glExtensions
220 && strstr(eglExtensions, "EGL_KHR_image_base")
221 && strstr(eglExtensions, "EGL_KHR_gl_texture_2D_image")
222 && strstr(glExtensions, "GL_OES_EGL_image");
223 }
224
isEGLFenceSyncSupported()225 bool GLUtils::isEGLFenceSyncSupported()
226 {
227 const char* eglExtensions = eglQueryString(eglGetCurrentDisplay(), EGL_EXTENSIONS);
228 return eglExtensions && strstr(eglExtensions, "EGL_KHR_fence_sync");
229 }
230
231 /////////////////////////////////////////////////////////////////////////////////////////
232 // Textures utilities
233 /////////////////////////////////////////////////////////////////////////////////////////
234
getInternalFormat(SkBitmap::Config config)235 static GLenum getInternalFormat(SkBitmap::Config config)
236 {
237 switch (config) {
238 case SkBitmap::kA8_Config:
239 return GL_ALPHA;
240 case SkBitmap::kARGB_4444_Config:
241 return GL_RGBA;
242 case SkBitmap::kARGB_8888_Config:
243 return GL_RGBA;
244 case SkBitmap::kRGB_565_Config:
245 return GL_RGB;
246 default:
247 return -1;
248 }
249 }
250
getType(SkBitmap::Config config)251 static GLenum getType(SkBitmap::Config config)
252 {
253 switch (config) {
254 case SkBitmap::kA8_Config:
255 return GL_UNSIGNED_BYTE;
256 case SkBitmap::kARGB_4444_Config:
257 return GL_UNSIGNED_SHORT_4_4_4_4;
258 case SkBitmap::kARGB_8888_Config:
259 return GL_UNSIGNED_BYTE;
260 case SkBitmap::kIndex8_Config:
261 return -1; // No type for compressed data.
262 case SkBitmap::kRGB_565_Config:
263 return GL_UNSIGNED_SHORT_5_6_5;
264 default:
265 return -1;
266 }
267 }
268
defaultPbufferConfig(EGLDisplay display)269 static EGLConfig defaultPbufferConfig(EGLDisplay display)
270 {
271 EGLConfig config;
272 EGLint numConfigs;
273
274 static const EGLint configAttribs[] = {
275 EGL_SURFACE_TYPE, EGL_PBUFFER_BIT,
276 EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
277 EGL_NONE
278 };
279
280 eglChooseConfig(display, configAttribs, &config, 1, &numConfigs);
281 GLUtils::checkEglError("eglPbufferConfig");
282 if (numConfigs != 1)
283 ALOGI("eglPbufferConfig failed (%d)\n", numConfigs);
284
285 return config;
286 }
287
createPbufferSurface(EGLDisplay display,const EGLConfig & config,EGLint * errorCode)288 static EGLSurface createPbufferSurface(EGLDisplay display, const EGLConfig& config,
289 EGLint* errorCode)
290 {
291 const EGLint attribList[] = {
292 EGL_WIDTH, 1,
293 EGL_HEIGHT, 1,
294 EGL_NONE
295 };
296 EGLSurface surface = eglCreatePbufferSurface(display, config, attribList);
297
298 if (errorCode)
299 *errorCode = eglGetError();
300 else
301 GLUtils::checkEglError("eglCreatePbufferSurface");
302
303 if (surface == EGL_NO_SURFACE)
304 return EGL_NO_SURFACE;
305
306 return surface;
307 }
308
deleteTexture(GLuint * texture)309 void GLUtils::deleteTexture(GLuint* texture)
310 {
311 glDeleteTextures(1, texture);
312 GLUtils::checkGlError("glDeleteTexture");
313 *texture = 0;
314 }
315
createSampleColorTexture(int r,int g,int b)316 GLuint GLUtils::createSampleColorTexture(int r, int g, int b)
317 {
318 GLuint texture;
319 glGenTextures(1, &texture);
320 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
321 GLubyte pixels[4 *3] = {
322 r, g, b,
323 r, g, b,
324 r, g, b,
325 r, g, b
326 };
327 glBindTexture(GL_TEXTURE_2D, texture);
328 GLUtils::checkGlError("glBindTexture");
329 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 2, 2, 0, GL_RGB, GL_UNSIGNED_BYTE, pixels);
330 GLUtils::checkGlError("glTexImage2D");
331 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
332 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
333 return texture;
334 }
335
createSampleTexture()336 GLuint GLUtils::createSampleTexture()
337 {
338 GLuint texture;
339 glGenTextures(1, &texture);
340 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
341 GLubyte pixels[4 *3] = {
342 255, 0, 0,
343 0, 255, 0,
344 0, 0, 255,
345 255, 255, 0
346 };
347 glBindTexture(GL_TEXTURE_2D, texture);
348 GLUtils::checkGlError("glBindTexture");
349 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 2, 2, 0, GL_RGB, GL_UNSIGNED_BYTE, pixels);
350 GLUtils::checkGlError("glTexImage2D");
351 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
352 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
353 return texture;
354 }
355
createTileGLTexture(int width,int height)356 GLuint GLUtils::createTileGLTexture(int width, int height)
357 {
358 GLuint texture;
359 glGenTextures(1, &texture);
360 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
361 GLubyte* pixels = 0;
362 #ifdef DEBUG
363 int length = width * height * 4;
364 pixels = new GLubyte[length];
365 for (int i = 0; i < length; i++)
366 pixels[i] = i % 256;
367 #endif
368 glBindTexture(GL_TEXTURE_2D, texture);
369 GLUtils::checkGlError("glBindTexture");
370 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, pixels);
371 GLUtils::checkGlError("glTexImage2D");
372 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
373 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
374 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
375 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
376 #ifdef DEBUG
377 delete pixels;
378 #endif
379 return texture;
380 }
381
isPureColorBitmap(const SkBitmap & bitmap,Color & pureColor)382 bool GLUtils::isPureColorBitmap(const SkBitmap& bitmap, Color& pureColor)
383 {
384 // If the bitmap is the pure color, skip the transfer step, and update the Tile Info.
385 // This check is taking < 1ms if we do full bitmap check per tile.
386 // TODO: use the SkPicture to determine whether or not a tile is single color.
387 TRACE_METHOD();
388 pureColor = Color(Color::transparent);
389 bitmap.lockPixels();
390 bool sameColor = true;
391 int bitmapWidth = bitmap.width();
392
393 // Create a row of pure color using the first pixel.
394 // TODO: improve the perf here, by either picking a random pixel, or
395 // creating an array of rows with pre-defined commonly used color, add
396 // smart LUT to speed things up if possible.
397 int* firstPixelPtr = static_cast<int*> (bitmap.getPixels());
398 int* pixelsRow = new int[bitmapWidth];
399 for (int i = 0; i < bitmapWidth; i++)
400 pixelsRow[i] = (*firstPixelPtr);
401
402 // Then compare the pure color row with each row of the bitmap.
403 for (int j = 0; j < bitmap.height(); j++) {
404 if (memcmp(pixelsRow, &firstPixelPtr[bitmapWidth * j], 4 * bitmapWidth)) {
405 sameColor = false;
406 break;
407 }
408 }
409 delete pixelsRow;
410 pixelsRow = 0;
411
412 if (sameColor) {
413 unsigned char* rgbaPtr = static_cast<unsigned char*>(bitmap.getPixels());
414 pureColor = Color(rgbaPtr[0], rgbaPtr[1], rgbaPtr[2], rgbaPtr[3]);
415 ALOGV("sameColor tile found , %x at (%d, %d, %d, %d)",
416 *firstPixelPtr, rgbaPtr[0], rgbaPtr[1], rgbaPtr[2], rgbaPtr[3]);
417 }
418 bitmap.unlockPixels();
419
420 return sameColor;
421 }
422
423 // Return true when the tile is pure color.
skipTransferForPureColor(const TileRenderInfo * renderInfo,const SkBitmap & bitmap)424 bool GLUtils::skipTransferForPureColor(const TileRenderInfo* renderInfo,
425 const SkBitmap& bitmap)
426 {
427 bool skipTransfer = false;
428 Tile* tilePtr = renderInfo->baseTile;
429
430 if (tilePtr) {
431 TileTexture* tileTexture = tilePtr->backTexture();
432 // Check the bitmap, and make everything ready here.
433 if (tileTexture && renderInfo->isPureColor) {
434 // update basetile's info
435 // Note that we are skipping the whole TransferQueue.
436 renderInfo->textureInfo->m_width = bitmap.width();
437 renderInfo->textureInfo->m_height = bitmap.height();
438 renderInfo->textureInfo->m_internalFormat = GL_RGBA;
439
440 TilesManager::instance()->transferQueue()->addItemInPureColorQueue(renderInfo);
441
442 skipTransfer = true;
443 }
444 }
445 return skipTransfer;
446 }
447
paintTextureWithBitmap(const TileRenderInfo * renderInfo,SkBitmap & bitmap)448 void GLUtils::paintTextureWithBitmap(const TileRenderInfo* renderInfo,
449 SkBitmap& bitmap)
450 {
451 if (!renderInfo)
452 return;
453 const SkSize& requiredSize = renderInfo->tileSize;
454 TextureInfo* textureInfo = renderInfo->textureInfo;
455
456 if (skipTransferForPureColor(renderInfo, bitmap))
457 return;
458
459 if (requiredSize.equals(textureInfo->m_width, textureInfo->m_height))
460 GLUtils::updateQueueWithBitmap(renderInfo, bitmap);
461 else {
462 if (!requiredSize.equals(bitmap.width(), bitmap.height())) {
463 ALOGV("The bitmap size (%d,%d) does not equal the texture size (%d,%d)",
464 bitmap.width(), bitmap.height(),
465 requiredSize.width(), requiredSize.height());
466 }
467 GLUtils::updateQueueWithBitmap(renderInfo, bitmap);
468
469 textureInfo->m_width = bitmap.width();
470 textureInfo->m_height = bitmap.height();
471 textureInfo->m_internalFormat = GL_RGBA;
472 }
473 }
474
updateQueueWithBitmap(const TileRenderInfo * renderInfo,SkBitmap & bitmap)475 void GLUtils::updateQueueWithBitmap(const TileRenderInfo* renderInfo, SkBitmap& bitmap)
476 {
477 if (!renderInfo
478 || !renderInfo->textureInfo
479 || !renderInfo->baseTile)
480 return;
481
482 TilesManager::instance()->transferQueue()->updateQueueWithBitmap(renderInfo, bitmap);
483 }
484
updateSharedSurfaceTextureWithBitmap(ANativeWindow * anw,const SkBitmap & bitmap)485 bool GLUtils::updateSharedSurfaceTextureWithBitmap(ANativeWindow* anw, const SkBitmap& bitmap)
486 {
487 TRACE_METHOD();
488 SkAutoLockPixels alp(bitmap);
489 if (!bitmap.getPixels())
490 return false;
491 ANativeWindow_Buffer buffer;
492 if (ANativeWindow_lock(anw, &buffer, 0))
493 return false;
494 if (buffer.width < bitmap.width() || buffer.height < bitmap.height()) {
495 ALOGW("bitmap (%dx%d) too large for buffer (%dx%d)!",
496 bitmap.width(), bitmap.height(),
497 buffer.width, buffer.height);
498 ANativeWindow_unlockAndPost(anw);
499 return false;
500 }
501 uint8_t* img = (uint8_t*)buffer.bits;
502 int row;
503 int bpp = 4; // Now we only deal with RGBA8888 format.
504 bitmap.lockPixels();
505 uint8_t* bitmapOrigin = static_cast<uint8_t*>(bitmap.getPixels());
506
507 if (buffer.stride != bitmap.width())
508 // Copied line by line since we need to handle the offsets and stride.
509 for (row = 0 ; row < bitmap.height(); row ++) {
510 uint8_t* dst = &(img[buffer.stride * row * bpp]);
511 uint8_t* src = &(bitmapOrigin[bitmap.width() * row * bpp]);
512 memcpy(dst, src, bpp * bitmap.width());
513 }
514 else
515 memcpy(img, bitmapOrigin, bpp * bitmap.width() * bitmap.height());
516
517 bitmap.unlockPixels();
518 ANativeWindow_unlockAndPost(anw);
519 return true;
520 }
521
createTextureWithBitmap(GLuint texture,const SkBitmap & bitmap,GLint filter)522 void GLUtils::createTextureWithBitmap(GLuint texture, const SkBitmap& bitmap, GLint filter)
523 {
524 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
525 glBindTexture(GL_TEXTURE_2D, texture);
526 GLUtils::checkGlError("glBindTexture");
527 SkBitmap::Config config = bitmap.getConfig();
528 int internalformat = getInternalFormat(config);
529 int type = getType(config);
530 bitmap.lockPixels();
531 glTexImage2D(GL_TEXTURE_2D, 0, internalformat, bitmap.width(), bitmap.height(),
532 0, internalformat, type, bitmap.getPixels());
533 bitmap.unlockPixels();
534 if (GLUtils::checkGlError("glTexImage2D")) {
535 #ifndef DEBUG
536 if (allowGLLog())
537 #endif
538 ALOGE("GL ERROR: glTexImage2D parameters are : textureId %d,"
539 " bitmap.width() %d, bitmap.height() %d,"
540 " internalformat 0x%x, type 0x%x, bitmap.getPixels() %p",
541 texture, bitmap.width(), bitmap.height(), internalformat, type,
542 bitmap.getPixels());
543 }
544 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, filter);
545 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, filter);
546 }
547
updateTextureWithBitmap(GLuint texture,const SkBitmap & bitmap,const IntRect & inval,GLint filter)548 void GLUtils::updateTextureWithBitmap(GLuint texture, const SkBitmap& bitmap,
549 const IntRect& inval, GLint filter)
550 {
551 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
552 glBindTexture(GL_TEXTURE_2D, texture);
553 GLUtils::checkGlError("glBindTexture");
554 SkBitmap::Config config = bitmap.getConfig();
555 int internalformat = getInternalFormat(config);
556 int type = getType(config);
557 bitmap.lockPixels();
558 if (inval.isEmpty()) {
559 glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, bitmap.width(), bitmap.height(),
560 internalformat, type, bitmap.getPixels());
561 } else {
562 glTexSubImage2D(GL_TEXTURE_2D, 0, inval.x(), inval.y(), inval.width(), inval.height(),
563 internalformat, type, bitmap.getPixels());
564 }
565 bitmap.unlockPixels();
566 if (GLUtils::checkGlError("glTexSubImage2D")) {
567 #ifndef DEBUG
568 if (allowGLLog())
569 #endif
570 ALOGE("GL ERROR: glTexSubImage2D parameters are : textureId %d,"
571 " bitmap.width() %d, bitmap.height() %d,"
572 " internalformat 0x%x, type 0x%x, bitmap.getPixels() %p",
573 texture, bitmap.width(), bitmap.height(), internalformat, type,
574 bitmap.getPixels());
575 }
576 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, filter);
577 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, filter);
578 }
579
createEGLImageFromTexture(GLuint texture,EGLImageKHR * image)580 void GLUtils::createEGLImageFromTexture(GLuint texture, EGLImageKHR* image)
581 {
582 EGLClientBuffer buffer = reinterpret_cast<EGLClientBuffer>(texture);
583 static const EGLint attr[] = { EGL_IMAGE_PRESERVED_KHR, EGL_TRUE, EGL_NONE };
584 *image = eglCreateImageKHR(eglGetCurrentDisplay(), eglGetCurrentContext(),
585 EGL_GL_TEXTURE_2D_KHR, buffer, attr);
586 GLUtils::checkEglError("eglCreateImage", (*image != EGL_NO_IMAGE_KHR));
587 }
588
createTextureFromEGLImage(GLuint texture,EGLImageKHR image,GLint filter)589 void GLUtils::createTextureFromEGLImage(GLuint texture, EGLImageKHR image, GLint filter)
590 {
591 glBindTexture(GL_TEXTURE_2D, texture);
592 GLUtils::checkGlError("glBindTexture");
593 glEGLImageTargetTexture2DOES(GL_TEXTURE_2D, (GLeglImageOES)image);
594 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, filter);
595 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, filter);
596 }
597
convertToTransformationMatrix(const float * matrix,TransformationMatrix & transformMatrix)598 void GLUtils::convertToTransformationMatrix(const float* matrix, TransformationMatrix& transformMatrix)
599 {
600 transformMatrix.setMatrix(
601 matrix[0], matrix[1], matrix[2], matrix[3],
602 matrix[4], matrix[5], matrix[6], matrix[7],
603 matrix[8], matrix[9], matrix[10], matrix[11],
604 matrix[12], matrix[13], matrix[14], matrix[15]);
605 }
606
clearBackgroundIfOpaque(const Color * backgroundColor)607 void GLUtils::clearBackgroundIfOpaque(const Color* backgroundColor)
608 {
609 if (!backgroundColor->hasAlpha()) {
610 if (TilesManager::instance()->invertedScreen()) {
611 float color = 1.0 - ((((float) backgroundColor->red() / 255.0) +
612 ((float) backgroundColor->green() / 255.0) +
613 ((float) backgroundColor->blue() / 255.0)) / 3.0);
614 glClearColor(color, color, color, 1);
615 } else {
616 glClearColor((float)backgroundColor->red() / 255.0,
617 (float)backgroundColor->green() / 255.0,
618 (float)backgroundColor->blue() / 255.0, 1);
619 }
620 glClear(GL_COLOR_BUFFER_BIT);
621 }
622 }
623
deepCopyBitmapSubset(const SkBitmap & sourceBitmap,SkBitmap & subset,int leftOffset,int topOffset)624 bool GLUtils::deepCopyBitmapSubset(const SkBitmap& sourceBitmap,
625 SkBitmap& subset, int leftOffset, int topOffset)
626 {
627 sourceBitmap.lockPixels();
628 subset.lockPixels();
629 char* srcPixels = (char*) sourceBitmap.getPixels();
630 char* dstPixels = (char*) subset.getPixels();
631 if (!dstPixels || !srcPixels || !subset.lockPixelsAreWritable()) {
632 ALOGD("no pixels :( %p, %p (writable=%d)", srcPixels, dstPixels,
633 subset.lockPixelsAreWritable());
634 subset.unlockPixels();
635 sourceBitmap.unlockPixels();
636 return false;
637 }
638 int srcRowSize = sourceBitmap.rowBytes();
639 int destRowSize = subset.rowBytes();
640 for (int i = 0; i < subset.height(); i++) {
641 int srcOffset = (i + topOffset) * srcRowSize;
642 srcOffset += (leftOffset * sourceBitmap.bytesPerPixel());
643 int dstOffset = i * destRowSize;
644 memcpy(dstPixels + dstOffset, srcPixels + srcOffset, destRowSize);
645 }
646 subset.unlockPixels();
647 sourceBitmap.unlockPixels();
648 return true;
649 }
650
651 } // namespace WebCore
652
653 #endif // USE(ACCELERATED_COMPOSITING)
654