vdr/libsi/section.c
Klaus Schmidinger 5a4eb3f104 Version 1.3.5
- Fixed reading the EPG preferred language parameter from 'setup.conf'.
- Fixed switching to a visible programme in case the current channel has neither
  a video nor an audio PID.
- Fixed editing channels (SID now range checked) and creating new channels (NID,
  TID and RID are now set to 0).
- Fixed transponder handling to make it work with satellites that provide two
  transponders on the same frequency, with different polarization, like Hispasat
  at S30.0W (thanks to Thomas Bergwinkl for pointing this out). See man vdr(5)
  for details about the enhanced channel ID format.
- Since there appears to be no general solution for the UPT error yet, a recording
  now initiates an "emergency exit" if the number of UPT errors during one
  recording exceeds 10 (suggested by Gregoire Favre). Since the UPT error doesn't
  happen on my system, this has not been explicitly tested.
  The "preliminary fix" for the UPT error in VDR/dvbdevice.c has been disabled
  by default, since it makes channel switching unpleasently slow. If you want
  to have that workaround back, you can uncomment the line
  //#define WAIT_FOR_LOCK_AFTER_TUNING 1
  in VDR/dvbdevice.c.
- Adapted the 'sky' plugin to use the actual channel IDs, and to fetch EPG data
  from www.bleb.org.
- Limited automatic retuning to devices that actually provide the transponder
  (necessary for the 'sky' plugin).
- Fixed handling receivers in the 'sky' plugin, so that a recording on the same
  channel won't interrupt an ongoing Transfer Mode.
- Added subtable ID and TSDT handling to 'libsi' (thanks to Marcel Wiesweg).
- Fixed some Russian OSD texts (thanks to Vyacheslav Dikonov).
- Added the 'running status' to the EPG events. This is necessary for implementing
  the VPS function for recording.
- Removed the obsolete 'present' and 'following' handling from the EPG data.
- The EPG data is now always kept sorted chronologically in the internal data
  structures. This also means that any EPG data retrieved through the SVRDP
  command LSTE is guaranteed to be sorted by start time.
- Now using the 'running status' in the channel display, so that a programme
  that has an end time that is before the current time, but is still running,
  will still be shown in the display (provided the broadcasters handle the
  'running status' flag correctly). This also applies to programmes that have
  a start time that is in the future, but are already running.
- Implemented an "EPG linger time", which can be set to have older EPG information
  still displayed in the "Schedule" menu (thanks to Jaakko Hyvätti).
- Added PDCDescriptor handling to 'libsi'.
- Implemented handling the VPS timestamps (aka "Programme Identification Label")
  for full VPS support for timers (provided the tv stations actually broadcast
  this information). The VPS time is displayed in the event info page if it exists
  and is different than the event's start time.
- Extended the SVDRP command LSTE to allow limiting the listed data to a given
  channel, the present or following events, or events at a given time (thanks to
  Thomas Heiligenmann).
- Fixed a typo in libsi/si.h (thanks to Stéphane Esté-Gracias).
- Timers can now be set to use the VPS information to control recording a programme.
  The new setup options "Recording/Use VPS" and "Recording/VPS margin", as well as
  the "VPS" option in the individual timers, can be used to control this feature
  (see MANUAL for details).
  Note that this feature will certainly need a lot of testing before it can be
  called "safe"!
- The "Schedule" and "What's on now/next?" menus now have an additional column
  which displays information on whether there is a timer defined for an event,
  whether an event has a VPS time that's different than its start time, and
  whether an event is currently running (see MANUAL under "The "Schedule" Menu"
  for details).
2004-02-29 18:00:00 +01:00

345 lines
9.1 KiB
C

