视频打包为MP4格式并存储到TF卡的实现

一、MP4文件格式解析

1、视频文件总体介绍

(1)视频文件的本质:记录压缩后的视频帧并且能被播放器还原 解码播放(关键是找到视频的I帧,P帧)

(2)视频文件的关键:高效率记录信息、兼容性

(3)视频文件的信息:索引信息、有效信息

2、MP4格式总体介绍

(1)MP4由MP3(纯audio)升级而来,包含video和audio在内

(2)MP4是h.264的最主流打包格式

(3)MP4文件内部**采用网络字节序(大端模式) **

3、MP4学习路线

(1)学习MP4的组织形式和box解析

(2)移植和使用mp4v2开源库来打包MP4

(3)进一步研究MP4解包播放和mp4v2源码

(4)自己编程进行MP4的打包、解包、分割等。

4、正式开始MP4的组织形式的学习

参考学习(必读):https://www.cnblogs.com/ranson7zop/p/7889272.html

  MP4视频文件封装格式是基于QuickTime容器格式定义的,因此参考QuickTime的格式定义对理解MP4文件格式很有帮助。MP4文件格式是一个十分开放的容器,几乎可以用来描述所有的媒体结构,MP4文件中的媒体描述与媒体数据是分开的,并且媒体数据的组织也很自由,不一定要按照时间顺序排列,甚至媒体数据可以直接引用其他文件。同时,MP4也支持流媒体(通过网络来边下载边播放)。MP4目前被广泛用于封装h.264视频和AAC音频,是高清视频的代表。

(1)整个MP4文件由若干个各种不同的box组成,打包和解包时都是以box为单位的

(2)MP4中有且只有一个ftyp box(file type box),该box位于整个MP4的开头位置

二、MP4Info工具使用

所用到的软件及资料链接:

链接:https://pan.baidu.com/s/1DFbZPI4ch_y33RUAE6whiA 
提取码:pu15 
--来自百度网盘超级会员V5的分享

第一步:打开该软件
在这里插入图片描述
第二步:使用该软件打开一个MP4视频文件,就会出现下面的情况:
在这里插入图片描述
图中右半部分的信息通过解析box得到。
在这里插入图片描述
该软件无法打开太大的视频文件,只能打开一些小的视频文件用于分析。

三、mp4v2移植和播放实战

  mp4v2 库是一个专用于处理mp4容器的开源项目,其使用c++编写,并提供c语言接口。

1、下载mp4v2

(1)https://launchpad.net/ubuntu/+source/mp4v2/2.0.0~dfsg0-6

(2)解压,并在目录内创建_install目录作为安装目录

2、配置并编译

(1)导出环境变量,使得可以找到交叉编译链:
sudo PATH=$PATH:/opt/hisi-linux/x86-arm/arm-hisiv300-linux/target/bin CC=arm-hisiv300-linux-gcc CXX=arm-hisiv300-linux-gcc 

进行配置:
./configure --host=arm-hisiv300-linux --prefix=/home/aston/sambashare/mp4v2-2.0.0/_install  --disable-option-checking --disable-debug --disable-optimize --disable-fvisibility --disable-gch --disable-largefile --disable-util --disable-dependency-tracking --disable-libtool-lock

注:_install  这个文件夹需要我们手动创建,最终编译生成的文件会存放在这个文件夹中
(2)make
(3)make install
(4)检查各必要文件

3、部署

(1)生成的lib加到mpp lib里面

(2)生成的include下文件添加到海思SDK 中的mpp/include中去

(3)生成的lib/的动态库文(* so *)件加到开发板lib目录下(通过开发板挂载到主机的那个目录)

  若出现存储空间不够用的情况,可将一些不用的文件(如usb网卡驱动)删除掉或者放在开发板挂载的/mnt目录中。

cp ~/sambashare/mp4v2-2.0.0/_install/lib/* ./

cp ~/sambashare/mp4v2-2.0.0/_install/include/* ./ -rf

4、编译sample

(1)用提供的sample文件夹替换掉原来的sample,或者给提供的文件夹起个新的名字sample_mp4

链接:https://pan.baidu.com/s/1LMNPxBzWeI-94L8zKCVhpA 
提取码:gu90 
--来自百度网盘超级会员V5的分享

(2)make
在这里插入图片描述

6、准备TF卡

参考学习(了解扩充):https://blog.csdn.net/li_wen01/article/details/79929730
(1)TF卡格式化为FAT32文件系统,若失败可试试低层格式化软件如SDFormat之类
在这里插入图片描述
(2)开机后将TF卡进行挂载

 mount -t vfat /dev/mmcblk0p1 /usr/mmc

在这里插入图片描述
  要是内存卡太大,如16G则并不一定能用,无法识别到分区,不能工作。我用了一张8G的成功了。
在这里插入图片描述
(3)先检测下TF卡可用,如在电脑创建一个文件,在开发板去读写。

7、运行和测试

(1)运行上边步骤中make编译生成的sample_venc可执行文件,使用rtsp方式测试,确认有图像

(2)终止程序,取出TF卡用读卡器接电脑,通过电脑的视频播放软件查看

注:在进行功能测试试,rtsp并未正常工作,但mp4视频打包成功了,分析猜测原因可能是进行视频打包时打印信息太多影响了rtsp的正常工作。

四、MP4打包源码解析

源码:即上边那个网盘链接,将mpp文件中的所有文件通过sourceinsight软件建立工程进行分析。
在这里插入图片描述
参考学习:https://blog.csdn.net/weixin_42462202/article/details/90108485

HI_S32 SAMPLE_COMM_VENC_MP4(VENC_STREAM_S *stStream)//该函数运行一次只负责将一帧数据打包
{
	static int nRecordFlag = 0x00;//这些static变量的值只有在第一次程序运行被初始化
	static int recording = 0x1;   //之后的每次调用,都会使用上次保留下来的值
	static int spsflag = 0;
	static int ppsflag = 0;
	static MP4TrackId video = 0;
	static MP4FileHandle hMP4File = NULL;

	static char recordfish = 0x1;
	
	int j = 0;
	int len = 0;
	char *pData = NULL;
	char isSyncSample = 0;
	
	if(recordfish == 0x00){
		return 0;
	}

	if(hMP4File == NULL){//第一次进入的时候
		hMP4File = MP4CreateEx("/usr/mmc/test.mp4",0, 1, 1, 0, 0, 0, 0);//创建一个mp4文件
		if (hMP4File == MP4_INVALID_FILE_HANDLE)	{
			printf("open file fialed.\n");
			return -1;
		}
	
		MP4SetTimeScale(hMP4File, 90000);//设置时间戳
	}

	//recording:表示正在记录,正在打包,一个序列由1sps+1pps+sei+1I帧+29P帧
	if(recording && stStream->u32Seq > 30){	//丢弃前30帧,也可以不丢弃,都是为了确保从I帧开始
		if(stStream->u32PackCount >= 3){//sps,pps,sei,从I帧开始编码,保证文件开始就能播放
			nRecordFlag = 1;
		}
	
		if(nRecordFlag){
			for(j = 0;j < stStream->u32PackCount;j++){
				len 	= stStream->pstPack[j].u32Len - stStream->pstPack[j].u32Offset;
				pData	= (stStream->pstPack[j].pu8Addr + stStream->pstPack[j].u32Offset);						
	
				if(stStream->pstPack[j].DataType.enH264EType == H264E_NALU_SPS){
					if(spsflag == 0x00){
						spsflag = 0x1;
						//写sps
						printf("Write sps =================\n");	
	
						//添加track,相当于添加一路流	
						video = MP4AddH264VideoTrack(hMP4File, 90000, 90000 / 30, 1280, 720,
																pData[4+1], //sps[1] AVCProfileIndication
																pData[4+2], //sps[2] profile_compat
																pData[4+3], //sps[3] AVCLevelIndication
																3); // 4 bytes length before each NAL unit
							MP4SetVideoProfileLevel(hMP4File, 0x7F);//视频的级别
							MP4AddH264SequenceParameterSet(hMP4File, video, pData+4, len-4); //添加sps到mp4文件										
					}
					
					continue;
				}
				
				if(stStream->pstPack[j].DataType.enH264EType == H264E_NALU_PPS){
					if(ppsflag == 0x00){
						ppsflag = 0x1;
						//添加pps									
						printf("Write pps -------------------\n");										
						MP4AddH264PictureParameterSet(hMP4File, video, pData+4, len-4);//添加pps到mp4文件
					}
					
					continue;
				}
	
				isSyncSample = (stStream->pstPack[j].DataType.enH264EType == H264E_NALU_ISLICE)	?  (1) : (0);
				pData[0] = (len - 4) >> 24;//构建网络字节序,减四,去掉了分隔符0001的长度
				pData[1] = (len - 4) >> 16;
				pData[2] = (len - 4) >> 8;
				pData[3] = len - 4; 							
	
				printf("Write date type = %d  isSyncSample = %d\n",stStream->pstPack[j].DataType.enH264EType,isSyncSample);								
	
				MP4WriteSample(hMP4File, video, pData, len , MP4_INVALID_DURATION, 0, isSyncSample);//写入nalu
	
				
			}					
		}
	}
	
	if((recording && stStream->u32Seq > 900)){//控制文件时长
		recording = 0x00;
		printf("Close mp4 file\n");						
		MP4Close(hMP4File, 0);//关闭mp4文件,在写入mp4文件后必须调用此函数,不然mp4文件会损坏
		hMP4File = NULL;
		video = 0;
		recordfish = 0x00;
	}

}

五、mp4v2结合MP4Info学习分析

1、思路

(1)修改MP4打包源码,用MP4Info查看录制的MP4细节

(2)再深度:修改mp4v2源码中细节,编译执行,再打包MP4视频查看分析

2、实践1:去掉sps

  通过测试,去掉sps后的视频VLC播放器以及其他影视播放器(腾讯等),均无法播放。

3、实践2:去掉pps

  通过测试,去掉pps后的视频VLC播放器以及其他影视播放器(腾讯等),均无法播放。
在这里插入图片描述
图中自上而下依次是完整MP4视频,去掉sps,去掉pps。

六、添加网络telnet调试

1、为什么添加telnet调试

(1)嵌入式linux系统的用户界面就是commandline,本质上由busybox提供

(2)busybox的命令行只有1个,一旦前台被占用就无法做其他操作

(3)解决方案有2个:一个是建立多个commandline,一个是开放其他用户界面。

2、telnet调试的原理

在这里插入图片描述
(1)在开发板中提前运行telnetd(这个软件在海思提供的系统中已经自带了)
在这里插入图片描述
telnet:客户端程序,是开发板远程登陆别的服务器使用的
telnetd:服务器程序,是一个守护进程,是其他客户端登陆开发板使用的

(2)远程通过telnet的client连接server,构建一个用户界面

(3)这是非常传统典型的远程登录的方式···其实用过的

3、在HI3518E开发板上telnet远程登录调试实战

(1)命令行执行 telnetd &,然后Windows打开SecureCRT(其他具有相同功能的软件也可,如MobaXterm)配置SSH至192.168.1.10,即开发板网口
在这里插入图片描述
在这里插入图片描述
(2)在CRT界面输入:

用户名ssid:root
password:因为我的开发板没有设置密码,直接回车,进入

在这里插入图片描述
(3)问题:若遇到不断重启,将etc/profile 中的加载项移至/etc/init.d中的rcS文件中。这是因为因为开启telnet服务会多次加载profile(挂载上一次后无法再次挂载)
在这里插入图片描述
profile文件:
在这里插入图片描述
可以将这些内容移至/etc/init.d中的rcS文件中。

(4)可以将telnetd &加入rcS中开机默认加载

七、海思proc文件系统调试接口(现在大部分被sys文件系统取代了)

1、proc文件系统的原理

参考学习:
https://www.cnblogs.com/lidabo/p/5628020.html
https://blog.csdn.net/weixin_29202687/article/details/116963940
在这里插入图片描述

2、海思proc文件系统调试的文档说明

链接:https://pan.baidu.com/s/1zDDRvTl3gx4GhHE4kgx-zA 
提取码:pfp8 
--来自百度网盘超级会员V5的分享

在这里插入图片描述

3、额外提供的调试经验文档

链接:https://pan.baidu.com/s/1Eu3LmkO3-y3YDsEARnikbw 
提取码:oifi 
--来自百度网盘超级会员V5的分享

注:本文章参考了《朱老师物联网大讲堂》课程笔记,并结合了自己的实际开发经历以及网上他人的技术文章,综合整理得到。如有侵权,联系删除!水平有限,欢迎各位在评论区交流。

  • 2
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小嵌同学

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值