1 /*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 #include "FrameBuffer.h"
17 #include "NativeSubWindow.h"
18 #include "FBConfig.h"
19 #include "EGLDispatch.h"
20 #include "GLDispatch.h"
21 #include "GL2Dispatch.h"
22 #include "ThreadInfo.h"
23 #include <stdio.h>
24 #include "TimeUtils.h"
25
26 FrameBuffer *FrameBuffer::s_theFrameBuffer = NULL;
27 HandleType FrameBuffer::s_nextHandle = 0;
28
29 #ifdef WITH_GLES2
getGLES2ExtensionString(EGLDisplay p_dpy)30 static const char *getGLES2ExtensionString(EGLDisplay p_dpy)
31 {
32 EGLConfig config;
33 EGLSurface surface;
34
35 GLint configAttribs[] = {
36 EGL_SURFACE_TYPE, EGL_WINDOW_BIT | EGL_PBUFFER_BIT,
37 EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
38 EGL_NONE
39 };
40
41 int n;
42 if (!s_egl.eglChooseConfig(p_dpy, configAttribs,
43 &config, 1, &n)) {
44 return NULL;
45 }
46
47 EGLint pbufAttribs[] = {
48 EGL_WIDTH, 1,
49 EGL_HEIGHT, 1,
50 EGL_NONE
51 };
52
53 surface = s_egl.eglCreatePbufferSurface(p_dpy, config, pbufAttribs);
54 if (surface == EGL_NO_SURFACE) {
55 return NULL;
56 }
57
58 GLint gl2ContextAttribs[] = {
59 EGL_CONTEXT_CLIENT_VERSION, 2,
60 EGL_NONE
61 };
62
63 EGLContext ctx = s_egl.eglCreateContext(p_dpy, config,
64 EGL_NO_CONTEXT,
65 gl2ContextAttribs);
66 if (ctx == EGL_NO_CONTEXT) {
67 s_egl.eglDestroySurface(p_dpy, surface);
68 return NULL;
69 }
70
71 if (!s_egl.eglMakeCurrent(p_dpy, surface, surface, ctx)) {
72 s_egl.eglDestroySurface(p_dpy, surface);
73 s_egl.eglDestroyContext(p_dpy, ctx);
74 return NULL;
75 }
76
77 const char *extString = (const char *)s_gl2.glGetString(GL_EXTENSIONS);
78 if (!extString) {
79 extString = "";
80 }
81
82 s_egl.eglMakeCurrent(p_dpy, NULL, NULL, NULL);
83 s_egl.eglDestroyContext(p_dpy, ctx);
84 s_egl.eglDestroySurface(p_dpy, surface);
85
86 return extString;
87 }
88 #endif
89
finalize()90 void FrameBuffer::finalize(){
91 if(s_theFrameBuffer){
92 s_theFrameBuffer->removeSubWindow();
93 s_theFrameBuffer->m_colorbuffers.clear();
94 s_theFrameBuffer->m_windows.clear();
95 s_theFrameBuffer->m_contexts.clear();
96 s_egl.eglMakeCurrent(s_theFrameBuffer->m_eglDisplay, NULL, NULL, NULL);
97 s_egl.eglDestroyContext(s_theFrameBuffer->m_eglDisplay,s_theFrameBuffer->m_eglContext);
98 s_egl.eglDestroyContext(s_theFrameBuffer->m_eglDisplay,s_theFrameBuffer->m_pbufContext);
99 s_egl.eglDestroySurface(s_theFrameBuffer->m_eglDisplay,s_theFrameBuffer->m_pbufSurface);
100 s_theFrameBuffer = NULL;
101 }
102 }
103
initialize(int width,int height)104 bool FrameBuffer::initialize(int width, int height)
105 {
106 if (s_theFrameBuffer != NULL) {
107 return true;
108 }
109
110 //
111 // allocate space for the FrameBuffer object
112 //
113 FrameBuffer *fb = new FrameBuffer(width, height);
114 if (!fb) {
115 ERR("Failed to create fb\n");
116 return false;
117 }
118
119 #ifdef WITH_GLES2
120 //
121 // Try to load GLES2 Plugin, not mandatory
122 //
123 if (getenv("ANDROID_NO_GLES2")) {
124 fb->m_caps.hasGL2 = false;
125 }
126 else {
127 fb->m_caps.hasGL2 = s_gl2_enabled;
128 }
129 #else
130 fb->m_caps.hasGL2 = false;
131 #endif
132
133 //
134 // Initialize backend EGL display
135 //
136 fb->m_eglDisplay = s_egl.eglGetDisplay(EGL_DEFAULT_DISPLAY);
137 if (fb->m_eglDisplay == EGL_NO_DISPLAY) {
138 ERR("Failed to Initialize backend EGL display\n");
139 delete fb;
140 return false;
141 }
142
143 if (!s_egl.eglInitialize(fb->m_eglDisplay, &fb->m_caps.eglMajor, &fb->m_caps.eglMinor)) {
144 ERR("Failed to eglInitialize\n");
145 delete fb;
146 return false;
147 }
148
149 DBG("egl: %d %d\n", fb->m_caps.eglMajor, fb->m_caps.eglMinor);
150 s_egl.eglBindAPI(EGL_OPENGL_ES_API);
151
152 //
153 // if GLES2 plugin has loaded - try to make GLES2 context and
154 // get GLES2 extension string
155 //
156 const char *gl2Extensions = NULL;
157 #ifdef WITH_GLES2
158 if (fb->m_caps.hasGL2) {
159 gl2Extensions = getGLES2ExtensionString(fb->m_eglDisplay);
160 if (!gl2Extensions) {
161 // Could not create GLES2 context - drop GL2 capability
162 fb->m_caps.hasGL2 = false;
163 }
164 }
165 #endif
166
167 //
168 // Create EGL context for framebuffer post rendering.
169 //
170 #if 0
171 GLint configAttribs[] = {
172 EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
173 EGL_RENDERABLE_TYPE, EGL_OPENGL_ES_BIT,
174 EGL_NONE
175 };
176 #else
177 GLint configAttribs[] = {
178 EGL_RED_SIZE, 1,
179 EGL_GREEN_SIZE, 1,
180 EGL_BLUE_SIZE, 1,
181 EGL_SURFACE_TYPE, EGL_WINDOW_BIT | EGL_PBUFFER_BIT,
182 EGL_NONE
183 };
184 #endif
185
186 int n;
187 if (!s_egl.eglChooseConfig(fb->m_eglDisplay, configAttribs,
188 &fb->m_eglConfig, 1, &n)) {
189 ERR("Failed on eglChooseConfig\n");
190 delete fb;
191 return false;
192 }
193
194 GLint glContextAttribs[] = {
195 EGL_CONTEXT_CLIENT_VERSION, 1,
196 EGL_NONE
197 };
198
199 fb->m_eglContext = s_egl.eglCreateContext(fb->m_eglDisplay, fb->m_eglConfig,
200 EGL_NO_CONTEXT,
201 glContextAttribs);
202 if (fb->m_eglContext == EGL_NO_CONTEXT) {
203 printf("Failed to create Context 0x%x\n", s_egl.eglGetError());
204 delete fb;
205 return false;
206 }
207
208 //
209 // Create another context which shares with the eglContext to be used
210 // when we bind the pbuffer. That prevent switching drawable binding
211 // back and forth on framebuffer context.
212 // The main purpose of it is to solve a "blanking" behaviour we see on
213 // on Mac platform when switching binded drawable for a context however
214 // it is more efficient on other platforms as well.
215 //
216 fb->m_pbufContext = s_egl.eglCreateContext(fb->m_eglDisplay, fb->m_eglConfig,
217 fb->m_eglContext,
218 glContextAttribs);
219 if (fb->m_pbufContext == EGL_NO_CONTEXT) {
220 printf("Failed to create Pbuffer Context 0x%x\n", s_egl.eglGetError());
221 delete fb;
222 return false;
223 }
224
225 //
226 // create a 1x1 pbuffer surface which will be used for binding
227 // the FB context.
228 // The FB output will go to a subwindow, if one exist.
229 //
230 EGLint pbufAttribs[] = {
231 EGL_WIDTH, 1,
232 EGL_HEIGHT, 1,
233 EGL_NONE
234 };
235
236 fb->m_pbufSurface = s_egl.eglCreatePbufferSurface(fb->m_eglDisplay,
237 fb->m_eglConfig,
238 pbufAttribs);
239 if (fb->m_pbufSurface == EGL_NO_SURFACE) {
240 printf("Failed to create pbuf surface for FB 0x%x\n", s_egl.eglGetError());
241 delete fb;
242 return false;
243 }
244
245 // Make the context current
246 if (!fb->bind_locked()) {
247 ERR("Failed to make current\n");
248 delete fb;
249 return false;
250 }
251
252 //
253 // Initilize framebuffer capabilities
254 //
255 const char *glExtensions = (const char *)s_gl.glGetString(GL_EXTENSIONS);
256 bool has_gl_oes_image = false;
257 if (glExtensions) {
258 has_gl_oes_image = strstr(glExtensions, "GL_OES_EGL_image") != NULL;
259 }
260
261 if (fb->m_caps.hasGL2 && has_gl_oes_image) {
262 has_gl_oes_image &= (strstr(gl2Extensions, "GL_OES_EGL_image") != NULL);
263 }
264
265 const char *eglExtensions = s_egl.eglQueryString(fb->m_eglDisplay,
266 EGL_EXTENSIONS);
267
268 if (eglExtensions && has_gl_oes_image) {
269 fb->m_caps.has_eglimage_texture_2d =
270 strstr(eglExtensions, "EGL_KHR_gl_texture_2D_image") != NULL;
271 fb->m_caps.has_eglimage_renderbuffer =
272 strstr(eglExtensions, "EGL_KHR_gl_renderbuffer_image") != NULL;
273 }
274 else {
275 fb->m_caps.has_eglimage_texture_2d = false;
276 fb->m_caps.has_eglimage_renderbuffer = false;
277 }
278
279 //
280 // Fail initialization if not all of the following extensions
281 // exist:
282 // EGL_KHR_gl_texture_2d_image
283 // GL_OES_EGL_IMAGE (by both GLES implementations [1 and 2])
284 //
285 if (!fb->m_caps.has_eglimage_texture_2d) {
286 ERR("Failed: Missing egl_image related extension(s)\n");
287 delete fb;
288 return false;
289 }
290
291 //
292 // Initialize set of configs
293 //
294 InitConfigStatus configStatus = FBConfig::initConfigList(fb);
295 if (configStatus == INIT_CONFIG_FAILED) {
296 ERR("Failed: Initialize set of configs\n");
297 delete fb;
298 return false;
299 }
300
301 //
302 // Check that we have config for each GLES and GLES2
303 //
304 int nConfigs = FBConfig::getNumConfigs();
305 int nGLConfigs = 0;
306 int nGL2Configs = 0;
307 for (int i=0; i<nConfigs; i++) {
308 GLint rtype = FBConfig::get(i)->getRenderableType();
309 if (0 != (rtype & EGL_OPENGL_ES_BIT)) {
310 nGLConfigs++;
311 }
312 if (0 != (rtype & EGL_OPENGL_ES2_BIT)) {
313 nGL2Configs++;
314 }
315 }
316
317 //
318 // Fail initialization if no GLES configs exist
319 //
320 if (nGLConfigs == 0) {
321 delete fb;
322 return false;
323 }
324
325 //
326 // If no GLES2 configs exist - not GLES2 capability
327 //
328 if (nGL2Configs == 0) {
329 fb->m_caps.hasGL2 = false;
330 }
331
332 //
333 // Initialize some GL state in the pbuffer context
334 //
335 fb->initGLState();
336
337 //
338 // Cache the GL strings so we don't have to think about threading or
339 // current-context when asked for them.
340 //
341 fb->m_glVendor = (const char*)s_gl.glGetString(GL_VENDOR);
342 fb->m_glRenderer = (const char*)s_gl.glGetString(GL_RENDERER);
343 fb->m_glVersion = (const char*)s_gl.glGetString(GL_VERSION);
344
345 // release the FB context
346 fb->unbind_locked();
347
348 //
349 // Keep the singleton framebuffer pointer
350 //
351 s_theFrameBuffer = fb;
352 return true;
353 }
354
FrameBuffer(int p_width,int p_height)355 FrameBuffer::FrameBuffer(int p_width, int p_height) :
356 m_width(p_width),
357 m_height(p_height),
358 m_eglDisplay(EGL_NO_DISPLAY),
359 m_eglSurface(EGL_NO_SURFACE),
360 m_eglContext(EGL_NO_CONTEXT),
361 m_pbufContext(EGL_NO_CONTEXT),
362 m_prevContext(EGL_NO_CONTEXT),
363 m_prevReadSurf(EGL_NO_SURFACE),
364 m_prevDrawSurf(EGL_NO_SURFACE),
365 m_subWin((EGLNativeWindowType)0),
366 m_subWinDisplay(NULL),
367 m_lastPostedColorBuffer(0),
368 m_zRot(0.0f),
369 m_eglContextInitialized(false),
370 m_statsNumFrames(0),
371 m_statsStartTime(0LL),
372 m_onPost(NULL),
373 m_onPostContext(NULL),
374 m_fbImage(NULL),
375 m_glVendor(NULL),
376 m_glRenderer(NULL),
377 m_glVersion(NULL)
378 {
379 m_fpsStats = getenv("SHOW_FPS_STATS") != NULL;
380 }
381
~FrameBuffer()382 FrameBuffer::~FrameBuffer()
383 {
384 free(m_fbImage);
385 }
386
setPostCallback(OnPostFn onPost,void * onPostContext)387 void FrameBuffer::setPostCallback(OnPostFn onPost, void* onPostContext)
388 {
389 android::Mutex::Autolock mutex(m_lock);
390 m_onPost = onPost;
391 m_onPostContext = onPostContext;
392 if (m_onPost && !m_fbImage) {
393 m_fbImage = (unsigned char*)malloc(4 * m_width * m_height);
394 if (!m_fbImage) {
395 ERR("out of memory, cancelling OnPost callback");
396 m_onPost = NULL;
397 m_onPostContext = NULL;
398 return;
399 }
400 }
401 }
402
setupSubWindow(FBNativeWindowType p_window,int p_x,int p_y,int p_width,int p_height,float zRot)403 bool FrameBuffer::setupSubWindow(FBNativeWindowType p_window,
404 int p_x, int p_y,
405 int p_width, int p_height, float zRot)
406 {
407 bool success = false;
408
409 if (s_theFrameBuffer) {
410 s_theFrameBuffer->m_lock.lock();
411 FrameBuffer *fb = s_theFrameBuffer;
412 if (!fb->m_subWin) {
413
414 // create native subwindow for FB display output
415 fb->m_subWin = createSubWindow(p_window,
416 &fb->m_subWinDisplay,
417 p_x,p_y,p_width,p_height);
418 if (fb->m_subWin) {
419 fb->m_nativeWindow = p_window;
420
421 // create EGLSurface from the generated subwindow
422 fb->m_eglSurface = s_egl.eglCreateWindowSurface(fb->m_eglDisplay,
423 fb->m_eglConfig,
424 fb->m_subWin,
425 NULL);
426
427 if (fb->m_eglSurface == EGL_NO_SURFACE) {
428 ERR("Failed to create surface\n");
429 destroySubWindow(fb->m_subWinDisplay, fb->m_subWin);
430 fb->m_subWin = (EGLNativeWindowType)0;
431 }
432 else if (fb->bindSubwin_locked()) {
433 // Subwin creation was successfull,
434 // update viewport and z rotation and draw
435 // the last posted color buffer.
436 s_gl.glViewport(0, 0, p_width, p_height);
437 fb->m_zRot = zRot;
438 fb->post( fb->m_lastPostedColorBuffer, false );
439 fb->unbind_locked();
440 success = true;
441 }
442 }
443 }
444 s_theFrameBuffer->m_lock.unlock();
445 }
446
447 return success;
448 }
449
removeSubWindow()450 bool FrameBuffer::removeSubWindow()
451 {
452 bool removed = false;
453 if (s_theFrameBuffer) {
454 s_theFrameBuffer->m_lock.lock();
455 if (s_theFrameBuffer->m_subWin) {
456 s_egl.eglMakeCurrent(s_theFrameBuffer->m_eglDisplay, NULL, NULL, NULL);
457 s_egl.eglDestroySurface(s_theFrameBuffer->m_eglDisplay,
458 s_theFrameBuffer->m_eglSurface);
459 destroySubWindow(s_theFrameBuffer->m_subWinDisplay,
460 s_theFrameBuffer->m_subWin);
461
462 s_theFrameBuffer->m_eglSurface = EGL_NO_SURFACE;
463 s_theFrameBuffer->m_subWin = (EGLNativeWindowType)0;
464 removed = true;
465 }
466 s_theFrameBuffer->m_lock.unlock();
467 }
468 return removed;
469 }
470
genHandle()471 HandleType FrameBuffer::genHandle()
472 {
473 HandleType id;
474 do {
475 id = ++s_nextHandle;
476 } while( id == 0 ||
477 m_contexts.find(id) != m_contexts.end() ||
478 m_windows.find(id) != m_windows.end() );
479
480 return id;
481 }
482
createColorBuffer(int p_width,int p_height,GLenum p_internalFormat)483 HandleType FrameBuffer::createColorBuffer(int p_width, int p_height,
484 GLenum p_internalFormat)
485 {
486 android::Mutex::Autolock mutex(m_lock);
487 HandleType ret = 0;
488
489 ColorBufferPtr cb( ColorBuffer::create(p_width, p_height, p_internalFormat) );
490 if (cb.Ptr() != NULL) {
491 ret = genHandle();
492 m_colorbuffers[ret].cb = cb;
493 m_colorbuffers[ret].refcount = 1;
494 }
495 return ret;
496 }
497
createRenderContext(int p_config,HandleType p_share,bool p_isGL2)498 HandleType FrameBuffer::createRenderContext(int p_config, HandleType p_share,
499 bool p_isGL2)
500 {
501 android::Mutex::Autolock mutex(m_lock);
502 HandleType ret = 0;
503
504 RenderContextPtr share(NULL);
505 if (p_share != 0) {
506 RenderContextMap::iterator s( m_contexts.find(p_share) );
507 if (s == m_contexts.end()) {
508 return 0;
509 }
510 share = (*s).second;
511 }
512
513 RenderContextPtr rctx( RenderContext::create(p_config, share, p_isGL2) );
514 if (rctx.Ptr() != NULL) {
515 ret = genHandle();
516 m_contexts[ret] = rctx;
517 }
518 return ret;
519 }
520
createWindowSurface(int p_config,int p_width,int p_height)521 HandleType FrameBuffer::createWindowSurface(int p_config, int p_width, int p_height)
522 {
523 android::Mutex::Autolock mutex(m_lock);
524
525 HandleType ret = 0;
526 WindowSurfacePtr win( WindowSurface::create(p_config, p_width, p_height) );
527 if (win.Ptr() != NULL) {
528 ret = genHandle();
529 m_windows[ret] = win;
530 }
531
532 return ret;
533 }
534
DestroyRenderContext(HandleType p_context)535 void FrameBuffer::DestroyRenderContext(HandleType p_context)
536 {
537 android::Mutex::Autolock mutex(m_lock);
538 m_contexts.erase(p_context);
539 }
540
DestroyWindowSurface(HandleType p_surface)541 void FrameBuffer::DestroyWindowSurface(HandleType p_surface)
542 {
543 android::Mutex::Autolock mutex(m_lock);
544 m_windows.erase(p_surface);
545 }
546
openColorBuffer(HandleType p_colorbuffer)547 void FrameBuffer::openColorBuffer(HandleType p_colorbuffer)
548 {
549 android::Mutex::Autolock mutex(m_lock);
550 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
551 if (c == m_colorbuffers.end()) {
552 // bad colorbuffer handle
553 return;
554 }
555 (*c).second.refcount++;
556 }
557
closeColorBuffer(HandleType p_colorbuffer)558 void FrameBuffer::closeColorBuffer(HandleType p_colorbuffer)
559 {
560 android::Mutex::Autolock mutex(m_lock);
561 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
562 if (c == m_colorbuffers.end()) {
563 // bad colorbuffer handle
564 return;
565 }
566 if (--(*c).second.refcount == 0) {
567 m_colorbuffers.erase(c);
568 }
569 }
570
flushWindowSurfaceColorBuffer(HandleType p_surface)571 bool FrameBuffer::flushWindowSurfaceColorBuffer(HandleType p_surface)
572 {
573 android::Mutex::Autolock mutex(m_lock);
574
575 WindowSurfaceMap::iterator w( m_windows.find(p_surface) );
576 if (w == m_windows.end()) {
577 // bad surface handle
578 return false;
579 }
580
581 (*w).second->flushColorBuffer();
582
583 return true;
584 }
585
setWindowSurfaceColorBuffer(HandleType p_surface,HandleType p_colorbuffer)586 bool FrameBuffer::setWindowSurfaceColorBuffer(HandleType p_surface,
587 HandleType p_colorbuffer)
588 {
589 android::Mutex::Autolock mutex(m_lock);
590
591 WindowSurfaceMap::iterator w( m_windows.find(p_surface) );
592 if (w == m_windows.end()) {
593 // bad surface handle
594 return false;
595 }
596
597 ColorBufferMap::iterator c( m_colorbuffers.find(p_colorbuffer) );
598 if (c == m_colorbuffers.end()) {
599 // bad colorbuffer handle
600 return false;
601 }
602
603 (*w).second->setColorBuffer( (*c).second.cb );
604
605 return true;
606 }
607
updateColorBuffer(HandleType p_colorbuffer,int x,int y,int width,int height,GLenum format,GLenum type,void * pixels)608 bool FrameBuffer::updateColorBuffer(HandleType p_colorbuffer,
609 int x, int y, int width, int height,
610 GLenum format, GLenum type, void *pixels)
611 {
612 android::Mutex::Autolock mutex(m_lock);
613
614 ColorBufferMap::iterator c( m_colorbuffers.find(p_colorbuffer) );
615 if (c == m_colorbuffers.end()) {
616 // bad colorbuffer handle
617 return false;
618 }
619
620 (*c).second.cb->subUpdate(x, y, width, height, format, type, pixels);
621
622 return true;
623 }
624
bindColorBufferToTexture(HandleType p_colorbuffer)625 bool FrameBuffer::bindColorBufferToTexture(HandleType p_colorbuffer)
626 {
627 android::Mutex::Autolock mutex(m_lock);
628
629 ColorBufferMap::iterator c( m_colorbuffers.find(p_colorbuffer) );
630 if (c == m_colorbuffers.end()) {
631 // bad colorbuffer handle
632 return false;
633 }
634
635 return (*c).second.cb->bindToTexture();
636 }
637
bindColorBufferToRenderbuffer(HandleType p_colorbuffer)638 bool FrameBuffer::bindColorBufferToRenderbuffer(HandleType p_colorbuffer)
639 {
640 android::Mutex::Autolock mutex(m_lock);
641
642 ColorBufferMap::iterator c( m_colorbuffers.find(p_colorbuffer) );
643 if (c == m_colorbuffers.end()) {
644 // bad colorbuffer handle
645 return false;
646 }
647
648 return (*c).second.cb->bindToRenderbuffer();
649 }
650
bindContext(HandleType p_context,HandleType p_drawSurface,HandleType p_readSurface)651 bool FrameBuffer::bindContext(HandleType p_context,
652 HandleType p_drawSurface,
653 HandleType p_readSurface)
654 {
655 android::Mutex::Autolock mutex(m_lock);
656
657 WindowSurfacePtr draw(NULL), read(NULL);
658 RenderContextPtr ctx(NULL);
659
660 //
661 // if this is not an unbind operation - make sure all handles are good
662 //
663 if (p_context || p_drawSurface || p_readSurface) {
664 RenderContextMap::iterator r( m_contexts.find(p_context) );
665 if (r == m_contexts.end()) {
666 // bad context handle
667 return false;
668 }
669
670 ctx = (*r).second;
671 WindowSurfaceMap::iterator w( m_windows.find(p_drawSurface) );
672 if (w == m_windows.end()) {
673 // bad surface handle
674 return false;
675 }
676 draw = (*w).second;
677
678 if (p_readSurface != p_drawSurface) {
679 WindowSurfaceMap::iterator w( m_windows.find(p_readSurface) );
680 if (w == m_windows.end()) {
681 // bad surface handle
682 return false;
683 }
684 read = (*w).second;
685 }
686 else {
687 read = draw;
688 }
689 }
690
691 if (!s_egl.eglMakeCurrent(m_eglDisplay,
692 draw ? draw->getEGLSurface() : EGL_NO_SURFACE,
693 read ? read->getEGLSurface() : EGL_NO_SURFACE,
694 ctx ? ctx->getEGLContext() : EGL_NO_CONTEXT)) {
695 ERR("eglMakeCurrent failed\n");
696 return false;
697 }
698
699 //
700 // Bind the surface(s) to the context
701 //
702 RenderThreadInfo *tinfo = RenderThreadInfo::get();
703 WindowSurfacePtr bindDraw, bindRead;
704 if (draw.Ptr() == NULL && read.Ptr() == NULL) {
705 // Unbind the current read and draw surfaces from the context
706 bindDraw = tinfo->currDrawSurf;
707 bindRead = tinfo->currReadSurf;
708 } else {
709 bindDraw = draw;
710 bindRead = read;
711 }
712
713 if (bindDraw.Ptr() != NULL && bindRead.Ptr() != NULL) {
714 if (bindDraw.Ptr() != bindRead.Ptr()) {
715 bindDraw->bind(ctx, SURFACE_BIND_DRAW);
716 bindRead->bind(ctx, SURFACE_BIND_READ);
717 }
718 else {
719 bindDraw->bind(ctx, SURFACE_BIND_READDRAW);
720 }
721 }
722
723 //
724 // update thread info with current bound context
725 //
726 tinfo->currContext = ctx;
727 tinfo->currDrawSurf = draw;
728 tinfo->currReadSurf = read;
729 if (ctx) {
730 if (ctx->isGL2()) tinfo->m_gl2Dec.setContextData(&ctx->decoderContextData());
731 else tinfo->m_glDec.setContextData(&ctx->decoderContextData());
732 }
733 else {
734 tinfo->m_glDec.setContextData(NULL);
735 tinfo->m_gl2Dec.setContextData(NULL);
736 }
737 return true;
738 }
739
740 //
741 // The framebuffer lock should be held when calling this function !
742 //
bind_locked()743 bool FrameBuffer::bind_locked()
744 {
745 EGLContext prevContext = s_egl.eglGetCurrentContext();
746 EGLSurface prevReadSurf = s_egl.eglGetCurrentSurface(EGL_READ);
747 EGLSurface prevDrawSurf = s_egl.eglGetCurrentSurface(EGL_DRAW);
748
749 if (!s_egl.eglMakeCurrent(m_eglDisplay, m_pbufSurface,
750 m_pbufSurface, m_pbufContext)) {
751 ERR("eglMakeCurrent failed\n");
752 return false;
753 }
754
755 m_prevContext = prevContext;
756 m_prevReadSurf = prevReadSurf;
757 m_prevDrawSurf = prevDrawSurf;
758 return true;
759 }
760
bindSubwin_locked()761 bool FrameBuffer::bindSubwin_locked()
762 {
763 EGLContext prevContext = s_egl.eglGetCurrentContext();
764 EGLSurface prevReadSurf = s_egl.eglGetCurrentSurface(EGL_READ);
765 EGLSurface prevDrawSurf = s_egl.eglGetCurrentSurface(EGL_DRAW);
766
767 if (!s_egl.eglMakeCurrent(m_eglDisplay, m_eglSurface,
768 m_eglSurface, m_eglContext)) {
769 ERR("eglMakeCurrent failed\n");
770 return false;
771 }
772
773 //
774 // initialize GL state in eglContext if not yet initilaized
775 //
776 if (!m_eglContextInitialized) {
777 initGLState();
778 m_eglContextInitialized = true;
779 }
780
781 m_prevContext = prevContext;
782 m_prevReadSurf = prevReadSurf;
783 m_prevDrawSurf = prevDrawSurf;
784 return true;
785 }
786
unbind_locked()787 bool FrameBuffer::unbind_locked()
788 {
789 if (!s_egl.eglMakeCurrent(m_eglDisplay, m_prevDrawSurf,
790 m_prevReadSurf, m_prevContext)) {
791 return false;
792 }
793
794 m_prevContext = EGL_NO_CONTEXT;
795 m_prevReadSurf = EGL_NO_SURFACE;
796 m_prevDrawSurf = EGL_NO_SURFACE;
797 return true;
798 }
799
post(HandleType p_colorbuffer,bool needLock)800 bool FrameBuffer::post(HandleType p_colorbuffer, bool needLock)
801 {
802 if (needLock) m_lock.lock();
803 bool ret = false;
804
805 ColorBufferMap::iterator c( m_colorbuffers.find(p_colorbuffer) );
806 if (c != m_colorbuffers.end()) {
807
808 m_lastPostedColorBuffer = p_colorbuffer;
809 if (!m_subWin) {
810 // no subwindow created for the FB output
811 // cannot post the colorbuffer
812 if (needLock) m_lock.unlock();
813 return ret;
814 }
815
816
817 // bind the subwindow eglSurface
818 if (!bindSubwin_locked()) {
819 ERR("FrameBuffer::post eglMakeCurrent failed\n");
820 if (needLock) m_lock.unlock();
821 return false;
822 }
823
824 //
825 // render the color buffer to the window
826 //
827 s_gl.glPushMatrix();
828 s_gl.glRotatef(m_zRot, 0.0f, 0.0f, 1.0f);
829 if (m_zRot != 0.0f) {
830 s_gl.glClear(GL_COLOR_BUFFER_BIT);
831 }
832 ret = (*c).second.cb->post();
833 s_gl.glPopMatrix();
834
835 if (ret) {
836 //
837 // output FPS statistics
838 //
839 if (m_fpsStats) {
840 long long currTime = GetCurrentTimeMS();
841 m_statsNumFrames++;
842 if (currTime - m_statsStartTime >= 1000) {
843 float dt = (float)(currTime - m_statsStartTime) / 1000.0f;
844 printf("FPS: %5.3f\n", (float)m_statsNumFrames / dt);
845 m_statsStartTime = currTime;
846 m_statsNumFrames = 0;
847 }
848 }
849
850 s_egl.eglSwapBuffers(m_eglDisplay, m_eglSurface);
851 }
852
853 // restore previous binding
854 unbind_locked();
855
856 //
857 // Send framebuffer (without FPS overlay) to callback
858 //
859 if (m_onPost) {
860 (*c).second.cb->readback(m_fbImage);
861 m_onPost(m_onPostContext, m_width, m_height, -1,
862 GL_RGBA, GL_UNSIGNED_BYTE, m_fbImage);
863 }
864
865 }
866
867 if (needLock) m_lock.unlock();
868 return ret;
869 }
870
repost()871 bool FrameBuffer::repost()
872 {
873 if (m_lastPostedColorBuffer) {
874 return post( m_lastPostedColorBuffer );
875 }
876 return false;
877 }
878
initGLState()879 void FrameBuffer::initGLState()
880 {
881 s_gl.glMatrixMode(GL_PROJECTION);
882 s_gl.glLoadIdentity();
883 s_gl.glOrthof(-1.0, 1.0, -1.0, 1.0, -1.0, 1.0);
884 s_gl.glMatrixMode(GL_MODELVIEW);
885 s_gl.glLoadIdentity();
886 }
887