⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 atsc_stt.c

📁 Dvbstreamer 用在解析MPTS的部分内容
💻 C
字号:
/*Copyright (C) 2006  Adam CharrettThis program is free software; you can redistribute it and/ormodify it under the terms of the GNU General Public Licenseas published by the Free Software Foundation; either version 2of the License, or (at your option) any later version.This program is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY; without even the implied warranty ofMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See theGNU General Public License for more details.You should have received a copy of the GNU General Public Licensealong with this program; if not, write to the Free SoftwareFoundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USAstt.cDecode PSIP System Time Table.*/#include "config.h"#include <stdio.h>#include <stdlib.h>#include <string.h>#include <stdint.h>#include "dvbpsi.h"#include "dvbpsi_private.h"#include "psi.h"#include "descriptor.h"#include "demux.h"#include "atsc/stt.h"#include "objects.h"typedef struct dvbpsi_atsc_stt_decoder_s{  dvbpsi_atsc_stt_callback      pf_callback;  void *                        p_cb_data;} dvbpsi_atsc_stt_decoder_t;dvbpsi_descriptor_t *dvbpsi_atsc_STTAddDescriptor(                                               dvbpsi_atsc_stt_t *p_stt,                                               uint8_t i_tag, uint8_t i_length,                                               uint8_t *p_data);void dvbpsi_atsc_GatherSTTSections(dvbpsi_decoder_t * p_psi_decoder,                              void * p_private_decoder,                              dvbpsi_psi_section_t * p_section);void dvbpsi_atsc_DecodeSTTSections(dvbpsi_atsc_stt_t* p_stt,                              dvbpsi_psi_section_t* p_section);/***************************************************************************** * dvbpsi_atsc_AttachSTT ***************************************************************************** * Initialize a STT subtable decoder. *****************************************************************************/int dvbpsi_atsc_AttachSTT(dvbpsi_decoder_t * p_psi_decoder, uint8_t i_table_id,                           dvbpsi_atsc_stt_callback pf_callback, void* p_cb_data){  dvbpsi_demux_t* p_demux = (dvbpsi_demux_t*)p_psi_decoder->p_private_decoder;  dvbpsi_demux_subdec_t* p_subdec;  dvbpsi_atsc_stt_decoder_t*  p_stt_decoder;  if(dvbpsi_demuxGetSubDec(p_demux, i_table_id, 0))  {    DVBPSI_ERROR_ARG("STT decoder",                     "Already a decoder for (table_id == 0x%02x)",                     i_table_id);    return 1;  }  p_subdec = (dvbpsi_demux_subdec_t*)malloc(sizeof(dvbpsi_demux_subdec_t));  if(p_subdec == NULL)  {    return 1;  }  p_stt_decoder = (dvbpsi_atsc_stt_decoder_t*)malloc(sizeof(dvbpsi_atsc_stt_decoder_t));  if(p_stt_decoder == NULL)  {    free(p_subdec);    return 1;  }  /* subtable decoder configuration */  p_subdec->pf_callback = &dvbpsi_atsc_GatherSTTSections;  p_subdec->p_cb_data = p_stt_decoder;  p_subdec->i_id = (uint32_t)i_table_id << 16;  p_subdec->pf_detach = dvbpsi_atsc_DetachSTT;  /* Attach the subtable decoder to the demux */  p_subdec->p_next = p_demux->p_first_subdec;  p_demux->p_first_subdec = p_subdec;  /* STT decoder information */  p_stt_decoder->pf_callback = pf_callback;  p_stt_decoder->p_cb_data = p_cb_data;  return 0;}/***************************************************************************** * dvbpsi_atsc_DetachSTT ***************************************************************************** * Close a STT decoder. *****************************************************************************/void dvbpsi_atsc_DetachSTT(dvbpsi_demux_t * p_demux, uint8_t i_table_id, uint16_t i_extension){  dvbpsi_demux_subdec_t* p_subdec;  dvbpsi_demux_subdec_t** pp_prev_subdec;  dvbpsi_atsc_stt_decoder_t* p_stt_decoder;  p_subdec = dvbpsi_demuxGetSubDec(p_demux, i_table_id, 0);  if(p_demux == NULL)  {    DVBPSI_ERROR_ARG("STT Decoder",                     "No such STT decoder (table_id == 0x%02x,"                     "extension == 0x00)",                     i_table_id);    return;  }  p_stt_decoder = (dvbpsi_atsc_stt_decoder_t*)p_subdec->p_cb_data;  free(p_subdec->p_cb_data);  pp_prev_subdec = &p_demux->p_first_subdec;  while(*pp_prev_subdec != p_subdec)    pp_prev_subdec = &(*pp_prev_subdec)->p_next;  *pp_prev_subdec = p_subdec->p_next;  free(p_subdec);}/***************************************************************************** * dvbpsi_atsc_InitSTT ***************************************************************************** * Initialize a pre-allocated dvbpsi_atsc_stt_t structure. *****************************************************************************/void dvbpsi_atsc_InitSTT(dvbpsi_atsc_stt_t *p_stt, uint8_t i_protocol){  p_stt->i_protocol = i_protocol;  p_stt->p_first_descriptor = NULL;}/***************************************************************************** * dvbpsi_atsc_EmptySTT ***************************************************************************** * Clean a dvbpsi_atsc_stt_t structure. *****************************************************************************/void dvbpsi_atsc_EmptySTT(dvbpsi_atsc_stt_t* p_stt){  dvbpsi_DeleteDescriptors(p_stt->p_first_descriptor);      p_stt->p_first_descriptor = NULL;}/***************************************************************************** * dvbpsi_atsc_STTAddDescriptor ***************************************************************************** * Add a descriptor to the STT table. *****************************************************************************/dvbpsi_descriptor_t *dvbpsi_atsc_STTAddDescriptor(                                               dvbpsi_atsc_stt_t *p_stt,                                               uint8_t i_tag, uint8_t i_length,                                               uint8_t *p_data){  dvbpsi_descriptor_t * p_descriptor                        = dvbpsi_NewDescriptor(i_tag, i_length, p_data);  if(p_descriptor)  {    if(p_stt->p_first_descriptor == NULL)    {      p_stt->p_first_descriptor = p_descriptor;    }    else    {      dvbpsi_descriptor_t * p_last_descriptor = p_stt->p_first_descriptor;      while(p_last_descriptor->p_next != NULL)        p_last_descriptor = p_last_descriptor->p_next;      p_last_descriptor->p_next = p_descriptor;    }  }  return p_descriptor;}/***************************************************************************** * dvbpsi_atsc_GatherSTTSections ***************************************************************************** * Callback for the subtable demultiplexor. *****************************************************************************/void dvbpsi_atsc_GatherSTTSections(dvbpsi_decoder_t * p_psi_decoder,                              void * p_private_decoder,                              dvbpsi_psi_section_t * p_section){  dvbpsi_atsc_stt_decoder_t * p_stt_decoder                        = (dvbpsi_atsc_stt_decoder_t*)p_private_decoder;  dvbpsi_atsc_stt_t *p_stt;    if(!p_section->b_syntax_indicator)  {    /* Invalid section_syntax_indicator */    DVBPSI_ERROR("STT decoder",                 "invalid section (section_syntax_indicator == 0)");      }  else  {      dvbpsi_atsc_NewSTT(p_stt, p_section->p_payload_start[0]);      if (p_stt)      {          /* Decode the sections */          dvbpsi_atsc_DecodeSTTSections(p_stt,                                  p_section);          /* Delete the sections */          dvbpsi_ReleasePSISections(p_psi_decoder,p_section);          /* signal the new STT */          p_stt_decoder->pf_callback(p_stt_decoder->p_cb_data,                                     p_stt);      }  }}/***************************************************************************** * dvbpsi_DecodeSTTSection ***************************************************************************** * STT decoder. *****************************************************************************/void dvbpsi_atsc_DecodeSTTSections(dvbpsi_atsc_stt_t* p_stt,                              dvbpsi_psi_section_t* p_section){    uint8_t *p_byte, *p_end;    uint16_t i_length = 0;    p_byte = p_section->p_payload_start + 1;     p_stt->i_system_time = (((uint32_t)p_byte[0]) << 24) |                           (((uint32_t)p_byte[1]) << 16) |                           (((uint32_t)p_byte[2]) <<  8) |                           ((uint32_t)p_byte[3]);    p_stt->i_gps_utc_offset = p_byte[4];    p_stt->i_daylight_savings = (((uint16_t)p_byte[5]) << 16) |                                 ((uint16_t)p_byte[6]);    p_byte += 7;    /* Table descriptors */    i_length = (p_section->i_length - 17);    p_end = p_byte + i_length;    while(p_byte + 2 <= p_end)    {        uint8_t i_tag = p_byte[0];        uint8_t i_length = p_byte[1];        if(i_length + 2 <= p_end - p_byte)          dvbpsi_atsc_STTAddDescriptor(p_stt, i_tag, i_length, p_byte + 2);        p_byte += 2 + i_length;    }}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -