Android codec2 编码 -- 基于录屏

2023-09-13 20:32:34

前言

本篇文章主要是理解Android 12编码的流程, 首先从上层的应用出发理解mediacodec提供给外部API的用法。然后针对每个api 分析编码各个流程中框架中的流程。

熟悉一个框架的流程 可以从简单到复杂、从整体到局部去展开。 同时在理解过中会产生各种各样的问题,各种问题的解决就是一个知识经验的形成过程。

android 原生的应用srcreenrecord

  • 应用和代码路径

代码路径:frameworks\av\cmds\screenrecord\screenrecord.cpp

编译生成的是screenrecord在system/bin目录,默认在android系统都会携带。

使用命令:这个命令会将屏幕的操作录制到/sdcard/test.mp4下。

 screenrecord /sdcard/test.mp4
  • 应用流程

    • 首先在编码器mediacodec调用createInputSurface创建一个inputSurface。这个inputSurface传递出来到显示 作为虚拟显示的bufferProducer。
    • 在surfaceFlinger 端,inputSurface作为prepareVirtualDisplay的参数, 使得surfaceFlinger从这个surface中获取bufffer, 然后将屏幕合成后的数据写到这个buffer里面。
    • 在编码端将这个buffer 作为编码的输入进行处理。mediacodec编码完成之后调用dequeueOutputBuffer 将编码之后的数据取出来写到文件,然后调用releaseOutputBuffer将这个buffer释放回去。
    • 在编码器这边,surfaceflinger 是生产端,mediacodec是消费端。其他有关屏幕录制或者surface 直接到编码的流程大概都是这样的。
    创建编码器,创建输入的surface,配置format,启动编码器
    sp<AMessage> format = new AMessage;
    format->setInt32(KEY_WIDTH, gVideoWidth);
    format->setInt32(KEY_HEIGHT, gVideoHeight);
    .....
    codec = MediaCodec::CreateByType(looper, kMimeTypeAvc, true);    
    err = codec->configure(format, NULL, NULL,
               MediaCodec::CONFIGURE_FLAG_ENCODE);
    err = codec->createInputSurface(&bufferProducer);
    err = codec->start();
    
    err = prepareVirtualDisplay(displayState, bufferProducer, &dpy);
    
    从编码器中取出buffer,后续就是将这个buffer写入到mp4文件中。
    Vector<sp<MediaCodecBuffer> > buffers;
    err = encoder->getOutputBuffers(&buffers);
    err = encoder->dequeueOutputBuffer(&bufIndex, &offset, &size, &ptsUsec,
    &flags, kTimeout);
    

上述流程的疑问?

  1. mediacodec是如何将surface的数据取出来 然后进行编码的?

    相对应于解码的流程,会有一个queueInbufferbuffer 将未解码的数据喂给mediacodec,而在编码这边编码器只有一个从codec创建出来的Surface,这个surface会配置到surfaceFlinger那边的虚拟显示中。

MediaCodec获取编码数据流程

代码路径:
frameworks\av\media\libstagefright\MediaCodec.cpp
frameworks\av\media\codec2\sfplugin\CCodec.cpp
frameworks\av\media\libstagefright\bqhelper\GraphicBufferSource.cpp
frameworks\av\media\codec2\sfplugin\C2OMXNode.cpp

简单的理解可以分为这两个路径:

  1. 生产者: surfaceFlinger从MediaCodec创建的InputSurface中申请buffer,然后将各个图层的数据合成到这块buffer中,合成后 通知到消费者 也就是componet这一端。
  2. 消费者:componet收到生产者surfaceFlinger的通知后,将这个合成的buffer给到硬解或者软解编码器进行编码。编码后的数据,外部应用通过dequeueOutputBuffer可以获取到。

