OpenGL播放yuv数据流(着色器SHADER)-android

Posted 朱韦刚

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了OpenGL播放yuv数据流(着色器SHADER)-android相关的知识,希望对你有一定的参考价值。

OpenGL播放yuv数据流(着色器SHADER)-android(一)

可以参考:http://blog.csdn.net/ueryueryuery/article/details/17608185这篇文章很有帮助。


这个和windows还有ios略有不同,下面将步骤整理一下以做记录:

1:在avtivity_main.xml中添加用于显示的GLsurfaceView

    <android.opengl.GLSurfaceView
        android:id="@+id/lvsPlaySurfaceView"
        android:layout_width="match_parent"
        android:layout_height="400dp"/>

2:将GLsurfaceView传到里面

//得到opengal渲染用的surfaceView
openglsurfaceView = (GLSurfaceView) findViewById(R.id.lvsPlaySurfaceView);

3:需要添加权限在AndroidMainfest.xml中:
    <!--为了能使用OpenGLES 2.0 API,你必须在你的manifest中添加以下声明:-->
    <uses-feature android:glEsVersion="0x00020000" android:required="true" />
    <!-- 如果你的应用要使用纹理压缩功能,你必须还要声明设备需要支持什么样的压缩格式-->
    <supports-gl-texture android:name="GL_OES_compressed_ETC1_RGB8_texture" />
    <supports-gl-texture android:name="GL_OES_compressed_paletted_texture" />
4:下面就是具体的实现代码
//.java
package com.example.zhuweigang.lvsandroidplay;

import android.content.Context;
import android.opengl.GLES20;
import android.opengl.GLSurfaceView;
import android.support.v4.app.NavUtils;
import android.util.AttributeSet;
import android.util.Log;
import android.view.SurfaceHolder;
import javax.microedition.khronos.egl.EGLConfig;
import javax.microedition.khronos.opengles.GL10;

import java.nio.Buffer;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;

/**
 * Created by zhuweigang on 2016/12/26.
 */

public class Lvs_OpenGl_Interface_Android implements GLSurfaceView.Renderer
{
    private static final String TAG = "lvs_OpenGL";

    //顶点数组(物体表面坐标取值范围是-1到1,数组坐标:左下,右下,左上,右上)
    private static float[] vertexVertices = {
        -1.0f, -1.0f,
        1.0f, -1.0f,
        -1.0f,  1.0f,
        1.0f,  1.0f,
    };

    //像素,纹理数组(纹理坐标取值范围是0-1,坐标原点位于左下角,数组坐标:左上,右上,左下,右下,如果先左下,图像会倒过来)
    private static float[] textureVertices = {
        0.0f,  1.0f,
        1.0f,  1.0f,
        0.0f,  0.0f,
        1.0f,  0.0f,
    };

    //shader的vsh源码字符串
    private static final String vertexShaderString =
            "attribute vec4 vertexIn;" +
            "attribute vec2 textureIn;" +
            "varying vec2 textureOut;" +
            "void main() {" +
                "gl_Position = vertexIn;" +
                "textureOut = textureIn;" +
            "}";

    //shader的fsh源码字符串
    private static final String yuvFragmentShaderString =
            "precision mediump float;" +
            "uniform sampler2D tex_y;" +
            "uniform sampler2D tex_u;" +
            "uniform sampler2D tex_v;" +
            "varying vec2 textureOut;" +
            "void main() {" +
                "vec4 c = vec4((texture2D(tex_y, textureOut).r - 16./255.) * 1.164);" +
                "vec4 U = vec4(texture2D(tex_u, textureOut).r - 128./255.);" +
                "vec4 V = vec4(texture2D(tex_v, textureOut).r - 128./255.);" +
                "c += V * vec4(1.596, -0.813, 0, 0);" +
                "c += U * vec4(0, -0.392, 2.017, 0);" +
                "c.a = 1.0;" +
                "gl_FragColor = c;" +
            "}";

    //着色器用的顶点属性索引 position是由3个(x,y,z)组成,
    public int ATTRIB_VERTEX = 0;
    //着色器用的像素,纹理属性索引 而颜色是4个(r,g,b,a)
    public int ATTRIB_TEXTURE = 0;

