专栏首页曾大稳的博客OpenGLES渲染画面通过MediaCodec录制

OpenGLES渲染画面通过MediaCodec录制

录制原理

  • 预览

通过fbo处理视频数据,通过samplerExternalOES纹理来创建SurfaceTexture,这样的话摄像头数据就和fbo相关联,具体可以看OpenGLES通过SurfaceTexture预览摄像头画面

  • 录制

通过MediaCodec创建一个surface,然后通过创建一个新的egl环境共享预览的EglContext和这个surface绑定,渲染fbo绑定的纹理,即可录制。 egl环境配置: Android配置EGL环境 Android自定义GLSurfaceView

流程如下图所示:

MediaCodec录制主要代码

 private MediaMuxer mMediaMuxer;
 private MediaCodec.BufferInfo mBuffInfo;
 private MediaCodec mVideoEncodec;
 private int width, height;


  //初始化
 public void initEncoder(EGLContext eglContext,String savePath,String mineType,int width,int height){
        this.width = width;
        this.height = height;
        this.mEGLContext = eglContext;
        initMediaEncoder(savePath,mineType,width,height);
    }

    private void initMediaEncoder(String savePath, String mineType, int width, int height) {
        try {
            mMediaMuxer = new MediaMuxer(savePath,MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);
            initVideoEncoder(mineType,width,height);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    private void initVideoEncoder(String mineType, int width, int height) {
        try {
            mVideoEncodec= MediaCodec.createEncoderByType(mineType);

            MediaFormat videoFormat = MediaFormat.createVideoFormat(mineType,width,height);
            videoFormat.setInteger(MediaFormat.KEY_COLOR_FORMAT, MediaCodecInfo.CodecCapabilities.COLOR_FormatSurface);
            videoFormat.setInteger(MediaFormat.KEY_FRAME_RATE,30);//30帧
            videoFormat.setInteger(MediaFormat.KEY_BIT_RATE,width*height*4);//RGBA
            videoFormat.setInteger(MediaFormat.KEY_BIT_RATE,width*height*4);//RGBA
            //设置压缩等级  默认是baseline
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
                videoFormat.setInteger(MediaFormat.KEY_PROFILE,MediaCodecInfo.CodecProfileLevel.AVCProfileMain);
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                    videoFormat.setInteger(MediaFormat.KEY_LEVEL, MediaCodecInfo.CodecProfileLevel.AVCLevel3);
                }
            }

            mVideoEncodec.configure(videoFormat,null,null,MediaCodec.CONFIGURE_FLAG_ENCODE);

            mBuffInfo = new MediaCodec.BufferInfo();
            mSurface = mVideoEncodec.createInputSurface();
        } catch (IOException e) {
            e.printStackTrace();
            mVideoEncodec=null;
            mBuffInfo=null;
            mSurface=null;
        }
    }


//开始录制
 public void startRecode(){
    videoEncodec.start();
    int outputBufferIndex = videoEncodec.dequeueOutputBuffer(videoBufferinfo, 0);
    if (outputBufferIndex == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED) {
        videoTrackIndex = mediaMuxer.addTrack(videoEncodec.getOutputFormat());
        mediaMuxer.start();
    }else {
        while (outputBufferIndex>=0){
            ByteBuffer outputBuffer= videoEncodec.getOutputBuffers()[outputBufferIndex];
            outputBuffer.position(videoBufferinfo.offset);
            outputBuffer.limit(videoBufferinfo.offset + videoBufferinfo.size);

            //设置时间戳
            if(pts==0){
                pts = videoBufferinfo.presentationTimeUs;
            }
            videoBufferinfo.presentationTimeUs = videoBufferinfo.presentationTimeUs - pts;
            //写入数据
            mediaMuxer.writeSampleData(videoTrackIndex,outputBuffer,videoBufferinfo);
            if(encoderWeakReference.get().onMediaInfoListener!=null){
                encoderWeakReference.get().onMediaInfoListener.onMediaTime((int) (videoBufferinfo.presentationTimeUs/1000000));
            }
            videoEncodec.releaseOutputBuffer(outputBufferIndex,false);
            outputBufferIndex = videoEncodec.dequeueOutputBuffer(videoBufferinfo, 0);
        }
    }
 }
 
//停止录制
public void stopRecode(){
    videoEncodec.stop();
    videoEncodec.release();
    videoEncodec =null;

    mediaMuxer.stop();
    mediaMuxer.release();
    mediaMuxer = null;
}

具体示例请看: https://github.com/ChinaZeng/SurfaceRecodeDemo

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • c++基础语法

    注意:在开发过程中,cpp或者c会被编译为dll或者so供其调用者使用,一般把public的函数定义到h文件,不然调用者都不知道有哪些函数。

    曾大稳
  • Android配置EGL环境

    Android配置egl环境我们根据GLSurfaceView源码来实现。在GLSurfaceView源码里面,当调用setRenderer的时候会开启一个线程...

    曾大稳
  • ffmpeg 音频播放器相关

    因为每一个AVframe的pts不一定都有,所以就需要自己手维护一个当前时间的变量

    曾大稳
  • acm C语言求两个数最大值

    kevinfaith
  • 更新c++学习笔记 第二章

    **先说说什么是重载:**C++ 允许多个函数拥有相同的名字,只要它们的参数列表不同就可以,这就是函数的重载(Function Overloading)。借助重...

    互联网CEO
  • 枚举+优化(4)——哈希表优化实例2

    mathor
  • 百练-2456-Aggressive cows

    ACM模版 描述 ? 题解 二分 + 贪心。水题。 代码 #include <iostream> #include <cstdio> #include <alg...

    f_zyj
  • POJ 3259 Wormholes(spfa判负环)

           题意是有n个点,m条边,k个虫洞(权值为负),输入完m条无向边后输入k条有向边,问能不能找到一个点,从这个点出发,最后回到这个点的时候权值是负的(...

    Ch_Zaqdt
  • 程序员进阶之算法练习(八)附两道搜狐笔试题

    前言 前面讲了那么多算法的重要性。口说无凭,这次带上两道搜狐今年的笔试题。 这里先附上两道搜狐题目的大意: 题目一: 《宝石》 有一串宝石首尾相连,用...

    落影
  • SpringMVC下获取验证码

    zcqshine

扫码关注云+社区

领取腾讯云代金券