205 lines
5.3 KiB
C
205 lines
5.3 KiB
C
// ITU-T H.222.0(10/2014)
|
|
// Information technology - Generic coding of moving pictures and associated audio information: Systems
|
|
// 2.4.4.3 Program association table(p65)
|
|
|
|
#include "mpeg-ts-internal.h"
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <assert.h>
|
|
|
|
struct pmt_t* pat_alloc_pmt(struct pat_t* pat)
|
|
{
|
|
void* ptr;
|
|
unsigned int n;
|
|
|
|
if (NULL == pat->pmts)
|
|
{
|
|
assert(0 == pat->pmt_count);
|
|
assert(0 == pat->pmt_capacity);
|
|
pat->pmts = pat->pmt_default;
|
|
pat->pmt_capacity = sizeof(pat->pmt_default) / sizeof(pat->pmt_default[0]);
|
|
}
|
|
|
|
if (pat->pmt_count >= pat->pmt_capacity)
|
|
{
|
|
if (pat->pmt_count + 1 > 65535)
|
|
{
|
|
assert(0);
|
|
return NULL;
|
|
}
|
|
|
|
n = pat->pmt_capacity + pat->pmt_capacity / 4 + 4;
|
|
ptr = realloc(pat->pmts == pat->pmt_default ? NULL : pat->pmts, sizeof(pat->pmts[0]) * n);
|
|
if (!ptr)
|
|
return NULL;
|
|
|
|
assert(ptr != pat->pmt_default);
|
|
if (pat->pmts == pat->pmt_default)
|
|
memmove(ptr, pat->pmt_default, sizeof(pat->pmt_default));
|
|
pat->pmts = (struct pmt_t*)ptr;
|
|
pat->pmt_capacity = n;
|
|
}
|
|
|
|
// new pmt
|
|
memset(&pat->pmts[pat->pmt_count], 0, sizeof(pat->pmts[0]));
|
|
return &pat->pmts[pat->pmt_count];
|
|
}
|
|
|
|
static struct pmt_t* pat_fetch(struct pat_t* pat, uint16_t pid)
|
|
{
|
|
unsigned int i;
|
|
struct pmt_t* pmt;
|
|
for(i = 0; i < pat->pmt_count; i++)
|
|
{
|
|
if(pat->pmts[i].pid == pid)
|
|
return &pat->pmts[i];
|
|
}
|
|
|
|
// new pmt
|
|
pmt = pat_alloc_pmt(pat);
|
|
pat->pmt_count++;
|
|
return pmt;
|
|
}
|
|
|
|
size_t pat_read(struct pat_t *pat, const uint8_t* data, size_t bytes)
|
|
{
|
|
// Table 2-30 Program association section(p65)
|
|
|
|
struct pmt_t* pmt;
|
|
uint32_t i = 0;
|
|
uint16_t pn, pid;
|
|
// uint32_t crc = 0;
|
|
uint32_t section_length, transport_stream_id, version_number;
|
|
|
|
if(bytes < 8)
|
|
return 0; // invalid data length
|
|
// printf("PAT: %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]);
|
|
assert(PAT_TID_PAS == data[0]); // table_id
|
|
assert(1 == ((data[1] >> 7) & 0x01)); // section_syntax_indicator
|
|
// uint32_t zero = (data[1] >> 6) & 0x01;
|
|
// uint32_t reserved = (data[1] >> 4) & 0x03;
|
|
section_length = ((data[1] & 0x0F) << 8) | data[2];
|
|
transport_stream_id = (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;
|
|
// uint32_t sector_number = data[6];
|
|
// uint32_t last_sector_number = data[7];
|
|
|
|
if (PAT_TID_PAS != data[0] || section_length + 3 < 8 + 4 /*crc32*/ || section_length + 3 > bytes)
|
|
{
|
|
assert(0);
|
|
return 0; // invalid data length
|
|
}
|
|
|
|
assert(bytes >= section_length + 3); // PMT = section_length + 3
|
|
//if(pat->ver != version_number)
|
|
// pat_clear(pat); // fix pat.pmt[i].pes.pkt.data memory leak
|
|
pat->tsid = transport_stream_id;
|
|
pat->ver = version_number;
|
|
|
|
// TODO: version_number change, reload pmts
|
|
|
|
// 4:CRC, 5:follow section_length item
|
|
for(i = 8; i + 4 <= section_length + 8 - 5 - 4/*CRC32*/ && section_length + 3 <= bytes; i += 4)
|
|
{
|
|
pn = (data[i] << 8) | data[i+1];
|
|
pid = ((data[i+2] & 0x1F) << 8) | data[i+3];
|
|
// printf("PAT: pn: 0x%0x, pid: 0x%0x\n", (unsigned int)pn, (unsigned int)pid);
|
|
|
|
if(0 == pn)
|
|
continue; // ignore NIT info
|
|
pmt = pat_fetch(pat, pid);
|
|
if(NULL == pmt)
|
|
continue;
|
|
|
|
pmt->pn = pn;
|
|
pmt->pid = pid;
|
|
}
|
|
|
|
//assert(i+4 == bytes);
|
|
//crc = (data[i] << 24) | (data[i+1] << 16) | (data[i+2] << 8) | data[i+3];
|
|
//crc = mpeg_crc32(-1, data, bytes-4);
|
|
// assert(0 == mpeg_crc32(0xffffffff, data, section_length+3));
|
|
return section_length + 3;
|
|
}
|
|
|
|
size_t pat_write(const struct pat_t *pat, uint8_t *data)
|
|
{
|
|
// Table 2-30 Program association section(p65)
|
|
|
|
uint32_t i = 0;
|
|
uint32_t len = 0;
|
|
uint32_t crc = 0;
|
|
|
|
len = pat->pmt_count * 4 + 5 + 4; // 5 bytes remain header and 4 bytes crc32
|
|
|
|
// shall not exceed 1021 (0x3FD).
|
|
assert(len <= 1021);
|
|
assert(len <= TS_PACKET_SIZE - 7);
|
|
|
|
data[0] = PAT_TID_PAS; // program association table
|
|
|
|
// section_syntax_indicator = '1'
|
|
// '0'
|
|
// reserved '11'
|
|
nbo_w16(data + 1, (uint16_t)(0xb000 | len));
|
|
|
|
// transport_stream_id
|
|
nbo_w16(data + 3, (uint16_t)pat->tsid);
|
|
|
|
// reserved '11'
|
|
// version_number 'xxxxx'
|
|
// current_next_indicator '1'
|
|
data[5] = (uint8_t)(0xC1 | (pat->ver << 1));
|
|
|
|
// section_number/last_section_number
|
|
data[6] = 0x00;
|
|
data[7] = 0x00;
|
|
|
|
for(i = 0; i < pat->pmt_count; i++)
|
|
{
|
|
nbo_w16(data + 8 + i * 4 + 0, (uint16_t)pat->pmts[i].pn);
|
|
nbo_w16(data + 8 + i * 4 + 2, (uint16_t)(0xE000 | pat->pmts[i].pid));
|
|
}
|
|
|
|
// crc32
|
|
crc = mpeg_crc32(0xffffffff, data, len-1);
|
|
//put32(data + section_length - 1, crc);
|
|
data[len - 1 + 3] = (crc >> 24) & 0xFF;
|
|
data[len - 1 + 2] = (crc >> 16) & 0xFF;
|
|
data[len - 1 + 1] = (crc >> 8) & 0xFF;
|
|
data[len - 1 + 0] = crc & 0xFF;
|
|
|
|
return len + 3; // total length
|
|
}
|
|
|
|
struct pmt_t* pat_find(struct pat_t* pat, uint16_t pn)
|
|
{
|
|
unsigned int i;
|
|
for(i = 0; i < pat->pmt_count; i++)
|
|
{
|
|
if(pat->pmts[i].pn == pn)
|
|
return &pat->pmts[i];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void pat_clear(struct pat_t* pat)
|
|
{
|
|
unsigned int i;
|
|
for (i = 0; i < pat->pmt_count; i++)
|
|
{
|
|
pmt_clear(&pat->pmts[i]);
|
|
}
|
|
|
|
if (pat->pmts && pat->pmts != pat->pmt_default)
|
|
{
|
|
free(pat->pmts);
|
|
pat->pmts = NULL;
|
|
}
|
|
|
|
pat->pmt_count = 0;
|
|
pat->pmt_capacity = 0;
|
|
}
|