164 lines
5.5 KiB
C
164 lines
5.5 KiB
C
#include "mpeg-ps-internal.h"
|
|
#include "mpeg-util.h"
|
|
#include <assert.h>
|
|
|
|
// 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;
|
|
}
|