blob: cf131b11db0b36fce68848b569ade75939a35a52 [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 Agopianc7d14e22011-08-01 16:32:21 -070043#include "SurfaceFlinger.h"
Mathias Agopian1f7bec62010-06-25 18:02:21 -070044
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080045using namespace android;
46
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080047
48static __attribute__((noinline))
49void checkGLErrors()
50{
Mathias Agopiancbb288b2009-09-07 16:32:45 -070051 do {
52 // there could be more than one error flag
53 GLenum error = glGetError();
54 if (error == GL_NO_ERROR)
55 break;
Steve Blocke6f43dd2012-01-06 19:20:56 +000056 ALOGE("GL error 0x%04x", int(error));
Mathias Agopiancbb288b2009-09-07 16:32:45 -070057 } while(true);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080058}
59
60static __attribute__((noinline))
61void checkEGLErrors(const char* token)
62{
63 EGLint error = eglGetError();
Mathias Agopiancbb288b2009-09-07 16:32:45 -070064 if (error && error != EGL_SUCCESS) {
Steve Blocke6f43dd2012-01-06 19:20:56 +000065 ALOGE("%s: EGL error 0x%04x (%s)",
Mathias Agopian0928e312009-08-07 16:38:10 -070066 token, int(error), EGLUtils::strerror(error));
Mathias Agopiancbb288b2009-09-07 16:32:45 -070067 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080068}
69
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080070/*
71 * Initialize the display to the specified values.
72 *
73 */
74
75DisplayHardware::DisplayHardware(
76 const sp<SurfaceFlinger>& flinger,
77 uint32_t dpy)
Mathias Agopian1f7bec62010-06-25 18:02:21 -070078 : DisplayHardwareBase(flinger, dpy),
Mathias Agopianc7d14e22011-08-01 16:32:21 -070079 mFlinger(flinger), mFlags(0), mHwc(0)
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -080080{
81 init(dpy);
82}
83
84DisplayHardware::~DisplayHardware()
85{
86 fini();
87}
88
89float DisplayHardware::getDpiX() const { return mDpiX; }
90float DisplayHardware::getDpiY() const { return mDpiY; }
91float DisplayHardware::getDensity() const { return mDensity; }
92float DisplayHardware::getRefreshRate() const { return mRefreshRate; }
93int DisplayHardware::getWidth() const { return mWidth; }
94int DisplayHardware::getHeight() const { return mHeight; }
95PixelFormat DisplayHardware::getFormat() const { return mFormat; }
Mathias Agopianca99fb82010-04-14 16:43:44 -070096uint32_t DisplayHardware::getMaxTextureSize() const { return mMaxTextureSize; }
Mathias Agopian3d64e732011-04-18 15:59:24 -070097
98uint32_t DisplayHardware::getMaxViewportDims() const {
99 return mMaxViewportDims[0] < mMaxViewportDims[1] ?
100 mMaxViewportDims[0] : mMaxViewportDims[1];
101}
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800102
Mathias Agopian61630912011-07-06 16:35:30 -0700103static status_t selectConfigForPixelFormat(
104 EGLDisplay dpy,
105 EGLint const* attrs,
106 PixelFormat format,
107 EGLConfig* outConfig)
108{
109 EGLConfig config = NULL;
110 EGLint numConfigs = -1, n=0;
111 eglGetConfigs(dpy, NULL, 0, &numConfigs);
112 EGLConfig* const configs = new EGLConfig[numConfigs];
113 eglChooseConfig(dpy, attrs, configs, numConfigs, &n);
114 for (int i=0 ; i<n ; i++) {
115 EGLint nativeVisualId = 0;
116 eglGetConfigAttrib(dpy, configs[i], EGL_NATIVE_VISUAL_ID, &nativeVisualId);
117 if (nativeVisualId>0 && format == nativeVisualId) {
118 *outConfig = configs[i];
119 delete [] configs;
120 return NO_ERROR;
121 }
122 }
123 delete [] configs;
124 return NAME_NOT_FOUND;
125}
126
127
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800128void DisplayHardware::init(uint32_t dpy)
129{
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700130 mNativeWindow = new FramebufferNativeWindow();
Mathias Agopian0928e312009-08-07 16:38:10 -0700131 framebuffer_device_t const * fbDev = mNativeWindow->getDevice();
Mathias Agopian1f339ff2011-07-01 17:08:43 -0700132 if (!fbDev) {
Steve Blocke6f43dd2012-01-06 19:20:56 +0000133 ALOGE("Display subsystem failed to initialize. check logs. exiting...");
Mathias Agopian1f339ff2011-07-01 17:08:43 -0700134 exit(0);
135 }
136
Mathias Agopian61630912011-07-06 16:35:30 -0700137 int format;
138 ANativeWindow const * const window = mNativeWindow.get();
139 window->query(window, NATIVE_WINDOW_FORMAT, &format);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700140 mDpiX = mNativeWindow->xdpi;
141 mDpiY = mNativeWindow->ydpi;
142 mRefreshRate = fbDev->fps;
Mathias Agopiand0566bc2011-11-17 17:49:17 -0800143 mNextFakeVSync = 0;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700144
Mathias Agopian385977f2011-11-04 18:46:11 -0700145
146/* FIXME: this is a temporary HACK until we are able to report the refresh rate
147 * properly from the HAL. The WindowManagerService now relies on this value.
148 */
149#ifndef REFRESH_RATE
150 mRefreshRate = fbDev->fps;
151#else
152 mRefreshRate = REFRESH_RATE;
153#warning "refresh rate set via makefile to REFRESH_RATE"
154#endif
155
Mathias Agopiand0566bc2011-11-17 17:49:17 -0800156 mRefreshPeriod = nsecs_t(1e9 / mRefreshRate);
157
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700158 EGLint w, h, dummy;
159 EGLint numConfigs=0;
160 EGLSurface surface;
161 EGLContext context;
Mathias Agopian61630912011-07-06 16:35:30 -0700162 EGLBoolean result;
163 status_t err;
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700164
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800165 // initialize EGL
Mathias Agopiana5b02e02009-09-04 18:49:03 -0700166 EGLint attribs[] = {
Mathias Agopian61630912011-07-06 16:35:30 -0700167 EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
168 EGL_NONE, 0,
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800169 EGL_NONE
170 };
Mathias Agopiana5b02e02009-09-04 18:49:03 -0700171
172 // debug: disable h/w rendering
173 char property[PROPERTY_VALUE_MAX];
174 if (property_get("debug.sf.hw", property, NULL) > 0) {
175 if (atoi(property) == 0) {
Steve Block32397c12012-01-05 23:22:43 +0000176 ALOGW("H/W composition disabled");
Mathias Agopiana5b02e02009-09-04 18:49:03 -0700177 attribs[2] = EGL_CONFIG_CAVEAT;
178 attribs[3] = EGL_SLOW_CONFIG;
179 }
180 }
181
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800182 // TODO: all the extensions below should be queried through
183 // eglGetProcAddress().
184
185 EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
186 eglInitialize(display, NULL, NULL);
187 eglGetConfigs(display, NULL, 0, &numConfigs);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700188
Mathias Agopian61630912011-07-06 16:35:30 -0700189 EGLConfig config = NULL;
190 err = selectConfigForPixelFormat(display, attribs, format, &config);
Steve Blocke6f43dd2012-01-06 19:20:56 +0000191 ALOGE_IF(err, "couldn't find an EGLConfig matching the screen format");
Mathias Agopian6cf50a72009-08-06 16:05:39 -0700192
Mathias Agopian0928e312009-08-07 16:38:10 -0700193 EGLint r,g,b,a;
194 eglGetConfigAttrib(display, config, EGL_RED_SIZE, &r);
195 eglGetConfigAttrib(display, config, EGL_GREEN_SIZE, &g);
196 eglGetConfigAttrib(display, config, EGL_BLUE_SIZE, &b);
197 eglGetConfigAttrib(display, config, EGL_ALPHA_SIZE, &a);
198
Mathias Agopian1e16b132009-05-07 17:40:23 -0700199 if (mNativeWindow->isUpdateOnDemand()) {
Mathias Agopian95a666b2009-09-24 14:57:26 -0700200 mFlags |= PARTIAL_UPDATES;
Mathias Agopian1e16b132009-05-07 17:40:23 -0700201 }
202
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800203 if (eglGetConfigAttrib(display, config, EGL_CONFIG_CAVEAT, &dummy) == EGL_TRUE) {
204 if (dummy == EGL_SLOW_CONFIG)
205 mFlags |= SLOW_CONFIG;
206 }
207
208 /*
209 * Create our main surface
210 */
211
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700212 surface = eglCreateWindowSurface(display, config, mNativeWindow.get(), NULL);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700213 eglQuerySurface(display, surface, EGL_WIDTH, &mWidth);
214 eglQuerySurface(display, surface, EGL_HEIGHT, &mHeight);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800215
Mathias Agopian95a666b2009-09-24 14:57:26 -0700216 if (mFlags & PARTIAL_UPDATES) {
217 // if we have partial updates, we definitely don't need to
218 // preserve the backbuffer, which may be costly.
Mathias Agopian0928bee2009-09-16 20:15:42 -0700219 eglSurfaceAttrib(display, surface,
220 EGL_SWAP_BEHAVIOR, EGL_BUFFER_DESTROYED);
221 }
222
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800223 if (eglQuerySurface(display, surface, EGL_SWAP_BEHAVIOR, &dummy) == EGL_TRUE) {
224 if (dummy == EGL_BUFFER_PRESERVED) {
225 mFlags |= BUFFER_PRESERVED;
226 }
227 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800228
David 'Digit' Turnerae71acc2009-06-19 04:41:12 +0200229 /* Read density from build-specific ro.sf.lcd_density property
Mathias Agopian24e5f522009-08-12 21:18:15 -0700230 * except if it is overridden by qemu.sf.lcd_density.
David 'Digit' Turnerae71acc2009-06-19 04:41:12 +0200231 */
232 if (property_get("qemu.sf.lcd_density", property, NULL) <= 0) {
233 if (property_get("ro.sf.lcd_density", property, NULL) <= 0) {
Steve Block32397c12012-01-05 23:22:43 +0000234 ALOGW("ro.sf.lcd_density not defined, using 160 dpi by default.");
David 'Digit' Turnerae71acc2009-06-19 04:41:12 +0200235 strcpy(property, "160");
David 'Digit' Turner694e10b2009-06-18 04:30:32 +0200236 }
David 'Digit' Turner31469e12009-07-29 00:38:58 +0200237 } else {
238 /* for the emulator case, reset the dpi values too */
239 mDpiX = mDpiY = atoi(property);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800240 }
241 mDensity = atoi(property) * (1.0f/160.0f);
242
243
244 /*
245 * Create our OpenGL ES context
246 */
247
Mathias Agopian67226812010-10-11 17:54:43 -0700248
249 EGLint contextAttributes[] = {
250#ifdef EGL_IMG_context_priority
251#ifdef HAS_CONTEXT_PRIORITY
252#warning "using EGL_IMG_context_priority"
253 EGL_CONTEXT_PRIORITY_LEVEL_IMG, EGL_CONTEXT_PRIORITY_HIGH_IMG,
254#endif
255#endif
256 EGL_NONE, EGL_NONE
257 };
258 context = eglCreateContext(display, config, NULL, contextAttributes);
259
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800260 mDisplay = display;
261 mConfig = config;
262 mSurface = surface;
263 mContext = context;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700264 mFormat = fbDev->format;
265 mPageFlipCount = 0;
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700266
267 /*
268 * Gather OpenGL ES extensions
269 */
270
Mathias Agopian61630912011-07-06 16:35:30 -0700271 result = eglMakeCurrent(display, surface, surface, context);
272 if (!result) {
Steve Blocke6f43dd2012-01-06 19:20:56 +0000273 ALOGE("Couldn't create a working GLES context. check logs. exiting...");
Mathias Agopian61630912011-07-06 16:35:30 -0700274 exit(0);
275 }
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700276
277 GLExtensions& extensions(GLExtensions::getInstance());
278 extensions.initWithGLStrings(
279 glGetString(GL_VENDOR),
280 glGetString(GL_RENDERER),
281 glGetString(GL_VERSION),
282 glGetString(GL_EXTENSIONS),
283 eglQueryString(display, EGL_VENDOR),
284 eglQueryString(display, EGL_VERSION),
285 eglQueryString(display, EGL_EXTENSIONS));
286
287 glGetIntegerv(GL_MAX_TEXTURE_SIZE, &mMaxTextureSize);
Mathias Agopian3d64e732011-04-18 15:59:24 -0700288 glGetIntegerv(GL_MAX_VIEWPORT_DIMS, mMaxViewportDims);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700289
Steve Blocka19954a2012-01-04 20:05:49 +0000290 ALOGI("EGL informations:");
291 ALOGI("# of configs : %d", numConfigs);
292 ALOGI("vendor : %s", extensions.getEglVendor());
293 ALOGI("version : %s", extensions.getEglVersion());
294 ALOGI("extensions: %s", extensions.getEglExtension());
295 ALOGI("Client API: %s", eglQueryString(display, EGL_CLIENT_APIS)?:"Not Supported");
296 ALOGI("EGLSurface: %d-%d-%d-%d, config=%p", r, g, b, a, config);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700297
Steve Blocka19954a2012-01-04 20:05:49 +0000298 ALOGI("OpenGL informations:");
299 ALOGI("vendor : %s", extensions.getVendor());
300 ALOGI("renderer : %s", extensions.getRenderer());
301 ALOGI("version : %s", extensions.getVersion());
302 ALOGI("extensions: %s", extensions.getExtension());
303 ALOGI("GL_MAX_TEXTURE_SIZE = %d", mMaxTextureSize);
304 ALOGI("GL_MAX_VIEWPORT_DIMS = %d x %d", mMaxViewportDims[0], mMaxViewportDims[1]);
305 ALOGI("flags = %08x", mFlags);
Mathias Agopian1f7bec62010-06-25 18:02:21 -0700306
307 // Unbind the context from this thread
308 eglMakeCurrent(display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
Mathias Agopiana350ff92010-08-10 17:14:02 -0700309
310
311 // initialize the H/W composer
Mathias Agopianc7d14e22011-08-01 16:32:21 -0700312 mHwc = new HWComposer(mFlinger);
Mathias Agopiana350ff92010-08-10 17:14:02 -0700313 if (mHwc->initCheck() == NO_ERROR) {
314 mHwc->setFrameBuffer(mDisplay, mSurface);
315 }
316}
317
318HWComposer& DisplayHardware::getHwComposer() const {
319 return *mHwc;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800320}
321
322/*
323 * Clean up. Throw out our local state.
324 *
325 * (It's entirely possible we'll never get here, since this is meant
326 * for real hardware, which doesn't restart.)
327 */
328
329void DisplayHardware::fini()
330{
331 eglMakeCurrent(mDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
332 eglTerminate(mDisplay);
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800333}
334
335void DisplayHardware::releaseScreen() const
336{
337 DisplayHardwareBase::releaseScreen();
Antti Hatalaf5f27122010-09-09 02:33:05 -0700338 if (mHwc->initCheck() == NO_ERROR) {
339 mHwc->release();
340 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800341}
342
343void DisplayHardware::acquireScreen() const
344{
345 DisplayHardwareBase::acquireScreen();
346}
347
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800348uint32_t DisplayHardware::getPageFlipCount() const {
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700349 return mPageFlipCount;
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800350}
351
Mathias Agopiand0566bc2011-11-17 17:49:17 -0800352// this needs to be thread safe
353nsecs_t DisplayHardware::waitForVSync() const {
354 nsecs_t timestamp;
355 if (mVSync.wait(&timestamp) < 0) {
356 // vsync not supported!
357 usleep( getDelayToNextVSyncUs(&timestamp) );
358 }
359 return timestamp;
360}
361
362int32_t DisplayHardware::getDelayToNextVSyncUs(nsecs_t* timestamp) const {
363 Mutex::Autolock _l(mFakeVSyncMutex);
364 const nsecs_t period = mRefreshPeriod;
365 const nsecs_t now = systemTime(CLOCK_MONOTONIC);
366 nsecs_t next_vsync = mNextFakeVSync;
367 nsecs_t sleep = next_vsync - now;
368 if (sleep < 0) {
369 // we missed, find where the next vsync should be
370 sleep = (period - ((now - next_vsync) % period));
371 next_vsync = now + sleep;
372 }
373 mNextFakeVSync = next_vsync + period;
374 timestamp[0] = next_vsync;
375
376 // round to next microsecond
377 int32_t sleep_us = (sleep + 999LL) / 1000LL;
378
379 // guaranteed to be > 0
380 return sleep_us;
381}
382
Mathias Agopian74faca22009-09-17 16:18:16 -0700383status_t DisplayHardware::compositionComplete() const {
384 return mNativeWindow->compositionComplete();
385}
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800386
Mathias Agopian35b48d12010-09-13 22:57:58 -0700387int DisplayHardware::getCurrentBufferIndex() const {
388 return mNativeWindow->getCurrentBufferIndex();
389}
390
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800391void DisplayHardware::flip(const Region& dirty) const
392{
393 checkGLErrors();
394
395 EGLDisplay dpy = mDisplay;
396 EGLSurface surface = mSurface;
397
Mathias Agopian5e78e092009-06-11 17:19:54 -0700398#ifdef EGL_ANDROID_swap_rectangle
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700399 if (mFlags & SWAP_RECTANGLE) {
Mathias Agopianb8a55602009-06-26 19:06:36 -0700400 const Region newDirty(dirty.intersect(bounds()));
401 const Rect b(newDirty.getBounds());
Mathias Agopiandf3ca302009-05-04 19:29:25 -0700402 eglSetSwapRectangleANDROID(dpy, surface,
403 b.left, b.top, b.width(), b.height());
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800404 }
Mathias Agopian5e78e092009-06-11 17:19:54 -0700405#endif
406
Mathias Agopian95a666b2009-09-24 14:57:26 -0700407 if (mFlags & PARTIAL_UPDATES) {
Mathias Agopian29d06ac2009-06-29 18:49:56 -0700408 mNativeWindow->setUpdateRectangle(dirty.getBounds());
Mathias Agopian1e16b132009-05-07 17:40:23 -0700409 }
410
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700411 mPageFlipCount++;
Mathias Agopiana350ff92010-08-10 17:14:02 -0700412
413 if (mHwc->initCheck() == NO_ERROR) {
414 mHwc->commit();
415 } else {
416 eglSwapBuffers(dpy, surface);
417 }
The Android Open Source Projectedbf3b62009-03-03 19:31:44 -0800418 checkEGLErrors("eglSwapBuffers");
419
420 // for debugging
421 //glClearColor(1,0,0,0);
422 //glClear(GL_COLOR_BUFFER_BIT);
423}
424
425uint32_t DisplayHardware::getFlags() const
426{
427 return mFlags;
428}
429
430void DisplayHardware::makeCurrent() const
431{
432 eglMakeCurrent(mDisplay, mSurface, mSurface, mContext);
433}
Erik Gilling1d21a9c2010-12-01 16:38:01 -0800434
435void DisplayHardware::dump(String8& res) const
436{
437 mNativeWindow->dump(res);
438}