Older/MediaServer/libmpeg/source/mpeg-pmt.c

355 lines
9.9 KiB
C
Raw Permalink Normal View History

2024-10-01 00:12:57 +08:00
// ITU-T H.222.0(10/2014)
// Information technology - Generic coding of moving pictures and associated audio information: Systems
// 2.4.4.8 Program map table(p68)
#include "mpeg-ts-internal.h"
#include "mpeg-ts-opus.h"
#include "mpeg-util.h"
#include <stdlib.h>
#include <string.h>
#include <assert.h>
static struct pes_t* pmt_fetch(struct pmt_t* pmt, uint16_t pid)
{
unsigned int i;
for(i = 0; i < pmt->stream_count; i++)
{
if(pmt->streams[i].pid == pid)
return &pmt->streams[i];
}
if(pmt->stream_count >= sizeof(pmt->streams) / sizeof(pmt->streams[0]))
{
assert(0);
return NULL;
}
// new stream
return &pmt->streams[pmt->stream_count++];
}
static int pmt_read_program_descriptor(struct pmt_t* pmt, const uint8_t* data, uint16_t bytes)
{
uint8_t tag;
uint8_t len;
uint8_t channels;
// Registration descriptor
while (bytes > 2)
{
tag = data[0];
len = data[1];
if (len + 2 > bytes)
return -1; // invalid len
// ISO/IEC 13818-1:2018 (E) Table 2-45 Program and program element descriptors (p90)
switch (tag)
{
case 0x05: // 2.6.8 Registration descriptor(p94)
if (len >= 4 && 'C' == data[2] && 'U' == data[3] && 'E' == data[4] && 'I' == data[5])
{
memcpy(pmt->proginfo, data+2, 4);
}
break;
}
data += len + 2;
bytes -= len + 2;
}
assert(0 == bytes);
return 0;
}
static int pmt_read_descriptor(struct pes_t* stream, const uint8_t* data, uint16_t bytes)
{
uint8_t tag;
uint8_t len;
uint8_t channels;
// Registration descriptor
while (bytes > 2)
{
tag = data[0];
len = data[1];
if (len + 2 > bytes)
return -1; // invalid len
// ISO/IEC 13818-1:2018 (E) Table 2-45 ?Program and program element descriptors (p90)
switch (tag)
{
case 0x05: // 2.6.8 Registration descriptor(p94)
if (len >= 4 && 'O' == data[2] && 'p' == data[3] && 'u' == data[4] && 's' == data[5])
{
assert(PSI_STREAM_PRIVATE_DATA == stream->codecid);
stream->codecid = PSI_STREAM_AUDIO_OPUS;
}
else if (len >= 4 && 'A' == data[2] && 'V' == data[3] && '0' == data[4] && '1' == data[5])
{
// https://aomediacodec.github.io/av1-mpeg2-ts/
// Constraints on AV1 streams in MPEG-2 TS
assert(PSI_STREAM_PRIVATE_DATA == stream->codecid);
stream->codecid = PSI_STREAM_AV1;
}
else if (len >= 4 && 'A' == data[2] && 'V' == data[3] && 'S' == data[4] && '3' == data[5])
{
stream->codecid = PSI_STREAM_VIDEO_AVS3;
}
break;
case 0x7f: // DVB-Service Information: 6.1 Descriptor identification and location (p38)
// 2.6.90 Extension descriptor
if (len >= 2 && data[3] <= sizeof(opus_channel_map)/sizeof(opus_channel_map[0]) && PSI_STREAM_AUDIO_OPUS == stream->codecid && OPUS_EXTENSION_DESCRIPTOR_TAG == data[2]) // User defined (provisional Opus)
{
channels = data[3];
stream->esinfo = (uint8_t*)calloc(1, sizeof(opus_default_extradata));
if (stream->esinfo)
{
stream->esinfo_len = sizeof(opus_default_extradata);
memcpy(stream->esinfo, opus_default_extradata, stream->esinfo_len);
stream->esinfo[9] = channels ? channels : 2;
stream->esinfo[18] = channels ? (channels > 2 ? 1 : 0) : /* Dual Mono */ 255;
stream->esinfo[19] = opus_stream_cnt[channels];
stream->esinfo[20] = opus_coupled_stream_cnt[channels];
memcpy(stream->esinfo + 21, opus_channel_map[(channels ? channels : 2) - 1], channels ? channels : 2);
}
}
}
data += len + 2;
bytes -= len + 2;
}
assert(0 == bytes);
return 0;
}
static int pmt_write_descriptor(const struct pes_t* stream, uint8_t* data, int bytes)
{
uint8_t* p;
p = data;
if (PSI_STREAM_AUDIO_OPUS == stream->codecid && bytes > 2 + 4 /*fourcc*/ + 4 /*DVI OPUS*/ )
{
*p++ = 0x05; // 2.6.8 Registration descriptor(p94)
*p++ = 4;
memcpy(p, "Opus", 4);
p += 4;
*p++ = 0x7f; // DVB-Service Information: 6.1 Descriptor identification and location (p38)
*p++ = 2;
*p++ = OPUS_EXTENSION_DESCRIPTOR_TAG;
*p++ = stream->esinfo_len > 8 ? stream->esinfo[9] : 2 /*0xFF*/ ; // default 2-channels
}
return (int)(p - data);
}
size_t pmt_read(struct pmt_t *pmt, const uint8_t* data, size_t bytes)
{
struct pes_t* stream;
uint16_t pid, len;
uint32_t i, section_length, program_number, version_number;
uint32_t PCR_PID, program_info_length;
if (bytes < 12)
return 0; // invalid data length
// printf("PMT: %0x %0x %0x %0x %0x %0x %0x %0x, %0x, %0x, %0x, %0x\n", (unsigned int)data[0], (unsigned int)data[1], (unsigned int)data[2], (unsigned int)data[3], (unsigned int)data[4], (unsigned int)data[5], (unsigned int)data[6],(unsigned int)data[7],(unsigned int)data[8],(unsigned int)data[9],(unsigned int)data[10],(unsigned int)data[11]);
assert(PAT_TID_PMS == data[0]);
assert(1 == ((data[1] >> 7) & 0x01));
section_length = ((data[1] & 0x0F) << 8) | data[2];
program_number = (data[3] << 8) | data[4];
// uint32_t reserved2 = (data[5] >> 6) & 0x03;
version_number = (data[5] >> 1) & 0x1F;
// uint32_t current_next_indicator = data[5] & 0x01;
assert(0 == data[6]); // sector_number
assert(0 == data[7]); // last_sector_number
// uint32_t reserved3 = (data[8] >> 5) & 0x07;
PCR_PID = ((data[8] & 0x1F) << 8) | data[9];
// uint32_t reserved4 = (data[10] >> 4) & 0x0F;
program_info_length = ((data[10] & 0x0F) << 8) | data[11];
if (PAT_TID_PMS != data[0] || section_length + 3 < 12 + 4 /*crc32*/ || section_length + 3 > bytes
|| program_info_length + 12 /*head*/ > section_length + 3 /*head*/ - 4 /*crc32*/ )
{
assert(0);
return 0; // invalid data length
}
//if(pmt->ver != version_number)
// pmt_clear(pmt); // fix pmt.pes.pkt.data memory leak
pmt->PCR_PID = PCR_PID;
pmt->pn = program_number;
//pmt->ver = version_number;
pmt->pminfo_len = program_info_length;
if(program_info_length > 2)
{
// descriptor(data + 12, program_info_length)
pmt_read_program_descriptor(pmt, data + 12, program_info_length);
}
PMT_VERSION_CHANGE:
assert(bytes >= section_length + 3); // PMT = section_length + 3
for (i = 12 + program_info_length; i + 5 <= section_length + 3 - 4/*CRC32*/ && section_length + 3 <= bytes; i += len + 5) // 9: follow section_length item
{
pid = ((data[i+1] & 0x1F) << 8) | data[i+2];
len = ((data[i+3] & 0x0F) << 8) | data[i+4];
//printf("PMT: pn: 0x%0x, pid: 0x%0x, codec: 0x%0x, eslen: %d\n", (unsigned int)pmt->pn, (unsigned int)pid, (unsigned int)data[i], (unsigned int)len);
if (i + len + 5 > section_length + 3 - 4/*CRC32*/)
break; // mark error ?
assert(pmt->stream_count <= sizeof(pmt->streams)/sizeof(pmt->streams[0]));
stream = pmt_fetch(pmt, pid);
if (NULL == stream)
{
if (pmt->ver != version_number)
{
pmt->ver = version_number; // once only
pmt_clear(pmt);
goto PMT_VERSION_CHANGE;
}
continue;
}
stream->pn = (uint16_t)pmt->pn;
stream->pid = pid;
stream->codecid = data[i];
stream->esinfo_len = 0; // default nothing
if (len > 2)
{
//descriptor(data + i + 5, len)
pmt_read_descriptor(stream, data + i + 5, len);
}
}
pmt->ver = version_number;
//assert(j+4 == bytes);
//crc = (data[j] << 24) | (data[j+1] << 16) | (data[j+2] << 8) | data[j+3];
// assert(0 == mpeg_crc32(0xffffffff, data, section_length+3));
return section_length + 3;
}
size_t pmt_write(const struct pmt_t *pmt, uint8_t *data)
{
// 2.4.4.8 Program map table (p68)
// Table 2-33
uint32_t i = 0;
uint32_t crc = 0;
ptrdiff_t len = 0;
uint8_t *p = NULL;
data[0] = PAT_TID_PMS; // program map table
// skip section_length
// program_number
nbo_w16(data + 3, (uint16_t)pmt->pn);
// reserved '11'
// version_number 'xxxxx'
// current_next_indicator '1'
data[5] = (uint8_t)(0xC1 | (pmt->ver << 1));
// section_number/last_section_number
data[6] = 0x00;
data[7] = 0x00;
// reserved '111'
// PCR_PID 13-bits 0x1FFF
nbo_w16(data + 8, (uint16_t)(0xE000 | pmt->PCR_PID));
// reserved '1111'
// program_info_length 12-bits, the first two bits of which shall be '00'.
assert(pmt->pminfo_len < 0x400);
nbo_w16(data + 10, (uint16_t)(0xF000 | pmt->pminfo_len));
if(pmt->pminfo_len > 0 && pmt->pminfo_len < 0x400)
{
// fill program info
assert(pmt->pminfo);
memcpy(data + 12, pmt->pminfo, pmt->pminfo_len);
}
// streams
p = data + 12 + pmt->pminfo_len;
for(i = 0; i < pmt->stream_count && p - data < 1021 - 4 - 5 - pmt->streams[i].esinfo_len; i++)
{
// stream_type
*p = (uint8_t)(PSI_STREAM_AUDIO_OPUS == pmt->streams[i].codecid) ? PSI_STREAM_PRIVATE_DATA : pmt->streams[i].codecid;
// reserved '111'
// elementary_PID 13-bits
nbo_w16(p + 1, 0xE000 | pmt->streams[i].pid);
len = 0;
// fill elementary stream info
//if(PSI_STREAM_AUDIO_OPUS == pmt->streams[i].codecid || (pmt->streams[i].esinfo_len > 0 && pmt->streams[i].esinfo))
{
//assert(pmt->streams[i].esinfo);
//memcpy(p, pmt->streams[i].esinfo, pmt->streams[i].esinfo_len);
//p += pmt->streams[i].esinfo_len;
len = pmt_write_descriptor(&pmt->streams[i], p + 5, 1021 - (int)(p + 5 - data));
}
// reserved '1111'
// ES_info_lengt 12-bits
nbo_w16(p + 3, 0xF000 | (uint16_t)len);
p += 5 + len;
}
// section_length
len = p + 4 - (data + 3); // 4 bytes crc32
assert(len <= 1021); // shall not exceed 1021 (0x3FD).
assert(len <= TS_PACKET_SIZE - 7);
// section_syntax_indicator '1'
// '0'
// reserved '11'
nbo_w16(data + 1, (uint16_t)(0xb000 | len));
// crc32
crc = mpeg_crc32(0xffffffff, data, (uint32_t)(p-data));
//put32(p, crc);
p[3] = (crc >> 24) & 0xFF;
p[2] = (crc >> 16) & 0xFF;
p[1] = (crc >> 8) & 0xFF;
p[0] = crc & 0xFF;
return (p - data) + 4; // total length
}
void pmt_clear(struct pmt_t* pmt)
{
unsigned int i;
struct pes_t* pes;
for (i = 0; i < pmt->stream_count; i++)
{
pes = &pmt->streams[i];
if (pes->pkt.data)
{
free(pes->pkt.data);
pes->pkt.data = NULL;
}
pes->pkt.size = 0;
if (pes->esinfo)
{
free(pes->esinfo);
pes->esinfo = NULL;
}
pes->esinfo_len = 0;
}
pmt->stream_count = 0;
if (pmt->pminfo)
{
free(pmt->pminfo);
pmt->pminfo = NULL;
}
pmt->pminfo_len = 0;
}