#include "mpeg-ps-internal.h" #include "mpeg-util.h" #include // 2.5.3.5 System header (p79) // Table 2-40 - Program stream system header #if 1 int system_header_read(struct ps_system_header_t* h, struct mpeg_bits_t* reader) { uint8_t v8; uint16_t v16; size_t i, len, end; len = mpeg_bits_read16(reader); end = mpeg_bits_tell(reader) + len; if (mpeg_bits_error(reader) || end > mpeg_bits_length(reader)) return MPEG_ERROR_NEED_MORE_DATA; //assert((0x80 & data[6]) == 0x80); // '1xxxxxxx' //assert((0x01 & data[8]) == 0x01); // 'xxxxxxx1' h->rate_bound = (mpeg_bits_read8(reader) & 0x7F) << 15; h->rate_bound |= mpeg_bits_read15(reader); v8 = mpeg_bits_read8(reader); h->audio_bound = (v8 >> 2) & 0x3F; h->fixed_flag = (v8 >> 1) & 0x01; h->CSPS_flag = (v8 >> 0) & 0x01; v8 = mpeg_bits_read8(reader); assert((0x20 & v8) == 0x20); // 'xx1xxxxx' h->system_audio_lock_flag = (v8 >> 7) & 0x01; h->system_video_lock_flag = (v8 >> 6) & 0x01; h->video_bound = v8 & 0x1F; // assert((0x7F & data[11]) == 0x00); // 'x0000000' h->packet_rate_restriction_flag = (mpeg_bits_read8(reader) >> 7) & 0x01; for (i = 0; 0 == mpeg_bits_error(reader) && mpeg_bits_tell(reader) + 1 < end && i < sizeof(h->streams) / sizeof(h->streams[0]); i++) { v8 = mpeg_bits_read8(reader); if ((v8 & 0x80) != 0x80) break; h->streams[i].stream_id = v8; if (h->streams[i].stream_id == PES_SID_EXTENSION) // '10110111' { v8 = mpeg_bits_read8(reader); assert(v8 == 0xC0); // '11000000' h->streams[i].stream_id = mpeg_bits_read8(reader) & 0x7F; v8 = mpeg_bits_read8(reader); assert(v8 == 0xB6); // '10110110' } v16 = mpeg_bits_read16(reader); assert((v16 & 0xC000) == 0xC000); // '11xxxxxx' h->streams[i].buffer_bound_scale = (v16 >> 13) & 0x01; h->streams[i].buffer_size_bound = v16 & 0x1FFF; } assert(0 == mpeg_bits_error(reader)); //assert(end == mpeg_bits_tell(reader)); return MPEG_ERROR_OK; } #else size_t system_header_read(struct ps_system_header_t *h, const uint8_t* data, size_t bytes) { size_t i, j; size_t len; if (bytes < 12) return 0; assert(0x00 == data[0] && 0x00 == data[1] && 0x01 == data[2] && PES_SID_SYS == data[3]); len = (data[4] << 8) | data[5]; if(len + 6 > bytes) { assert(0); return 0; } assert((0x80 & data[6]) == 0x80); // '1xxxxxxx' assert((0x01 & data[8]) == 0x01); // 'xxxxxxx1' h->rate_bound = ((data[6] & 0x7F) << 15) | (data[7] << 7) | ((data[8] >> 1) & 0x7F); h->audio_bound = (data[9] >> 2) & 0x3F; h->fixed_flag = (data[9] >> 1) & 0x01; h->CSPS_flag = (data[9] >> 0) & 0x01; assert((0x20 & data[10]) == 0x20); // 'xx1xxxxx' h->system_audio_lock_flag = (data[10] >> 7) & 0x01; h->system_video_lock_flag = (data[10] >> 6) & 0x01; h->video_bound = data[10] & 0x1F; // assert((0x7F & data[11]) == 0x00); // 'x0000000' h->packet_rate_restriction_flag = (data[11] >> 7) & 0x01; i = 12; for (j = 0; (data[i] & 0x80) == 0x80 && j < sizeof(h->streams) / sizeof(h->streams[0]) && i < bytes; j++) { h->streams[j].stream_id = data[i++]; if (h->streams[j].stream_id == PES_SID_EXTENSION) // '10110111' { assert(data[i] == 0xC0); // '11000000' assert((data[i + 1] & 80) == 0); // '1xxxxxxx' h->streams[j].stream_id = (h->streams[j].stream_id << 7) | (data[i + 1] & 0x7F); assert(data[i + 2] == 0xB6); // '10110110' i += 3; } assert((data[i] & 0xC0) == 0xC0); // '11xxxxxx' h->streams[j].buffer_bound_scale = (data[i] >> 5) & 0x01; h->streams[j].buffer_size_bound = (data[i] & 0x1F) | data[i + 1]; i += 2; } return len + 4 + 2; } #endif size_t system_header_write(const struct ps_system_header_t *h, uint8_t *data) { size_t i, j; // system_header_start_code nbo_w32(data, 0x000001BB); // header length //put16(data + 4, 6 + h->stream_count*3); // rate_bound // 1xxxxxxx xxxxxxxx xxxxxxx1 data[6] = 0x80 | ((h->rate_bound >> 15) & 0x7F); data[7] = (h->rate_bound >> 7) & 0xFF; data[8] = 0x01 | ((h->rate_bound & 0x7F) << 1); // 6-audio_bound + 1-fixed_flag + 1-CSPS_flag data[9] = ((h->audio_bound & 0x3F) << 2) | ((h->fixed_flag & 0x01) << 1) | (h->CSPS_flag & 0x01); // 1-system_audio_lock_flag + 1-system_video_lock_flag + 1-maker + 5-video_bound data[10] = 0x20 | ((h->system_audio_lock_flag & 0x01) << 7) | ((h->system_video_lock_flag & 0x01) << 6) | (h->video_bound & 0x1F); // 1-packet_rate_restriction_flag + 7-reserved data[11] = 0x7F | ((h->packet_rate_restriction_flag & 0x01) << 7); i = 12; for (j = 0; j < h->stream_count; j++) { data[i++] = (uint8_t)h->streams[j].stream_id; if (PES_SID_EXTENSION == h->streams[j].stream_id) // '10110111' { data[i++] = 0xD0; // '11000000' data[i++] = h->streams[j].stream_extid & 0x7F; // '0xxxxxxx' data[i++] = 0xB6; // '10110110' } // '11' + 1-P-STD_buffer_bound_scale + 13-P-STD_buffer_size_bound // '11xxxxxx xxxxxxxx' data[i++] = 0xC0 | ((h->streams[j].buffer_bound_scale & 0x01) << 5) | ((h->streams[j].buffer_size_bound >> 8) & 0x1F); data[i++] = h->streams[j].buffer_size_bound & 0xFF; } // header length nbo_w16(data + 4, (uint16_t)(i - 6)); return i; }