📄 sdt.c
字号:
else { /* Perform a few sanity checks */ if(p_sdt_decoder->p_building_sdt) { if(p_sdt_decoder->p_building_sdt->i_ts_id != p_section->i_extension) { /* transport_stream_id */ DVBPSI_ERROR("SDT decoder", "'transport_stream_id' differs" " whereas no TS discontinuity has occured"); b_reinit = 1; } else if(p_sdt_decoder->p_building_sdt->i_version != p_section->i_version) { /* version_number */ DVBPSI_ERROR("SDT decoder", "'version_number' differs" " whereas no discontinuity has occured"); b_reinit = 1; } else if(p_sdt_decoder->i_last_section_number != p_section->i_last_number) { /* last_section_number */ DVBPSI_ERROR("SDT decoder", "'last_section_number' differs" " whereas no discontinuity has occured"); b_reinit = 1; } } else { if( (p_sdt_decoder->b_current_valid) && (p_sdt_decoder->current_sdt.i_version == p_section->i_version)) { /* Signal a new SDT if the previous one wasn't active */ if( (!p_sdt_decoder->current_sdt.b_current_next) && (p_section->b_current_next)) { dvbpsi_sdt_t * p_sdt = (dvbpsi_sdt_t*)ObjectCreateType(dvbpsi_sdt_t); p_sdt_decoder->current_sdt.b_current_next = 1; *p_sdt = p_sdt_decoder->current_sdt; p_sdt_decoder->pf_callback(p_sdt_decoder->p_cb_data, p_sdt); } /* Don't decode since this version is already decoded */ b_append = 0; } } } } /* Reinit the decoder if wanted */ if(b_reinit) { /* Force redecoding */ p_sdt_decoder->b_current_valid = 0; /* Free structures */ if(p_sdt_decoder->p_building_sdt) { ObjectRefDec(p_sdt_decoder->p_building_sdt); p_sdt_decoder->p_building_sdt = NULL; } /* Clear the section array */ for(i = 0; i <= 255; i++) { if(p_sdt_decoder->ap_sections[i] != NULL) { dvbpsi_ReleasePSISections(p_psi_decoder, p_sdt_decoder->ap_sections[i]); p_sdt_decoder->ap_sections[i] = NULL; } } } /* Append the section to the list if wanted */ if(b_append) { int b_complete; /* Initialize the structures if it's the first section received */ if(!p_sdt_decoder->p_building_sdt) { dvbpsi_NewSDT(p_sdt_decoder->p_building_sdt, p_section->i_extension, p_section->i_version, p_section->b_current_next, ((uint16_t)(p_section->p_payload_start[0]) << 8) | p_section->p_payload_start[1]); p_sdt_decoder->i_last_section_number = p_section->i_last_number; } /* Fill the section array */ if(p_sdt_decoder->ap_sections[p_section->i_number] != NULL) { DVBPSI_DEBUG_ARG("SDT decoder", "overwrite section number %d", p_section->i_number); dvbpsi_ReleasePSISections(p_psi_decoder, p_sdt_decoder->ap_sections[p_section->i_number]); } p_sdt_decoder->ap_sections[p_section->i_number] = p_section; /* Check if we have all the sections */ b_complete = 0; for(i = 0; i <= p_sdt_decoder->i_last_section_number; i++) { if(!p_sdt_decoder->ap_sections[i]) break; if(i == p_sdt_decoder->i_last_section_number) b_complete = 1; } if(b_complete) { /* Save the current information */ p_sdt_decoder->current_sdt = *p_sdt_decoder->p_building_sdt; p_sdt_decoder->b_current_valid = 1; /* Chain the sections */ if(p_sdt_decoder->i_last_section_number) { for(i = 0; i <= p_sdt_decoder->i_last_section_number - 1; i++) p_sdt_decoder->ap_sections[i]->p_next = p_sdt_decoder->ap_sections[i + 1]; } /* Decode the sections */ dvbpsi_DecodeSDTSections(p_sdt_decoder->p_building_sdt, p_sdt_decoder->ap_sections[0]); /* Delete the sections */ dvbpsi_ReleasePSISections(p_psi_decoder,p_sdt_decoder->ap_sections[0]); /* signal the new SDT */ p_sdt_decoder->pf_callback(p_sdt_decoder->p_cb_data, p_sdt_decoder->p_building_sdt); /* Reinitialize the structures */ p_sdt_decoder->p_building_sdt = NULL; for(i = 0; i <= p_sdt_decoder->i_last_section_number; i++) p_sdt_decoder->ap_sections[i] = NULL; } } else { dvbpsi_ReleasePSISections(p_psi_decoder,p_section); }}/***************************************************************************** * dvbpsi_DecodeSDTSection ***************************************************************************** * SDT decoder. *****************************************************************************/void dvbpsi_DecodeSDTSections(dvbpsi_sdt_t* p_sdt, dvbpsi_psi_section_t* p_section){ uint8_t *p_byte, *p_end; while(p_section) { for(p_byte = p_section->p_payload_start + 3; p_byte + 4 < p_section->p_payload_end;) { uint16_t i_service_id = ((uint16_t)(p_byte[0]) << 8) | p_byte[1]; int b_eit_schedule = (int)((p_byte[2] & 0x2) >> 1); int b_eit_present = (int)((p_byte[2]) & 0x1); uint8_t i_running_status = (uint8_t)(p_byte[3]) >> 5; int b_free_ca = (int)((p_byte[3] & 0x10) >> 4); uint16_t i_length = ((uint16_t)(p_byte[3] & 0xf) <<8) | p_byte[4]; dvbpsi_sdt_service_t* p_service = dvbpsi_SDTAddService(p_sdt, i_service_id, b_eit_schedule, b_eit_present, i_running_status, b_free_ca); /* Service descriptors */ p_byte += 5; p_end = p_byte + i_length; if( p_end > p_section->p_payload_end ) break; 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_SDTServiceAddDescriptor(p_service, i_tag, i_length, p_byte + 2); p_byte += 2 + i_length; } } p_section = p_section->p_next; }}/***************************************************************************** * dvbpsi_GenSDTSections ***************************************************************************** * Generate SDT sections based on the dvbpsi_sdt_t structure. *****************************************************************************/dvbpsi_psi_section_t *dvbpsi_GenSDTSections(dvbpsi_sdt_t* p_sdt){ dvbpsi_psi_section_t * p_result = dvbpsi_NewPSISection(1024); dvbpsi_psi_section_t * p_current = p_result; dvbpsi_psi_section_t * p_prev; dvbpsi_sdt_service_t * p_service = p_sdt->p_first_service; p_current->i_table_id = 0x42; p_current->b_syntax_indicator = 1; p_current->b_private_indicator = 1; p_current->i_length = 12; /* header + CRC_32 */ p_current->i_extension = p_sdt->i_ts_id; p_current->i_version = p_sdt->i_version; p_current->b_current_next = p_sdt->b_current_next; p_current->i_number = 0; p_current->p_payload_end += 11; /* just after the header */ p_current->p_payload_start = p_current->p_data + 8; /* Original Network ID */ p_current->p_data[8] = (p_sdt->i_network_id >> 8) ; p_current->p_data[9] = p_sdt->i_network_id; p_current->p_data[10] = 0xff; /* SDT service */ while(p_service != NULL) { uint8_t * p_service_start = p_current->p_payload_end; uint16_t i_service_length = 5; dvbpsi_descriptor_t * p_descriptor = p_service->p_first_descriptor; while ( (p_descriptor != NULL)&& ((p_service_start - p_current->p_data) + i_service_length <= 1020) ) { i_service_length += p_descriptor->i_length + 2; p_descriptor = p_descriptor->p_next; } if ( (p_descriptor != NULL) && (p_service_start - p_current->p_data != 11) && (i_service_length <= 1009) ) { /* will put more descriptors in an empty section */ DVBPSI_DEBUG("SDT generator","create a new section to carry more Service descriptors"); p_prev = p_current; p_current = dvbpsi_NewPSISection(1024); p_prev->p_next = p_current; p_current->i_table_id = 0x42; p_current->b_syntax_indicator = 1; p_current->b_private_indicator = 1; p_current->i_length = 12; /* header + CRC_32 */ p_current->i_extension = p_sdt->i_ts_id;; p_current->i_version = p_sdt->i_version; p_current->b_current_next = p_sdt->b_current_next; p_current->i_number = p_prev->i_number + 1; p_current->p_payload_end += 11; /* just after the header */ p_current->p_payload_start = p_current->p_data + 8; /* Original Network ID */ p_current->p_data[8] = (p_sdt->i_network_id >> 8) ; p_current->p_data[9] = p_sdt->i_network_id; p_current->p_data[10] = 0xff; p_service_start = p_current->p_payload_end; } p_service_start[0] = (p_service->i_service_id >>8); p_service_start[1] = (p_service->i_service_id ); p_service_start[2] = 0xfc | (p_service-> b_eit_schedule ? 0x2 : 0x0) | (p_service->b_eit_present ? 0x01 : 0x00); p_service_start[3] = ((p_service->i_running_status & 0x07) << 5 ) | ((p_service->b_free_ca & 0x1) << 4); /* Increase the length by 5 */ p_current->p_payload_end += 5; p_current->i_length += 5; /* ES descriptors */ p_descriptor = p_service->p_first_descriptor; while ( (p_descriptor != NULL) && ( (p_current->p_payload_end - p_current->p_data) + p_descriptor->i_length <= 1018) ) { /* p_payload_end is where the descriptor begins */ p_current->p_payload_end[0] = p_descriptor->i_tag; p_current->p_payload_end[1] = p_descriptor->i_length; memcpy(p_current->p_payload_end + 2, p_descriptor->p_data, p_descriptor->i_length); /* Increase length by descriptor_length + 2 */ p_current->p_payload_end += p_descriptor->i_length + 2; p_current->i_length += p_descriptor->i_length + 2; p_descriptor = p_descriptor->p_next; } if(p_descriptor != NULL) DVBPSI_ERROR("SDT generator", "unable to carry all the descriptors"); /* ES_info_length */ i_service_length = p_current->p_payload_end - p_service_start - 5; p_service_start[3] |= ((i_service_length >> 8) & 0x0f); p_service_start[4] = i_service_length; p_service = p_service->p_next; } /* Finalization */ p_prev = p_result; while(p_prev != NULL) { p_prev->i_last_number = p_current->i_number; dvbpsi_BuildPSISection(p_prev); p_prev = p_prev->p_next; } return p_result;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -