关于s3c6410 实现opengl的分析

发布时间:2024-09-06  

1. 什么是EGL
EGL是用来管理绘图表面的(Drawing surfaces),并且提供了如下的机制
(1) 与本地窗口系统进行通信
(2) 查找绘图表面可用的类型和配置信息
(3) 创建绘图表面
(4) 同步OpenGL ES 2.0和其他的渲染API(Open VG、本地窗口系统的绘图命令等)
(5) 管理渲染资源,比如材质

2. EGL 和 OpenGL ES API的联系
(1) 通过解析OpenGL ES API函数库 libGLES_android.so来获取函数指针,进行调用。
(2) 通过线程局部存储机制进行联系

关于通过函数指针进行联系在前面已经分析过了。下面着重分析通过线程局部存储机制进行联系分析一下。

2.1什么是线程局部存储(TLS)
TLS 让多线程程序设计更容易一些。TLS 是一个机制,经由它,程序可以拥有全域变量,但处于「每一线程各不相同」的状态。也就是说,进程中的所有线程都可以拥有全域变量,但这些变量只对特定对某个线程才有意义。http://xianjunzhang.blog.sohu.com/21537031.html
2.2 TLS的好处
你可能有一个多线程程序,每一个线程都对不同的文件写文件(也因此它们使用不同的文件handle)。这种情况下,把每一个线程所使用的文件handle 储存在TLS 中,将会十分方便。当线程需要知道所使用的handle,它可以从TLS 获得。重点在于:线程用来取得文件handle 的那一段代码在任何情况下都是相同的,而从TLS中取出的文件handle 却各不相同。非常灵巧,不是吗?有全域变数的便利,却又分属各线程。 http://xianjunzhang.blog.sohu.com/21537031.html
2.3 OpenGL ES中的TLS
2.3.1 TLS的初始化
应用程序通过EGL调用eglGetDisplay()函数时,会调用到libEGL.so中,通过看其源码egl.so可以发现,其中有条语句

static pthread_once_t once_control = PTHREAD_ONCE_INIT;
static int sEarlyInitState = pthread_once(&once_control, &early_egl_init);
这两条语句会先于eglGetDisplay函数执行。第二条语句中将函数指针early_egl_init作为参数传入,会执行回调,并且保证单个线程只会执行一次。在early_egl_init()中,对TLS机制进行初始化。将TLS里放入一个结构体指针,这个指针指向gHooksNoContext(gl_hooks_t类型),这个结构体里的每个函数指针被初始化为了gl_no_context。也就是现在如果通过TLS调用的OpenGL ES API都会调到gl_no_context这个函数中。
综上,这两条语句完成了TLS的初始化。
另一处初始化时在eglInitialize函数中,同样设置成了gHooksNoContext。
2.3.2 TLS的赋值
在eglMakeCurrent中,会将渲染上下文绑定到渲染面。在EGL中首先会处理完一系列和本地窗口系统的变量后,调用libGLES_android.so中的eglMakeCurrent,调用成功的话会设置TLS。将TLS指针指向前面已经初始化化好的gl_hooks_t结构体指针,这个结构体里的成员都已经指向了libGLES_android.so中的OpenGL API函数,至此EGL的大部分初始化工作就已经完成了。基本就可以使用OpenGL ES API进行绘图了。
static inline void setGlThreadSpecific(gl_hooks_t const *value) {
    gl_hooks_t const * volatile * tls_hooks = get_tls_hooks();
    tls_hooks[TLS_SLOT_OPENGL_API] = value;
}

3. 调用OpenGL ES API函数
在通过EGL对本地窗口系统做一系列初始化之后,就需要调用真正的OpenGL ES API进行3D绘图了,对于很多没有接触过OpenGL和计算机图形学的人来说,这部分可能是比较困难的(很多算法我都不懂。。。)。下面脱离具体的算法实现,看看要调用OpenGL ES API,Android 的3D系统做了什么。
具体分析可以参考Android源码中OpenGL ES的测试代码。
在应用程序中要调用一个OpenGL ES API,需要包含对应的头文件。比如OpenGL ES 1.1对应的头文件是。但是在具体的执行中,调用到了那个库中呢?
分析tests中的tritex测试案例。它的Andoird.mk是这样的。
LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)

LOCAL_SRC_FILES:=
tritex.cpp

LOCAL_SHARED_LIBRARIES :=
libcutils
    libEGL
    libGLESv1_CM
    libui

LOCAL_MODULE:= test-opengl-tritex

PRODUCT_COPY_FILES :=
    out/target/product/generic/system/bin/test-opengl-tritex:/nfs/gltest/

LOCAL_MODULE_TAGS := optional

include $(BUILD_EXECUTABLE)   
从中可以看出,它链接的时候使用的libGLESv1_CM.so
而生成libGLESV1_CM.so的Android.mk是这样写的。
include $(CLEAR_VARS)

LOCAL_SRC_FILES:=   
GLES_CM/gl.cpp.arm
#

LOCAL_SHARED_LIBRARIES += libcutils libEGL
LOCAL_LDLIBS := -lpthread -ldl
LOCAL_MODULE:= libGLESv1_CM

# needed on sim build because of weird logging issues
ifeq ($(TARGET_SIMULATOR),true)
else
    LOCAL_SHARED_LIBRARIES += libdl
    # we need to access the private Bionic header
    LOCAL_C_INCLUDES += bionic/libc/private
endif

LOCAL_CFLAGS += -DLOG_TAG='libGLESv1'
LOCAL_CFLAGS += -DGL_GLEXT_PROTOTYPES -DEGL_EGLEXT_PROTOTYPES
LOCAL_CFLAGS += -fvisibility=hidden