    private GLSurfaceView mTargetSurface;                                                              //外部传入的GLSurfaceView
    public int p = 0;                                                                                    //Program着色器程序的id
    ByteBuffer vertexVertices_buffer = null;                                                           //定义顶点数组
    ByteBuffer textureVertices_buffer = null;                                                          //定义像素纹理数组
    public int m_IsInitShaders = 0;                                                                    //是否已经InitShaders,onSurfaceCreated
    public Lvs_Play_Interface_Sdk_Android.OpenGl_DisplayCallBackInterface m_displaydatack = null;      //用于显示回调函数,参数数据及时间戳
    public byte m_yuvbuf[] = new byte[640*480*3];												       //存放yuv数据的buf指针,申请buffer在外面
    public ByteBuffer yuvplaner_y = null;                                                               //分用于渲染的变量
    public ByteBuffer yuvplaner_u = null;                                                               //分用于渲染的变量
    public ByteBuffer yuvplaner_v = null;                                                               //分用于渲染的变量;
    public int[] m_millis_realtime = new int[1];                                                      //实时的时间戳,每次回调会更新
    public int m_yuvdata_width = 0;											                            //数据宽
    public int m_yuvdata_height = 0;										                            //数据高
    public int m_frameBuffer = 0;                                                                       //framebuffer
    public int m_renderBuffer = 0;                                                                      //renderbuffer
    public int m_textureid_y, m_textureid_u, m_textureid_v;                                         //纹理的名称,并且,该纹理的名称在当前的应用中不能被再次使用。
    public int m_textureUniformY, m_textureUniformU,m_textureUniformV;                            //用于纹理渲染的变量

    //构造方法
    public Lvs_OpenGl_Interface_Android(GLSurfaceView paramGLSurfaceView)
    {
        //将surfaceview传进来用于显示数据时候刷新
        mTargetSurface = paramGLSurfaceView;

        //应用GlsurfaceView版本号2.0
        mTargetSurface.setEGLContextClientVersion(2);
    }

    @Override
    public void onSurfaceCreated(GL10 gl10, EGLConfig eglConfig)
    {
        Log.i(TAG, "onSurfaceCreated :");
        //这个必须在onSurfaceCreated中,否则失败
        //初始化着色器,类似于告GPU当传进去数据的时候采用什么样的规则。
        InitShaders();
        m_IsInitShaders = 1;
    }

    @Override
    public void onSurfaceChanged(GL10 gl10, int width, int height)
    {
        GLES20.glViewport(0, 0, width, height);
    }

    @Override
    public void onDrawFrame(GL10 gl10)
    {
        //这里做具体的处理
        //具体的显示
        if (mTargetSurface != null)
        {
            DisplayImage(0);
        }
    }

    //接口初始化
    int lvs_opengl_interface_init(GLSurfaceView surface,int yuvdata_width,int yuvdata_height,
                                  Lvs_Play_Interface_Sdk_Android.OpenGl_DisplayCallBackInterface displaydatack)
    {
        int ret  = 0;

        //初始化
        ret = initopengl(yuvdata_width,yuvdata_height,displaydatack);
        if (ret != 1)
        {
            return -1;
        }

        ret = 1;

        return ret ;
    }

    //接口渲染数据(定时器,渲染时间,毫秒),数据及渲染定时时间在回调里面做处理
    void lvs_opengl_interface_write(int value)
    {
        //这里如果有可能则调成类的成员函数,以后处理,暂时不知道怎么解决类成员函数递归
        TimerFunc1();
    }

