Version 7.6.3.2-android, tag libreoffice-7.6.3.2-android
[LibreOffice.git] / android / source / src / java / org / mozilla / gecko / gfx / GLController.java
blob3dcfbbdfdbfff5b41ea9460cd9f5cbf950a9e72c
1 /* -*- Mode: Java; c-basic-offset: 4; tab-width: 20; indent-tabs-mode: nil; -*-
2 * This Source Code Form is subject to the terms of the Mozilla Public
3 * License, v. 2.0. If a copy of the MPL was not distributed with this
4 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
6 package org.mozilla.gecko.gfx;
8 import javax.microedition.khronos.egl.EGL10;
9 import javax.microedition.khronos.egl.EGL11;
10 import javax.microedition.khronos.egl.EGLConfig;
11 import javax.microedition.khronos.egl.EGLContext;
12 import javax.microedition.khronos.egl.EGLDisplay;
13 import javax.microedition.khronos.egl.EGLSurface;
14 import javax.microedition.khronos.opengles.GL;
15 import javax.microedition.khronos.opengles.GL10;
17 public class GLController {
18 private static final int EGL_CONTEXT_CLIENT_VERSION = 0x3098;
19 private static final String LOGTAG = "GeckoGLController";
21 private LayerView mView;
22 private int mGLVersion;
23 private boolean mSurfaceValid;
24 private int mWidth, mHeight;
26 private EGL10 mEGL;
27 private EGLDisplay mEGLDisplay;
28 private EGLConfig mEGLConfig;
29 private EGLContext mEGLContext;
30 private EGLSurface mEGLSurface;
32 private GL mGL;
34 private static final int LOCAL_EGL_OPENGL_ES2_BIT = 4;
36 private static final int[] CONFIG_SPEC = {
37 EGL10.EGL_RED_SIZE, 5,
38 EGL10.EGL_GREEN_SIZE, 6,
39 EGL10.EGL_BLUE_SIZE, 5,
40 EGL10.EGL_SURFACE_TYPE, EGL10.EGL_WINDOW_BIT,
41 EGL10.EGL_RENDERABLE_TYPE, LOCAL_EGL_OPENGL_ES2_BIT,
42 EGL10.EGL_NONE
45 public GLController(LayerView view) {
46 mView = view;
47 mGLVersion = 2;
48 mSurfaceValid = false;
51 public void setGLVersion(int version) {
52 mGLVersion = version;
55 /** You must call this on the same thread you intend to use OpenGL on. */
56 public void initGLContext() {
57 initEGLContext();
58 createEGLSurface();
61 public void disposeGLContext() {
62 if (mEGL == null) {
63 return;
66 if (!mEGL.eglMakeCurrent(mEGLDisplay, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_SURFACE,
67 EGL10.EGL_NO_CONTEXT)) {
68 throw new GLControllerException("EGL context could not be released! " +
69 getEGLError());
72 if (mEGLSurface != null) {
73 if (!mEGL.eglDestroySurface(mEGLDisplay, mEGLSurface)) {
74 throw new GLControllerException("EGL surface could not be destroyed! " +
75 getEGLError());
78 mEGLSurface = null;
81 if (mEGLContext != null) {
82 if (!mEGL.eglDestroyContext(mEGLDisplay, mEGLContext)) {
83 throw new GLControllerException("EGL context could not be destroyed! " +
84 getEGLError());
87 mGL = null;
88 mEGLContext = null;
92 public GL getGL() { return mEGLContext.getGL(); }
93 public EGLDisplay getEGLDisplay() { return mEGLDisplay; }
94 public EGLConfig getEGLConfig() { return mEGLConfig; }
95 public EGLContext getEGLContext() { return mEGLContext; }
96 public EGLSurface getEGLSurface() { return mEGLSurface; }
97 public LayerView getView() { return mView; }
99 public boolean hasSurface() {
100 return mEGLSurface != null;
103 public boolean swapBuffers() {
104 return mEGL.eglSwapBuffers(mEGLDisplay, mEGLSurface);
107 public boolean checkForLostContext() {
108 if (mEGL.eglGetError() != EGL11.EGL_CONTEXT_LOST) {
109 return false;
112 mEGLDisplay = null;
113 mEGLConfig = null;
114 mEGLContext = null;
115 mEGLSurface = null;
116 mGL = null;
117 return true;
120 // This function is invoked by JNI
121 public synchronized void resumeCompositorIfValid() {
122 if (mSurfaceValid) {
123 mView.getListener().compositionResumeRequested(mWidth, mHeight);
127 // Wait until we are allowed to use EGL functions on the Surface backing
128 // this window. This function is invoked by JNI
129 public synchronized void waitForValidSurface() {
130 while (!mSurfaceValid) {
131 try {
132 wait();
133 } catch (InterruptedException e) {
134 throw new RuntimeException(e);
139 public synchronized int getWidth() {
140 return mWidth;
143 public synchronized int getHeight() {
144 return mHeight;
147 synchronized void surfaceDestroyed() {
148 mSurfaceValid = false;
149 notifyAll();
152 synchronized void surfaceChanged(int newWidth, int newHeight) {
153 mWidth = newWidth;
154 mHeight = newHeight;
155 mSurfaceValid = true;
156 notifyAll();
159 private void initEGL() {
160 mEGL = (EGL10)EGLContext.getEGL();
162 mEGLDisplay = mEGL.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);
163 if (mEGLDisplay == EGL10.EGL_NO_DISPLAY) {
164 throw new GLControllerException("eglGetDisplay() failed");
167 int[] version = new int[2];
168 if (!mEGL.eglInitialize(mEGLDisplay, version)) {
169 throw new GLControllerException("eglInitialize() failed " + getEGLError());
172 mEGLConfig = chooseConfig();
175 private void initEGLContext() {
176 initEGL();
178 int[] attribList = { EGL_CONTEXT_CLIENT_VERSION, mGLVersion, EGL10.EGL_NONE };
179 mEGLContext = mEGL.eglCreateContext(mEGLDisplay, mEGLConfig, EGL10.EGL_NO_CONTEXT,
180 attribList);
181 if (mEGLContext == null || mEGLContext == EGL10.EGL_NO_CONTEXT) {
182 throw new GLControllerException("createContext() failed " +
183 getEGLError());
186 mGL = mEGLContext.getGL();
188 if (mView.getRenderer() != null) {
189 mView.getRenderer().onSurfaceCreated((GL10)mGL, mEGLConfig);
190 mView.getRenderer().onSurfaceChanged((GL10)mGL, mWidth, mHeight);
194 private EGLConfig chooseConfig() {
195 int[] numConfigs = new int[1];
196 if (!mEGL.eglChooseConfig(mEGLDisplay, CONFIG_SPEC, null, 0, numConfigs) ||
197 numConfigs[0] <= 0) {
198 throw new GLControllerException("No available EGL configurations " +
199 getEGLError());
202 EGLConfig[] configs = new EGLConfig[numConfigs[0]];
203 if (!mEGL.eglChooseConfig(mEGLDisplay, CONFIG_SPEC, configs, numConfigs[0], numConfigs)) {
204 throw new GLControllerException("No EGL configuration for that specification " +
205 getEGLError());
208 // Select the first 565 RGB configuration.
209 int[] red = new int[1], green = new int[1], blue = new int[1];
210 for (EGLConfig config : configs) {
211 mEGL.eglGetConfigAttrib(mEGLDisplay, config, EGL10.EGL_RED_SIZE, red);
212 mEGL.eglGetConfigAttrib(mEGLDisplay, config, EGL10.EGL_GREEN_SIZE, green);
213 mEGL.eglGetConfigAttrib(mEGLDisplay, config, EGL10.EGL_BLUE_SIZE, blue);
214 if (red[0] == 5 && green[0] == 6 && blue[0] == 5) {
215 return config;
219 // if there's no 565 RGB configuration, select another one that fulfils the specification
220 return configs[0];
223 private void createEGLSurface() {
224 Object window = mView.getNativeWindow();
225 mEGLSurface = mEGL.eglCreateWindowSurface(mEGLDisplay, mEGLConfig, window, null);
226 if (mEGLSurface == null || mEGLSurface == EGL10.EGL_NO_SURFACE) {
227 throw new GLControllerException("EGL window surface could not be created! " +
228 getEGLError());
231 if (!mEGL.eglMakeCurrent(mEGLDisplay, mEGLSurface, mEGLSurface, mEGLContext)) {
232 throw new GLControllerException("EGL surface could not be made into the current " +
233 "surface! " + getEGLError());
236 mGL = mEGLContext.getGL();
238 if (mView.getRenderer() != null) {
239 mView.getRenderer().onSurfaceCreated((GL10)mGL, mEGLConfig);
240 mView.getRenderer().onSurfaceChanged((GL10)mGL, mView.getWidth(), mView.getHeight());
245 * Provides an EGLSurface without assuming ownership of this surface.
246 * This class does not keep a reference to the provided EGL surface; the
247 * caller assumes ownership of the surface once it is returned.
249 private EGLSurface provideEGLSurface() {
250 if (mEGL == null) {
251 initEGL();
254 Object window = mView.getNativeWindow();
255 EGLSurface surface = mEGL.eglCreateWindowSurface(mEGLDisplay, mEGLConfig, window, null);
256 if (surface == null || surface == EGL10.EGL_NO_SURFACE) {
257 throw new GLControllerException("EGL window surface could not be created! " +
258 getEGLError());
261 return surface;
264 private String getEGLError() {
265 return "Error " + mEGL.eglGetError();
268 public static class GLControllerException extends RuntimeException {
269 public static final long serialVersionUID = 1L;
271 GLControllerException(String e) {
272 super(e);