Android EGL環(huán)境的搭建

Android EGL環(huán)境的搭建

1.概述

1.1 EGL是什么?

???????想要了解EGL是什么叼丑,我覺得我們首先就必須要了解什么是OpenGLES扯罐。通俗上講,OpenGL是一個操作GPU的API肥照,它通過驅(qū)動向GPU發(fā)送相關(guān)指令脚仔,控制圖形渲染管線狀態(tài)機的運行狀態(tài)。但OpenGL需要本地視窗系統(tǒng)進行交互舆绎,這就需要一個中間控制層鲤脏,最好與平臺無關(guān)。EGL——因此被獨立的設(shè)計出來吕朵,它作為OpenGL ES和本地窗口的橋梁猎醇。EGL 是 OpenGL ES(嵌入式)和底層 Native 平臺視窗系統(tǒng)之間的接口。EGL API 是獨立于OpenGL ES各版本標(biāo)準(zhǔn)的獨立API 努溃,其主要作用是為OpenGL指令創(chuàng)建 Context 硫嘶、繪制目標(biāo)Surface 、配置Framebuffer屬性梧税、Swap提交繪制結(jié)果等沦疾。此外称近,EGL為GPU廠商和OS窗口系統(tǒng)之間提供了一個標(biāo)準(zhǔn)配置接口。一般來說哮塞,OpenGL ES 圖形管線的狀態(tài)被存儲于 EGL 管理的一個Context中刨秆。而Frame Buffers 和其他繪制 Surfaces 通過 EGL API進行創(chuàng)建、管理和銷毀彻桃。 EGL 同時也控制和提供了對設(shè)備顯示和可能的設(shè)備渲染配置的訪問坛善。

1.2 為什要搭建EGL環(huán)境。

???????目前在Android系統(tǒng)上已經(jīng)有了GLSurfaceView供我們?nèi)粘J褂昧诰欤窃谝恍┨囟ǖ沫h(huán)境下系統(tǒng)提供的GLSurfaceView就無能為力了眠屎,比如說一個畫面同時渲染到多個SurfaceView上,這個時候就需要我們自己搭建EGL環(huán)境肆饶,進行EGLContext的共享來達到畫面在多個SurfaceView之間共享的目的改衩。除了這種需求之外,從0搭建EGL環(huán)境也是最快了解EGL的方式之一驯镊,我們要有造輪子的精神葫督。

2.EGL環(huán)境的搭建

2.1 獲取EGL實例

//1.得到Egl實例
EGL10 mEgl = (EGL10) EGLContext.getEGL();

2.2 獲取EGL實例

//2.得到默認的顯示設(shè)備
EGLDisplay mEglDisplay = mEgl.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);

2.3 初始化EGL

 int[] version=new int[2];//該數(shù)組用于存放EGL初始化完畢后系統(tǒng)返回的EGL的主版本號和次版本號,int[0]為主版本號板惑,int[1]為子版本號
 boolean initialize = mEgl.eglInitialize(mEglDisplay, version);

2.4 設(shè)置顯示設(shè)備的屬性

/4.設(shè)置顯示設(shè)備的屬性

int[] attribute=new int[] {
                EGL10.EGL_RED_SIZE, 8,//紅色
                EGL10.EGL_GREEN_SIZE, 8,//綠色
                EGL10.EGL_BLUE_SIZE, 8,//藍色
                EGL10.EGL_ALPHA_SIZE, 8,//通明度
                EGL10.EGL_DEPTH_SIZE, 8,//深度
                EGL10.EGL_STENCIL_SIZE, 4,//模板(和3d有關(guān))
                EGL10.EGL_NONE};//屬性結(jié)束符號
//根據(jù)屬性信息從系統(tǒng)的所有的配置信息中橄镜,獲取支持該屬性列表的配置信息的個數(shù)。一般來說就選取一個就好了
 int[] num_config = new int[1];
 boolean chooseConfig = mEgl.eglChooseConfig(mEglDisplay, attribute,
 null, //null的意思就是我們獲取的配置信息不進行存儲
1, //獲取所需屬性的個數(shù)
num_config //存儲屬性返回值個數(shù)的數(shù)組
);

2.5 從系統(tǒng)中獲取對應(yīng)屬性的配置

EGLConfig[] eglConfigs=new EGLConfig[num_config[0]];
boolean eglChooseConfig = mEgl.eglChooseConfig(mEglDisplay, attribute, eglConfigs, num_config[0], num_config);

2.6 創(chuàng)建EglContext

    //如果eglContext==null則創(chuàng)建新的egl上下文
    //設(shè)置EGL的客戶端的版本號冯乘,如果不設(shè)置無法繪制出紋理
    int[] attrib_list = {EGL_CONTEXT_CLIENT_VERSION, 2,
            EGL10.EGL_NONE };
    if (eglContext==null){
        mEglContext = mEgl.eglCreateContext(mEglDisplay, eglConfigs[0], EGL_NO_CONTEXT, null);

    }else {
        //根據(jù)傳入eglContext創(chuàng)建可以共享的egl上下文
        mEglContext = mEgl.eglCreateContext(mEglDisplay, eglConfigs[0], eglContext, null);
    }