    //渲染数据(定时器,渲染时间,毫秒),数据及渲染定时时间在回调里面做处理
    void TimerFunc1()
    {
        int ret = 0;

        //因为glut的定时器是调用一次才产生一次定时,所以如果要持续产生定时的话,
        //在定时函数末尾再次调用glutTimerFunc

        //调用回调函数获取数据
        if (m_displaydatack != null && m_IsInitShaders == 1)
        {
            ret= m_displaydatack.OpenGl_DisplayDataCallback(m_yuvbuf,m_millis_realtime);
            if (ret > 0)
            {
                //刷新让他能显示在onDrawFrame中处理
                mTargetSurface.requestRender();
                try {
                    Thread.sleep(m_millis_realtime[0]);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            else
            {
                try {
                    Thread.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            //递归调用自身,java用递归调用本身有问题了,外面for循环调用处理
            //TimerFunc1(opengl_interface);
        }
    }

    //初始化
    int initopengl(int yuvdata_width,int yuvdata_height, Lvs_Play_Interface_Sdk_Android.OpenGl_DisplayCallBackInterface displaydatack)
    {
        int ret = 0;

        m_yuvdata_width = yuvdata_width;
        m_yuvdata_height = yuvdata_height;
        m_displaydatack = displaydatack;

        //分配内存
        if (yuvplaner_y == null)
        {
            yuvplaner_y = ByteBuffer.allocate(m_yuvdata_width*m_yuvdata_height + 100);
        }
        if (yuvplaner_u == null)
        {
            yuvplaner_u = ByteBuffer.allocate(m_yuvdata_width*m_yuvdata_height + 100);
        }
        if (yuvplaner_v == null)
        {
            yuvplaner_v = ByteBuffer.allocate(m_yuvdata_width*m_yuvdata_height + 100);
        }

        ret = 1;
        return ret;
    }
    //初始化着色器,类似于告GPU当传进去数据的时候采用什么样的规则。
    void InitShaders()
    {
        int error = 0;

        createBuffers(vertexVertices, textureVertices);

        p = createProgram(vertexShaderString, yuvFragmentShaderString);
        ATTRIB_VERTEX = GLES20.glGetAttribLocation(p, "vertexIn");
        if (ATTRIB_VERTEX == -1)
        {
            Log.i(TAG, "glGetAttribLocation : " + error);
        }
        ATTRIB_TEXTURE = GLES20.glGetAttribLocation(p, "textureIn");
        if (ATTRIB_TEXTURE == -1)
        {
            Log.i(TAG, "glGetAttribLocation : " + error);
        }

        //Program:  在链接了程序以后,我们可以使用glUseProgram()函数来加载并使用链接好的程序
        GLES20.glUseProgram(p);

        //获取片源着色器源码中的变量,用于纹理渲染
        m_textureUniformY = GLES20.glGetUniformLocation(p, "tex_y");
        m_textureUniformU = GLES20.glGetUniformLocation(p, "tex_u");
        m_textureUniformV = GLES20.glGetUniformLocation(p, "tex_v");

        //初始化纹理
        int[] textures_y = new int[1];
        GLES20.glGenTextures(1, textures_y,0);
        m_textureid_y = textures_y[0];
        textures_y = null;
        //绑定纹理
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, m_textureid_y);
        //设置该纹理的一些属性
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D,GLES20.GL_TEXTURE_MAG_FILTER,GLES20.GL_LINEAR);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D,GLES20.GL_TEXTURE_MIN_FILTER,GLES20.GL_LINEAR);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_CLAMP_TO_EDGE);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_CLAMP_TO_EDGE);