这里我们关注消费者这一端的实现。

  • mediacodec creatInputSurface

    • 调用流程
      遵循 mediacodec—>ccodec这样的流程,ccodec调用的是codec2client。client 通过HIDL调用到componentstore端(在IComponetSotore.hal中有定义了这样的接口C2PlatformComponentStore–>componentStore–>IComponetSotore 具体用vendor定义的还是default google实现的 是在之前service端创建服务的时候决定的)。

    • createInputSurface

      创建了GraphicBufferSource, 在这个类的初始化中调用BufferQueue::createBufferQueue
      创建Producer和Consumer,通过将GraphicBufferSource监听注册到mConsumer中,
      这里就是onFrameAvailable注册的地方。Producer和GraphicBufferSource会封装到InputSurface 返回到codec2client。

    • GraphicBufferSourceWrapper的connect
      创建好之后的InputSurface会强制转换为GraphicBufferSourceWrapper,然后会调用这个类的connect。connect中是创建了C2OMXNode,传递的参数是之前MediaCodec::CreateByType
      创建的componet。同时通过调用GraphicBufferSource::configure,将这个C2OMXNode配置到GraphicBufferSource的mComponent。

    CCodec::createInputSurface()
            int32_t width = 0;
            (void)outputFormat->findInt32("width", &width);
            int32_t height = 0;
            (void)outputFormat->findInt32("height", &height);
            err = setupInputSurface(std::make_shared<GraphicBufferSourceWrapper>(
                    gbs, width, height, usage));
            bufferProducer = persistentSurface->getBufferProducer();
    
    CCodec::setupInputSurface:
       status_t err = mChannel->setInputSurface(surface);
    
    CCodecBufferChannel::setInputSurface:
         mInputSurface->connect(mComponent);
    
    class GraphicBufferSourceWrapper : public InputSurfaceWrapper {
    connect(const std::shared_ptr<Codec2Client::Component> &comp) {
            mNode = new C2OMXNode(comp);
            mOmxNode = new hardware::media::omx::V1_0::utils::TWOmxNode(mNode);
            mNode->setFrameSize(mWidth, mHeight);
            // Usage is queried during configure(), so setting it beforehand.
            OMX_U32 usage = mConfig.mUsage & 0xFFFFFFFF;
            (void)mNode->setParameter(
                    (OMX_INDEXTYPE)OMX_IndexParamConsumerUsageBits,
                    &usage, sizeof(usage));
            mSource->configure(
                    mOmxNode, static_cast<hardware::graphics::common::V1_0::Dataspace>(mDataSpace));
            return OK;
    }
    }
    
    status_t GraphicBufferSource::configure(
            const sp<ComponentWrapper>& component,
            int32_t dataSpace,
            int32_t bufferCount,
            uint32_t frameWidth,
            uint32_t frameHeight,
            uint32_t consumerUsage)
    {
        mComponent = component;
    }
    
  • onFrameAvailable
    • 通过之前在GraphicBufferSource注册onFrameAvailable到producer中lister,当合成后又buffer 可用的时候,会回调到GraphicBufferSource的onFrameAvailable。
    • onFrameAvailable经过一系列的处理 会调用到mComponent->submitBuffer,这个调用C2OMXNode->emptyBuffer。
      c2OMXNode这边将这块omxBuf 封装成c2Buffer,然后queue到c2OMXNode 的队列中去。C2OMXNode有专门的mQueueThread来把队列中c2works queue到Codec2Client中。
    • 在client中的Codec2Client::Component::queue()在调用 mComponent->queue_nb(&c2works)。
      mComponet 是simpleC2Componet, 在其中的queue_nb会把上面传递的items 放到componet的mWorkQueue中,然后发送kWhatProcess消息, 收到消息后调用processQueue。然后调用各个组件的process。
// BufferQueue::ConsumerListener callback
void GraphicBufferSource::onFrameAvailable(const BufferItem& item __unused) 
   onBufferAcquired_l(buffer);

void GraphicBufferSource::onBufferAcquired_l(const VideoBuffer &buffer)
	fillCodecBuffer_l();