2.7 創(chuàng)建surface

    eglSurface = mEgl.eglCreateWindowSurface(mEglDisplay, eglConfigs[0], surface, null);

2.8.綁定EGLContext和surface到顯示設(shè)備

    boolean makeCurrent = mEgl.eglMakeCurrent(mEglDisplay, eglSurface, eglSurface, mEglContext);
    if (!makeCurrent){
        Log.e("ike","eglMakeCurrent failed");
    }

2.9 刷新數(shù)據(jù)

mEgl.eglSwapBuffers(mEglDisplay, mEglSurface);

3 完整示例代碼


import android.icu.text.UFormat;
import android.util.Log;
import android.view.Surface;
import android.view.SurfaceView;

import javax.microedition.khronos.egl.EGL;
import javax.microedition.khronos.egl.EGL10;
import javax.microedition.khronos.egl.EGLConfig;
import javax.microedition.khronos.egl.EGLContext;
import javax.microedition.khronos.egl.EGLDisplay;
import javax.microedition.khronos.egl.EGLSurface;

import static javax.microedition.khronos.egl.EGL10.EGL_NO_CONTEXT;

/**
* author ike
* create time 21:16 2019/9/25
* function: EGL Helper
**/
public class EGLHelper {
    private EGL10 mEgl;
    private EGLDisplay mEglDisplay;
    private EGLContext mEglContext;
    private EGLSurface eglSurface;

    public void init(Surface surface, EGLContext eglContext){
        //1.得到Egl實例
        mEgl = (EGL10) EGLContext.getEGL();
        //2.得到默認的顯示設(shè)備(就是窗口)
        mEglDisplay = mEgl.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);
        //判斷獲取默認顯示設(shè)備是否成功
        if (mEglDisplay==EGL10.EGL_NO_DISPLAY){
            throw new RuntimeException("eglGetDisplay failed");
        }
        //3.初始化默認顯示設(shè)備(初始化EGL)
        //主版本號和次版本號
       int[] version=new int[2];
        boolean initialize = mEgl.eglInitialize(mEglDisplay, version);
        if (!initialize){
            throw new RuntimeException("eglInitialize failed");
        }
        Log.e("ike","version:"+version[0]);
        Log.e("ike","version:"+version[1]);

        //4.設(shè)置顯示設(shè)備的屬性
        int[] attribute=new int[] {
                EGL10.EGL_RED_SIZE, 8,
                EGL10.EGL_GREEN_SIZE, 8,
                EGL10.EGL_BLUE_SIZE, 8,
                EGL10.EGL_ALPHA_SIZE, 8,
                EGL10.EGL_DEPTH_SIZE, 8,
                EGL10.EGL_STENCIL_SIZE, 4,
                EGL10.EGL_NONE};
        //根據(jù)屬性信息從系統(tǒng)的所有的配置信息中洽胶,獲取支持該屬性列表的配置信息的個數(shù)。一般來說就選取一個就好了
        int[] num_config = new int[1];
        boolean chooseConfig = mEgl.eglChooseConfig(mEglDisplay, attribute, null, 1, num_config);
        if (!chooseConfig){
            throw new RuntimeException("eglChooseConfig failed");

        }
        //判斷是否選擇到符合傳入?yún)?shù)的配置信息
        if (num_config[0]<=0){
            throw new IllegalArgumentException(
                    "No configs match configSpec");
        }
        //5.從系統(tǒng)中獲取對應(yīng)屬性的配置
        EGLConfig[] eglConfigs=new EGLConfig[num_config[0]];
        boolean eglChooseConfig = mEgl.eglChooseConfig(mEglDisplay, attribute, eglConfigs, num_config[0], num_config);
        if (!eglChooseConfig){
            throw new RuntimeException("eglChooseConfig$2 failed");
        }
        //6. 創(chuàng)建EglContext
        //如果eglContext==null則創(chuàng)建新的egl上下文
        int[] attrib_list = {EGL_CONTEXT_CLIENT_VERSION, 2,
            EGL10.EGL_NONE };
        if (eglContext==null){
            mEglContext = mEgl.eglCreateContext(mEglDisplay, eglConfigs[0], EGL_NO_CONTEXT, null);

        }else {
            //根據(jù)傳入eglContext創(chuàng)建可以共享的egl上下文
            mEglContext = mEgl.eglCreateContext(mEglDisplay, eglConfigs[0], eglContext, null);
        }
        if (mEglContext==null||mEglContext== EGL_NO_CONTEXT){
            mEglContext=null;
            throw new RuntimeException("eglCreateContext failed");
        }
        //7.創(chuàng)建surface
        eglSurface = mEgl.eglCreateWindowSurface(mEglDisplay, eglConfigs[0], surface, null);
        if (eglSurface==null||eglSurface==EGL10.EGL_NO_SURFACE){
            eglSurface=null;
            throw new RuntimeException("eglCreateWindowSurface failed");
        }
        //8.綁定EGLContext和surface到顯示設(shè)備
        boolean makeCurrent = mEgl.eglMakeCurrent(mEglDisplay, eglSurface, eglSurface, mEglContext);
        if (!makeCurrent){
            Log.e("ike","eglMakeCurrent failed");
        }
    }

    /**
     * 刷新數(shù)據(jù)
     */
    public void swapBuffers(){
        if (mEgl!=null){
            boolean eglSwapBuffers = mEgl.eglSwapBuffers(mEglDisplay, eglSurface);
            if (!eglSwapBuffers){
                Log.e("ike","eglSwapBuffers failed");

            }
        }
    }

    /**
     * 獲取EGL上下文
     * @return
     */
    public EGLContext getEGLCOntext(){
        return mEglContext;
    }
    public void destoryEgl(){
       if (mEgl!=null){
           //與顯示設(shè)備解綁裆馒,銷毀eglsurface
           mEgl.eglMakeCurrent(mEglDisplay,EGL10.EGL_NO_SURFACE,EGL10.EGL_NO_SURFACE,EGL_NO_CONTEXT);
           mEgl.eglDestroySurface(mEglDisplay,eglSurface);
           eglSurface=null;
           //銷毀上下文
           mEgl.eglDestroyContext(mEglDisplay,mEglContext);
           mEglContext=null;
           //銷毀顯示設(shè)備
           mEgl.eglTerminate(mEglDisplay);
           mEglDisplay=null;
           mEgl=null;
       }
    }
}

