下面分析 NALread.cpp 函數和代碼。
void read(InputNALUnit& nalu, vector<uint8_t>& nalUnitBuf)
{/* perform anti-emulation prevention */TComInputBitstream *pcBitstream = new TComInputBitstream(NULL);convertPayloadToRBSP(nalUnitBuf, (nalUnitBuf[0] & 64) == 0);nalu.m_Bitstream = new TComInputBitstream(&nalUnitBuf);delete pcBitstream;readNalUnitHeader(nalu);
}
nalUnitBuf? 是從文件流中讀出的一個nal 單元,我們知道,H264,HEVC/H265? 都是采用 Annex? B 中? Byte stream NAL unit syntax的要求格式,進行封裝,好處是通過前面
的一兩個字節就能夠適用網絡環境傳輸,通過?convertPayloadToRBSP? 函數的目的就是去除偽起始碼,NAL 包分為 VCL(videocoding layer) 和 非VCL包(sps,vps,pps等頭部信息包)。
enum NalUnitType
{NAL_UNIT_CODED_SLICE_TRAIL_N = 0, // 0NAL_UNIT_CODED_SLICE_TRAIL_R, // 1NAL_UNIT_CODED_SLICE_TSA_N, // 2NAL_UNIT_CODED_SLICE_TLA, // 3 // Current name in the spec: TSA_RNAL_UNIT_CODED_SLICE_STSA_N, // 4NAL_UNIT_CODED_SLICE_STSA_R, // 5NAL_UNIT_CODED_SLICE_RADL_N, // 6NAL_UNIT_CODED_SLICE_DLP, // 7 // Current name in the spec: RADL_RNAL_UNIT_CODED_SLICE_RASL_N, // 8NAL_UNIT_CODED_SLICE_TFD, // 9 // Current name in the spec: RASL_RNAL_UNIT_RESERVED_10,NAL_UNIT_RESERVED_11,NAL_UNIT_RESERVED_12,NAL_UNIT_RESERVED_13,NAL_UNIT_RESERVED_14,NAL_UNIT_RESERVED_15,NAL_UNIT_CODED_SLICE_BLA, // 16 // Current name in the spec: BLA_W_LPNAL_UNIT_CODED_SLICE_BLANT, // 17 // Current name in the spec: BLA_W_DLPNAL_UNIT_CODED_SLICE_BLA_N_LP, // 18NAL_UNIT_CODED_SLICE_IDR, // 19 // Current name in the spec: IDR_W_DLPNAL_UNIT_CODED_SLICE_IDR_N_LP, // 20NAL_UNIT_CODED_SLICE_CRA, // 21NAL_UNIT_RESERVED_22,NAL_UNIT_RESERVED_23,NAL_UNIT_RESERVED_24,NAL_UNIT_RESERVED_25,NAL_UNIT_RESERVED_26,NAL_UNIT_RESERVED_27,NAL_UNIT_RESERVED_28,NAL_UNIT_RESERVED_29,NAL_UNIT_RESERVED_30,NAL_UNIT_RESERVED_31,NAL_UNIT_VPS, // 32NAL_UNIT_SPS, // 33NAL_UNIT_PPS, // 34NAL_UNIT_ACCESS_UNIT_DELIMITER, // 35NAL_UNIT_EOS, // 36NAL_UNIT_EOB, // 37NAL_UNIT_FILLER_DATA, // 38NAL_UNIT_SEI, // 39 Prefix SEINAL_UNIT_SEI_SUFFIX, // 40 Suffix SEINAL_UNIT_RESERVED_41,NAL_UNIT_RESERVED_42,NAL_UNIT_RESERVED_43,NAL_UNIT_RESERVED_44,NAL_UNIT_RESERVED_45,NAL_UNIT_RESERVED_46,NAL_UNIT_RESERVED_47,NAL_UNIT_UNSPECIFIED_48,NAL_UNIT_UNSPECIFIED_49,NAL_UNIT_UNSPECIFIED_50,NAL_UNIT_UNSPECIFIED_51,NAL_UNIT_UNSPECIFIED_52,NAL_UNIT_UNSPECIFIED_53,NAL_UNIT_UNSPECIFIED_54,NAL_UNIT_UNSPECIFIED_55,NAL_UNIT_UNSPECIFIED_56,NAL_UNIT_UNSPECIFIED_57,NAL_UNIT_UNSPECIFIED_58,NAL_UNIT_UNSPECIFIED_59,NAL_UNIT_UNSPECIFIED_60,NAL_UNIT_UNSPECIFIED_61,NAL_UNIT_UNSPECIFIED_62,NAL_UNIT_UNSPECIFIED_63,NAL_UNIT_INVALID,
};
以下是nal_unit_header的描述,(nalUnitBuf[0]& 64) == 0? 就是? NalUnitType? 從0 到31的, NalUnitType 從32到64的,(nalUnitBuf[0] & 64)== 1, 在語法描述中有
forbidden_zero_bit 為 0,例如:NAL_UNIT_VPS? 是32 , 0(forbidden_zero_bit)? 100000(nal_unit_type),?? 0(nuh_reserved_zero_6bits), 所以NAL_UNIT_VPS的nalUnitBuf[0] 為64,。? 通過 NalUnitType 得知,slice相關的(nalUnitBuf[0] & 64)== 0? , slice不相關的(nalUnitBuf[0] & 64)==? 1.?
在slice中可能就有cabac_zero_word字節,不需要進行解碼的。
nal_unit_header(?) { | Descriptor |
???? forbidden_zero_bit | f(1) |
???? nal_unit_type | u(6) |
???? nuh_reserved_zero_6bits | u(6) |
???? nuh_temporal_id_plus1 | u(3) |
} | ? |
convertPayloadToRBSP(nalUnitBuf, (nalUnitBuf[0] & 64) == 0);? 函數就是去偽起始碼,readNalUnitHeader 就是判斷nalu的類型信息。