blob: a774841ddb6fcc7dfcff0883fc961ae474762ec3 [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>
Mathias Agopian6cf50a72009-08-06 16:05:39 -070029#include <ui/EGLUtils.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080030
31#include <GLES/gl.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070032#include <EGL/egl.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080033#include <EGL/eglext.h>
34
Mathias Agopian076b1cc2009-04-10 14:24:30 -070035#include <pixelflinger/pixelflinger.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080036
37#include "DisplayHardware/DisplayHardware.h"
38
Mathias Agopian076b1cc2009-04-10 14:24:30 -070039#include <hardware/gralloc.h>
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080040
Mathias Agopian1f7bec62010-06-25 18:02:21 -070041#include "GLExtensions.h"
Mathias Agopiana350ff92010-08-10 17:14:02 -070042#include "HWComposer.h"
Mathias Agopian1f7bec62010-06-25 18:02:21 -070043
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080044using namespace android;
45
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080046
47static __attribute__((noinline))
48void checkGLErrors()
49{
Mathias Agopiancbb288b2009-09-07 16:32:45 -070050 do {
51 // there could be more than one error flag
52 GLenum error = glGetError();
53 if (error == GL_NO_ERROR)
54 break;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080055 LOGE("GL error 0x%04x", int(error));
Mathias Agopiancbb288b2009-09-07 16:32:45 -070056 } while(true);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080057}
58
59static __attribute__((noinline))
60void checkEGLErrors(const char* token)
61{
62 EGLint error = eglGetError();
Mathias Agopiancbb288b2009-09-07 16:32:45 -070063 if (error && error != EGL_SUCCESS) {
64 LOGE("%s: EGL error 0x%04x (%s)",
Mathias Agopian0928e312009-08-07 16:38:10 -070065 token, int(error), EGLUtils::strerror(error));
Mathias Agopiancbb288b2009-09-07 16:32:45 -070066 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080067}
68
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080069/*
70 * Initialize the display to the specified values.
71 *
72 */
73
74DisplayHardware::DisplayHardware(
75 const sp<SurfaceFlinger>& flinger,
76 uint32_t dpy)
Mathias Agopian1f7bec62010-06-25 18:02:21 -070077 : DisplayHardwareBase(flinger, dpy),
Mathias Agopiana350ff92010-08-10 17:14:02 -070078 mFlags(0), mHwc(0)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080079{
80 init(dpy);
81}
82
83DisplayHardware::~DisplayHardware()
84{
85 fini();
86}
87
88float DisplayHardware::getDpiX() const { return mDpiX; }
89float DisplayHardware::getDpiY() const { return mDpiY; }
90float DisplayHardware::getDensity() const { return mDensity; }
91float DisplayHardware::getRefreshRate() const { return mRefreshRate; }
92int DisplayHardware::getWidth() const { return mWidth; }
93int DisplayHardware::getHeight() const { return mHeight; }
94PixelFormat DisplayHardware::getFormat() const { return mFormat; }
Mathias Agopianca99fb82010-04-14 16:43:44 -070095uint32_t DisplayHardware::getMaxTextureSize() const { return mMaxTextureSize; }
Mathias Agopian3d64e732011-04-18 15:59:24 -070096
97uint32_t DisplayHardware::getMaxViewportDims() const {
98 return mMaxViewportDims[0] < mMaxViewportDims[1] ?
99 mMaxViewportDims[0] : mMaxViewportDims[1];
100}
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800101
102void DisplayHardware::init(uint32_t dpy)
103{
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700104 mNativeWindow = new FramebufferNativeWindow();
Mathias Agopian0928e312009-08-07 16:38:10 -0700105 framebuffer_device_t const * fbDev = mNativeWindow->getDevice();
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700106 mDpiX = mNativeWindow->xdpi;
107 mDpiY = mNativeWindow->ydpi;
108 mRefreshRate = fbDev->fps;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700109
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700110 EGLint w, h, dummy;
111 EGLint numConfigs=0;
112 EGLSurface surface;
113 EGLContext context;
114
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800115 // initialize EGL
Mathias Agopiana5b02e02009-09-04 18:49:03 -0700116 EGLint attribs[] = {
Mathias Agopian0928e312009-08-07 16:38:10 -0700117 EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
Mathias Agopiana5b02e02009-09-04 18:49:03 -0700118 EGL_NONE, 0,
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800119 EGL_NONE
120 };
Mathias Agopiana5b02e02009-09-04 18:49:03 -0700121
122 // debug: disable h/w rendering
123 char property[PROPERTY_VALUE_MAX];
124 if (property_get("debug.sf.hw", property, NULL) > 0) {
125 if (atoi(property) == 0) {
126 LOGW("H/W composition disabled");
127 attribs[2] = EGL_CONFIG_CAVEAT;
128 attribs[3] = EGL_SLOW_CONFIG;
129 }
130 }
131
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800132 // TODO: all the extensions below should be queried through
133 // eglGetProcAddress().
134
135 EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
136 eglInitialize(display, NULL, NULL);
137 eglGetConfigs(display, NULL, 0, &numConfigs);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700138
Mathias Agopian6cf50a72009-08-06 16:05:39 -0700139 EGLConfig config;
Mathias Agopian0928e312009-08-07 16:38:10 -0700140 status_t err = EGLUtils::selectConfigForNativeWindow(
141 display, attribs, mNativeWindow.get(), &config);
Mathias Agopian6cf50a72009-08-06 16:05:39 -0700142 LOGE_IF(err, "couldn't find an EGLConfig matching the screen format");
143
Mathias Agopian0928e312009-08-07 16:38:10 -0700144 EGLint r,g,b,a;
145 eglGetConfigAttrib(display, config, EGL_RED_SIZE, &r);
146 eglGetConfigAttrib(display, config, EGL_GREEN_SIZE, &g);
147 eglGetConfigAttrib(display, config, EGL_BLUE_SIZE, &b);
148 eglGetConfigAttrib(display, config, EGL_ALPHA_SIZE, &a);
149
Mathias Agopian1e16b132009-05-07 17:40:23 -0700150 if (mNativeWindow->isUpdateOnDemand()) {
Mathias Agopian95a666b2009-09-24 14:57:26 -0700151 mFlags |= PARTIAL_UPDATES;
Mathias Agopian1e16b132009-05-07 17:40:23 -0700152 }
153
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800154 if (eglGetConfigAttrib(display, config, EGL_CONFIG_CAVEAT, &dummy) == EGL_TRUE) {
155 if (dummy == EGL_SLOW_CONFIG)
156 mFlags |= SLOW_CONFIG;
157 }
158
159 /*
160 * Create our main surface
161 */
162
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700163 surface = eglCreateWindowSurface(display, config, mNativeWindow.get(), NULL);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700164 eglQuerySurface(display, surface, EGL_WIDTH, &mWidth);
165 eglQuerySurface(display, surface, EGL_HEIGHT, &mHeight);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800166
Mathias Agopian95a666b2009-09-24 14:57:26 -0700167 if (mFlags & PARTIAL_UPDATES) {
168 // if we have partial updates, we definitely don't need to
169 // preserve the backbuffer, which may be costly.
Mathias Agopian0928bee2009-09-16 20:15:42 -0700170 eglSurfaceAttrib(display, surface,
171 EGL_SWAP_BEHAVIOR, EGL_BUFFER_DESTROYED);
172 }
173
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800174 if (eglQuerySurface(display, surface, EGL_SWAP_BEHAVIOR, &dummy) == EGL_TRUE) {
175 if (dummy == EGL_BUFFER_PRESERVED) {
176 mFlags |= BUFFER_PRESERVED;
177 }
178 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800179
David 'Digit' Turnerae71acc2009-06-19 04:41:12 +0200180 /* Read density from build-specific ro.sf.lcd_density property
Mathias Agopian24e5f522009-08-12 21:18:15 -0700181 * except if it is overridden by qemu.sf.lcd_density.
David 'Digit' Turnerae71acc2009-06-19 04:41:12 +0200182 */
183 if (property_get("qemu.sf.lcd_density", property, NULL) <= 0) {
184 if (property_get("ro.sf.lcd_density", property, NULL) <= 0) {
185 LOGW("ro.sf.lcd_density not defined, using 160 dpi by default.");
186 strcpy(property, "160");
David 'Digit' Turner694e10b2009-06-18 04:30:32 +0200187 }
David 'Digit' Turner31469e12009-07-29 00:38:58 +0200188 } else {
189 /* for the emulator case, reset the dpi values too */
190 mDpiX = mDpiY = atoi(property);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800191 }
192 mDensity = atoi(property) * (1.0f/160.0f);
193
194
195 /*
196 * Create our OpenGL ES context
197 */
198
Mathias Agopian67226812010-10-11 17:54:43 -0700199
200 EGLint contextAttributes[] = {
201#ifdef EGL_IMG_context_priority
202#ifdef HAS_CONTEXT_PRIORITY
203#warning "using EGL_IMG_context_priority"
204 EGL_CONTEXT_PRIORITY_LEVEL_IMG, EGL_CONTEXT_PRIORITY_HIGH_IMG,
205#endif
206#endif
207 EGL_NONE, EGL_NONE
208 };
209 context = eglCreateContext(display, config, NULL, contextAttributes);
210
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800211 mDisplay = display;
212 mConfig = config;
213 mSurface = surface;
214 mContext = context;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700215 mFormat = fbDev->format;
216 mPageFlipCount = 0;
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700217
218 /*
219 * Gather OpenGL ES extensions
220 */
221
222 eglMakeCurrent(display, surface, surface, context);
223
224 GLExtensions& extensions(GLExtensions::getInstance());
225 extensions.initWithGLStrings(
226 glGetString(GL_VENDOR),
227 glGetString(GL_RENDERER),
228 glGetString(GL_VERSION),
229 glGetString(GL_EXTENSIONS),
230 eglQueryString(display, EGL_VENDOR),
231 eglQueryString(display, EGL_VERSION),
232 eglQueryString(display, EGL_EXTENSIONS));
233
234 glGetIntegerv(GL_MAX_TEXTURE_SIZE, &mMaxTextureSize);
Mathias Agopian3d64e732011-04-18 15:59:24 -0700235 glGetIntegerv(GL_MAX_VIEWPORT_DIMS, mMaxViewportDims);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700236
237
238#ifdef EGL_ANDROID_swap_rectangle
239 if (extensions.hasExtension("EGL_ANDROID_swap_rectangle")) {
240 if (eglSetSwapRectangleANDROID(display, surface,
241 0, 0, mWidth, mHeight) == EGL_TRUE) {
242 // This could fail if this extension is not supported by this
243 // specific surface (of config)
244 mFlags |= SWAP_RECTANGLE;
245 }
246 }
247 // when we have the choice between PARTIAL_UPDATES and SWAP_RECTANGLE
248 // choose PARTIAL_UPDATES, which should be more efficient
249 if (mFlags & PARTIAL_UPDATES)
250 mFlags &= ~SWAP_RECTANGLE;
251#endif
252
253 LOGI("EGL informations:");
254 LOGI("# of configs : %d", numConfigs);
255 LOGI("vendor : %s", extensions.getEglVendor());
256 LOGI("version : %s", extensions.getEglVersion());
257 LOGI("extensions: %s", extensions.getEglExtension());
258 LOGI("Client API: %s", eglQueryString(display, EGL_CLIENT_APIS)?:"Not Supported");
259 LOGI("EGLSurface: %d-%d-%d-%d, config=%p", r, g, b, a, config);
260
261 LOGI("OpenGL informations:");
262 LOGI("vendor : %s", extensions.getVendor());
263 LOGI("renderer : %s", extensions.getRenderer());
264 LOGI("version : %s", extensions.getVersion());
265 LOGI("extensions: %s", extensions.getExtension());
266 LOGI("GL_MAX_TEXTURE_SIZE = %d", mMaxTextureSize);
Mathias Agopian3d64e732011-04-18 15:59:24 -0700267 LOGI("GL_MAX_VIEWPORT_DIMS = %d x %d", mMaxViewportDims[0], mMaxViewportDims[1]);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700268 LOGI("flags = %08x", mFlags);
269
270 // Unbind the context from this thread
271 eglMakeCurrent(display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
Mathias Agopiana350ff92010-08-10 17:14:02 -0700272
273
274 // initialize the H/W composer
275 mHwc = new HWComposer();
276 if (mHwc->initCheck() == NO_ERROR) {
277 mHwc->setFrameBuffer(mDisplay, mSurface);
278 }
279}
280
281HWComposer& DisplayHardware::getHwComposer() const {
282 return *mHwc;
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}
297
298void DisplayHardware::releaseScreen() const
299{
300 DisplayHardwareBase::releaseScreen();
Antti Hatalaf5f27122010-09-09 02:33:05 -0700301 if (mHwc->initCheck() == NO_ERROR) {
302 mHwc->release();
303 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800304}
305
306void DisplayHardware::acquireScreen() const
307{
308 DisplayHardwareBase::acquireScreen();
309}
310
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800311uint32_t DisplayHardware::getPageFlipCount() const {
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700312 return mPageFlipCount;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800313}
314
Mathias Agopian74faca22009-09-17 16:18:16 -0700315status_t DisplayHardware::compositionComplete() const {
316 return mNativeWindow->compositionComplete();
317}
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800318
Mathias Agopian35b48d12010-09-13 22:57:58 -0700319int DisplayHardware::getCurrentBufferIndex() const {
320 return mNativeWindow->getCurrentBufferIndex();
321}
322
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800323void DisplayHardware::flip(const Region& dirty) const
324{
325 checkGLErrors();
326
327 EGLDisplay dpy = mDisplay;
328 EGLSurface surface = mSurface;
329
Mathias Agopian5e78e092009-06-11 17:19:54 -0700330#ifdef EGL_ANDROID_swap_rectangle
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700331 if (mFlags & SWAP_RECTANGLE) {
Mathias Agopianb8a55602009-06-26 19:06:36 -0700332 const Region newDirty(dirty.intersect(bounds()));
333 const Rect b(newDirty.getBounds());
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700334 eglSetSwapRectangleANDROID(dpy, surface,
335 b.left, b.top, b.width(), b.height());
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800336 }
Mathias Agopian5e78e092009-06-11 17:19:54 -0700337#endif
338
Mathias Agopian95a666b2009-09-24 14:57:26 -0700339 if (mFlags & PARTIAL_UPDATES) {
Mathias Agopian29d06ac2009-06-29 18:49:56 -0700340 mNativeWindow->setUpdateRectangle(dirty.getBounds());
Mathias Agopian1e16b132009-05-07 17:40:23 -0700341 }
342
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700343 mPageFlipCount++;
Mathias Agopiana350ff92010-08-10 17:14:02 -0700344
345 if (mHwc->initCheck() == NO_ERROR) {
346 mHwc->commit();
347 } else {
348 eglSwapBuffers(dpy, surface);
349 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800350 checkEGLErrors("eglSwapBuffers");
351
352 // for debugging
353 //glClearColor(1,0,0,0);
354 //glClear(GL_COLOR_BUFFER_BIT);
355}
356
357uint32_t DisplayHardware::getFlags() const
358{
359 return mFlags;
360}
361
362void DisplayHardware::makeCurrent() const
363{
364 eglMakeCurrent(mDisplay, mSurface, mSurface, mContext);
365}
Erik Gilling1d21a9c2010-12-01 16:38:01 -0800366
367void DisplayHardware::dump(String8& res) const
368{
369 mNativeWindow->dump(res);
370}