        int[] textures_u = new int[1];
        GLES20.glGenTextures(1, textures_u,0);
        m_textureid_u = textures_u[0];
        textures_u = null;
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, m_textureid_u);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D,GLES20.GL_TEXTURE_MAG_FILTER,GLES20.GL_LINEAR);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D,GLES20.GL_TEXTURE_MIN_FILTER,GLES20.GL_LINEAR);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_CLAMP_TO_EDGE);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_CLAMP_TO_EDGE);

        int[] textures_v = new int[1];
        GLES20.glGenTextures(1, textures_v,0);
        m_textureid_v = textures_v[0];
        textures_v = null;
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, m_textureid_v);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D,GLES20.GL_TEXTURE_MAG_FILTER,GLES20.GL_LINEAR);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D,GLES20.GL_TEXTURE_MIN_FILTER,GLES20.GL_LINEAR);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_CLAMP_TO_EDGE);
        GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_CLAMP_TO_EDGE);
        return;
    }
    //具体显示图像的函数(参数是指针)
    int DisplayImage(long parm)
    {
        int ret = 0;

        //关联到yuv数据的分量数组
        if (yuvplaner_y != null)
        {
            yuvplaner_y.clear();
            yuvplaner_y.put(m_yuvbuf,0,m_yuvdata_width*m_yuvdata_height);
            yuvplaner_y.position(0);
        }
        if (yuvplaner_u != null)
        {
            yuvplaner_u.clear();
            yuvplaner_u.put(m_yuvbuf,m_yuvdata_width*m_yuvdata_height,m_yuvdata_width*m_yuvdata_height/4);
            yuvplaner_u.position(0);
        }
        if (yuvplaner_v != null)
        {
            yuvplaner_v.clear();
            yuvplaner_v.put(m_yuvbuf,m_yuvdata_width*m_yuvdata_height + m_yuvdata_width*m_yuvdata_height/4,m_yuvdata_width*m_yuvdata_height/4);
            yuvplaner_v.position(0);
        }

        //Clear
        //清除颜色设为黑色,把整个窗口清除为当前的清除颜色,glClear()的唯一参数表示需要被清除的缓冲区。
        GLES20.glClearColor(0.0f,0.0f,0.0f,1.0f);
        GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT);

        //定义顶点数组,android平台要在这里做其他平台在initshader中,否则显示不出来图像
        GLES20.glVertexAttribPointer(ATTRIB_VERTEX, 2, GLES20.GL_FLOAT, false, 0, vertexVertices_buffer);
        //启用属性数组,android平台要在这里做其他平台在initshader中,否则显示不出来图像
        GLES20.glEnableVertexAttribArray(ATTRIB_VERTEX);
        //定义像素纹理数组,android平台要在这里做其他平台在initshader中,否则显示不出来图像
        GLES20. glVertexAttribPointer(ATTRIB_TEXTURE, 2, GLES20.GL_FLOAT, false, 0, textureVertices_buffer);
        //启用属性数组,android平台要在这里做其他平台在initshader中,否则显示不出来图像
        GLES20.glEnableVertexAttribArray(ATTRIB_TEXTURE);

        //显卡中有N个纹理单元(具体数目依赖你的显卡能力),每个纹理单元(GL_TEXTURE0、GL_TEXTURE1等)都有GL_TEXTURE_1D、GL_TEXTURE_2D等
        //Y
        //选择当前活跃的纹理单元
        GLES20.glActiveTexture(GLES20.GL_TEXTURE0);
        //允许建立一个绑定到目标纹理的有名称的纹理
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, m_textureid_y);
        //根据指定的参数,生成一个2D纹理(Texture)。相似的函数还有glTexImage1D、glTexImage3D。
        GLES20.glTexImage2D(GLES20.GL_TEXTURE_2D, 0, GLES20.GL_LUMINANCE, m_yuvdata_width, m_yuvdata_height, 0, GLES20.GL_LUMINANCE, GLES20.GL_UNSIGNED_BYTE, yuvplaner_y);
        GLES20.glUniform1i(m_textureUniformY, 0);     //设置纹理,按照前面设置的规则怎样将图像或纹理贴上(参数和选择的活跃纹理单元对应,GL_TEXTURE0)

        //U
        GLES20.glActiveTexture(GLES20.GL_TEXTURE1);
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, m_textureid_u);
        GLES20.glTexImage2D(GLES20.GL_TEXTURE_2D, 0, GLES20.GL_LUMINANCE, m_yuvdata_width/2, m_yuvdata_height/2, 0, GLES20.GL_LUMINANCE, GLES20.GL_UNSIGNED_BYTE, yuvplaner_u);
        GLES20.glUniform1i(m_textureUniformU, 1);
        //V
        GLES20.glActiveTexture(GLES20.GL_TEXTURE2);
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, m_textureid_v);
        GLES20.glTexImage2D(GLES20.GL_TEXTURE_2D, 0, GLES20.GL_LUMINANCE, m_yuvdata_width/2, m_yuvdata_height/2, 0, GLES20.GL_LUMINANCE, GLES20.GL_UNSIGNED_BYTE, yuvplaner_v);
        GLES20.glUniform1i(m_textureUniformV, 2);

        // Draw
        // 绘制
        GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4);
        //单缓冲显示
        GLES20.glFlush();

        GLES20.glDisableVertexAttribArray(ATTRIB_VERTEX);
        GLES20.glDisableVertexAttribArray(ATTRIB_TEXTURE);

        return 1;
    }

    /**
     * create program and load shaders, fragment shader is very important.
     */
    public int createProgram(String vertexSource, String fragmentSource) {
        // create shaders
        int vertexShader = loadShader(GLES20.GL_VERTEX_SHADER, vertexSource);
        int pixelShader = loadShader(GLES20.GL_FRAGMENT_SHADER, fragmentSource);
        // just check

        int program = GLES20.glCreateProgram();
        if (program != 0) {
            GLES20.glAttachShader(program, vertexShader);
            GLES20.glAttachShader(program, pixelShader);
            GLES20.glLinkProgram(program);
            int[] linkStatus = new int[1];
            GLES20.glGetProgramiv(program, GLES20.GL_LINK_STATUS, linkStatus, 0);
            if (linkStatus[0] != GLES20.GL_TRUE) {
                GLES20.glDeleteProgram(program);
                program = 0;
            }
        }
        return program;
    }
    /**
     * create shader with given source.
     */
    private int loadShader(int shaderType, String source)
    {
        int shader = GLES20.glCreateShader(shaderType);
        if (shader != 0) {
            GLES20.glShaderSource(shader, source);
            GLES20.glCompileShader(shader);
            int[] compiled = new int[1];
            GLES20.glGetShaderiv(shader, GLES20.GL_COMPILE_STATUS, compiled, 0);
            if (compiled[0] == 0)
            {
                GLES20.glDeleteShader(shader);
                shader = 0;
            }
        }
        return shader;
    }

    /**
     * these two buffers are used for holding vertices, screen vertices and texture vertices.
     */
    private void createBuffers(float[] vert, float[] coord) {
        vertexVertices_buffer = ByteBuffer.allocateDirect(vert.length * 4);
        vertexVertices_buffer.order(ByteOrder.nativeOrder());
        vertexVertices_buffer.asFloatBuffer().put(vert);
        vertexVertices_buffer.position(0);

        if (textureVertices_buffer == null) {
            textureVertices_buffer = ByteBuffer.allocateDirect(coord.length * 4);
            textureVertices_buffer.order(ByteOrder.nativeOrder());
            textureVertices_buffer.asFloatBuffer().put(coord);
            textureVertices_buffer.position(0);
        }
    }
}


