H264解码过滤花屏视频帧

H264解码过滤花屏视频帧众所周知视频在各个领域占有极为重要的地位,安防领域,互联网,医药,教育等等等等。扯淡我就尽量不多扯了,现主要扯安防领域吧,安防领域尤其是视频分析领域,视频质量要求比较苛刻。下面介绍一下场景比较苛刻的图片情况:1.这种2.这种花屏现象,在视频接入解码过程中尤为常见,(比如28181接入,rtsp等等),解码大家都考虑使用ffmpeg进行解码,首先考虑的可能是解码错误直接从解码过程…

大家好,又见面了,我是你们的朋友全栈君。

众所周知视频在各个领域占有极为重要的地位,安防领域,互联网,医药,教育等等等等。扯淡我就尽量不多扯了,现主要扯安防领域吧,安防领域尤其是视频分析领域,视频质量要求比较苛刻。下面介绍一下场景比较苛刻的图片情况:

1.这种

H264解码过滤花屏视频帧

2.这种

H264解码过滤花屏视频帧

花屏现象,在视频接入解码过程中尤为常见,(比如28181接入,rtsp等等),解码大家都考虑使用ffmpeg进行解码,首先考虑的可能是解码错误直接从解码过程中就把这种错误的帧给干掉,看了好多博客大概也就是这个思想。

1.如果解码错误抛帧。2.如果是I帧从下一个IDR帧开始解码。想法不错当然我也在做了这一部分,具体部分代码示例如下:

