/****************************************************************************** * * Copyright (C) 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ***************************************************************************** * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore */ /** ****************************************************************************** * @file ihevce_bitstream.c * * @brief * This file contains function definitions related to bitstream generation * * @author * ittiam * * @List of Functions * ihevce_bitstrm_init() * ihevce_put_bits() * ihevce_put_bit() * ihevce_put_rbsp_trailing_bits() * ihevce_put_uev() * ihevce_put_sev() * ihevce_put_nal_start_code_prefix() * ****************************************************************************** */ /*****************************************************************************/ /* File Includes */ /*****************************************************************************/ /* System include files */ #include <assert.h> #include <math.h> #include <stdarg.h> #include <stdio.h> #include <stdlib.h> #include <string.h> /* User include files */ #include "ihevc_typedefs.h" #include "ihevc_debug.h" #include "ihevc_platform_macros.h" #include "ihevce_error_codes.h" #include "ihevce_bitstream.h" #include "ihevce_defs.h" /*****************************************************************************/ /* Function Definitions */ /*****************************************************************************/ /** ****************************************************************************** * * @brief Initializes the encoder bitstream engine * * @par Description * This routine needs to be called at start of slice/frame encode * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @param[in] p1_bitstrm_buf * bitstream buffer pointer where the encoded stream is generated in byte order * * @param[in] u4_max_bitstrm_size * indicates maximum bitstream buffer size. (in bytes) * If actual stream size exceeds the maximum size, encoder should * 1. Not corrput data beyond u4_max_bitstrm_size bytes * 2. Report an error back to application indicating overflow * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_bitstrm_init(bitstrm_t *ps_bitstrm, UWORD8 *pu1_bitstrm_buf, UWORD32 u4_max_bitstrm_size) { ps_bitstrm->pu1_strm_buffer = pu1_bitstrm_buf; ps_bitstrm->u4_max_strm_size = u4_max_bitstrm_size; /* Default init values for other members of bitstream context */ ps_bitstrm->u4_strm_buf_offset = 0; ps_bitstrm->u4_cur_word = 0; ps_bitstrm->i4_bits_left_in_cw = WORD_SIZE; ps_bitstrm->i4_zero_bytes_run = 0; return (IHEVCE_SUCCESS); } /** ****************************************************************************** * * @brief puts a code with specified number of bits into the bitstream * * @par Description * inserts code_len number of bits from lsb of code_val into the * bitstream. updates context members like u4_cur_word, u4_strm_buf_offset and * i4_bits_left_in_cw. If the total words (u4_strm_buf_offset) exceeds max * available size (u4_max_strm_size), returns error without corrupting data * beyond it * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @param[in] u4_code_val * code value that needs to be inserted in the stream. * * @param[in] code_len * indicates code length (in bits) of code_val that would be inserted in * bitstream buffer size. Range of length[1:WORD_SIZE] * * @remarks Assumptions: all bits from bit position code_len to msb of * code_val shall be zero * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_put_bits(bitstrm_t *ps_bitstrm, UWORD32 u4_code_val, WORD32 code_len) { UWORD32 u4_cur_word = ps_bitstrm->u4_cur_word; WORD32 bits_left_in_cw = ps_bitstrm->i4_bits_left_in_cw; /* check assumptions made in the module */ ASSERT(code_len > 0 && code_len <= WORD_SIZE); if(code_len < WORD_SIZE) ASSERT((u4_code_val >> code_len) == 0); /* sanity check on the bitstream engine state */ ASSERT(bits_left_in_cw > 0 && bits_left_in_cw <= WORD_SIZE); ASSERT(ps_bitstrm->i4_zero_bytes_run <= EPB_ZERO_BYTES); ASSERT(ps_bitstrm->pu1_strm_buffer != NULL); if(bits_left_in_cw > code_len) { /*******************************************************************/ /* insert the code in local bitstream word and return */ /* code is inserted in position of bits left (post decrement) */ /*******************************************************************/ bits_left_in_cw -= code_len; u4_cur_word |= (u4_code_val << bits_left_in_cw); ps_bitstrm->u4_cur_word = u4_cur_word; ps_bitstrm->i4_bits_left_in_cw = bits_left_in_cw; return (IHEVCE_SUCCESS); } else { /********************************************************************/ /* 1. insert parital code corresponding to bits left in cur word */ /* 2. flush all the bits of cur word to bitstream */ /* 3. insert emulation prevention bytes while flushing the bits */ /* 4. insert remaining bits of code starting from msb of cur word */ /* 5. update bitsleft in current word and stream buffer offset */ /********************************************************************/ UWORD32 u4_strm_buf_offset = ps_bitstrm->u4_strm_buf_offset; UWORD32 u4_max_strm_size = ps_bitstrm->u4_max_strm_size; WORD32 zero_run = ps_bitstrm->i4_zero_bytes_run; UWORD8 *pu1_strm_buf = ps_bitstrm->pu1_strm_buffer; WORD32 i, rem_bits = (code_len - bits_left_in_cw); /*********************************************************************/ /* Bitstream overflow check */ /* NOTE: corner case of epb bytes (max 2 for 32bit word) not handled */ /*********************************************************************/ if((u4_strm_buf_offset + (WORD_SIZE >> 3)) >= u4_max_strm_size) { /* return without corrupting the buffer beyond its size */ return (IHEVCE_BITSTREAM_BUFFER_OVERFLOW); } /* insert parital code corresponding to bits left in cur word */ u4_cur_word |= u4_code_val >> rem_bits; for(i = WORD_SIZE; i > 0; i -= 8) { /* flush the bits in cur word byte by byte and copy to stream */ UWORD8 u1_next_byte = (u4_cur_word >> (i - 8)) & 0xFF; PUTBYTE_EPB(pu1_strm_buf, u4_strm_buf_offset, u1_next_byte, zero_run); } /* insert the remaining bits from code val into current word */ u4_cur_word = rem_bits ? (u4_code_val << (WORD_SIZE - rem_bits)) : 0; /* update the state variables and return success */ ps_bitstrm->u4_cur_word = u4_cur_word; ps_bitstrm->i4_bits_left_in_cw = WORD_SIZE - rem_bits; ps_bitstrm->i4_zero_bytes_run = zero_run; ps_bitstrm->u4_strm_buf_offset = u4_strm_buf_offset; return (IHEVCE_SUCCESS); } } /** ****************************************************************************** * * @brief inserts a 1-bit code into the bitstream * * @par Description * inserts 1bit lsb of code_val into the bitstream * updates context members like u4_cur_word, u4_strm_buf_offset and * i4_bits_left_in_cw. If the total words (u4_strm_buf_offset) exceeds max * available size (u4_max_strm_size), returns error without corrupting data * beyond it * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @param[in] u4_code_val * code value that needs to be inserted in the stream. * * @remarks Assumptions: all bits from bit position 1 to msb of code_val * shall be zero * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_put_bit(bitstrm_t *ps_bitstrm, UWORD32 u4_code_val) { /* call the put bits function for 1 bit and return */ return (ihevce_put_bits(ps_bitstrm, u4_code_val, 1)); } /** ****************************************************************************** * * @brief inserts rbsp trailing bits at the end of stream buffer (NAL) * * @par Description * inserts rbsp trailing bits, updates context members like u4_cur_word and * i4_bits_left_in_cw and flushes the same in the bitstream buffer. If the * total words (u4_strm_buf_offset) exceeds max available size * (u4_max_strm_size), returns error without corrupting data beyond it * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_put_rbsp_trailing_bits(bitstrm_t *ps_bitstrm) { WORD32 i; UWORD32 u4_cur_word = ps_bitstrm->u4_cur_word; WORD32 bits_left_in_cw = ps_bitstrm->i4_bits_left_in_cw; WORD32 bytes_left_in_cw = (bits_left_in_cw - 1) >> 3; UWORD32 u4_strm_buf_offset = ps_bitstrm->u4_strm_buf_offset; UWORD32 u4_max_strm_size = ps_bitstrm->u4_max_strm_size; WORD32 zero_run = ps_bitstrm->i4_zero_bytes_run; UWORD8 *pu1_strm_buf = ps_bitstrm->pu1_strm_buffer; /*********************************************************************/ /* Bitstream overflow check */ /* NOTE: corner case of epb bytes (max 2 for 32bit word) not handled */ /*********************************************************************/ if((u4_strm_buf_offset + (WORD_SIZE >> 3) - bytes_left_in_cw) >= u4_max_strm_size) { /* return without corrupting the buffer beyond its size */ return (IHEVCE_BITSTREAM_BUFFER_OVERFLOW); } /* insert a 1 at the end of current word and flush all the bits */ u4_cur_word |= (1U << (bits_left_in_cw - 1)); /* get the bits to be inserted in msbdb of the word */ // u4_cur_word <<= (WORD_SIZE - bytes_left_in_cw + 1); for(i = WORD_SIZE; i > (bytes_left_in_cw * 8); i -= 8) { /* flush the bits in cur word byte by byte and copy to stream */ UWORD8 u1_next_byte = (u4_cur_word >> (i - 8)) & 0xFF; PUTBYTE_EPB(pu1_strm_buf, u4_strm_buf_offset, u1_next_byte, zero_run); } /* update the stream offset */ ps_bitstrm->u4_strm_buf_offset = u4_strm_buf_offset; /* Default init values for scratch variables of bitstream context */ ps_bitstrm->u4_cur_word = 0; ps_bitstrm->i4_bits_left_in_cw = WORD_SIZE; ps_bitstrm->i4_zero_bytes_run = 0; return (IHEVCE_SUCCESS); } /** ****************************************************************************** * * @brief puts exponential golomb code of a unsigned integer into bitstream * * @par Description * computes uev code for given syntax element and inserts the same into * bitstream by calling ihevce_put_bits() interface. * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @param[in] u4_code_num * unsigned integer input whose golomb code is written in stream * * @remarks Assumptions: code value can be represented in less than 16bits * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_put_uev(bitstrm_t *ps_bitstrm, UWORD32 u4_code_num) { UWORD32 u4_bit_str, u4_range; IHEVCE_ERROR_T e_error; /* convert the codenum to exp-golomb bit code: Table 9-2 JCTVC-J1003_d7 */ u4_bit_str = u4_code_num + 1; /* get range of the bit string and put using put_bits() */ GETRANGE(u4_range, u4_bit_str); e_error = ihevce_put_bits(ps_bitstrm, u4_bit_str, (2 * u4_range - 1)); return (e_error); } /** ****************************************************************************** * * @brief puts exponential golomb code of a signed integer into bitstream * * @par Description * computes sev code for given syntax element and inserts the same into * bitstream by calling ihevce_put_bits() interface. * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @param[in] syntax_elem * signed integer input whose golomb code is written in stream * * @remarks Assumptions: code value can be represented in less than 16bits * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_put_sev(bitstrm_t *ps_bitstrm, WORD32 syntax_elem) { UWORD32 u4_code_num, u4_bit_str, u4_range; IHEVCE_ERROR_T e_error; /************************************************************************/ /* convert the codenum to exp-golomb bit code for signed syntax element */ /* See Table9-2 and Table 9-3 of standard JCTVC-J1003_d7 */ /************************************************************************/ if(syntax_elem <= 0) { /* codeNum for non-positive integer = 2*abs(x) : Table9-3 */ u4_code_num = ((-syntax_elem) << 1); } else { /* codeNum for positive integer = 2x-1 : Table9-3 */ u4_code_num = (syntax_elem << 1) - 1; } /* convert the codenum to exp-golomb bit code: Table 9-2 JCTVC-J1003_d7 */ u4_bit_str = u4_code_num + 1; /* get range of the bit string and put using put_bits() */ GETRANGE(u4_range, u4_bit_str); e_error = ihevce_put_bits(ps_bitstrm, u4_bit_str, (2 * u4_range - 1)); return (e_error); } /** ****************************************************************************** * * @brief insert NAL start code prefix (0x000001) into bitstream with an option * of inserting leading_zero_8bits (which makes startcode prefix as 0x00000001) * * @par Description * Although start code prefix could have been put by calling ihevce_put_bits(), * ihevce_put_nal_start_code_prefix() is specially added to make sure emulation * prevention insertion is not done for the NAL start code prefix which will * surely happen otherwise by calling ihevce_put_bits() interface. * * @param[in] ps_bitstrm * pointer to bitstream context (handle) * * @param[in] insert_leading_zero_8bits * flag indicating if one more zero bytes needs to prefixed before start code * * @return success or failure error code * ****************************************************************************** */ IHEVCE_ERROR_T ihevce_put_nal_start_code_prefix(bitstrm_t *ps_bitstrm, WORD32 insert_leading_zero_8bits) { UWORD32 u4_strm_buf_offset = ps_bitstrm->u4_strm_buf_offset; UWORD8 *pu1_strm_buf = ps_bitstrm->pu1_strm_buffer; WORD32 num_nals = ps_bitstrm->i4_num_nal; /* Bitstream buffer overflow check assuming worst case of 4 bytes */ if((u4_strm_buf_offset + 4) > ps_bitstrm->u4_max_strm_size) { return (IHEVCE_BITSTREAM_BUFFER_OVERFLOW); } /* Update the current NAL start ptr and increment counter */ ASSERT(num_nals >= 0); ASSERT(num_nals < MAX_NALS_IN_AU); if(num_nals < MAX_NALS_IN_AU) { ps_bitstrm->apu1_nal_start[num_nals] = pu1_strm_buf + u4_strm_buf_offset; ps_bitstrm->i4_num_nal++; } /* Insert leading zero 8 bits conditionally */ if(insert_leading_zero_8bits) { pu1_strm_buf[u4_strm_buf_offset] = 0x00; u4_strm_buf_offset++; } /* Insert NAL start code prefix 0x00 00 01 */ pu1_strm_buf[u4_strm_buf_offset] = 0x00; u4_strm_buf_offset++; pu1_strm_buf[u4_strm_buf_offset] = 0x00; u4_strm_buf_offset++; pu1_strm_buf[u4_strm_buf_offset] = 0x01; u4_strm_buf_offset++; /* update the stream offset */ ps_bitstrm->u4_strm_buf_offset = u4_strm_buf_offset; return (IHEVCE_SUCCESS); }