/***************************************************************************
* Copyright (c) 2003 by Marcel Wiesweg *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* $Id: section.c 1.3 2004/02/20 13:44:59 kls Exp $
* *
***************************************************************************/
#include "section.h"
#include <stdio.h>
namespace SI {
/*********************** PAT ***********************/
void PAT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const pat>(s, offset);
associationLoop.setData(data+offset, getLength()-offset-4);
}
int PAT::getTransportStreamId() const {
return HILO(s->transport_stream_id);
}
int PAT::Association::getServiceId() const {
return HILO(s->program_number);
}
int PAT::Association::getPid() const {
return HILO(s->network_pid);
}
void PAT::Association::Parse() {
s=data.getData<pat_prog>();
}
/*********************** CAT ***********************/
void CAT::Parse() {
loop.setData(data+sizeof(cat), getLength()-sizeof(cat)-4);
}
/*********************** PMT ***********************/
void PMT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const pmt>(s, offset);
commonDescriptors.setDataAndOffset(data+offset, HILO(s->program_info_length), offset);
streamLoop.setData(data+offset, getLength()-offset-4);
}
int PMT::getServiceId() const {
return HILO(s->program_number);
}
int PMT::getPCRPid() const {
return HILO(s->PCR_PID);
}
int PMT::Stream::getPid() const {
return HILO(s->elementary_PID);
}
int PMT::Stream::getStreamType() const {
return s->stream_type;
}
void PMT::Stream::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const pmt_info>(s, offset);
streamDescriptors.setData(data+offset, HILO(s->ES_info_length));
}
/*********************** TSDT ***********************/
void TSDT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const tsdt>(s, offset);
transportStreamDescriptors.setDataAndOffset(data+offset, getLength()-offset-4, offset);
}
/*********************** NIT ***********************/
int NIT::getNetworkId() const {
return HILO(s->network_id);
}
void NIT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const nit>(s, offset);
commonDescriptors.setDataAndOffset(data+offset, HILO(s->network_descriptor_length), offset);
const nit_mid *mid;
data.setPointerAndOffset<const nit_mid>(mid, offset);
transportStreamLoop.setData(data+offset, HILO(mid->transport_stream_loop_length));
}
int NIT::TransportStream::getTransportStreamId() const {
return HILO(s->transport_stream_id);
}
int NIT::TransportStream::getOriginalNetworkId() const {
return HILO(s->original_network_id);
}
void NIT::TransportStream::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const ni_ts>(s, offset);
transportStreamDescriptors.setData(data+offset, HILO(s->transport_descriptors_length));
}
/*********************** SDT ***********************/
void SDT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const sdt>(s, offset);
serviceLoop.setData(data+offset, getLength()-offset-4); //4 is for CRC
}
int SDT::getTransportStreamId() const {
return HILO(s->transport_stream_id);
}
int SDT::getOriginalNetworkId() const {
return HILO(s->original_network_id);
}
int SDT::Service::getServiceId() const {
return HILO(s->service_id);
}
int SDT::Service::getEITscheduleFlag() const {
return s->eit_schedule_flag;
}
int SDT::Service::getEITpresentFollowingFlag() const {
return s->eit_present_following_flag;
}
RunningStatus SDT::Service::getRunningStatus() const {
return (RunningStatus)s->running_status;
}
int SDT::Service::getFreeCaMode() const {
return s->free_ca_mode;
}
void SDT::Service::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const sdt_descr>(s, offset);
serviceDescriptors.setData(data+offset, HILO(s->descriptors_loop_length));
}
/*********************** EIT ***********************/
int EIT::getServiceId() const {
return HILO(s->service_id);
}
int EIT::getTransportStreamId() const {
return HILO(s->transport_stream_id);
}
int EIT::getOriginalNetworkId() const {
return HILO(s->original_network_id);
}
int EIT::getSegmentLastSectionNumber() const {
return s->segment_last_section_number;
}
int EIT::getLastTableId() const {
return s->last_table_id;
}
bool EIT::isPresentFollowing() const {
return getTableId() == TableIdEIT_presentFollowing || getTableId() == TableIdEIT_presentFollowing_other;
}
bool EIT::isActualTS() const {
return
(getTableId() ==TableIdEIT_presentFollowing)
|| (TableIdEIT_schedule_first <= getTableId() && getTableId() <= TableIdEIT_schedule_last);
}
void EIT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const eit>(s, offset);
//printf("%d %d %d %d %d\n", getServiceId(), getTransportStreamId(), getOriginalNetworkId(), isPresentFollowing(), isActualTS());
eventLoop.setData(data+offset, getLength()-offset-4); //4 is for CRC
}
time_t EIT::Event::getStartTime() const {
return DVBTime::getTime(s->mjd_hi, s->mjd_lo, s->start_time_h, s->start_time_m, s->start_time_s);
}
time_t EIT::Event::getDuration() const {
return DVBTime::getDuration(s->duration_h, s->duration_m, s->duration_s);
}
int EIT::Event::getEventId() const {
return HILO(s->event_id);
}
int EIT::Event::getMJD() const {
return HILO(s->mjd);
}
int EIT::Event::getStartTimeHour() const {
return DVBTime::bcdToDec(s->start_time_h);
}
int EIT::Event::getStartTimeMinute() const {
return DVBTime::bcdToDec(s->start_time_m);
}
int EIT::Event::getStartTimeSecond() const {
return DVBTime::bcdToDec(s->start_time_s);
}
int EIT::Event::getDurationHour() const {
return DVBTime::bcdToDec(s->duration_h);
}
int EIT::Event::getDurationMinute() const {
return DVBTime::bcdToDec(s->duration_m);
}
int EIT::Event::getDurationSecond() const {
return DVBTime::bcdToDec(s->duration_s);
}
RunningStatus EIT::Event::getRunningStatus() const {
return (RunningStatus)s->running_status;
}
int EIT::Event::getFreeCaMode() const {
return s->free_ca_mode;
}
void EIT::Event::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const eit_event>(s, offset);
//printf("%d %d %d\n", getStartTime(), getDuration(), getRunningStatus());
eventDescriptors.setData(data+offset, HILO(s->descriptors_loop_length));
}
/*********************** TDT ***********************/
time_t TDT::getTime() const {
return DVBTime::getTime(s->utc_mjd_hi, s->utc_mjd_lo, s->utc_time_h, s->utc_time_m, s->utc_time_s);
}
void TDT::Parse() {
s=data.getData<const tdt>();
}
/*********************** TOT ***********************/
time_t TOT::getTime() const {
return DVBTime::getTime(s->utc_mjd_hi, s->utc_mjd_lo, s->utc_time_h, s->utc_time_m, s->utc_time_s);
}
void TOT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const tot>(s, offset);
descriptorLoop.setData(data+offset, getLength()-offset-4);
}
/*********************** RST ***********************/
void RST::Parse() {
unsigned int offset=0;
const rst *s;
data.setPointerAndOffset<const rst>(s, offset);
infoLoop.setData(data+offset, getLength()-offset);
}
int RST::RunningInfo::getTransportStreamId() const {
return HILO(s->transport_stream_id);
}
int RST::RunningInfo::getOriginalNetworkId() const {
return HILO(s->original_network_id);
}
int RST::RunningInfo::getServiceId() const {
return HILO(s->service_id);
}
int RST::RunningInfo::getEventId() const {
return HILO(s->event_id);
}
RunningStatus RST::RunningInfo::getRunningStatus() const {
return (RunningStatus)s->running_status;
}
void RST::RunningInfo::Parse() {
s=data.getData<const rst_info>();
}
/*********************** AIT ***********************/
int AIT::getApplicationType() const {
return HILO(first->application_type);
}
int AIT::getAITVersion() const {
return first->version_number;
}
void AIT::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const ait>(first, offset);
commonDescriptors.setDataAndOffset(data+offset, HILO(first->common_descriptors_length), offset);
const ait_mid *mid;
data.setPointerAndOffset<const ait_mid>(mid, offset);
applicationLoop.setData(data+offset, HILO(mid->application_loop_length));
}
long AIT::Application::getOrganisationId() const {
return data.FourBytes(0);
}
int AIT::Application::getApplicationId() const {
return HILO(s->application_id);
}
int AIT::Application::getControlCode() const {
return s->application_control_code;
}
void AIT::Application::Parse() {
unsigned int offset=0;
data.setPointerAndOffset<const ait_app>(s, offset);
applicationDescriptors.setData(data+offset, HILO(s->application_descriptors_length));
}
} //end of namespace