码迷,mamicode.com
首页 > 其他好文 > 详细

如何从H264原始数据SPS里面得到视频的长宽

时间:2015-06-16 17:07:35      阅读:195      评论:0      收藏:0      [点我收藏+]

标签:h264   sps   宽高   

有这样一种需求,给你一个h264原始数据文件,让你直接播放出来,如何实现?

思路是这样的,H264原始数据格式都是 0x00000001后面跟0x67 0x68 0x65 0x41这样的数据,解码需要一个完整的NAL数据单元,我们需要将每个0x00000001以及下一个0x00000001之前的数据读出来,交给解码器解码。

读文件我就不啰嗦了,本文主要讲解如何从SPS获取视频长宽,SPS即0x67开头数据,读到0x00000001 后一个字节是0x67的后面跟的数据就是SPS了。我们看一个封包里面sps的结构:

技术分享

我们可以看到 最下面的42开始就是0x67后面的SPS数据了,看H264的官方文档可以知道,SPS里面的参数都是按照bit的形式存储的。

我自己写了一个解析SPS的类,代码如下:

import android.util.Log;

/*
 * Author:Vincent Luo
 * Date: 20150615
 * Description:参考H264标准语法实现对SPS参数的解析
 */
public class H264SPSPaser {
	private static final String TAG = "H264SPSPaser";
	private static int startBit = 0;
	/*
	 * 从数据流data中第StartBit位开始读,读bitCnt位,以无符号整形返回
	 */
	public static short u(byte[] data,int bitCnt,int StartBit){
		short ret = 0;
		int start = StartBit;
		for(int i = 0;i < bitCnt;i++){
			ret<<=1;
			if ((data[start / 8] & (0x80 >> (start%8))) != 0)
			{
				ret += 1;
			}
			start++;
		}
		return ret;
	}
	/*
	 * 无符号指数哥伦布编码
	 * leadingZeroBits = ?1;
	 * for( b = 0; !b; leadingZeroBits++ )
	 *    b = read_bits( 1 )
	 * 变量codeNum 按照如下方式赋值:
	 * codeNum = 2^leadingZeroBits ? 1 + read_bits( leadingZeroBits )
	 * 这里read_bits( leadingZeroBits )的返回值使用高位在先的二进制无符号整数表示。
	 */
	public static short ue(byte[] data,int StartBit){
		short ret = 0;
		int leadingZeroBits = -1;
		int tempStartBit = (StartBit == -1)?startBit:StartBit;//如果传入-1,那么就用上次记录的静态变量
		 for( int b = 0; b != 1; leadingZeroBits++ ){//读到第一个不为0的数,计算前面0的个数
			 b = u(data,1,tempStartBit++);
		 }
		 Log.d(TAG,"ue leadingZeroBits = " + leadingZeroBits + ",Math.pow(2, leadingZeroBits) = " + Math.pow(2, leadingZeroBits) + ",tempStartBit = " + tempStartBit);
		 ret = (short) (Math.pow(2, leadingZeroBits) - 1 + u(data,leadingZeroBits,tempStartBit));
		 startBit = tempStartBit + leadingZeroBits;
		 Log.d(TAG,"ue startBit = " + startBit);
		return ret;
	}
	/*
	 * 有符号指数哥伦布编码
	 * 9.1.1 有符号指数哥伦布编码的映射过程
	 *按照9.1节规定,本过程的输入是codeNum。
	 *本过程的输出是se(v)的值。
	 *表9-3中给出了分配给codeNum的语法元素值的规则,语法元素值按照绝对值的升序排列,负值按照其绝对
	 *值参与排列,但列在绝对值相等的正值之后。
	 *表 9-3-有符号指数哥伦布编码语法元素se(v)值与codeNum的对应
	 *codeNum 语法元素值
	 *	0 		0
	 *	1		1
	 *	2		?1
	 *	3		2
	 *	4		?2
	 *	5		3
	 *	6		?3
	 *	k		(?1)^(k+1) Ceil( k÷2 )
	 */
	public static int se(byte[] data,int StartBit){
		int ret = 0;
		short codeNum = ue(data,StartBit);
		ret = (int) (Math.pow(-1, codeNum + 1)*Math.ceil(codeNum/2));
		return ret;
	}
}
只需调用如下接口即可得到宽高:

int width = (H264SPSPaser.ue(sps,34) + 1)*16;
int height = (H264SPSPaser.ue(sps,-1) + 1)*16;
其中参数sps就是42开始的那一串字节流,34指的是宽开始的位置(bit位)。


如何从H264原始数据SPS里面得到视频的长宽

标签:h264   sps   宽高   

原文地址:http://blog.csdn.net/vincent_blog/article/details/46517733

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!