ifeq ($(ARCH_ARM_HAVE_TLS_REGISTER),true)
LOCAL_CFLAGS += -DHAVE_ARM_TLS_REGISTER
endif

include $(BUILD_SHARED_LIBRARY)
源文件只有一个gl.cpp.
这个文件里的函数看起来只有两个函数(怎么可能!!!),其实不然。
其中一句
extern 'C' {
#include 'gl_api.in'
#include 'glext_api.in'
}
C语言中对#include的处理类似于宏的展开,直接把文件包含进来进行编译的。
gl_api.in中,实际上这些函数的定义。
……
void API_ENTRY(glClearColor)(GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha) {
    CALL_GL_API(glClearColor, red, green, blue, alpha);
}
void API_ENTRY(glClearDepthf)(GLclampf depth) {
    CALL_GL_API(glClearDepthf, depth);
}
……
gl.cpp中去掉条件编译
#define GET_TLS(reg)
            'mov   ' #reg ', #0xFFFF0FFF      n'
            'ldr   ' #reg ', [' #reg ', #-15] n'

#define API_ENTRY(_api) __attribute__((naked)) _api

#define CALL_GL_API(_api, ...)                             
         asm volatile(                                         
            GET_TLS(r12)                                       
            'ldr   r12, [r12, %[tls]] n'                      
            'cmp   r12, #0            n'                      
            'ldrne pc, [r12, %[api]] n'                      
            'bx    lr                 n'                      
            :                                                  
            : [tls] 'J'(TLS_SLOT_OPENGL_API*4),                
              [api] 'J'(__builtin_offsetof(gl_hooks_t, gl._api))   
            :                                                  
            );

    #define CALL_GL_API_RETURN(_api, ...)
        CALL_GL_API(_api, __VA_ARGS__)
        return 0; // placate gcc's warnings. never reached.
CALL_GL_API这个带参数的宏。它的意思是获取TLS_SLOT_OPENGL_API的TLS,如果它的值不是NULL,就跳到相应的OpenGL ES API的地址去执行。这个地方为什么会跳过去呢??
因为从线程局部存储保存的线程指针,指向了一个gl_hooks_t指针,而这个指针指向的结构体里的成员已经在EGL中被初始化为了libGLES_android.so里的函数地址了。所以就可以跳过去了。

从以上分析可以看出libGLESv1_CM.so只是一个wrapper,对OpenGL ES API进行了一个简单的包裹,真正的实现还是在libGLES_andoird.so中的。
对于libGLESV2.so是同样的道理。

这样做的好处:
1. 两套OpenGL ES API对应一个实现库,便于维护
2. 可以对OpenGL ES API进行更改,而不需要改变对外的接口,易于程序移植和兼容。
4. 需要注意的问题
OpenGL ES API很多函数并没有实现。软件加速库不支持OpenGL ES2.0

Loader.cpp中的init_api函数通过dlsym函数,对so文件进行解析,返回了函数的指针,在对每个函数进行跟踪的过程中发现,原来glShaderSource并没有在openGL ES的源码中实现,而且发现很多函数都没有在OpenGL ES的源码中实现。

I/libagl ( 2445): in eglChooseConfig
I/libagl ( 2445): in eglCreateWindowSurface
I/libagl ( 2445): in eglCreateContext
I/libagl ( 2445): in eglMakeCurrent
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API
E/libEGL ( 2445): called unimplemented OpenGL ES API

5. 学习OpenGL ES 应用程序,了解基本的OpenGL ES API
学习OpenGL ES API是为了了解基本的API函数的工作原理和一些3D术语,从而配合三星s3c6410(0718)文档,实现自己的OpenGL ES硬件加速库。

(1) Vertex Arrays/Buffer Objects
(2) Vertex Shader
实现对定点通用的一些操作
Vertex shaders can be used for traditional vertex-based operations such as transforming the position by a matrix, computing the lighting equation to generate a per-vertex color, and generating or transforming texture coordinates. Alternately, because the vertex shader is specified by the application, vertex shaders can be used to do custom vertex transformations
(3) Primitive Assembly
将Vertex Shader中的数据汇编成可以画出来的几何图形,比如三角形、直线等
(4) Rasterizition
将前一步的数据进行绘制,同时将前几个阶段的图元转换成二维的片段,传递给fragment shader
(5) Fragment Shader
对Fragment进行操作。
(6) Per-Fragment Shader
对每个片段进行判断(是否可见)和处理(混合、抖动处理)。
(7) 在FrameBuffer中进行显示。

6 s3c6410 驱动分析
6410的g3d的驱动代码位于Kernel/drivers/media/video/Samsung/g3d中的s3c_fimg3d.c中。
从s3c_fimg3d.c中我们可以获取什么知识??Almost Nothing!
我这里有从网上下载的另一个版本的6410的g3d的驱动,比较全。
点击打开

小小分析:
在驱动中的s3c_g3d_probe中,
主要做了get_resouce request_mem_region ioremap等操作
/* get the memory region for the post processor driver */
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
这句话获取了s3c-g3d的IO资源。
s3c_g3d probe() called
res.start=72000000,res.end=72ffffff,rest.name=s3c-g3d,res.flags=512
before ioremap:
s3c_g3d_mem->start=72000000,s3c_g3d_mem->end=72ffffff,s3c_g3d_mem->name=s3c-g3d,s3c_g3d_mem->flags=-2147483648
after io remap:
s3c_g3d_base=d5000000
s3c_g3d version : 0x1050000
S3C G3D Init : Done

文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>