/* The copyright in this software is being made available under the BSD * License, included below. This software may be subject to other third party * and contributor rights, including patent rights, and no such rights are * granted under this license. * * Copyright (c) 2010-2014, ITU/ISO/IEC * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * Neither the name of the ITU/ISO/IEC nor the names of its contributors may * be used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include "TLibCommon/NAL.h" #include "TLibCommon/TComBitStream.h" #include "NALwrite.h" using namespace std; //! \ingroup TLibEncoder //! \{ static const Char emulation_prevention_three_byte[] = {3}; Void writeNalUnitHeader(ostream& out, OutputNALUnit& nalu) // nal_unit_header() { TComOutputBitstream bsNALUHeader; bsNALUHeader.write(0,1); // forbidden_zero_bit bsNALUHeader.write(nalu.m_nalUnitType, 6); // nal_unit_type #if H_MV bsNALUHeader.write(nalu.m_layerId, 6); // layerId #else bsNALUHeader.write(nalu.m_reservedZero6Bits, 6); // nuh_reserved_zero_6bits #endif bsNALUHeader.write(nalu.m_temporalId+1, 3); // nuh_temporal_id_plus1 out.write(bsNALUHeader.getByteStream(), bsNALUHeader.getByteStreamLength()); } /** * write nalu to bytestream out, performing RBSP anti startcode * emulation as required. nalu.m_RBSPayload must be byte aligned. */ void write(ostream& out, OutputNALUnit& nalu) { writeNalUnitHeader(out, nalu); /* write out rsbp_byte's, inserting any required * emulation_prevention_three_byte's */ /* 7.4.1 ... * emulation_prevention_three_byte is a byte equal to 0x03. When an * emulation_prevention_three_byte is present in the NAL unit, it shall be * discarded by the decoding process. * The last byte of the NAL unit shall not be equal to 0x00. * Within the NAL unit, the following three-byte sequences shall not occur at * any byte-aligned position: * - 0x000000 * - 0x000001 * - 0x000002 * Within the NAL unit, any four-byte sequence that starts with 0x000003 * other than the following sequences shall not occur at any byte-aligned * position: * - 0x00000300 * - 0x00000301 * - 0x00000302 * - 0x00000303 */ vector& rbsp = nalu.m_Bitstream.getFIFO(); #if H_MV // This will handle situation when writing NAL with zero payload if (rbsp.size() == 0) return; #endif for (vector::iterator it = rbsp.begin(); it != rbsp.end();) { /* 1) find the next emulated 00 00 {00,01,02,03} * 2a) if not found, write all remaining bytes out, stop. * 2b) otherwise, write all non-emulated bytes out * 3) insert emulation_prevention_three_byte */ vector::iterator found = it; do { /* NB, end()-1, prevents finding a trailing two byte sequence */ found = search_n(found, rbsp.end()-1, 2, 0); found++; /* if not found, found == end, otherwise found = second zero byte */ if (found == rbsp.end()) break; if (*(++found) <= 3) break; } while (true); it = found; if (found != rbsp.end()) { it = rbsp.insert(found, emulation_prevention_three_byte[0]); } } out.write((Char*)&(*rbsp.begin()), rbsp.end() - rbsp.begin()); /* 7.4.1.1 * ... when the last byte of the RBSP data is equal to 0x00 (which can * only occur when the RBSP ends in a cabac_zero_word), a final byte equal * to 0x03 is appended to the end of the data. */ if (rbsp.back() == 0x00) { out.write(emulation_prevention_three_byte, 1); } } /** * Write rbsp_trailing_bits to bs causing it to become byte-aligned */ void writeRBSPTrailingBits(TComOutputBitstream& bs) { bs.write( 1, 1 ); bs.writeAlignZero(); } /** * Copy NALU from naluSrc to naluDest */ void copyNaluData(OutputNALUnit& naluDest, const OutputNALUnit& naluSrc) { naluDest.m_nalUnitType = naluSrc.m_nalUnitType; #if H_MV naluDest.m_layerId = naluSrc.m_layerId; #else naluDest.m_reservedZero6Bits = naluSrc.m_reservedZero6Bits; #endif naluDest.m_temporalId = naluSrc.m_temporalId; naluDest.m_Bitstream = naluSrc.m_Bitstream; } //! \}