blob: 21f87e3778948de30466f1b57b885601791ace0f [file] [log] [blame]
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007 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
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080017#include <stdlib.h>
18#include <stdio.h>
19#include <string.h>
20#include <math.h>
21
22#include <cutils/properties.h>
23
Mathias Agopian076b1cc2009-04-10 14:24:30 -070024#include <utils/RefBase.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080025#include <utils/Log.h>
26
Mathias Agopian076b1cc2009-04-10 14:24:30 -070027#include <ui/PixelFormat.h>
Mathias Agopian0926f502009-05-04 14:17:04 -070028#include <ui/FramebufferNativeWindow.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080029
30#include <GLES/gl.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070031#include <EGL/egl.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080032#include <EGL/eglext.h>
33
Mathias Agopian076b1cc2009-04-10 14:24:30 -070034#include <pixelflinger/pixelflinger.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080035
36#include "DisplayHardware/DisplayHardware.h"
37
38#include <hardware/copybit.h>
39#include <hardware/overlay.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070040#include <hardware/gralloc.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080041
42using namespace android;
43
44static __attribute__((noinline))
45const char *egl_strerror(EGLint err)
46{
47 switch (err){
48 case EGL_SUCCESS: return "EGL_SUCCESS";
49 case EGL_NOT_INITIALIZED: return "EGL_NOT_INITIALIZED";
50 case EGL_BAD_ACCESS: return "EGL_BAD_ACCESS";
51 case EGL_BAD_ALLOC: return "EGL_BAD_ALLOC";
52 case EGL_BAD_ATTRIBUTE: return "EGL_BAD_ATTRIBUTE";
53 case EGL_BAD_CONFIG: return "EGL_BAD_CONFIG";
54 case EGL_BAD_CONTEXT: return "EGL_BAD_CONTEXT";
55 case EGL_BAD_CURRENT_SURFACE: return "EGL_BAD_CURRENT_SURFACE";
56 case EGL_BAD_DISPLAY: return "EGL_BAD_DISPLAY";
57 case EGL_BAD_MATCH: return "EGL_BAD_MATCH";
58 case EGL_BAD_NATIVE_PIXMAP: return "EGL_BAD_NATIVE_PIXMAP";
59 case EGL_BAD_NATIVE_WINDOW: return "EGL_BAD_NATIVE_WINDOW";
60 case EGL_BAD_PARAMETER: return "EGL_BAD_PARAMETER";
61 case EGL_BAD_SURFACE: return "EGL_BAD_SURFACE";
62 case EGL_CONTEXT_LOST: return "EGL_CONTEXT_LOST";
63 default: return "UNKNOWN";
64 }
65}
66
67static __attribute__((noinline))
68void checkGLErrors()
69{
70 GLenum error = glGetError();
71 if (error != GL_NO_ERROR)
72 LOGE("GL error 0x%04x", int(error));
73}
74
75static __attribute__((noinline))
76void checkEGLErrors(const char* token)
77{
78 EGLint error = eglGetError();
79 // GLESonGL seems to be returning 0 when there is no errors?
80 if (error && error != EGL_SUCCESS)
81 LOGE("%s error 0x%04x (%s)",
82 token, int(error), egl_strerror(error));
83}
84
85
86/*
87 * Initialize the display to the specified values.
88 *
89 */
90
91DisplayHardware::DisplayHardware(
92 const sp<SurfaceFlinger>& flinger,
93 uint32_t dpy)
94 : DisplayHardwareBase(flinger, dpy)
95{
96 init(dpy);
97}
98
99DisplayHardware::~DisplayHardware()
100{
101 fini();
102}
103
104float DisplayHardware::getDpiX() const { return mDpiX; }
105float DisplayHardware::getDpiY() const { return mDpiY; }
106float DisplayHardware::getDensity() const { return mDensity; }
107float DisplayHardware::getRefreshRate() const { return mRefreshRate; }
108int DisplayHardware::getWidth() const { return mWidth; }
109int DisplayHardware::getHeight() const { return mHeight; }
110PixelFormat DisplayHardware::getFormat() const { return mFormat; }
111
112void DisplayHardware::init(uint32_t dpy)
113{
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700114 hw_module_t const* module;
115
116 mNativeWindow = new FramebufferNativeWindow();
117
118 mOverlayEngine = NULL;
119 if (hw_get_module(OVERLAY_HARDWARE_MODULE_ID, &module) == 0) {
120 overlay_control_open(module, &mOverlayEngine);
121 }
122
123 framebuffer_device_t const * fbDev = mNativeWindow->getDevice();
124
125 PixelFormatInfo fbFormatInfo;
126 getPixelFormatInfo(PixelFormat(fbDev->format), &fbFormatInfo);
127
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800128 // initialize EGL
129 const EGLint attribs[] = {
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700130 EGL_BUFFER_SIZE, fbFormatInfo.bitsPerPixel,
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800131 EGL_DEPTH_SIZE, 0,
132 EGL_NONE
133 };
134 EGLint w, h, dummy;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700135 EGLint numConfigs=0, n=0;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800136 EGLSurface surface;
137 EGLContext context;
138 mFlags = 0;
139
140 // TODO: all the extensions below should be queried through
141 // eglGetProcAddress().
142
143 EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
144 eglInitialize(display, NULL, NULL);
145 eglGetConfigs(display, NULL, 0, &numConfigs);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700146
147 // Get all the "potential match" configs...
148 EGLConfig* const configs = new EGLConfig[numConfigs];
149 eglChooseConfig(display, attribs, configs, numConfigs, &n);
150 LOGE_IF(n<=0, "no EGLConfig available!");
151 EGLConfig config = configs[0];
152 if (n > 1) {
153 // if there is more than one candidate, go through the list
154 // and pick one that matches our framebuffer format
155 int fbSzA = fbFormatInfo.getSize(PixelFormatInfo::INDEX_ALPHA);
156 int fbSzR = fbFormatInfo.getSize(PixelFormatInfo::INDEX_RED);
157 int fbSzG = fbFormatInfo.getSize(PixelFormatInfo::INDEX_GREEN);
158 int fbSzB = fbFormatInfo.getSize(PixelFormatInfo::INDEX_BLUE);
159 for (int i=0 ; i<n ; i++) {
160 EGLint r,g,b,a;
161 eglGetConfigAttrib(display, configs[i], EGL_RED_SIZE, &r);
162 eglGetConfigAttrib(display, configs[i], EGL_GREEN_SIZE, &g);
163 eglGetConfigAttrib(display, configs[i], EGL_BLUE_SIZE, &b);
164 eglGetConfigAttrib(display, configs[i], EGL_ALPHA_SIZE, &a);
165 if (fbSzA == a && fbSzR == r && fbSzG == g && fbSzB == b) {
166 config = configs[i];
Mathias Agopiane92c7132009-06-03 14:49:08 -0700167 break;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700168 }
169 }
170 }
171 delete [] configs;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800172
173 /*
174 * Gather EGL extensions
175 */
176
177 const char* const egl_extensions = eglQueryString(
178 display, EGL_EXTENSIONS);
179
180 LOGI("EGL informations:");
181 LOGI("# of configs : %d", numConfigs);
182 LOGI("vendor : %s", eglQueryString(display, EGL_VENDOR));
183 LOGI("version : %s", eglQueryString(display, EGL_VERSION));
184 LOGI("extensions: %s", egl_extensions);
185 LOGI("Client API: %s", eglQueryString(display, EGL_CLIENT_APIS)?:"Not Supported");
186
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800187
Mathias Agopian1e16b132009-05-07 17:40:23 -0700188 if (mNativeWindow->isUpdateOnDemand()) {
189 mFlags |= UPDATE_ON_DEMAND;
190 }
191
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800192 if (eglGetConfigAttrib(display, config, EGL_CONFIG_CAVEAT, &dummy) == EGL_TRUE) {
193 if (dummy == EGL_SLOW_CONFIG)
194 mFlags |= SLOW_CONFIG;
195 }
196
197 /*
198 * Create our main surface
199 */
200
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700201 surface = eglCreateWindowSurface(display, config, mNativeWindow.get(), NULL);
Mathias Agopian5e78e092009-06-11 17:19:54 -0700202 checkEGLErrors("eglCreateWindowSurface");
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800203
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800204 if (eglQuerySurface(display, surface, EGL_SWAP_BEHAVIOR, &dummy) == EGL_TRUE) {
205 if (dummy == EGL_BUFFER_PRESERVED) {
206 mFlags |= BUFFER_PRESERVED;
207 }
208 }
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700209
Mathias Agopian5e78e092009-06-11 17:19:54 -0700210#ifdef EGL_ANDROID_swap_rectangle
Mathias Agopiane6bf8b32009-05-06 23:47:08 -0700211 if (strstr(egl_extensions, "EGL_ANDROID_swap_rectangle")) {
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700212 mFlags |= SWAP_RECTANGLE;
213 }
Mathias Agopian2dd67272009-06-29 18:53:53 -0700214 // when we have the choice between UPDATE_ON_DEMAND and SWAP_RECTANGLE
215 // choose UPDATE_ON_DEMAND, which is more efficient
216 if (mFlags & UPDATE_ON_DEMAND)
217 mFlags &= ~SWAP_RECTANGLE;
Mathias Agopian5e78e092009-06-11 17:19:54 -0700218#endif
Mathias Agopian2dd67272009-06-29 18:53:53 -0700219
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700220
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700221 mDpiX = mNativeWindow->xdpi;
Mathias Agopian80d7a762009-07-09 22:11:57 -0700222 mDpiY = mNativeWindow->ydpi;
Mathias Agopian1e16b132009-05-07 17:40:23 -0700223 mRefreshRate = fbDev->fps;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800224
225 char property[PROPERTY_VALUE_MAX];
David 'Digit' Turnerae71acc2009-06-19 04:41:12 +0200226 /* Read density from build-specific ro.sf.lcd_density property
227 * except if it is overriden by qemu.sf.lcd_density.
228 */
229 if (property_get("qemu.sf.lcd_density", property, NULL) <= 0) {
230 if (property_get("ro.sf.lcd_density", property, NULL) <= 0) {
231 LOGW("ro.sf.lcd_density not defined, using 160 dpi by default.");
232 strcpy(property, "160");
David 'Digit' Turner694e10b2009-06-18 04:30:32 +0200233 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800234 }
235 mDensity = atoi(property) * (1.0f/160.0f);
236
237
238 /*
239 * Create our OpenGL ES context
240 */
241
242 context = eglCreateContext(display, config, NULL, NULL);
243 //checkEGLErrors("eglCreateContext");
244
245 eglQuerySurface(display, surface, EGL_WIDTH, &mWidth);
246 eglQuerySurface(display, surface, EGL_HEIGHT, &mHeight);
247
248
249 /*
250 * Gather OpenGL ES extensions
251 */
252
253 eglMakeCurrent(display, surface, surface, context);
254 const char* const gl_extensions = (const char*)glGetString(GL_EXTENSIONS);
255 LOGI("OpenGL informations:");
256 LOGI("vendor : %s", glGetString(GL_VENDOR));
257 LOGI("renderer : %s", glGetString(GL_RENDERER));
258 LOGI("version : %s", glGetString(GL_VERSION));
259 LOGI("extensions: %s", gl_extensions);
260
261 if (strstr(gl_extensions, "GL_ARB_texture_non_power_of_two")) {
262 mFlags |= NPOT_EXTENSION;
263 }
264 if (strstr(gl_extensions, "GL_OES_draw_texture")) {
265 mFlags |= DRAW_TEXTURE_EXTENSION;
266 }
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700267 if (strstr( gl_extensions, "GL_OES_EGL_image") &&
Mathias Agopiane6bf8b32009-05-06 23:47:08 -0700268 (strstr(egl_extensions, "EGL_KHR_image_base") ||
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700269 strstr(egl_extensions, "EGL_KHR_image")) &&
270 strstr(egl_extensions, "EGL_ANDROID_image_native_buffer")) {
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800271 mFlags |= DIRECT_TEXTURE;
272 }
273
274 // Unbind the context from this thread
275 eglMakeCurrent(display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
276
277 mDisplay = display;
278 mConfig = config;
279 mSurface = surface;
280 mContext = context;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700281 mFormat = fbDev->format;
282 mPageFlipCount = 0;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800283}
284
285/*
286 * Clean up. Throw out our local state.
287 *
288 * (It's entirely possible we'll never get here, since this is meant
289 * for real hardware, which doesn't restart.)
290 */
291
292void DisplayHardware::fini()
293{
294 eglMakeCurrent(mDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
295 eglTerminate(mDisplay);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800296 overlay_control_close(mOverlayEngine);
297}
298
299void DisplayHardware::releaseScreen() const
300{
301 DisplayHardwareBase::releaseScreen();
302}
303
304void DisplayHardware::acquireScreen() const
305{
306 DisplayHardwareBase::acquireScreen();
307}
308
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800309uint32_t DisplayHardware::getPageFlipCount() const {
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700310 return mPageFlipCount;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800311}
312
313/*
314 * "Flip" the front and back buffers.
315 */
316
317void DisplayHardware::flip(const Region& dirty) const
318{
319 checkGLErrors();
320
321 EGLDisplay dpy = mDisplay;
322 EGLSurface surface = mSurface;
323
Mathias Agopian5e78e092009-06-11 17:19:54 -0700324#ifdef EGL_ANDROID_swap_rectangle
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700325 if (mFlags & SWAP_RECTANGLE) {
Mathias Agopianb8a55602009-06-26 19:06:36 -0700326 const Region newDirty(dirty.intersect(bounds()));
327 const Rect b(newDirty.getBounds());
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700328 eglSetSwapRectangleANDROID(dpy, surface,
329 b.left, b.top, b.width(), b.height());
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800330 }
Mathias Agopian5e78e092009-06-11 17:19:54 -0700331#endif
332
Mathias Agopian1e16b132009-05-07 17:40:23 -0700333 if (mFlags & UPDATE_ON_DEMAND) {
Mathias Agopian29d06ac2009-06-29 18:49:56 -0700334 mNativeWindow->setUpdateRectangle(dirty.getBounds());
Mathias Agopian1e16b132009-05-07 17:40:23 -0700335 }
336
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700337 mPageFlipCount++;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800338 eglSwapBuffers(dpy, surface);
339 checkEGLErrors("eglSwapBuffers");
340
341 // for debugging
342 //glClearColor(1,0,0,0);
343 //glClear(GL_COLOR_BUFFER_BIT);
344}
345
346uint32_t DisplayHardware::getFlags() const
347{
348 return mFlags;
349}
350
351void DisplayHardware::makeCurrent() const
352{
353 eglMakeCurrent(mDisplay, mSurface, mSurface, mContext);
354}