android jni 用c++使用opengles和egl实现离屏渲染

最近在做视频有关的东西,用开源的ffmpeg解码视频,在通过opengl渲染出来,因为有使用到egl就试了试离屏渲染

 

EGLConfig eglConf;
EGLSurface eglSurface;
EGLContext eglCtx;
EGLDisplay eglDisp;

 

egl用到的就是这四个

先初始化

 

 

EGLint configSpec[] = { EGL_RED_SIZE, 8,
                        EGL_GREEN_SIZE, 8,
                        EGL_BLUE_SIZE, 8,
                        EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_NONE };
EGLint surfaceAttr[] = {
        EGL_WIDTH, width,
        EGL_HEIGHT, height,
        EGL_NONE
};
eglDisp = eglGetDisplay(EGL_DEFAULT_DISPLAY);
EGLint eglMajVers, eglMinVers;
EGLint numConfigs;
eglInitialize(eglDisp, &eglMajVers, &eglMinVers);
eglChooseConfig(eglDisp, configSpec, &eglConf, 1, &numConfigs);
eglSurface = eglCreatePbufferSurface(eglDisp, eglConf, surfaceAttr);
const EGLint ctxAttr[] = {
        EGL_CONTEXT_CLIENT_VERSION, 2,
        EGL_NONE
};
eglCtx = eglCreateContext(eglDisp, eglConf,EGL_NO_CONTEXT, ctxAttr);

 

上边的width和height是模拟窗口的宽高 最好和图片的比例一致,不然转成图片会拉伸

 

eglMakeCurrent(eglDisp, eglSurface, eglSurface, eglCtx);

 

设置显示的窗口,前一个窗口是显示窗口,后一个是写入窗口,通过调用这个来切换窗口

然后可以用opengl 进行画图了

画完后调用一下

 

eglSwapBuffers(eglDisp, eglSurface);

 

把东西显示在窗口上

最后加个释放

 

eglMakeCurrent(eglDisp, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
eglDestroyContext(eglDisp, eglCtx);
eglDestroySurface(eglDisp, eglSurface);
eglTerminate(eglDisp);

eglDisp = EGL_NO_DISPLAY;
eglSurface = EGL_NO_SURFACE;
eglCtx = EGL_NO_CONTEXT;

 

试了试双屏切换

 

glViewport(0, 0, w, h);
glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
eglSwapBuffers(eglDisp, eglSurface);

jclass jcl = env->GetObjectClass(ob);
jmethodID mid = env->GetMethodID(jcl, "getBitmap", "(II)V");
env->CallVoidMethod(ob, mid,w,h);

 

第一次是离屏渲染,渲染完成后调用java代码把图片显示出来

 

public void getBitmap(int width,int height){
    IntBuffer RGBABuffer = IntBuffer.allocate(width * height);
    RGBABuffer.position(0);
    GLES20.glReadPixels(0, 0, width, height,GLES20.GL_RGBA,GLES20.GL_UNSIGNED_BYTE,RGBABuffer);
    int[] pixls = RGBABuffer.array();
    for (int y = 0; y < height / 2; y++) {
        for (int x = 0; x < width; x++) {
            int pos1 = y * width + x;
            int pos2 = (height - 1 - y) * width + x;
            int tmp = pixls[pos1];
            pixls[pos1] = (pixls[pos2] & 0xFF00FF00) | ((pixls[pos2] >> 16) & 0xff) | ((pixls[pos2] << 16) & 0x00ff0000); // ABGR->ARGB
            pixls[pos2] = (tmp & 0xFF00FF00) | ((tmp >> 16) & 0xff) | ((tmp << 16) & 0x00ff0000);
        }
    }
    if (height % 2 == 1) { // 中间一行
        for (int x = 0; x < width; x++) {
            int pos = (height / 2 + 1) * width + x;
            pixls[pos] = (pixls[pos] & 0xFF00FF00) | ((pixls[pos] >> 16) & 0xff) | ((pixls[pos] << 16) & 0x00ff0000);
        }
    }
    final Bitmap modelBitmap = Bitmap.createBitmap(pixls, width, height, Bitmap.Config.ARGB_8888);
    runOnUiThread(new Runnable() {
        @Override
        public void run() {
            ImageView imageView = (ImageView) findViewById(R.id.iv_opengl);
            imageView.setImageBitmap(modelBitmap);
        }
    });
}

 

然后切换

 

eglMakeCurrent(eglDisp, eglWindow, eglSurface, eglCtx);

glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
glViewport(left, top, viewWidth, viewHeight);
glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
eglSwapBuffers(eglDisp, eglWindow);

 

切换的时候要再先清屏在画,不然两个纹理会重叠

 

 

 

 

 

 

 

 

 

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值