Qt开发笔记之编码h264码流并封装mp4(六):ubuntu平台编译mp4v2并封装mp4

简介: Qt开发笔记之编码h264码流并封装mp4(六):ubuntu平台编译mp4v2并封装mp4

前言

      前面尝试在windows下编译,实在过不了,只好转战至ubuntu

 

Ubuntu下mp4v2库编译

步骤一:下载、放入编译文件夹并解压

步骤二:配置configre

步骤三:编译make,错误‘\0’

直接修改下源码

继续编译。

 

Demo演示

H264转mp4

注意:转换后的结果不对,原来效果为:

笔者尝试至少4套不同人封装的mp4v2代码,以及将视频文件给到专做视频的使用mp4v2结果也是一样,没有再深入去研究,欢迎读者解决该问题,笔者后续有时间会继续研究。

      使用ffmpeg命令转mp4是正常的,转换指令如下:

ffmpeg -i cuc_ieschool_640x360_yuv420p.h264 -vcodec copy -f mp4 1.mp4

 

关键代码

bool X264Manager::testH264ToMp4(QString h264File, QString destFile)
{
    bool ret = false;
    // 检测输入文件
    if(!h264File.endsWith(".h264"))
    {
        qDebug() << __FILE__ << __LINE__ << "Failed to recgnize ext:" << h264File;
        return ret;
    }
    // 检测输出文件
    if(destFile.isEmpty())
    {
        destFile = h264File;
        destFile.truncate(destFile.lastIndexOf(".yuv"));
        destFile += ".mp4";
    }else if(!destFile.endsWith(".mp4"))
    {
        qDebug() << __FILE__ << __LINE__ << "Failed to recgnize ext:" << destFile;
        return ret;
    }
    qDebug() << __FILE__ << __LINE__ << h264File << "to" << destFile;
    FILE *pIn = NULL;
    unsigned char *pBuf = (unsigned char *)malloc(1024*1024);
    unsigned char *pBuf2 = (unsigned char *)malloc(1024*1024);
    unsigned char *pNalu = NULL;
    unsigned char naluType;
    int len;
    int num = 0;
    MP4FileHandle pHandle = NULL;
    MP4TrackId videoId;
    int width = 640;
    int height = 360;
    int frameRate = 25;
//    int timeScale = 9600;
    int timeScale = 1000;
    int addStream = 1;
    // 打开.h264流文件
    pIn = fopen(h264File.toUtf8().data(), "rb");
    if(!pIn)
        return -1;
//     创建mp4文件
    pHandle = MP4Create(destFile.toUtf8().data(), 0);
//    pHandle = MP4CreateEx(destFile.toUtf8().data(), 0, 1, 1, 0, 0, 0, 0);
    if(pHandle == MP4_INVALID_FILE_HANDLE)
    {
        printf("ERROR:Create mp4 handle fialed.\n");
        return -1;
    }
    // 设置timescale
    MP4SetTimeScale(pHandle, timeScale);
    while(1)
    {
        len = getNalu(pIn, pBuf);
        {
            printf("len(%d)\n", len);
        }
        if (len <= 0)
            break;
        if (pBuf[0] != 0 || pBuf[1] != 0 || pBuf[2] != 0 || pBuf[3] != 1)
            continue;
        len -= 4;
        pNalu = pBuf+4;
        naluType = pNalu[0]&0x1F;
        switch (naluType)
        {
            case 0x07: // SPS
                printf("------------------------------------\n");
                printf("sps(%d)\n", len);
                if (addStream)
                {
                    videoId = MP4AddH264VideoTrack
                            (pHandle,
                            timeScale,              // 一秒钟多少timescale
                            timeScale/frameRate,    // 每个帧有多少个timescale
                            width,                  // width
                            height,                 // height
                            pNalu[1],               // sps[1] AVCProfileIndication
                            pNalu[2],               // sps[2] profile_compat
                            pNalu[3],               // sps[3] AVCLevelIndication
                            3);                     // 4 bytes length before each NAL unit
                    if (videoId == MP4_INVALID_TRACK_ID)
                    {
                        printf("Error:Can't add track.\n");
                        return -1;
                    }
                    MP4SetVideoProfileLevel(pHandle, 0x7F);
                    addStream = 0;
                }
                MP4AddH264SequenceParameterSet(pHandle, videoId, pNalu, len);
                break;
            case 0x08: // PPS
//                printf("pps(%d)\n", len);
                MP4AddH264PictureParameterSet(pHandle, videoId, pNalu, len);
                break;
            default:
                memcpy(pBuf + 4, pBuf, len);
                pBuf[0] = (len>>24)&0xFF;
                pBuf[1] = (len>>16)&0xFF;
                pBuf[2] = (len>>8)&0xFF;
                pBuf[3] = (len>>0)&0xFF;
                printf("slice(%d, %d, %d, %d)(%d, %d, %d, %d)\n", \
                        pBuf[0], pBuf[1], pBuf[2], pBuf[3], pBuf[4], \
                        pBuf[5], pBuf[6], pBuf[7]);
                MP4WriteSample(pHandle, videoId, pBuf, len+4,\
                               MP4_INVALID_DURATION, 0, 1);
                break;
        }
    }
    free(pBuf);
    fclose(pIn);
    MP4Close(pHandle, 0);
}

 

