我想通过EGL创建OpenGL上下文。因为我实际上不会画画,所以我想将Pbuffers与GBM平台结合使用。这是代码(C99):

#include <stdlib.h>
#include <assert.h>

#include <fcntl.h>
#include <unistd.h>

#include <EGL/egl.h>
#include <EGL/eglext.h>

#include <gbm.h>


int main( void )
{
    assert( eglBindAPI( EGL_OPENGL_API ) == EGL_TRUE );

    int fd = open("/dev/dri/card0", O_RDWR);
    struct gbm_device * gbm = gbm_create_device( fd );

    EGLDisplay dpy = eglGetDisplay( gbm );
    eglInitialize( dpy , NULL , NULL );

    EGLConfig config;
    EGLint n_of_configs;
    assert( eglGetConfigs( dpy , &config , 1 , &n_of_configs ) == EGL_TRUE );

    EGLSurface srf = eglCreatePbufferSurface( dpy , config , NULL );
    assert( srf != EGL_NO_SURFACE );

    EGLContext ctx = eglCreateContext( dpy , config , EGL_NO_CONTEXT , NULL );
    assert( ctx != EGL_NO_CONTEXT );

    assert( eglMakeCurrent( dpy , srf , srf , ctx ) == EGL_TRUE );

    eglDestroySurface( dpy , srf );
    eglDestroyContext( dpy , ctx );
    eglTerminate( dpy );

    gbm_device_destroy( gbm );
    close( fd );

    return EXIT_SUCCESS;
}

它失败,并显示以下错误:
test.c: In function ‘main’:
test.c:20:2: error: passing argument 1 of ‘eglGetDisplay’ from incompatible pointer type [-Werror]
  EGLDisplay dpy = eglGetDisplay( gbm );
  ^
In file included from test.c:7:0:
/usr/include/EGL/egl.h:251:31: note: expected ‘EGLNativeDisplayType’ but argument is of type ‘struct gbm_device *’
 EGLAPI EGLDisplay EGLAPIENTRY eglGetDisplay(EGLNativeDisplayType display_id);

This是我作为示例的页面。

我很惊讶,因为我是使用--with-egl-platforms=drm,wayland,x11构建Mesa的,即使声明EGL_DEFAULT_DISPLAY映射到指定的第一个平台here,它还是系统上_XDisplay *的别名。
--with-egl-platforms
List the platforms (window systems) to support. Its argument is a comma seprated string such as --with-egl-platforms=x11,drm. It decides the platforms a driver may support. The first listed platform is also used by the main library to decide the native platform: the platform the EGL native types such as EGLNativeDisplayType or EGLNativeWindowType defined for.

AFAIK,Weston在GBM之上创建了一个EGLDisplay,用于在裸KMS上绘图。我已经看了一下它的代码,并摸索了相关的系统头文件,以寻找解决方案,但似乎不存在。

FWIW,我在Radeon HD 3200,Linux 3.12.6,GCC 4.8.2上使用Mesa 10.0。

最佳答案

你说



EGL/GBM不支持Pbuffer。它也不支持Pixmaps。

要使用EGL/GBM创建屏幕外表面,必须将gbm_surface传递给eglCreateWindowSurface。 “窗口”是用词不当。没有真正的“窗口”被创建。除非您使用内核的KMS API将其发布到显示器上,否则所得的缓冲区将保持在屏幕外。

但这不是您的程序无法编译的原因。在调用eglGetDisplay和eglCreateWindowSurface时,您必须像这样进行转换:

eglGetDisplay((EGLNativeDisplayType)my_gbm_device);
eglCreateWindowSurface(egl_dpy, egl_config,
                       (EGLNativeWindowType)my_gbm_surface, NULL);

如果您正在使用包含commit 468cc86的Mesa,则可以通过使用eglGetPlatformDisplayEXT和eglCreatePlatformWindowSurface来避免进行强制转换,如下所示:
eglGetPlatformDisplayEXT(EGL_PLATFORM_GBM_MESA, my_gbm_device, NULL);
eglCreatePlatformWindowSurfaceEXT(egl_dpy, egl_config, my_gbm_surface, NULL);

无论您选择使用原始EGL函数还是使用较新的平台EGL函数,都应引用EGL_MESA_platform_gbm规范中的示例代码作为指南。

关于c - EGLDisplay在GBM上,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/20816844/

10-11 15:18