热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

stagefright架构分析(五)初始化VideoDecoder

VideoDecoder的初始化实际就是OMX的创建,Stagefright提供了OMX的封装OMXCodec在平台上stagefright-l可以看到所有的组件VideoDec

VideoDecoder的初始化实际就是OMX的创建,Stagefright提供了OMX的封装OMXCodec

在平台上stagefright -l 可以看到所有的组件


VideoDecoder的创建流程

status_t AwesomePlayer::initVideoDecoder(uint32_t flags) {

//最重要的创建mVideoSource

    sp decoder = OMXCodec::Create(
            mClient.interface(), mVideoTrack->getFormat(),
            false, // createEncoder
            mVideoTrack,
            NULL, flags, USE_SURFACE_ALLOC ? mNativeWindow : NULL);

        status_t err = mVideoSource->start();

}


sp OMXCodec::Create(
        const sp &omx,
        const sp &meta, bool createEncoder,
        const sp &source,
        const char *matchComponentName,
        uint32_t flags,
        const sp &nativeWindow) {

//得到MediaExtractor解析的MIME

    bool success = meta->findCString(kKeyMIMEType, &mime);

//找到对应的decoder name

    findMatchingCodecs(
            mime, createEncoder, matchComponentName, flags,
            &matchingCodecs, &matchingCodecQuirks);

//可能会找到多个匹配的codec

    for (size_t i = 0; i

//得到对应的component name

        const char *compOnentNameBase= matchingCodecs[i].string();

//创建软件codec

            softwareCodec = InstantiateSoftwareDecoder(componentName, source);

//根据component name创建OMX NODE

        status_t err = omx->allocateNode(componentName, observer, &node);

//创建OMXCodec

            sp codec = new OMXCodec(
                    omx, node, quirks, flags,
                    createEncoder, mime, componentName,
                    source, nativeWindow);

            observer->setCodec(codec);

//根据meta配置codec
            err = codec->configureCodec(meta);

//最终返回的是OMXCodec

return codec;

}

如何找到对应的codec name?

void OMXCodec::findMatchingCodecs(
        const char *mime,
        bool createEncoder, const char *matchComponentName,
        uint32_t flags,
        Vector *matchingCodecs,
        Vector *matchingCodecQuirks) {
//得到codec list

    const MediaCodecList *list = MediaCodecList::getInstance();

    for (;;) {

//从list中找到一个对应的codec name
        ssize_t matchIndex =
            list->findCodecByType(mime, createEncoder, index);

        if (matchIndex <0) {
            break;
        }

//找到后,就从找到的index的下一个继续查找
        index = matchIndex + 1;

//得到ComponentName,并把其放到matchingCodecs中
        const char *compOnentName= list->getCodecName(matchIndex);

            matchingCodecs->push(String8(componentName));

    }

}

MediaCodecList是怎么创建的?
//这是SINGLETON模型
const MediaCodecList *MediaCodecList::getInstance() {
    Mutex::Autolock autoLock(sInitMutex);
    if (sCodecList == NULL) {
        sCodecList = new MediaCodecList;
    }
    return sCodecList->initCheck() == OK ? sCodecList : NULL;
}
//这个list是从/etc/media_codecs.xml中解析读到的
MediaCodecList::MediaCodecList()
    : mInitCheck(NO_INIT) {
    FILE *file = fopen("/etc/media_codecs.xml", "r");
    if (file == NULL) {
        ALOGW("unable to open media codecs configuration xml file.");
        return;
    }
//解析的协议遵从xml标准,具体可以看device中对应的设备media_codecs.xml
    parseXMLFile(file);
    if (mInitCheck == OK) {
        // These are currently still used by the video editing suite.
        addMediaCodec(true /* encoder */, "AACEncoder", "audio/mp4a-latm");
        addMediaCodec(
                false /* encoder */, "OMX.google.raw.decoder", "audio/raw");
        Vector QcomAACQuirks;
        QcomAACQuirks.push(AString("requires-allocate-on-input-ports"));
        QcomAACQuirks.push(AString("requires-allocate-on-output-ports"));
        addMediaCodec(false, "OMX.qcom.audio.decoder.multiaac",
            "audio/mp4a-latm", getCodecSpecificQuirks(QcomAACQuirks));
    }
}
OMXCodec::Create中的OMX是哪来的?
在AwesomePlayer中调用OMXCodec::Create时,传入的OMX是mClient.interface()
OMXClient 的结构:
class OMXClient {
public:
    OMXClient();
    status_t connect();
    void disconnect();
    sp interface() {
        return mOMX;
    }
private:
    sp mOMX;
    OMXClient(const OMXClient &);
    OMXClient &operator=(const OMXClient &);
};

mClient的初始化是在AwesomePlayer构造时完成的,connect实际是其初始化。
AwesomePlayer::AwesomePlayer() {
    CHECK_EQ(mClient.connect(), (status_t)OK);
}

status_t OMXClient::connect() {
//OMX实际上是从MediaPlayerService中得到的
    sp service = interface_cast(binder);
    mOMX = service->getOMX();
}

//MediaPlayerService只有一个OMX实例
sp MediaPlayerService::getOMX() {
    Mutex::Autolock autoLock(mLock);
    if (mOMX.get() == NULL) {
        mOMX = new OMX;
    }
    return mOMX;
}

OMX如何创建codec对应的component
status_t OMX::allocateNode(
        const char *name, const sp &observer, node_id *node) {
//name是通过findMatchingCodecs找到的component name
    OMXNodeInstance *instance = new OMXNodeInstance(this, observer);
//得到实际的ComponentInstance
    OMX_ERRORTYPE err = mMaster->makeComponentInstance(
            name, &OMXNodeInstance::kCallbacks,
            instance, &handle);
}
OMX_ERRORTYPE OMXMaster::makeComponentInstance(
        const char *name,
        const OMX_CALLBACKTYPE *callbacks,
        OMX_PTR appData,
        OMX_COMPONENTTYPE **component) {
//根据name得到list中的index
    ssize_t index = mPluginByComponentName.indexOfKey(String8(name));
//得到对应的plugin
    OMXPluginBase *plugin = mPluginByComponentName.valueAt(index);
//创建ComponentInstance
    OMX_ERRORTYPE err =
        plugin->makeComponentInstance(name, callbacks, appData, component);
    if (err != OMX_ErrorNone) {
        return err;
    }
    mPluginByInstance.add(*component, plugin);
    return err;
}

codec 参数配置
//通过meta进行配置
err = codec->configureCodec(meta);
status_t OMXCodec::configureCodec(const sp &meta) {
//得到ESDS,解析出来得到codec_specific_data
        if (meta->findData(kKeyESDS, &type, &data, &size)) {
            ESDS esds((const char *)data, size);
            CHECK_EQ(esds.InitCheck(), (status_t)OK);


            const void *codec_specific_data;
            size_t codec_specific_data_size;
            esds.getCodecSpecificInfo(
                    &codec_specific_data, &codec_specific_data_size);


            meta->findCString(kKeyMIMEType, &mime_type);
            if (strncmp(mime_type, MEDIA_MIMETYPE_AUDIO_MPEG, 10)) {
            addCodecSpecificData(
                    codec_specific_data, codec_specific_data_size);
            }
//后面都是类似的操作,根据不同的codec,配置不同的信息,例如:
else if (!strcasecmp(MEDIA_MIMETYPE_AUDIO_AC3, mMIME)) {
        int32_t numChannels, sampleRate;
        CHECK(meta->findInt32(kKeyChannelCount, &numChannels));
        CHECK(meta->findInt32(kKeySampleRate, &sampleRate));
        setAC3Format(numChannels, sampleRate);
    }
//初始化OutputFormat
    initOutputFormat(meta);
//初始化NativeWindow
    if (mNativeWindow != NULL
        && !mIsEncoder
        && !strncasecmp(mMIME, "video/", 6)
        && !strncmp(mComponentName, "OMX.", 4)) {
        status_t err = initNativeWindow();
        if (err != OK) {
            return err;
        }
    }

推荐阅读
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Python爬虫中使用正则表达式的方法和注意事项
    本文介绍了在Python爬虫中使用正则表达式的方法和注意事项。首先解释了爬虫的四个主要步骤,并强调了正则表达式在数据处理中的重要性。然后详细介绍了正则表达式的概念和用法,包括检索、替换和过滤文本的功能。同时提到了re模块是Python内置的用于处理正则表达式的模块,并给出了使用正则表达式时需要注意的特殊字符转义和原始字符串的用法。通过本文的学习,读者可以掌握在Python爬虫中使用正则表达式的技巧和方法。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • http:my.oschina.netleejun2005blog136820刚看到群里又有同学在说HTTP协议下的Get请求参数长度是有大小限制的,最大不能超过XX ... [详细]
  • 关键词:Golang, Cookie, 跟踪位置, net/http/cookiejar, package main, golang.org/x/net/publicsuffix, io/ioutil, log, net/http, net/http/cookiejar ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • Java在运行已编译完成的类时,是通过java虚拟机来装载和执行的,java虚拟机通过操作系统命令JAVA_HOMEbinjava–option来启 ... [详细]
  • 怎么在PHP项目中实现一个HTTP断点续传功能发布时间:2021-01-1916:26:06来源:亿速云阅读:96作者:Le ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法
    本文介绍了解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法,包括检查location配置是否正确、pass_proxy是否需要加“/”等。同时,还介绍了修改nginx的error.log日志级别为debug,以便查看详细日志信息。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 图像因存在错误而无法显示 ... [详细]
  • 本文讨论了在VMWARE5.1的虚拟服务器Windows Server 2008R2上安装oracle 10g客户端时出现的问题,并提供了解决方法。错误日志显示了异常访问违例,通过分析日志中的问题帧,找到了解决问题的线索。文章详细介绍了解决方法,帮助读者顺利安装oracle 10g客户端。 ... [详细]
  • 本文详细介绍了使用C#实现Word模版打印的方案。包括添加COM引用、新建Word操作类、开启Word进程、加载模版文件等步骤。通过该方案可以实现C#对Word文档的打印功能。 ... [详细]
  • Ihaveaworkfolderdirectory.我有一个工作文件夹目录。holderDir.glob(*)>holder[ProjectOne, ... [详细]
author-avatar
f永远喜爱捉迷藏
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有