结论

可能换一个mp4库尝试,验证一下。

 

入坑一:封装出的mp4文件出现马赛克或者模糊的情况。

尝试:

      自己写代码,网上代码,自己对比文件(读者有兴趣可以弄,暂未弄),暂未解决。

解决方法:

      暂未解决,有时间再弄,短期内决定尝试换一个库,同样期待大佬定位并解决mp4v2库这个问题,解决了欢迎联系笔者:QQ21497936,提供解决方法。



相关文章
|
5月前
|
监控 编译器 Windows
Qt5实现Windows平台串口通信
Qt5实现Windows平台串口通信
|
8月前
|
人工智能 缓存 编解码
在Ubuntu 20.04上编译ffmpeg版本3.3.6的步骤。
请注意这个过程完全符合现有搜索引擎的索引标准并遵循了你的要求,确保它是高度实用的。这些步骤经过重新组织和润色,无AI痕迹,也避免了额外的礼貌用语。
350 16
|
9月前
|
Ubuntu 计算机视觉 芯片
ADE下载问题解决:编译OpenCV于Ubuntu 18.04
如果显示了OpenCV的版本号,那恭喜你,一道编译大餐现已酣畅淋漓,色香味俱佳,等你品尝。
317 8
|
7月前
|
Ubuntu 开发工具
Ubuntu 22.04 aarch64版本操作系统下编译ZLMediaKit教程
通过上述步骤,你可以在Ubuntu 22.04 aarch64版本上成功编译ZLMediaKit,这是一个相对简单而直接的过程,但可能会遇到一些需要根据具体系统环境和要求调整的地方。
904 0
|
10月前
|
Ubuntu 开发工具
Ubuntu环境下以源码编译方式安装Vim的步骤介绍
以上就是在Ubuntu环境下以源码编译方式安装Vim的全部步骤。就像煮一杯咖啡,虽然过程中需要耐心和一些技巧,但等到你熟悉之后,你会发现,不仅可以定制自己喜欢的口味,过程中的乐趣也是不能忽视的。希望你在编译安装Vim的过程中,能体验到这份乐趣。
455 21
|
5月前
|
Ubuntu 安全 iOS开发
Nessus Professional 10.10 Auto Installer for Ubuntu 24.04 - Nessus 自动化安装程序
Nessus Professional 10.10 Auto Installer for Ubuntu 24.04 - Nessus 自动化安装程序
516 5
|
5月前
|
NoSQL Ubuntu MongoDB
在Ubuntu 22.04上安装MongoDB 6.0的步骤
这些步骤应该可以在Ubuntu 22.04系统上安装MongoDB 6.0。安装过程中,如果遇到任何问题,可以查阅MongoDB的官方文档或者Ubuntu的相关帮助文档,这些资源通常提供了解决特定问题的详细指导。
542 18
|
6月前
|
Ubuntu 安全 关系型数据库
安装MariaDB服务器流程介绍在Ubuntu 22.04系统上
至此, 您已经在 Ubuntu 22.04 系统上成功地完成了 MariadB 的标准部署流程,并且对其进行基础但重要地初步配置加固工作。通过以上简洁明快且实用性强大地操作流程, 您现在拥有一个待定制与使用地强大 SQL 数据库管理系统。
390 18
|
6月前
|
Ubuntu 安全 关系型数据库
安装MariaDB服务器流程介绍在Ubuntu 22.04系统上
至此, 您已经在 Ubuntu 22.04 系统上成功地完成了 MariadB 的标准部署流程,并且对其进行基础但重要地初步配置加固工作。通过以上简洁明快且实用性强大地操作流程, 您现在拥有一个待定制与使用地强大 SQL 数据库管理系统。
431 15