5:调用代码
        //GLSurfaceView
        m_glsurfaceview = nsurfaceView;
        //opengl的view类
        pinterfaceOpenGL = new Lvs_OpenGl_Interface_Android(m_glsurfaceview);
        m_glsurfaceview.setRenderer(pinterfaceOpenGL);
        // 只有在绘制数据改变时才绘制view
        m_glsurfaceview.setRenderMode(GLSurfaceView.RENDERMODE_WHEN_DIRTY);


                //Opengl初始化
                ret = pinterfaceOpenGL.lvs_opengl_interface_init(m_glsurfaceview,m_opengl_width,m_opengl_height,
                        opengl_displaycallback);
                if (ret < 0)
                {
                    return;
                }

                for (;;)
                {
                    try {
                        Thread.sleep(1);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                    //渲染,带定时器,数据回调,及渲染时间回调,第一帧timer 40以后根据时间戳做调整
                   pinterfaceOpenGL.lvs_opengl_interface_write(40);
                }

6:数据传入在回掉函数
ret= m_displaydatack.OpenGl_DisplayDataCallback(m_yuvbuf,m_millis_realtime);中
7:实现效果
 

 

本demo还需完善。

如有错误请指正:

交流请加QQ群:62054820
QQ:379969650.


以上是关于OpenGL播放yuv数据流(着色器SHADER)-android的主要内容,如果未能解决你的问题,请参考以下文章

使用 GL_TEXTURE_2D 的 iOS YUV 420v 在 OpenGL 着色器中显示错误的颜色

OpenGL入门之着色器Shader

OpenGL入门之渲染管线pipeline,着色器Shader

着色器在 Opengl Shader Builder 上工作,但不在我的 OpenGL 应用程序中

OpenGL4.3新特性: 计算着色器 Compute Shader

基于OpenGL编写一个简易的2D渲染框架-09 重构渲染器-Shader