bool GraphicBufferSource::fillCodecBuffer_l() {
	err = submitBuffer_l(item); 

status_t GraphicBufferSource::submitBuffer_l(const VideoBuffer &item)
    status_t err = mComponent->submitBuffer(
            codecBufferId, graphicBuffer, codecTimeUs, buffer->getAcquireFenceFd());
      
class OmxComponentWrapper : public ComponentWrapper {
    status_t submitBuffer(
            int32_t bufferId, const sp<GraphicBuffer> &buffer,
            int64_t timestamp, int fenceFd) override {
        ALOGD("submitBuffer bufferId:%d", (int)bufferId);
        return mOmxNode->emptyBuffer(
                bufferId, OMX_BUFFERFLAG_ENDOFFRAME, buffer, timestamp, fenceFd);
    }
  
 status_t C2OMXNode::emptyBuffer(
        buffer_id buffer, const OMXBuffer &omxBuf,
        OMX_U32 flags, OMX_TICKS timestamp, int fenceFd) {
        mQueueThread->queue(comp, fenceFd, std::move(work), std::move(fd0), std::move(fd1));
    }
 
class C2OMXNode::QueueThread : public Thread {
 protected:
    bool threadLoop() override {
   	 comp->queue(&items);
    }

总结: Android 录屏编码这一部分 调用的路径非常长,主要连接surface和componet的是GraphicBufferSource类。在这里监听surface buffer的生成,并将其传递给编码的componet。

请添加图片描述

更多推荐

AIGC赋能教育 | 虚拟现实:下一代的教室在哪里?

在这个科技迅速进步的时代,全球教育领域正迎来一场前所未有的数字化和技术化变革。而AIGC则是代表了人工智能与教育的融合,它正在改变着学习方式、提高教育质量,并重新定义了教育的未来。我们正在亲眼目睹AIGC如何塑造着全球教育的格局,为学生、教育者和整个社会带来了广泛的机遇和挑战。这个崭新的时代不仅带来了无限可能性,也需要

DDoS是什么?

一.DDoS的定义DDos的前身DoS(DenialofService)攻击,其含义是拒绝服务攻击,这种攻击行为使网站服务器充斥大量的要求回复的信息,消耗网络带宽或系统资源,导致网络或系统不胜负荷而停止提供正常的网络服务。而DDoS分布式拒绝服务,则主要利用Internet上现有机器及系统的漏洞,攻占大量联网主机,使其

SpringMVC之JSON数据返回与异常处理机制

目录一.SpringMVC的JSON数据返回1.导入Maven依赖2.配置spring-mvc.xml3.@ResponseBody注解的使用3.1案例演示1.List集合转JSON2.Map集合转JSON3.返回指定格式String4.@ResponseBody用法5.Jackson5.1介绍5.2常用注解二.异常处

SwiftUI 5.0(iOS 17)TipKit 让用户更懂你的 App

概览作为我们秃头开发者来说,写出一款创意炸裂的App还不足以吸引用户眼球,更重要的是如何让用户用最短的时间掌握我们App的使用技巧。从iOS17开始,推出了全新的TipKit框架专注于此事。有了它,我们再也不用自己写App用户帮助以及使用指南的逻辑和界面了。使用TipKit非常简单,接下来就让我们一起走进TipKit

对于项目中爱抱怨成员,项目经理如何回应?

李佳刚被提拔成项目经理。他是技术部的骨干人员,工作能力很强,已经干了快五年了。公司规模中等,属于比较传统的制造型公司。随着公司升级转型,项目越来越多,难度也增加了。原来那种按部就班的订单式生产被更加灵活、个性化的客户需求所取代,不但技术难度提高了,完成一个项目所涉及的部门、协作单位也增多了。李佳由于出色的工作表现,因此

Java手写决策树和决策树应用拓展案例

Java手写决策树和决策树应用拓展案例1.算法思维导图以下是用mermaid代码表示的决策树算法实现原理的思维导图:#mermaid-svg-DWczpr7jTBRHS9bA{font-family:"trebuchetms",verdana,arial,sans-serif;font-size:16px;fill:#

JMeter:接口测试基础介绍

一、什么是接口接口是非常抽象的概念,先来看下中国最大的综合性辞典《辞海》是怎样定义接口的:两个不同系统或系统中两个不同特性部分的交接部分。一般分硬件接口和软件接口两种。前者是为连接计算机各部分之间、计算机与计算机之间、计算机与外部系统之间而专门设计的连接线及有关逻辑控制电路;后者是为连接两个程序层或块而专门设计的程序或

怎么在树莓派上搭建WordPress博客网站,并发布到外网可访问?

文章目录序幕概述1.安装PHP2.安装MySQL数据库3.安装Wordpress4.设置您的WordPress数据库设置MySQL/MariaDB创建WordPress数据库5.WordPressconfiguration6.将WordPress站点发布到公网安装相对URL插件修改config.php配置7.支持好友链

基于Java咖啡商品管理系统设计实现(源码+lw+部署文档+讲解等)

博主介绍:✌全网粉丝30W+,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌🍅文末获取源码联系🍅👇🏻精彩专栏推荐订阅👇🏻不然下次找不到哟2022-2024年最全的计算机软件毕业设计选题

WebLOAD: 一站式性能测试工具

WebLOAD是一款一站式前端性能测试工具,对测试人员来说使用非常方便。它可以帮助前端工程师和测试快速对网页进行性能测试和优化,提高网页加载速度,减少页面卡顿和闪烁。WebLOAD的特点、使用指南以及企业实际使用中的案列。WebLOAD的特点功能丰富:WebLOAD集成了众多前端性能测试工具,如前端性能分析、首屏时间预

算法刷题 week1 找出数组中重复的数字&不修改数组找出重复的数字

目录week11.找出数组中重复的数字题目数据范围样例题解(数组遍历)O(n)2.不修改数组找出重复的数字题目数据范围样例题解(分治,抽屉原理)O(nlogn)week11.找出数组中重复的数字题目给定一个长度为n的整数数组nums,数组中所有的数字都在0∼n−1的范围内。数组中某些数字是重复的,但不知道有几个数字重复

热文推荐