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
17 #include <stdio.h>
18 #include <stdlib.h>
19 #include <assert.h>
20 #include "RenderingThread.h"
21 #include "Renderer.h"
22
23 // include operating-system dependent windowing system impelemntation
24 #ifdef _WIN32
25 # error "WINDOWS IS NOT SUPPORTED AT THE MOMENT"
26 #elif defined __APPLE__
27 # error "Apple OS-X IS NOT SUPPORTED"
28 #elif defined (__unix__)
29 # include "X11Windowing.h"
30 #endif
31
32
33
34
35
36 Renderer * Renderer::m_instance = NULL;
37
instance()38 Renderer * Renderer::instance()
39 {
40 if (m_instance == NULL) m_instance = new Renderer;
41 return m_instance;
42 }
43
Renderer()44 Renderer::Renderer()
45 {
46 // Unix specific, use your platform specific windowing implementation
47 #ifdef __unix__
48 m_nw = new X11Windowing;
49 #endif
50
51 m_dpy = eglGetDisplay(m_nw->getNativeDisplay());
52 EGLint major, minor;
53 eglInitialize(m_dpy, &major, &minor);
54 fprintf(stderr, "egl initialized : %d.%d\n", major, minor);
55 }
56
createSurface(RenderingThread * thread,const ClientHandle & handle)57 int Renderer::createSurface(RenderingThread *thread, const ClientHandle & handle)
58 {
59 emugl::Mutex::AutoLock(this->m_mutex);
60
61 assert(m_surfaces.find(handle) == m_surfaces.end());
62 if (handle.handle == 0) {
63 fprintf(stderr, "trying to create surface for EGL_NO_SURFACE !!!\n");
64 return -1;
65 } else {
66 RendererSurface *surface = RendererSurface::create(m_dpy, RendererSurface::CONFIG_DEPTH, m_nw);
67 if (surface == NULL) {
68 printf("failed to create surface !!\n");
69 return -1;
70 }
71 m_surfaces.insert(SurfaceMap::value_type(handle, surface));
72 }
73 return 0;
74 }
75
destroySurface(RenderingThread * thread,const ClientHandle & handle)76 int Renderer::destroySurface(RenderingThread *thread, const ClientHandle &handle)
77 {
78 emugl::Mutex::AutoLock(this->m_mutex);
79
80 SurfaceMap::iterator i = m_surfaces.find(handle);
81 if (i == m_surfaces.end()) {
82 printf("removing surface that doesn't exists\n");
83 return -1;
84 }
85 if (i->second->destroy(m_nw)) {
86 m_surfaces.erase(handle);
87 }
88 return 0;
89 }
90
createContext(RenderingThread * thread,const ClientHandle & handle,ClientHandle shareCtx,int version)91 int Renderer::createContext(RenderingThread *thread, const ClientHandle &handle, ClientHandle shareCtx, int version)
92 {
93 emugl::Mutex::AutoLock(this->m_mutex);
94
95 assert(m_ctxs.find(handle) == m_ctxs.end());
96 RendererContext *shared = NULL;
97 if (shareCtx.handle != 0) {
98 ContextMap::iterator sctx = m_ctxs.find(shareCtx);
99 if (sctx != m_ctxs.end()) {
100 shared = sctx->second;
101 }
102 }
103
104 RendererContext *ctx =
105 RendererContext::create(m_dpy,
106 RendererSurface::getEglConfig(m_dpy, RendererSurface::CONFIG_DEPTH),
107 shared, version);
108 if (ctx == NULL) {
109 fprintf(stderr, "failed to create context\n");
110 return -1;
111 }
112 m_ctxs.insert(ContextMap::value_type(handle, ctx));
113 return 0;
114 }
115
destroyContext(RenderingThread * thread,const ClientHandle & handle)116 int Renderer::destroyContext(RenderingThread *thread, const ClientHandle &handle)
117 {
118 emugl::Mutex::AutoLock(this->m_mutex);
119
120 ContextMap::iterator i = m_ctxs.find(handle);
121 if (i == m_ctxs.end()) {
122 printf("removing context that doesn't exists\n");
123 return -1;
124 }
125 if (i->second->destroy()) {
126 m_ctxs.erase(handle);
127 }
128 return 0;
129 }
130
makeCurrent(RenderingThread * thread,const ClientHandle & drawSurface,const ClientHandle & readSurface,const ClientHandle & ctx)131 int Renderer::makeCurrent(RenderingThread *thread,
132 const ClientHandle &drawSurface,
133 const ClientHandle &readSurface,
134 const ClientHandle & ctx)
135 {
136 emugl::Mutex::AutoLock(this->m_mutex);
137
138 RendererContext *currentContext = thread->currentContext();
139
140 ContextMap::iterator c = m_ctxs.find(ctx);
141 EGLContext eglContext;
142 if (ctx.handle != 0 && c != m_ctxs.end()) {
143 if (c->second != currentContext) {
144 // new context is set
145 if (currentContext != NULL) currentContext->unref();
146 c->second->ref();
147 eglContext = c->second->eglContext();
148 thread->setCurrentContext(c->second);
149 thread->glDecoder().setContextData(&c->second->decoderContextData());
150 thread->gl2Decoder().setContextData(&c->second->decoderContextData());
151 } else {
152 // same context is already set
153 eglContext = c->second->eglContext();
154 }
155 } else {
156 eglContext = EGL_NO_CONTEXT;
157 if (currentContext != NULL) currentContext->unref();
158 thread->setCurrentContext(NULL);
159 thread->glDecoder().setContextData(NULL);
160 thread->gl2Decoder().setContextData(NULL);
161 }
162
163 EGLSurface draw = EGL_NO_SURFACE;
164 EGLSurface read = EGL_NO_SURFACE;
165 SurfaceMap::iterator i;
166 i = m_surfaces.find(drawSurface); if (i != m_surfaces.end()) draw = i->second->eglSurface();
167 i = m_surfaces.find(readSurface); if (i != m_surfaces.end()) read = i->second->eglSurface();
168
169 return eglMakeCurrent(m_dpy, draw, read, eglContext);
170 }
171
swapBuffers(RenderingThread * thread,const ClientHandle & surface)172 int Renderer::swapBuffers(RenderingThread *thread,
173 const ClientHandle &surface)
174 {
175 emugl::Mutex::AutoLock(this->m_mutex);
176
177 SurfaceMap::iterator s = m_surfaces.find(surface);
178 if (s == m_surfaces.end()) {
179 fprintf(stderr, "swapping buffers for non existing surface\n");
180 return -1;
181 }
182 return eglSwapBuffers(m_dpy, s->second->eglSurface());
183 }
184