Как получить SurfaceHolder с действительной поверхностью (необходимой для EGL.eglCreateWindowSurface)?

Я пытаюсь инициализировать GLES с помощью EGL (потому что я хочу рисовать из основного потока, а не использовать Renderer и рисовать изнутри onDrawFrame). Я получаю сообщение об ошибке: «Убедитесь, что SurfaceView или связанный с ним SurfaceHolder имеет допустимую поверхность». Очевидно, что mEgl.eglCreateWindowSurface терпит неудачу, потому что у SurfaceHolder нет допустимой поверхности. Как мне получить SurfaceHolder с действительной поверхностью для продолжения?

Мой метод Activity.onCreate:

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    GLSurfaceView surfaceView = new GLSurfaceView(this);
    setContentView(surfaceView);
    SurfaceHolder surfaceHolder = surfaceView.getHolder();
    Surface surface = surfaceHolder.getSurface();
    Log.v("HelloAndroid", "surface.isValid()= " + Boolean.toString(surface.isValid()));

    EGL10 mEgl = (EGL10) EGLContext.getEGL();
    EGLDisplay mEglDisplay = mEgl.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);
    int[] version = new int[2];
    mEgl.eglInitialize(mEglDisplay, version);
    EGLConfig[] configs = new EGLConfig[1];
    int[] num_config = new int[1];
    int[] configSpec = {
            EGL10.EGL_NONE
    };
    mEgl.eglChooseConfig(mEglDisplay, configSpec, configs, 1, num_config);
    EGLConfig mEglConfig = configs[0];
    EGLContext mEglContext = mEgl.eglCreateContext(mEglDisplay, mEglConfig,
            EGL10.EGL_NO_CONTEXT, null);
    EGLSurface mEglSurface = null;
    Log.v("HelloAndroid", "M");
    mEglSurface = mEgl.eglCreateWindowSurface(mEglDisplay,
            mEglConfig, surfaceHolder, null);
    Log.v("HelloAndroid", "N");
    mEgl.eglMakeCurrent(mEglDisplay, mEglSurface, mEglSurface,
            mEglContext);
    Log.v("HelloAndroid", "O");
}

Вывод журнала, который я получаю с помощью platform-tools/adb logcat:

V/HelloAndroid( 1861): surface.isValid()= false
D/libEGL  ( 1861): egl.cfg not found, using default config
D/libEGL  ( 1861): loaded /system/lib/egl/libGLES_android.so
V/HelloAndroid( 1861): M
D/AndroidRuntime( 1861): Shutting down VM
W/dalvikvm( 1861): threadid=3: thread exiting with uncaught exception (group=0x4001b188)
E/AndroidRuntime( 1861): Uncaught handler: thread main exiting due to uncaught exception
E/AndroidRuntime( 1861): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.helloandroid/com.example.helloandroid.HelloAndroid}: java.lang.IllegalArgumentException: Make sure the SurfaceView or associated SurfaceHolder has a valid Surface
E/AndroidRuntime( 1861):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2496)

Любая помощь приветствуется.


person Thomas Mertes    schedule 07.04.2011    source источник


Ответы (2)


Проверка, действительная для surface, должна выполняться при создании поверхности. Итак, surface.isValid() нужно вызывать в onSurfaceCreated или onSurfaceChanged, очевидно, надо добавить surfaceholder.callback.

person Lee    schedule 16.09.2011

Реализуйте SurfaceHolder.Callback и подождите, пока вам не сообщат, что поверхность создана. Также обратите внимание на то, когда он будет уничтожен. Вы должны полагаться только на то, что поверхность действительна между соответствующими вызовами создания и уничтожения.

person Laurence Gonsalves    schedule 07.04.2011
comment
Нормально ли, что владельцу требуется много времени, чтобы дать показания surface.is Valid()=true после запуска приложения? - person Ashwin; 21.10.2012