4 結(jié)語

本文代碼多數(shù)是從GLSurfaceView中提取而出姊氓,里面有完整的實例過程。如有錯誤的地方喷好,希望不吝指出翔横,要是有贊就更好了。睡覺9=痢:萄洹!
參考資料 https://blog.csdn.net/qq_38261174/article/details/84102154

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末无切,一起剝皮案震驚了整個濱河市蟀俊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌订雾,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,820評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件矛洞,死亡現(xiàn)場離奇詭異洼哎,居然都是意外死亡烫映,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,648評論 3 399
  • 文/潘曉璐 我一進店門噩峦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來锭沟,“玉大人,你說我怎么就攤上這事识补∽寤矗” “怎么了?”我有些...
    開封第一講書人閱讀 168,324評論 0 360
  • 文/不壞的土叔 我叫張陵凭涂,是天一觀的道長祝辣。 經(jīng)常有香客問我,道長切油,這世上最難降的妖魔是什么蝙斜? 我笑而不...
    開封第一講書人閱讀 59,714評論 1 297
  • 正文 為了忘掉前任,我火速辦了婚禮澎胡,結(jié)果婚禮上孕荠,老公的妹妹穿的比我還像新娘。我一直安慰自己攻谁,他們只是感情好稚伍,可當(dāng)我...
    茶點故事閱讀 68,724評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著戚宦,像睡著了一般个曙。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上阁苞,一...
    開封第一講書人閱讀 52,328評論 1 310
  • 那天困檩,我揣著相機與錄音,去河邊找鬼那槽。 笑死悼沿,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的骚灸。 我是一名探鬼主播糟趾,決...
    沈念sama閱讀 40,897評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼甚牲!你這毒婦竟也來了义郑?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,804評論 0 276
  • 序言:老撾萬榮一對情侶失蹤丈钙,失蹤者是張志新(化名)和其女友劉穎非驮,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體雏赦,經(jīng)...
    沈念sama閱讀 46,345評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡劫笙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,431評論 3 340
  • 正文 我和宋清朗相戀三年芙扎,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片填大。...
    茶點故事閱讀 40,561評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡戒洼,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出允华,到底是詐尸還是另有隱情圈浇,我是刑警寧澤,帶...
    沈念sama閱讀 36,238評論 5 350
  • 正文 年R本政府宣布靴寂,位于F島的核電站磷蜀,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏榨汤。R本人自食惡果不足惜蠕搜,卻給世界環(huán)境...
    茶點故事閱讀 41,928評論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望收壕。 院中可真熱鬧妓灌,春花似錦、人聲如沸蜜宪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,417評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽圃验。三九已至掉伏,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間澳窑,已是汗流浹背斧散。 一陣腳步聲響...
    開封第一講書人閱讀 33,528評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留摊聋,地道東北人鸡捐。 一個月前我還...
    沈念sama閱讀 48,983評論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像麻裁,于是被迫代替她去往敵國和親箍镜。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,573評論 2 359

推薦閱讀更多精彩內(nèi)容