|
- // ITU-T H.222.0(10/2014)
- // Information technology - Generic coding of moving pictures and associated audio information: Systems
- // 2.5.4 Program stream map(p82)
-
- #include "mpeg-ps-internal.h"
- #include "mpeg-pes-internal.h"
- #include "mpeg-element-descriptor.h"
- #include "mpeg-util.h"
- #include <assert.h>
- #include <string.h>
-
- static struct pes_t* psm_fetch(struct psm_t* psm, uint8_t sid)
- {
- size_t i;
- for (i = 0; i < psm->stream_count; i++)
- {
- if (psm->streams[i].sid == sid)
- return &psm->streams[i];
- }
-
- if (psm->stream_count >= sizeof(psm->streams) / sizeof(psm->streams[0]))
- {
- assert(0);
- return NULL;
- }
-
- // new stream
- return &psm->streams[psm->stream_count++];
- }
-
- #if 1
- int psm_read(struct psm_t* psm, struct mpeg_bits_t* reader)
- {
- uint8_t v8;
- size_t end, off;
- struct pes_t* stream;
- //uint8_t current_next_indicator;
- uint8_t single_extension_stream_flag;
- uint16_t program_stream_map_length;
- uint16_t program_stream_info_length;
- uint16_t element_stream_map_length;
- uint16_t element_stream_info_length;
- uint8_t cid, sid;
-
- // Table 2-41 - Program stream map(p79)
- program_stream_map_length = mpeg_bits_read16(reader); // (data[4] << 8) | data[5];
- end = mpeg_bits_tell(reader) + program_stream_map_length;
- if (mpeg_bits_error(reader) || end > mpeg_bits_length(reader))
- return MPEG_ERROR_NEED_MORE_DATA;
-
- v8 = mpeg_bits_read8(reader); // data[6]
- //assert((0x20 & data[6]) == 0x00); // 'xx0xxxxx'
- //current_next_indicator = (data[6] >> 7) & 0x01;
- single_extension_stream_flag = (uint8_t)(v8 >> 6) & 0x01; //(data[6] >> 6) & 0x01;
- psm->ver = v8 & 0x1F;
- mpeg_bits_read8(reader); //assert(data[7] == 0x01); // '00000001'
-
- // program stream descriptor
- program_stream_info_length = mpeg_bits_read16(reader); //(data[8] << 8) | data[9];
- if ((uint32_t)program_stream_info_length + 4 + 2 /*element_stream_map_length*/ > (uint32_t)program_stream_map_length)
- return MPEG_ERROR_INVALID_DATA;
-
- // TODO: parse descriptor
- //for (i = 10; i + 2 <= 10 + program_stream_info_length;)
- //{
- // // descriptor()
- // i += mpeg_elment_descriptor(data + i, 10 + program_stream_info_length - i);
- //}
- mpeg_bits_skip(reader, program_stream_info_length); // 10 + program_stream_info_length;
-
- // program element stream
- element_stream_map_length = mpeg_bits_read16(reader);
- /* Ignore es_map_length, trust psm_length */
- element_stream_map_length = program_stream_map_length - program_stream_info_length - 10;
- end = mpeg_bits_tell(reader) + element_stream_map_length;
-
- while (0 == mpeg_bits_error(reader)
- && mpeg_bits_tell(reader) + 4 /*element_stream_info_length*/ <= end
- && psm->stream_count < sizeof(psm->streams) / sizeof(psm->streams[0]))
- {
- cid = mpeg_bits_read8(reader);
- sid = mpeg_bits_read8(reader);
- element_stream_info_length = mpeg_bits_read16(reader);
- if (mpeg_bits_tell(reader) + element_stream_info_length > end)
- return MPEG_ERROR_INVALID_DATA;
-
- stream = psm_fetch(psm, sid); // sid
- if (NULL == stream)
- continue;
- stream->codecid = cid;
- stream->sid = sid;
- stream->pid = stream->sid; // for ts PID
-
- off = mpeg_bits_tell(reader);
- if (0xFD == stream->sid && 0 == single_extension_stream_flag)
- {
- if (element_stream_info_length < 3)
- return MPEG_ERROR_INVALID_DATA;
- //uint8_t pseudo_descriptor_tag = mpeg_bits_read8(reader);
- //uint8_t pseudo_descriptor_length = mpeg_bits_read8(reader)
- //uint8_t element_stream_id_extension = mpeg_bits_read8(reader) & 0x7F;
- //assert((0x80 & data[k + 2]) == 0x80); // '1xxxxxxx'
- mpeg_bits_skip(reader, 3);
- }
-
- while (0 == mpeg_bits_error(reader) && mpeg_bits_tell(reader) < off + element_stream_info_length)
- {
- // descriptor()
- mpeg_elment_descriptor(reader);
- }
-
- assert(mpeg_bits_tell(reader) == off + element_stream_info_length);
- mpeg_bits_seek(reader, off + element_stream_info_length); // make sure
- }
-
- mpeg_bits_read32(reader); // crc32
- // assert(j+4 == program_stream_map_length+6);
- // assert(0 == mpeg_crc32(0xffffffff, data, program_stream_map_length+6));
- assert(0 == mpeg_bits_error(reader));
- assert(end + 4 /*crc32*/ == mpeg_bits_tell(reader));
- return MPEG_ERROR_OK;
- }
-
- #else
- size_t psm_read(struct psm_t *psm, const uint8_t* data, size_t bytes)
- {
- size_t i, j, k;
- struct pes_t* stream;
- //uint8_t current_next_indicator;
- uint8_t single_extension_stream_flag;
- uint16_t program_stream_map_length;
- uint16_t program_stream_info_length;
- uint16_t element_stream_map_length;
- uint16_t element_stream_info_length;
-
- // Table 2-41 - Program stream map(p79)
- assert(0x00==data[0] && 0x00==data[1] && 0x01==data[2] && 0xBC==data[3]);
- program_stream_map_length = (data[4] << 8) | data[5];
- if (program_stream_map_length < 3 || bytes < (size_t)program_stream_map_length + 6)
- return 0; // invalid data length
-
- //assert((0x20 & data[6]) == 0x00); // 'xx0xxxxx'
- //current_next_indicator = (data[6] >> 7) & 0x01;
- single_extension_stream_flag = (data[6] >> 6) & 0x01;
- psm->ver = data[6] & 0x1F;
- //assert(data[7] == 0x01); // '00000001'
-
- // program stream descriptor
- program_stream_info_length = (data[8] << 8) | data[9];
- if ((size_t)program_stream_info_length + 4 + 2 /*element_stream_map_length*/ > (size_t)program_stream_map_length)
- return 0; // TODO: error
-
- // TODO: parse descriptor
- //for (i = 10; i + 2 <= 10 + program_stream_info_length;)
- //{
- // // descriptor()
- // i += mpeg_elment_descriptor(data + i, 10 + program_stream_info_length - i);
- //}
-
- // program element stream
- i = 10 + program_stream_info_length;
- element_stream_map_length = (data[i] << 8) | data[i+1];
- /* Ignore es_map_length, trust psm_length */
- element_stream_map_length = program_stream_map_length - program_stream_info_length - 10;
-
- i += 2;
- for(j = i; j + 4/*element_stream_info_length*/ <= i+element_stream_map_length && psm->stream_count < sizeof(psm->streams)/sizeof(psm->streams[0]); j += 4 + element_stream_info_length)
- {
- element_stream_info_length = (data[j + 2] << 8) | data[j + 3];
- if (j + 4 + element_stream_info_length > i + element_stream_map_length)
- return 0; // TODO: error
-
- stream = psm_fetch(psm, data[j + 1]); // sid
- if (NULL == stream)
- continue;
- stream->codecid = data[j];
- stream->sid = data[j+1];
- stream->pid = stream->sid; // for ts PID
-
- k = j + 4;
- if(0xFD == stream->sid && 0 == single_extension_stream_flag)
- {
- if(element_stream_info_length < 3)
- return 0; // TODO: error
- // uint8_t pseudo_descriptor_tag = data[k];
- // uint8_t pseudo_descriptor_length = data[k+1];
- // uint8_t element_stream_id_extension = data[k+2] & 0x7F;
- assert((0x80 & data[k+2]) == 0x80); // '1xxxxxxx'
- k += 3;
- }
-
- while(k + 2 <= j + 4 + element_stream_info_length)
- {
- // descriptor()
- k += mpeg_elment_descriptor(data+k, j + 4 + element_stream_info_length - k);
- }
-
- assert(k - j - 4 == element_stream_info_length);
- }
-
- // assert(j+4 == program_stream_map_length+6);
- // assert(0 == mpeg_crc32(0xffffffff, data, program_stream_map_length+6));
- return program_stream_map_length+6;
- }
- #endif
-
- size_t psm_write(const struct psm_t *psm, uint8_t *data)
- {
- // Table 2-41 - Program stream map(p79)
-
- size_t i,j;
- uint16_t extlen;
- unsigned int crc;
-
- nbo_w32(data, 0x00000100);
- data[3] = PES_SID_PSM;
-
- // program_stream_map_length 16-bits
- //nbo_w16(data+4, 6+4*psm->stream_count+4);
-
- // current_next_indicator '1'
- // single_extension_stream_flag '1'
- // reserved '0'
- // program_stream_map_version 'xxxxx'
- data[6] = 0xc0 | (psm->ver & 0x1F);
-
- // reserved '0000000'
- // marker_bit '1'
- data[7] = 0x01;
-
- extlen = 0;
- extlen += (uint16_t)service_extension_descriptor_write(data + 10 + extlen, 32);
- #if defined(MPEG_CLOCK_EXTENSION_DESCRIPTOR)
- extlen += (uint16_t)clock_extension_descriptor_write(data + 10 + extlen, 32, psm->clock);
- #endif
-
- // program_stream_info_length 16-bits
- nbo_w16(data + 8, extlen); // program_stream_info_length = 0
-
- // elementary_stream_map_length 16-bits
- //nbo_w16(data+10+extlen, psm->stream_count*4);
-
- j = 12 + extlen;
- for(i = 0; i < psm->stream_count; i++)
- {
- assert(PES_SID_EXTEND != psm->streams[i].sid);
-
- // stream_type:8
- data[j++] = psm->streams[i].codecid;
- // elementary_stream_id:8
- data[j++] = psm->streams[i].sid;
- // elementary_stream_info_length:16
- nbo_w16(data+j, psm->streams[i].esinfo_len);
- // descriptor()
- memcpy(data+j+2, psm->streams[i].esinfo, psm->streams[i].esinfo_len);
-
- j += 2 + psm->streams[i].esinfo_len;
- }
-
- // elementary_stream_map_length 16-bits
- nbo_w16(data + 10 + extlen, (uint16_t)(j - 12 - extlen));
- // program_stream_map_length:16
- nbo_w16(data + 4, (uint16_t)(j-6+4)); // 4-bytes crc32
-
- // crc32
- crc = mpeg_crc32(0xffffffff, data, (uint32_t)j);
- data[j+3] = (uint8_t)((crc >> 24) & 0xFF);
- data[j+2] = (uint8_t)((crc >> 16) & 0xFF);
- data[j+1] = (uint8_t)((crc >> 8) & 0xFF);
- data[j+0] = (uint8_t)(crc & 0xFF);
-
- return j+4;
- }
|