1/*
2 * Copyright (C) 2009 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
17package com.android.systemui;
18
19import static android.opengl.GLES20.*;
20import static javax.microedition.khronos.egl.EGL10.*;
21
22import android.app.ActivityManager;
23import android.app.WallpaperManager;
24import android.content.BroadcastReceiver;
25import android.content.ComponentCallbacks2;
26import android.content.Context;
27import android.content.Intent;
28import android.graphics.Bitmap;
29import android.graphics.Canvas;
30import android.graphics.Point;
31import android.graphics.Rect;
32import android.graphics.RectF;
33import android.graphics.Region.Op;
34import android.opengl.GLUtils;
35import android.os.SystemProperties;
36import android.renderscript.Matrix4f;
37import android.service.wallpaper.WallpaperService;
38import android.util.Log;
39import android.view.Display;
40import android.view.MotionEvent;
41import android.view.SurfaceHolder;
42import android.view.WindowManager;
43
44import java.io.IOException;
45import java.nio.ByteBuffer;
46import java.nio.ByteOrder;
47import java.nio.FloatBuffer;
48
49import javax.microedition.khronos.egl.EGL10;
50import javax.microedition.khronos.egl.EGLConfig;
51import javax.microedition.khronos.egl.EGLContext;
52import javax.microedition.khronos.egl.EGLDisplay;
53import javax.microedition.khronos.egl.EGLSurface;
54
55/**
56 * Default built-in wallpaper that simply shows a static image.
57 */
58@SuppressWarnings({"UnusedDeclaration"})
59public class ImageWallpaper extends WallpaperService {
60    private static final String TAG = "ImageWallpaper";
61    private static final String GL_LOG_TAG = "ImageWallpaperGL";
62    private static final boolean DEBUG = false;
63    private static final String PROPERTY_KERNEL_QEMU = "ro.kernel.qemu";
64
65    static final boolean FIXED_SIZED_SURFACE = true;
66    static final boolean USE_OPENGL = true;
67
68    WallpaperManager mWallpaperManager;
69
70    DrawableEngine mEngine;
71
72    boolean mIsHwAccelerated;
73
74    @Override
75    public void onCreate() {
76        super.onCreate();
77        mWallpaperManager = (WallpaperManager) getSystemService(WALLPAPER_SERVICE);
78
79        //noinspection PointlessBooleanExpression,ConstantConditions
80        if (FIXED_SIZED_SURFACE && USE_OPENGL) {
81            if (!isEmulator()) {
82                mIsHwAccelerated = ActivityManager.isHighEndGfx();
83            }
84        }
85    }
86
87    @Override
88    public void onTrimMemory(int level) {
89        if (mEngine != null) {
90            mEngine.trimMemory(level);
91        }
92    }
93
94    private static boolean isEmulator() {
95        return "1".equals(SystemProperties.get(PROPERTY_KERNEL_QEMU, "0"));
96    }
97
98    @Override
99    public Engine onCreateEngine() {
100        mEngine = new DrawableEngine();
101        return mEngine;
102    }
103
104    class DrawableEngine extends Engine {
105        static final int EGL_CONTEXT_CLIENT_VERSION = 0x3098;
106        static final int EGL_OPENGL_ES2_BIT = 4;
107
108        // TODO: Not currently used, keeping around until we know we don't need it
109        @SuppressWarnings({"UnusedDeclaration"})
110        private WallpaperObserver mReceiver;
111
112        Bitmap mBackground;
113        int mBackgroundWidth = -1, mBackgroundHeight = -1;
114        int mLastSurfaceWidth = -1, mLastSurfaceHeight = -1;
115        int mLastRotation = -1;
116        float mXOffset = 0.5f;
117        float mYOffset = 0.5f;
118        float mScale = 1f;
119
120        boolean mVisible = true;
121        boolean mRedrawNeeded;
122        boolean mOffsetsChanged;
123        int mLastXTranslation;
124        int mLastYTranslation;
125
126        private EGL10 mEgl;
127        private EGLDisplay mEglDisplay;
128        private EGLConfig mEglConfig;
129        private EGLContext mEglContext;
130        private EGLSurface mEglSurface;
131
132        private static final String sSimpleVS =
133                "attribute vec4 position;\n" +
134                "attribute vec2 texCoords;\n" +
135                "varying vec2 outTexCoords;\n" +
136                "uniform mat4 projection;\n" +
137                "\nvoid main(void) {\n" +
138                "    outTexCoords = texCoords;\n" +
139                "    gl_Position = projection * position;\n" +
140                "}\n\n";
141        private static final String sSimpleFS =
142                "precision mediump float;\n\n" +
143                "varying vec2 outTexCoords;\n" +
144                "uniform sampler2D texture;\n" +
145                "\nvoid main(void) {\n" +
146                "    gl_FragColor = texture2D(texture, outTexCoords);\n" +
147                "}\n\n";
148
149        private static final int FLOAT_SIZE_BYTES = 4;
150        private static final int TRIANGLE_VERTICES_DATA_STRIDE_BYTES = 5 * FLOAT_SIZE_BYTES;
151        private static final int TRIANGLE_VERTICES_DATA_POS_OFFSET = 0;
152        private static final int TRIANGLE_VERTICES_DATA_UV_OFFSET = 3;
153
154        class WallpaperObserver extends BroadcastReceiver {
155            @Override
156            public void onReceive(Context context, Intent intent) {
157                if (DEBUG) {
158                    Log.d(TAG, "onReceive");
159                }
160
161                mLastSurfaceWidth = mLastSurfaceHeight = -1;
162                mBackground = null;
163                mBackgroundWidth = -1;
164                mBackgroundHeight = -1;
165                mRedrawNeeded = true;
166                drawFrame();
167            }
168        }
169
170        public DrawableEngine() {
171            super();
172            setFixedSizeAllowed(true);
173        }
174
175        public void trimMemory(int level) {
176            if (level >= ComponentCallbacks2.TRIM_MEMORY_RUNNING_LOW &&
177                    mBackground != null && mIsHwAccelerated) {
178                if (DEBUG) {
179                    Log.d(TAG, "trimMemory");
180                }
181                mBackground.recycle();
182                mBackground = null;
183                mBackgroundWidth = -1;
184                mBackgroundHeight = -1;
185                mWallpaperManager.forgetLoadedWallpaper();
186            }
187        }
188
189        @Override
190        public void onCreate(SurfaceHolder surfaceHolder) {
191            if (DEBUG) {
192                Log.d(TAG, "onCreate");
193            }
194
195            super.onCreate(surfaceHolder);
196
197            // TODO: Don't need this currently because the wallpaper service
198            // will restart the image wallpaper whenever the image changes.
199            //IntentFilter filter = new IntentFilter(Intent.ACTION_WALLPAPER_CHANGED);
200            //mReceiver = new WallpaperObserver();
201            //registerReceiver(mReceiver, filter, null, mHandler);
202
203            updateSurfaceSize(surfaceHolder);
204
205            setOffsetNotificationsEnabled(false);
206        }
207
208        @Override
209        public void onDestroy() {
210            super.onDestroy();
211            if (mReceiver != null) {
212                unregisterReceiver(mReceiver);
213            }
214        }
215
216        void updateSurfaceSize(SurfaceHolder surfaceHolder) {
217            Point p = getDefaultDisplaySize();
218
219            // Load background image dimensions, if we haven't saved them yet
220            if (mBackgroundWidth <= 0 || mBackgroundHeight <= 0) {
221                // Need to load the image to get dimensions
222                mWallpaperManager.forgetLoadedWallpaper();
223                updateWallpaperLocked();
224                if (mBackgroundWidth <= 0 || mBackgroundHeight <= 0) {
225                    // Default to the display size if we can't find the dimensions
226                    mBackgroundWidth = p.x;
227                    mBackgroundHeight = p.y;
228                }
229            }
230
231            // Force the wallpaper to cover the screen in both dimensions
232            int surfaceWidth = Math.max(p.x, mBackgroundWidth);
233            int surfaceHeight = Math.max(p.y, mBackgroundHeight);
234
235            // If the surface dimensions haven't changed, then just return
236            final Rect frame = surfaceHolder.getSurfaceFrame();
237            if (frame != null) {
238                final int dw = frame.width();
239                final int dh = frame.height();
240                if (surfaceWidth == dw && surfaceHeight == dh) {
241                    return;
242                }
243            }
244
245            if (FIXED_SIZED_SURFACE) {
246                // Used a fixed size surface, because we are special.  We can do
247                // this because we know the current design of window animations doesn't
248                // cause this to break.
249                surfaceHolder.setFixedSize(surfaceWidth, surfaceHeight);
250            } else {
251                surfaceHolder.setSizeFromLayout();
252            }
253        }
254
255        @Override
256        public void onVisibilityChanged(boolean visible) {
257            if (DEBUG) {
258                Log.d(TAG, "onVisibilityChanged: mVisible, visible=" + mVisible + ", " + visible);
259            }
260
261            if (mVisible != visible) {
262                if (DEBUG) {
263                    Log.d(TAG, "Visibility changed to visible=" + visible);
264                }
265                mVisible = visible;
266                drawFrame();
267            }
268        }
269
270        @Override
271        public void onTouchEvent(MotionEvent event) {
272            super.onTouchEvent(event);
273        }
274
275        @Override
276        public void onOffsetsChanged(float xOffset, float yOffset,
277                float xOffsetStep, float yOffsetStep,
278                int xPixels, int yPixels) {
279            if (DEBUG) {
280                Log.d(TAG, "onOffsetsChanged: xOffset=" + xOffset + ", yOffset=" + yOffset
281                        + ", xOffsetStep=" + xOffsetStep + ", yOffsetStep=" + yOffsetStep
282                        + ", xPixels=" + xPixels + ", yPixels=" + yPixels);
283            }
284
285            if (mXOffset != xOffset || mYOffset != yOffset) {
286                if (DEBUG) {
287                    Log.d(TAG, "Offsets changed to (" + xOffset + "," + yOffset + ").");
288                }
289                mXOffset = xOffset;
290                mYOffset = yOffset;
291                mOffsetsChanged = true;
292            }
293            drawFrame();
294        }
295
296        @Override
297        public void onSurfaceChanged(SurfaceHolder holder, int format, int width, int height) {
298            if (DEBUG) {
299                Log.d(TAG, "onSurfaceChanged: width=" + width + ", height=" + height);
300            }
301
302            super.onSurfaceChanged(holder, format, width, height);
303
304            drawFrame();
305        }
306
307        @Override
308        public void onSurfaceDestroyed(SurfaceHolder holder) {
309            super.onSurfaceDestroyed(holder);
310            mLastSurfaceWidth = mLastSurfaceHeight = -1;
311        }
312
313        @Override
314        public void onSurfaceCreated(SurfaceHolder holder) {
315            super.onSurfaceCreated(holder);
316            mLastSurfaceWidth = mLastSurfaceHeight = -1;
317        }
318
319        @Override
320        public void onSurfaceRedrawNeeded(SurfaceHolder holder) {
321            if (DEBUG) {
322                Log.d(TAG, "onSurfaceRedrawNeeded");
323            }
324            super.onSurfaceRedrawNeeded(holder);
325
326            drawFrame();
327        }
328
329        private Point getDefaultDisplaySize() {
330            Point p = new Point();
331            Context c = ImageWallpaper.this.getApplicationContext();
332            WindowManager wm = (WindowManager)c.getSystemService(Context.WINDOW_SERVICE);
333            Display d = wm.getDefaultDisplay();
334            d.getRealSize(p);
335            return p;
336        }
337
338        void drawFrame() {
339            int newRotation = ((WindowManager) getSystemService(WINDOW_SERVICE)).
340                    getDefaultDisplay().getRotation();
341
342            // Sometimes a wallpaper is not large enough to cover the screen in one dimension.
343            // Call updateSurfaceSize -- it will only actually do the update if the dimensions
344            // should change
345            if (newRotation != mLastRotation) {
346                // Update surface size (if necessary)
347                updateSurfaceSize(getSurfaceHolder());
348            }
349            SurfaceHolder sh = getSurfaceHolder();
350            final Rect frame = sh.getSurfaceFrame();
351            final int dw = frame.width();
352            final int dh = frame.height();
353            boolean surfaceDimensionsChanged = dw != mLastSurfaceWidth || dh != mLastSurfaceHeight;
354
355            boolean redrawNeeded = surfaceDimensionsChanged || newRotation != mLastRotation;
356            if (!redrawNeeded && !mOffsetsChanged) {
357                if (DEBUG) {
358                    Log.d(TAG, "Suppressed drawFrame since redraw is not needed "
359                            + "and offsets have not changed.");
360                }
361                return;
362            }
363            mLastRotation = newRotation;
364
365            // Load bitmap if it is not yet loaded or if it was loaded at a different size
366            if (mBackground == null || surfaceDimensionsChanged) {
367                if (DEBUG) {
368                    Log.d(TAG, "Reloading bitmap: mBackground, bgw, bgh, dw, dh = " +
369                            mBackground + ", " +
370                            ((mBackground == null) ? 0 : mBackground.getWidth()) + ", " +
371                            ((mBackground == null) ? 0 : mBackground.getHeight()) + ", " +
372                            dw + ", " + dh);
373                }
374                mWallpaperManager.forgetLoadedWallpaper();
375                updateWallpaperLocked();
376                if (mBackground == null) {
377                    if (DEBUG) {
378                        Log.d(TAG, "Unable to load bitmap");
379                    }
380                    return;
381                }
382                if (DEBUG) {
383                    if (dw != mBackground.getWidth() || dh != mBackground.getHeight()) {
384                        Log.d(TAG, "Surface != bitmap dimensions: surface w/h, bitmap w/h: " +
385                                dw + ", " + dh + ", " + mBackground.getWidth() + ", " +
386                                mBackground.getHeight());
387                    }
388                }
389            }
390
391            // Center the scaled image
392            mScale = Math.max(1f, Math.max(dw / (float) mBackground.getWidth(),
393                    dh / (float) mBackground.getHeight()));
394            final int availw = dw - (int) (mBackground.getWidth() * mScale);
395            final int availh = dh - (int) (mBackground.getHeight() * mScale);
396            int xPixels = availw / 2;
397            int yPixels = availh / 2;
398
399            // Adjust the image for xOffset/yOffset values. If window manager is handling offsets,
400            // mXOffset and mYOffset are set to 0.5f by default and therefore xPixels and yPixels
401            // will remain unchanged
402            final int availwUnscaled = dw - mBackground.getWidth();
403            final int availhUnscaled = dh - mBackground.getHeight();
404            if (availwUnscaled < 0) xPixels += (int)(availwUnscaled * (mXOffset - .5f) + .5f);
405            if (availhUnscaled < 0) yPixels += (int)(availhUnscaled * (mYOffset - .5f) + .5f);
406
407            mOffsetsChanged = false;
408            mRedrawNeeded = false;
409            if (surfaceDimensionsChanged) {
410                mLastSurfaceWidth = dw;
411                mLastSurfaceHeight = dh;
412            }
413            if (!redrawNeeded && xPixels == mLastXTranslation && yPixels == mLastYTranslation) {
414                if (DEBUG) {
415                    Log.d(TAG, "Suppressed drawFrame since the image has not "
416                            + "actually moved an integral number of pixels.");
417                }
418                return;
419            }
420            mLastXTranslation = xPixels;
421            mLastYTranslation = yPixels;
422
423            if (DEBUG) {
424                Log.d(TAG, "Redrawing wallpaper");
425            }
426
427            if (mIsHwAccelerated) {
428                if (!drawWallpaperWithOpenGL(sh, availw, availh, xPixels, yPixels)) {
429                    drawWallpaperWithCanvas(sh, availw, availh, xPixels, yPixels);
430                }
431            } else {
432                drawWallpaperWithCanvas(sh, availw, availh, xPixels, yPixels);
433                if (FIXED_SIZED_SURFACE) {
434                    // If the surface is fixed-size, we should only need to
435                    // draw it once and then we'll let the window manager
436                    // position it appropriately.  As such, we no longer needed
437                    // the loaded bitmap.  Yay!
438                    // hw-accelerated path retains bitmap for faster rotation
439                    mBackground = null;
440                    mWallpaperManager.forgetLoadedWallpaper();
441                }
442            }
443
444        }
445
446        private void updateWallpaperLocked() {
447            Throwable exception = null;
448            try {
449                mBackground = null;
450                mBackgroundWidth = -1;
451                mBackgroundHeight = -1;
452                mBackground = mWallpaperManager.getBitmap();
453                mBackgroundWidth = mBackground.getWidth();
454                mBackgroundHeight = mBackground.getHeight();
455            } catch (RuntimeException e) {
456                exception = e;
457            } catch (OutOfMemoryError e) {
458                exception = e;
459            }
460
461            if (exception != null) {
462                mBackground = null;
463                mBackgroundWidth = -1;
464                mBackgroundHeight = -1;
465                // Note that if we do fail at this, and the default wallpaper can't
466                // be loaded, we will go into a cycle.  Don't do a build where the
467                // default wallpaper can't be loaded.
468                Log.w(TAG, "Unable to load wallpaper!", exception);
469                try {
470                    mWallpaperManager.clear();
471                } catch (IOException ex) {
472                    // now we're really screwed.
473                    Log.w(TAG, "Unable reset to default wallpaper!", ex);
474                }
475            }
476        }
477
478        private void drawWallpaperWithCanvas(SurfaceHolder sh, int w, int h, int left, int top) {
479            Canvas c = sh.lockCanvas();
480            if (c != null) {
481                try {
482                    if (DEBUG) {
483                        Log.d(TAG, "Redrawing: left=" + left + ", top=" + top);
484                    }
485
486                    final float right = left + mBackground.getWidth() * mScale;
487                    final float bottom = top + mBackground.getHeight() * mScale;
488                    if (w < 0 || h < 0) {
489                        c.save(Canvas.CLIP_SAVE_FLAG);
490                        c.clipRect(left, top, right, bottom,
491                                Op.DIFFERENCE);
492                        c.drawColor(0xff000000);
493                        c.restore();
494                    }
495                    if (mBackground != null) {
496                        RectF dest = new RectF(left, top, right, bottom);
497                        // add a filter bitmap?
498                        c.drawBitmap(mBackground, null, dest, null);
499                    }
500                } finally {
501                    sh.unlockCanvasAndPost(c);
502                }
503            }
504        }
505
506        private boolean drawWallpaperWithOpenGL(SurfaceHolder sh, int w, int h, int left, int top) {
507            if (!initGL(sh)) return false;
508
509            final float right = left + mBackground.getWidth() * mScale;
510            final float bottom = top + mBackground.getHeight() * mScale;
511
512            final Rect frame = sh.getSurfaceFrame();
513            final Matrix4f ortho = new Matrix4f();
514            ortho.loadOrtho(0.0f, frame.width(), frame.height(), 0.0f, -1.0f, 1.0f);
515
516            final FloatBuffer triangleVertices = createMesh(left, top, right, bottom);
517
518            final int texture = loadTexture(mBackground);
519            final int program = buildProgram(sSimpleVS, sSimpleFS);
520
521            final int attribPosition = glGetAttribLocation(program, "position");
522            final int attribTexCoords = glGetAttribLocation(program, "texCoords");
523            final int uniformTexture = glGetUniformLocation(program, "texture");
524            final int uniformProjection = glGetUniformLocation(program, "projection");
525
526            checkGlError();
527
528            glViewport(0, 0, frame.width(), frame.height());
529            glBindTexture(GL_TEXTURE_2D, texture);
530
531            glUseProgram(program);
532            glEnableVertexAttribArray(attribPosition);
533            glEnableVertexAttribArray(attribTexCoords);
534            glUniform1i(uniformTexture, 0);
535            glUniformMatrix4fv(uniformProjection, 1, false, ortho.getArray(), 0);
536
537            checkGlError();
538
539            if (w < 0 || h < 0) {
540                glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
541                glClear(GL_COLOR_BUFFER_BIT);
542            }
543
544            // drawQuad
545            triangleVertices.position(TRIANGLE_VERTICES_DATA_POS_OFFSET);
546            glVertexAttribPointer(attribPosition, 3, GL_FLOAT, false,
547                    TRIANGLE_VERTICES_DATA_STRIDE_BYTES, triangleVertices);
548
549            triangleVertices.position(TRIANGLE_VERTICES_DATA_UV_OFFSET);
550            glVertexAttribPointer(attribTexCoords, 3, GL_FLOAT, false,
551                    TRIANGLE_VERTICES_DATA_STRIDE_BYTES, triangleVertices);
552
553            glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
554
555            boolean status = mEgl.eglSwapBuffers(mEglDisplay, mEglSurface);
556            checkEglError();
557
558            finishGL();
559
560            return status;
561        }
562
563        private FloatBuffer createMesh(int left, int top, float right, float bottom) {
564            final float[] verticesData = {
565                    // X, Y, Z, U, V
566                     left,  bottom, 0.0f, 0.0f, 1.0f,
567                     right, bottom, 0.0f, 1.0f, 1.0f,
568                     left,  top,    0.0f, 0.0f, 0.0f,
569                     right, top,    0.0f, 1.0f, 0.0f,
570            };
571
572            final int bytes = verticesData.length * FLOAT_SIZE_BYTES;
573            final FloatBuffer triangleVertices = ByteBuffer.allocateDirect(bytes).order(
574                    ByteOrder.nativeOrder()).asFloatBuffer();
575            triangleVertices.put(verticesData).position(0);
576            return triangleVertices;
577        }
578
579        private int loadTexture(Bitmap bitmap) {
580            int[] textures = new int[1];
581
582            glActiveTexture(GL_TEXTURE0);
583            glGenTextures(1, textures, 0);
584            checkGlError();
585
586            int texture = textures[0];
587            glBindTexture(GL_TEXTURE_2D, texture);
588            checkGlError();
589
590            glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
591            glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
592
593            glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
594            glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
595
596            GLUtils.texImage2D(GL_TEXTURE_2D, 0, GL_RGBA, bitmap, GL_UNSIGNED_BYTE, 0);
597            checkGlError();
598
599            return texture;
600        }
601
602        private int buildProgram(String vertex, String fragment) {
603            int vertexShader = buildShader(vertex, GL_VERTEX_SHADER);
604            if (vertexShader == 0) return 0;
605
606            int fragmentShader = buildShader(fragment, GL_FRAGMENT_SHADER);
607            if (fragmentShader == 0) return 0;
608
609            int program = glCreateProgram();
610            glAttachShader(program, vertexShader);
611            checkGlError();
612
613            glAttachShader(program, fragmentShader);
614            checkGlError();
615
616            glLinkProgram(program);
617            checkGlError();
618
619            int[] status = new int[1];
620            glGetProgramiv(program, GL_LINK_STATUS, status, 0);
621            if (status[0] != GL_TRUE) {
622                String error = glGetProgramInfoLog(program);
623                Log.d(GL_LOG_TAG, "Error while linking program:\n" + error);
624                glDeleteShader(vertexShader);
625                glDeleteShader(fragmentShader);
626                glDeleteProgram(program);
627                return 0;
628            }
629
630            return program;
631        }
632
633        private int buildShader(String source, int type) {
634            int shader = glCreateShader(type);
635
636            glShaderSource(shader, source);
637            checkGlError();
638
639            glCompileShader(shader);
640            checkGlError();
641
642            int[] status = new int[1];
643            glGetShaderiv(shader, GL_COMPILE_STATUS, status, 0);
644            if (status[0] != GL_TRUE) {
645                String error = glGetShaderInfoLog(shader);
646                Log.d(GL_LOG_TAG, "Error while compiling shader:\n" + error);
647                glDeleteShader(shader);
648                return 0;
649            }
650
651            return shader;
652        }
653
654        private void checkEglError() {
655            int error = mEgl.eglGetError();
656            if (error != EGL_SUCCESS) {
657                Log.w(GL_LOG_TAG, "EGL error = " + GLUtils.getEGLErrorString(error));
658            }
659        }
660
661        private void checkGlError() {
662            int error = glGetError();
663            if (error != GL_NO_ERROR) {
664                Log.w(GL_LOG_TAG, "GL error = 0x" + Integer.toHexString(error), new Throwable());
665            }
666        }
667
668        private void finishGL() {
669            mEgl.eglMakeCurrent(mEglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
670            mEgl.eglDestroySurface(mEglDisplay, mEglSurface);
671            mEgl.eglDestroyContext(mEglDisplay, mEglContext);
672            mEgl.eglTerminate(mEglDisplay);
673        }
674
675        private boolean initGL(SurfaceHolder surfaceHolder) {
676            mEgl = (EGL10) EGLContext.getEGL();
677
678            mEglDisplay = mEgl.eglGetDisplay(EGL_DEFAULT_DISPLAY);
679            if (mEglDisplay == EGL_NO_DISPLAY) {
680                throw new RuntimeException("eglGetDisplay failed " +
681                        GLUtils.getEGLErrorString(mEgl.eglGetError()));
682            }
683
684            int[] version = new int[2];
685            if (!mEgl.eglInitialize(mEglDisplay, version)) {
686                throw new RuntimeException("eglInitialize failed " +
687                        GLUtils.getEGLErrorString(mEgl.eglGetError()));
688            }
689
690            mEglConfig = chooseEglConfig();
691            if (mEglConfig == null) {
692                throw new RuntimeException("eglConfig not initialized");
693            }
694
695            mEglContext = createContext(mEgl, mEglDisplay, mEglConfig);
696            if (mEglContext == EGL_NO_CONTEXT) {
697                throw new RuntimeException("createContext failed " +
698                        GLUtils.getEGLErrorString(mEgl.eglGetError()));
699            }
700
701            int attribs[] = {
702                EGL_WIDTH, 1,
703                EGL_HEIGHT, 1,
704                EGL_NONE
705            };
706            EGLSurface tmpSurface = mEgl.eglCreatePbufferSurface(mEglDisplay, mEglConfig, attribs);
707            mEgl.eglMakeCurrent(mEglDisplay, tmpSurface, tmpSurface, mEglContext);
708
709            int[] maxSize = new int[1];
710            Rect frame = surfaceHolder.getSurfaceFrame();
711            glGetIntegerv(GL_MAX_TEXTURE_SIZE, maxSize, 0);
712
713            mEgl.eglMakeCurrent(mEglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
714            mEgl.eglDestroySurface(mEglDisplay, tmpSurface);
715
716            if(frame.width() > maxSize[0] || frame.height() > maxSize[0]) {
717                mEgl.eglDestroyContext(mEglDisplay, mEglContext);
718                mEgl.eglTerminate(mEglDisplay);
719                Log.e(GL_LOG_TAG, "requested  texture size " +
720                    frame.width() + "x" + frame.height() + " exceeds the support maximum of " +
721                    maxSize[0] + "x" + maxSize[0]);
722                return false;
723            }
724
725            mEglSurface = mEgl.eglCreateWindowSurface(mEglDisplay, mEglConfig, surfaceHolder, null);
726            if (mEglSurface == null || mEglSurface == EGL_NO_SURFACE) {
727                int error = mEgl.eglGetError();
728                if (error == EGL_BAD_NATIVE_WINDOW || error == EGL_BAD_ALLOC) {
729                    Log.e(GL_LOG_TAG, "createWindowSurface returned " +
730                                         GLUtils.getEGLErrorString(error) + ".");
731                    return false;
732                }
733                throw new RuntimeException("createWindowSurface failed " +
734                        GLUtils.getEGLErrorString(error));
735            }
736
737            if (!mEgl.eglMakeCurrent(mEglDisplay, mEglSurface, mEglSurface, mEglContext)) {
738                throw new RuntimeException("eglMakeCurrent failed " +
739                        GLUtils.getEGLErrorString(mEgl.eglGetError()));
740            }
741
742            return true;
743        }
744
745
746        EGLContext createContext(EGL10 egl, EGLDisplay eglDisplay, EGLConfig eglConfig) {
747            int[] attrib_list = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE };
748            return egl.eglCreateContext(eglDisplay, eglConfig, EGL_NO_CONTEXT, attrib_list);
749        }
750
751        private EGLConfig chooseEglConfig() {
752            int[] configsCount = new int[1];
753            EGLConfig[] configs = new EGLConfig[1];
754            int[] configSpec = getConfig();
755            if (!mEgl.eglChooseConfig(mEglDisplay, configSpec, configs, 1, configsCount)) {
756                throw new IllegalArgumentException("eglChooseConfig failed " +
757                        GLUtils.getEGLErrorString(mEgl.eglGetError()));
758            } else if (configsCount[0] > 0) {
759                return configs[0];
760            }
761            return null;
762        }
763
764        private int[] getConfig() {
765            return new int[] {
766                    EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
767                    EGL_RED_SIZE, 8,
768                    EGL_GREEN_SIZE, 8,
769                    EGL_BLUE_SIZE, 8,
770                    EGL_ALPHA_SIZE, 0,
771                    EGL_DEPTH_SIZE, 0,
772                    EGL_STENCIL_SIZE, 0,
773                    EGL_CONFIG_CAVEAT, EGL_NONE,
774                    EGL_NONE
775            };
776        }
777    }
778}
779