#include "mpeg-ps-internal.h" #include // 2.5.3.3 Pack layer of program stream (p78) // Table 2-38 - Program stream pack // Table 2-39 - Program stream pack header #if 1 int pack_header_read(struct ps_pack_header_t* h, struct mpeg_bits_t* reader) { uint8_t v8; uint64_t v64; uint8_t stuffing_length; size_t header_length; v8 = mpeg_bits_read8(reader); // data[4] if (0 == (0xC0 & v8)) { // MPEG-1 // ISO/IEC 11172-1 /* pack() { pack_start_code 32 bslbf '0010' 4 bslbf system_clock_reference [32..30] 3 bslbf marker_bit 1 bslbf system_clock_reference [29..15] 15 bslbf marker_bit 1 bslbf system_clock_reference [14..0] 15 bslbf marker_bit 1 bslbf marker_bit 1 bslbf mux_rate 22 uimsbf marker_bit 1 bslbf if (nextbits() == system_header_start_code) system_header () while (nextbits() == packet_start_code) packet() } */ h->mpeg2 = 0; //assert(0x20 == (0xF0 & v8)); h->system_clock_reference_base = (((uint64_t)(v8 >> 1) & 0x07) << 30) | mpeg_bits_read30(reader); h->system_clock_reference_extension = 1; h->program_mux_rate = (mpeg_bits_read8(reader) & 0x7F) << 15; h->program_mux_rate |= mpeg_bits_read15(reader); return mpeg_bits_error(reader) ? MPEG_ERROR_NEED_MORE_DATA : MPEG_ERROR_OK; } else { h->mpeg2 = 1; v64 = mpeg_bits_read64(reader); // data[5-12] //assert((0x44 & data[4]) == 0x44); // '01xxx1xx' //assert((0x04 & data[6]) == 0x04); // 'xxxxx1xx' //assert((0x04 & data[8]) == 0x04); // 'xxxxx1xx' //assert((0x01 & data[9]) == 0x01); // 'xxxxxxx1' h->system_clock_reference_base = (((v8 >> 3) & 0x07) << 30) | ((v8 & 0x03) << 28) | (((v64 >> 51) & 0x1FFF) << 15) | ((v64 >> 35) & 0x7FFF); h->system_clock_reference_extension = (uint32_t)((v64 >> 25) & 0x1F); //assert((0x03 & v64) == 0x03); // 'xxxxxx11' h->program_mux_rate = (uint32_t)((v64 >> 2) & 0x3FFFFF); //assert((0xF8 & data[13]) == 0x00); // '00000xxx' stuffing_length = mpeg_bits_read8(reader) & 0x07; // stuffing header_length = 14 + stuffing_length; mpeg_bits_skip(reader, stuffing_length); return mpeg_bits_error(reader) ? MPEG_ERROR_NEED_MORE_DATA : MPEG_ERROR_OK; } } #else size_t pack_header_read(struct ps_pack_header_t *h, const uint8_t* data, size_t bytes) { uint8_t stuffing_length; size_t header_length; if (bytes < 14) return 0; assert(0x00 == data[0] && 0x00 == data[1] && 0x01 == data[2] && PES_SID_START == data[3]); if (0 == (0xC0 & data[4])) { // MPEG-1 h->mpeg2 = 0; assert(0x20 == (0xF0 & data[4])); h->system_clock_reference_base = (((uint64_t)(data[4] >> 1) & 0x07) << 30) | ((uint64_t)data[5] << 22) | (((uint64_t)data[6] >> 1) << 15) | ((uint64_t)data[7] << 7) | (data[8] >> 1); h->system_clock_reference_extension = 1; h->program_mux_rate = ((data[9] >> 1) << 15) | (data[10] << 7) | (data[11] >> 1); return 12; } else { h->mpeg2 = 1; assert((0x44 & data[4]) == 0x44); // '01xxx1xx' assert((0x04 & data[6]) == 0x04); // 'xxxxx1xx' assert((0x04 & data[8]) == 0x04); // 'xxxxx1xx' assert((0x01 & data[9]) == 0x01); // 'xxxxxxx1' h->system_clock_reference_base = (((uint64_t)(data[4] >> 3) & 0x07) << 30) | (((uint64_t)data[4] & 0x3) << 28) | ((uint64_t)data[5] << 20) | ((((uint64_t)data[6] >> 3) & 0x1F) << 15) | (((uint64_t)data[6] & 0x3) << 13) | ((uint64_t)data[7] << 5) | ((data[8] >> 3) & 0x1F); h->system_clock_reference_extension = ((data[8] & 0x3) << 7) | ((data[9] >> 1) & 0x7F); assert((0x03 & data[12]) == 0x03); // 'xxxxxx11' h->program_mux_rate = (data[10] << 14) | (data[11] << 6) | ((data[12] >> 2) & 0x3F); //assert((0xF8 & data[13]) == 0x00); // '00000xxx' stuffing_length = data[13] & 0x07; // stuffing header_length = 14 + stuffing_length; if (header_length > bytes) return 0; return header_length; } } #endif size_t pack_header_write(const struct ps_pack_header_t *h, uint8_t *data) { // pack_start_code nbo_w32(data, 0x000001BA); // 33-system_clock_reference_base + 9-system_clock_reference_extension // '01xxx1xx xxxxxxxx xxxxx1xx xxxxxxxx xxxxx1xx xxxxxxx1' data[4] = 0x44 | (((h->system_clock_reference_base >> 30) & 0x07) << 3) | ((h->system_clock_reference_base >> 28) & 0x03); data[5] = ((h->system_clock_reference_base >> 20) & 0xFF); data[6] = 0x04 | (((h->system_clock_reference_base >> 15) & 0x1F) << 3) | ((h->system_clock_reference_base >> 13) & 0x03); data[7] = ((h->system_clock_reference_base >> 5) & 0xFF); data[8] = 0x04 | ((h->system_clock_reference_base & 0x1F) << 3) | ((h->system_clock_reference_extension >> 7) & 0x03); data[9] = 0x01 | ((h->system_clock_reference_extension & 0x7F) << 1); // program_mux_rate // 'xxxxxxxx xxxxxxxx xxxxxx11' data[10] = (uint8_t)(h->program_mux_rate >> 14); data[11] = (uint8_t)(h->program_mux_rate >> 6); data[12] = (uint8_t)(0x03 | ((h->program_mux_rate & 0x3F) << 2)); // stuffing length // '00000xxx' data[13] = 0xF8; return 14; }