//伪代码......	
int len = av_parser_parse2(m_h264Parser, m_ctx, &avpkt.data, &avpkt.size, in_buf, in_buf_size, AV_NOPTS_VALUE, AV_NOPTS_VALUE, 0);
                //重要: 如果有解码错误,并且当前帧不是IDR就直接跳过
                //m_iErrorDeocde 表示是否有解码错误 
                //m_h264Parser->pict_type != AV_PICTURE_TYPE_I 表示当前帧是否是I帧
		if (m_iErrorDeocde /*&& m_iLastFrame*/ && m_h264Parser->pict_type != AV_PICTURE_TYPE_I)
			return 0;
		in_buf += len;
		in_buf_size -= len;

		if(avpkt.size > 0)
		{
#if 0
			decode_video(m_ctx, m_picture, &avpkt,out_buf,pout,lineSize);
			m_iLastFrame = (m_h264Parser->pict_type == AV_PICTURE_TYPE_I)?1:0;
#endif	
#if 1
			{
				ret = avcodec_decode_video2(m_ctx, m_picture, &got_picture, &avpkt);
				m_iLastFrame = (m_h264Parser->pict_type == AV_PICTURE_TYPE_I) ? 1 : 0;
				if (ret < 0)
				{
					printf("damage!!!!!!!!!!");
					avcodec_flush_buffers(m_ctx);
					goto finish;
				}
                   }

到这种情况其实已经过滤掉了很多坏图了,但是想上图展示的两种情况,就像是打不死的小强一下死了又来来了又死,怎么办?

当然前面的两张图你必须得把ffmpeg的错误隐藏给关掉,再就是另一个err_recognition这个东西,看解码那块的源码找到了个这么东西,具体干什么的,自己可以细细研究一下。

观察上面的图都有规律,是什么?对,没错!就是都有灰图,那灰图是怎么来的呢?于是乎我有看了看ffmpeg的h264解码,注意到了一个0x80这么数值,还是在alloc_pic的时候,难道这就是传说中的赋初始值?看着像,具体也没看太明白。。。。

H264解码过滤花屏视频帧

那么那些解码错误的灰色的图块吧,确实的东西是不是就是这个默认值呢?答案差不多,那我是不是就可以把这些看似解码正确的图片其实是花了的图片,直接判断这些坏块再做一遍过滤,剔除掉呢?

这里我补充一些色彩空间的知识,不再赘述了,大概就是Ycbcr经过偏置处理默认值128即0x80,大概就是为了和rgb的0~255在一个范围吧:

https://blog.csdn.net/asahinokawa/article/details/80596655

好了,到这一步骤,基本上就是单纯的过滤有灰块的图了,我的思想是判断这个值,或者这个值范围内的值,那么选择yuv哪个分量做过滤呢?当然是Y了,UV是色彩和饱和度,到了晚上这种值当然就是0x80了,看这张图。

H264解码过滤花屏视频帧

他的末尾全是这玩意,即0x80,这就是我选y的原因,因为他是亮度。。。

H264解码过滤花屏视频帧

我的过滤部分伪代码实现如下:

//只是思想没有做代码的整洁及优化,可以根据自己情况去增加删除代码
bool  CheckY(int iwidth, int iHeight, unsigned char*Buf)
{
	
	unsigned char * pNewPoint = Buf;
	int iCountu = 0, iCountv = 0, iCountY = 0;
	int iValues = 0;
	unsigned char uBytes = 0;

	//定位最后一行Y
	pNewPoint = Buf;
	//uBytes = pNewPoint[0];

	//获得新的像素位置
	pNewPoint = Buf + (iHeight - 8)*iwidth;

	//遍历Y信息的所有高
	for (int i = 0; i < 8; i++)
	{
		unsigned char *pNewPoint2 = pNewPoint + i*iwidth;
		//遍历Y信息的宽
		for (int j = 0; j < iwidth; j+=24)
		{
			int a = memcmp(pNewPoint2 + j, pNewPoint2 + j + 8, 8);
			int b = memcmp(pNewPoint2 + j + 8, pNewPoint2 + j + 16, 8);
			
	                
                        //判断连续两个8像素宏块,是否相同,并且在这个值范围内(0x7A~0x80)
			if (a == b && b == 0 && pNewPoint2[j] > 0x7A&& pNewPoint2[j] <= 0x80)
			{
				printf("%d%d%d%d%d%d", pNewPoint2[0], pNewPoint2[1], pNewPoint2[2], pNewPoint2[3], pNewPoint2[4], pNewPoint2[5]);
				iCountY++;
			}
			/*if (((unsigned char *)pNewPoint2)[j] == 0x80)
			{
				printf("uByte=%x,pNewPoint2=%x",uBytes,pNewPoint2[j]);
				
			}	*/
		}
	}
}

思想已经说完了,具体实现就看自己了,代码为商业代码,就不能提供了。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/149262.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • MyBatis–SqlSessionFactory概述及创建方式「建议收藏」

    MyBatis–SqlSessionFactory概述及创建方式「建议收藏」SqlSessionFactory概述使用MyBatis首先是使用配置或者代码去生产SqlSessionFactory,而MyBatis提供了构造器SqlSessionFactoryBuilder。MyBatis提供了一个类org.apache.ibatis.session.Configuration作为引导,采用的是Builder模式。具体的分步则是在Configurat…

  • 使用POI替换word中的特定字符/文字改进版

    推荐:http://www.cnblogs.com/roucheng/p/3504465.html

    2021年12月26日
  • js根据经纬度计算距离

    js根据经纬度计算距离varEARTH_RADIUS=6378137.0;//单位MvarPI=Math.PI;functiongetRad(d){returnd*PI/180.0;}/***caculatethegreatcircledistance*@param…

  • 关于platform_device一些讲解「建议收藏」

    关于platform_device一些讲解「建议收藏」从2.6版本开始引入了platform这个概念,在开发底层驱动程序时,首先要确认的就是设备的资源信息,例如设备的地址,在2.6内核中将每个设备的资源用结构platform_device来描述,该结构体定义在kernel\include\linux\platform_device.h中:structplatform_device{constchar*name;u32id;structdevicedev;u32num_resources;structresou

  • 课程表课程设计_工程经济学课程设计

    课程表课程设计_工程经济学课程设计一、需求分析由于大学上课教室分散,学生经常会忘记自己的课程或是上课的教室。这在一定程度上促进学生课表软件的开发。使其可以帮助学生记录自己的课程和对自己课程的掌握。因为手机相对笔记本电脑更加具有便携性

  • 嵌入式C语言面试题_c语言基础面试题

    嵌入式C语言面试题_c语言基础面试题预处理器(Preprocessor)1 . 用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽略闰年问题)         #define SECONDS_PER_YEAR (60 * 60 * 24 * 365)UL我在这想看到几件事情:1) #define 语法的基本知识(例如:不能以分号结束,括号的使用,等等)2)懂得预处理器将为你计算常数表达式的值,因此,直接写出你是如何…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号