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

181 lines
5.8 KiB
C
Raw Normal View History

2024-10-01 00:12:57 +08:00
// ETSI EN 300 468 V1.15.1 (2016-03)
// Digital Video Broadcasting (DVB); Specification for Service Information (SI) in DVB systems
// 5.2.3 Service Description Table (SDT) (p27)
#include "mpeg-ts-internal.h"
#include "mpeg-element-descriptor.h"
#include "mpeg-util.h"
#include <string.h>
#include <assert.h>
#define SERVICE_ENCODER "encoder"
/*
service_description_section(){
table_id 8 uimsbf
section_syntax_indicator 1 bslbf
reserved_future_use 1 bslbf
reserved 2 bslbf
section_length 12 uimsbf
transport_stream_id 16 uimsbf
reserved 2 bslbf
version_number 5 uimsbf
current_next_indicator 1 bslbf
section_number 8 uimsbf
last_section_number 8 uimsbf
original_network_id 16 uimsbf
reserved_future_use 8 bslbf
for (i=0;i<N;i++){
service_id 16 uimsbf
reserved_future_use 6 bslbf
EIT_schedule_flag 1 bslbf
EIT_present_following_flag 1 bslbf
running_status 3 uimsbf
free_CA_mode 1 bslbf
descriptors_loop_length 12 uimsbf
for (j=0;j<N;j++){
descriptor()
}
}
CRC_32 32 rpchof
}
*/
size_t sdt_read(struct pat_t *pat, const uint8_t* data, size_t bytes)
{
struct pmt_t* pmt;
uint16_t sid;
uint32_t i, k, n, section_length;
uint8_t tagid, taglen, tagn1, tagn2;
if (bytes < 11)
return 0;
// printf("SDT: %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]);
// uint32_t table_id = data[0];
// uint32_t section_syntax_indicator = (data[1] >> 7) & 0x01;
// uint32_t zero = (data[1] >> 6) & 0x01;
// uint32_t reserved = (data[1] >> 4) & 0x03;
section_length = ((data[1] & 0x0F) << 8) | data[2];
// uint32_t transport_stream_id = (data[3] << 8) | data[4];
// uint32_t reserved2 = (data[5] >> 6) & 0x03;
// uint32_t 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];
// uint32_t original_network_id = (data[8] << 8) | data[9];
// uint32_t reserved4 = data[10];
if(PAT_TID_SDT != data[0] || section_length + 3 > bytes)
return 0;
// TODO: version_number change, reload SDT
assert(bytes >= section_length + 3); // PMT = section_length + 3
for (i = 11; i + 5 <= section_length + 3 - 4/*CRC32*/ && section_length + 3 <= bytes; i += 5 + n) // 9: follow section_length item
{
sid = (data[i] << 8) | data[i + 1];
n = ((data[i+3] & 0x0F) << 8) | data[i+4];
// skip reserved/EIT data[i+2]
if(i + 5 + n > section_length + 3 - 4)
continue;
pmt = pat_find(pat, sid);
if(NULL == pmt)
continue; // pmt not found
for(k = i + 5; k + 2 <= i + 5 + n; k += 2 + taglen)
{
// 6.1 Descriptor identification and location
tagid = data[k];
taglen = data[k + 1];
// 6.2.33 Service descriptor (p77)
if(0x48 != tagid || k + taglen > i + 5 + n)
continue;
//service_type = data[k + 2];
tagn1 = data[k + 3];
if(tagn1 >= sizeof(pmt->provider) || k + 3 + tagn1 > i + 5 + n)
continue;
memcpy(pmt->provider, &data[k+4], tagn1);
pmt->provider[tagn1] = 0;
tagn2 = data[k + 4 + tagn1];
if(tagn2 >= sizeof(pmt->name) || k + 5 + tagn1 + tagn2 > i + 5 + n)
continue;
memcpy(pmt->name, &data[k+5+tagn1], tagn2);
pmt->name[tagn2] = 0;
}
}
//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 sdt_write(const struct pat_t* pat, uint8_t* data)
{
size_t i, j;
size_t len, s1, n1, v1;
uint32_t crc = 0;
n1 = strlen(SERVICE_ENCODER);
v1 = strlen(SERVICE_NAME);
s1 = 3 /*tag*/ + 1 + n1 + 1 + v1;
len = 3 /*nid*/ + s1 + 5 /*service head*/ + 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_SDT; // service_description_section
// section_syntax_indicator = '1'
// '0'
// reserved '11'
nbo_w16(data + 1, (uint16_t)(0xf000 | 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;
// original_network_id
nbo_w16(data + 8, (uint16_t)pat->tsid);
data[10] = 0xFF; // reserved
j = 11;
// only one
for (i = 0; i < 1; i++)
{
nbo_w16(data + j, (uint16_t)pat->tsid);
data[j + 2] = 0xfc | 0x00; // no EIT
assert(n1 < 255 && v1 < 255 && len < 255);
nbo_w16(data + j + 3, (uint16_t)(0x8000 | s1));
data[j + 5] = 0x48; // tag id
data[j + 6] = (uint8_t)(3 + n1 + v1); // tag len
data[j + 7] = 1; // service type
data[j + 8] = (uint8_t)n1;
memcpy(data + j + 9, SERVICE_NAME, n1);
data[j + 9 + n1] = (uint8_t)v1;
memcpy(data + j + 10 + n1, SERVICE_NAME, v1);
j += 10 + v1 + n1;
}
// 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;
}