/***************************************************************************
                          eit.c  -  description
                             -------------------
    begin                : Fri Aug 25 2000
    copyright            : (C) 2000 by Robert Schneider
    email                : Robert.Schneider@web.de
 ***************************************************************************/

/***************************************************************************
 *                                                                         *
 *   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: eit.c 1.18 2001/08/11 14:51:28 kls Exp $
 ***************************************************************************/

#include "eit.h"
#include <ctype.h>
#include <fcntl.h>
#include <fstream.h>
#include <iomanip.h>
#include <iostream.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/poll.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <time.h>
#include <unistd.h>
#include "config.h"
#include "videodir.h"

// --- cMJD ------------------------------------------------------------------

class cMJD {
public:
   cMJD();
   cMJD(u_char date_hi, u_char date_lo);
   cMJD(u_char date_hi, u_char date_lo, u_char timehr, u_char timemi, u_char timese);
   ~cMJD();
  /**  */
  void ConvertToTime();
  /**  */
  bool SetSystemTime();
  /**  */
  time_t GetTime_t();
protected: // Protected attributes
  /**  */
  time_t mjdtime;
protected: // Protected attributes
  /**  */
  u_char time_second;
protected: // Protected attributes
  /**  */
  u_char time_minute;
protected: // Protected attributes
  /**  */
  u_char time_hour;
protected: // Protected attributes
  /**  */
  u_short mjd;
};

cMJD::cMJD()
{
}

cMJD::cMJD(u_char date_hi, u_char date_lo)
{
   mjd = date_hi << 8 | date_lo;
   time_hour = time_minute = time_second = 0;
   ConvertToTime();
}

cMJD::cMJD(u_char date_hi, u_char date_lo, u_char timehr, u_char timemi, u_char timese)
{
   mjd = date_hi << 8 | date_lo;
   time_hour = timehr;
   time_minute = timemi;
   time_second = timese;
   ConvertToTime();
}

cMJD::~cMJD()
{
}

/**  */
void cMJD::ConvertToTime()
{
   struct tm t;

   t.tm_sec = time_second;
   t.tm_min = time_minute;
   t.tm_hour = time_hour;
   int k;

   t.tm_year = (int) ((mjd - 15078.2) / 365.25);
   t.tm_mon = (int) ((mjd - 14956.1 - (int)(t.tm_year * 365.25)) / 30.6001);
   t.tm_mday = (int) (mjd - 14956 - (int)(t.tm_year * 365.25) - (int)(t.tm_mon * 30.6001));
   k = (t.tm_mon == 14 || t.tm_mon == 15) ? 1 : 0;
   t.tm_year = t.tm_year + k;
   t.tm_mon = t.tm_mon - 1 - k * 12;
   t.tm_mon--;

   t.tm_isdst = -1;
   t.tm_gmtoff = 0;

   mjdtime = timegm(&t);

   //isyslog(LOG_INFO, "Time parsed = %s\n", ctime(&mjdtime));
}

/**  */
bool cMJD::SetSystemTime()
{
   struct tm *ptm;
   time_t loctim;

   ptm = localtime(&mjdtime);
   loctim = time(NULL);

   if (abs(mjdtime - loctim) > 2)
   {
      isyslog(LOG_INFO, "System Time = %s (%ld)\n", ctime(&loctim), loctim);
      isyslog(LOG_INFO, "Local Time  = %s (%ld)\n", ctime(&mjdtime), mjdtime);
      if (stime(&mjdtime) < 0)
         esyslog(LOG_ERR, "ERROR while setting system time: %m");
      return true;
   }

   return false;
}
/**  */
time_t cMJD::GetTime_t()
{
   return mjdtime;
}

// --- cTDT ------------------------------------------------------------------

typedef struct {
   u_char   table_id                   : 8;

#if BYTE_ORDER == BIG_ENDIAN
   u_char   section_syntax_indicator   : 1;
   u_char                              : 3;
   u_char   section_length_hi          : 4;
#else
   u_char   section_length_hi          : 4;
   u_char                              : 3;
   u_char   section_syntax_indicator   : 1;
#endif

   u_char   section_length_lo          : 8;


   u_char   utc_date_hi                : 8;
   u_char   utc_date_lo                : 8;
   u_char   utc_hour                   : 4;
   u_char   utc_hour_ten               : 4;
   u_char   utc_min                    : 4;
   u_char   utc_min_ten                : 4;
   u_char   utc_sec                    : 4;
   u_char   utc_sec_ten                : 4;
} tdt_t;

class cTDT {
public:
   cTDT(tdt_t *ptdt);
   ~cTDT();
  /**  */
  bool SetSystemTime();
protected: // Protected attributes
  /**  */
  tdt_t tdt;
  /**  */
  cMJD mjd; // kls 2001-03-02: made this a member instead of a pointer (it wasn't deleted in the destructor!)
};

cTDT::cTDT(tdt_t *ptdt)
:tdt(*ptdt)
,mjd(tdt.utc_date_hi, tdt.utc_date_lo, tdt.utc_hour_ten * 10 + tdt.utc_hour,
                                       tdt.utc_min_ten  * 10 + tdt.utc_min,
                                       tdt.utc_sec_ten  * 10 + tdt.utc_sec)
{
}

cTDT::~cTDT()
{
}
/**  */
bool cTDT::SetSystemTime()
{
   return mjd.SetSystemTime();
}

// --- cEventInfo ------------------------------------------------------------

cEventInfo::cEventInfo(unsigned short serviceid, unsigned short eventid)
{
   pTitle = NULL;
   pSubtitle = NULL;
   pExtendedDescription = NULL;
   bIsPresent = bIsFollowing = false;
   lDuration = 0;
   tTime = 0;
   uEventID = eventid;
   uServiceID = serviceid;
   cExtendedDescriptorNumber = 0;
   nChannelNumber = 0;
}

cEventInfo::~cEventInfo()
{
   delete pTitle;
   delete pSubtitle;
   delete pExtendedDescription;
}

/**  */
const char * cEventInfo::GetTitle() const
{
   return pTitle;
}
/**  */
const char * cEventInfo::GetSubtitle() const
{
   return pSubtitle;
}
/**  */
const char * cEventInfo::GetExtendedDescription() const
{
   return pExtendedDescription;
}
/**  */
bool cEventInfo::IsPresent() const
{
   return bIsPresent;
}
/**  */
void cEventInfo::SetPresent(bool pres)
{
   bIsPresent = pres;
}
/**  */
bool cEventInfo::IsFollowing() const
{
   return bIsFollowing;
}
/**  */
void cEventInfo::SetFollowing(bool foll)
{
   bIsFollowing = foll;
}
/**  */
const char * cEventInfo::GetDate() const
{
   static char szDate[25];

   strftime(szDate, sizeof(szDate), "%d.%m.%Y", localtime(&tTime));

   return szDate;
}
/**  */
const char * cEventInfo::GetTimeString() const
{
   static char szTime[25];

   strftime(szTime, sizeof(szTime), "%R", localtime(&tTime));

   return szTime;
}
/**  */
const char * cEventInfo::GetEndTimeString() const
{
   static char szEndTime[25];
   time_t tEndTime = tTime + lDuration;

   strftime(szEndTime, sizeof(szEndTime), "%R", localtime(&tEndTime));

   return szEndTime;
}
/**  */
time_t cEventInfo::GetTime() const
{
   return tTime;
}
/**  */
long cEventInfo::GetDuration() const
{
   return lDuration;
}
/**  */
unsigned short cEventInfo::GetEventID() const
{
   return uEventID;
}
/**  */
bool cEventInfo::SetTitle(char *string)
{
   if (string == NULL)
      return false;

   pTitle = strdup(string);
   if (pTitle == NULL)
      return false;

   return true;
}
/**  */
bool cEventInfo::SetSubtitle(char *string)
{
   if (string == NULL)
      return false;

   pSubtitle = strdup(string);
   if (pSubtitle == NULL)
      return false;

   return true;
}
/**  */
bool cEventInfo::AddExtendedDescription(char *string)
{
   int size = 0;
   bool first = true;
   char *p;

   if (string == NULL)
      return false;

   if (pExtendedDescription)
   {
      first = false;
      size += strlen(pExtendedDescription);
   }

   size += (strlen(string) + 1);

   p = (char *)realloc(pExtendedDescription, size);
   if (p == NULL)
      return false;

   if (first)
      *p = 0;

   strcat(p, string);

   pExtendedDescription = p;

   return true;
}
/**  */
void cEventInfo::SetTime(time_t t)
{
   tTime = t;
}
/**  */
void cEventInfo::SetDuration(long l)
{
   lDuration = l;
}
/**  */
void cEventInfo::SetEventID(unsigned short evid)
{
   uEventID = evid;
}
/**  */
void cEventInfo::SetServiceID(unsigned short servid)
{
   uServiceID = servid;
}
/**  */
u_char cEventInfo::GetExtendedDescriptorNumber() const
{
   return cExtendedDescriptorNumber;
}
/**  */
void cEventInfo::IncreaseExtendedDescriptorNumber()
{
   cExtendedDescriptorNumber++;
}

/**  */
unsigned short cEventInfo::GetServiceID() const
{
   return uServiceID;
}

/**  */
void cEventInfo::Dump(FILE *f, const char *Prefix) const
{
   if (tTime + lDuration >= time(NULL)) {
      fprintf(f, "%sE %u %ld %ld\n", Prefix, uEventID, tTime, lDuration);
      if (!isempty(pTitle))
         fprintf(f, "%sT %s\n", Prefix, pTitle);
      if (!isempty(pSubtitle))
         fprintf(f, "%sS %s\n", Prefix, pSubtitle);
      if (!isempty(pExtendedDescription))
         fprintf(f, "%sD %s\n", Prefix, pExtendedDescription);
      fprintf(f, "%se\n", Prefix);
      }
}

// --- cSchedule -------------------------------------------------------------

cSchedule::cSchedule(unsigned short servid)
{
   pPresent = pFollowing = NULL;
   uServiceID = servid;
}


cSchedule::~cSchedule()
{
}
/**  */
const cEventInfo * cSchedule::GetPresentEvent() const
{
   // checking temporal sanity of present event (kls 2000-11-01)
   time_t now = time(NULL);
   if (pPresent && !(pPresent->GetTime() <= now && now <= pPresent->GetTime() + pPresent->GetDuration()))
   {
      cEventInfo *pe = Events.First();
      while (pe != NULL)
      {
         if (pe->GetTime() <= now && now <= pe->GetTime() + pe->GetDuration())
            return pe;
         pe = Events.Next(pe);
      }
   }
   return pPresent;
}
/**  */
const cEventInfo * cSchedule::GetFollowingEvent() const
{
   // checking temporal sanity of following event (kls 2000-11-01)
   time_t now = time(NULL);
   const cEventInfo *pr = GetPresentEvent(); // must have it verified!
   if (pFollowing && !(pr && pr->GetTime() + pr->GetDuration() <= pFollowing->GetTime()))
   {
      int minDt = INT_MAX;
      cEventInfo *pe = Events.First(), *pf = NULL;
      while (pe != NULL)
      {
         int dt = pe->GetTime() - now;
         if (dt > 0 && dt < minDt)
         {
            minDt = dt;
            pf = pe;
         }
         pe = Events.Next(pe);
      }
      return pf;
   }
   return pFollowing;
}
/**  */
void cSchedule::SetServiceID(unsigned short servid)
{
   uServiceID = servid;
}
/**  */
unsigned short cSchedule::GetServiceID() const
{
   return uServiceID;
}
/**  */
const cEventInfo * cSchedule::GetEvent(unsigned short uEventID) const
{
   cEventInfo *pe = Events.First();
   while (pe != NULL)
   {
      if (pe->GetEventID() == uEventID)
         return pe;

      pe = Events.Next(pe);
   }

   return NULL;
}
/**  */
const cEventInfo * cSchedule::GetEvent(time_t tTime) const
{
   cEventInfo *pe = Events.First();
   while (pe != NULL)
   {
      if (pe->GetTime() == tTime)
         return pe;

      pe = Events.Next(pe);
   }

   return NULL;
}
/**  */
bool cSchedule::SetPresentEvent(cEventInfo *pEvent)
{
   if (pPresent != NULL)
      pPresent->SetPresent(false);
   pPresent = pEvent;
   pPresent->SetPresent(true);

   return true;
}

/**  */
bool cSchedule::SetFollowingEvent(cEventInfo *pEvent)
{
   if (pFollowing != NULL)
      pFollowing->SetFollowing(false);
   pFollowing = pEvent;
   pFollowing->SetFollowing(true);

   return true;
}

/**  */
void cSchedule::Cleanup()
{
   Cleanup(time(NULL));
}

/**  */
void cSchedule::Cleanup(time_t tTime)
{
   cEventInfo *pEvent;
   for (int a = 0; true ; a++)
   {
      pEvent = Events.Get(a);
      if (pEvent == NULL)
         break;
      if (pEvent->GetTime() + pEvent->GetDuration() < tTime)
      {
         Events.Del(pEvent);
         a--;
      }
   }
}

/**  */
void cSchedule::Dump(FILE *f, const char *Prefix) const
{
   cChannel *channel = Channels.GetByServiceID(uServiceID);
   if (channel)
   {
      fprintf(f, "%sC %u %s\n", Prefix, uServiceID, channel->name);
      for (cEventInfo *p = Events.First(); p; p = Events.Next(p))
         p->Dump(f, Prefix);
      fprintf(f, "%sc\n", Prefix);
   }
}

// --- cSchedules ------------------------------------------------------------

cSchedules::cSchedules()
{
   pCurrentSchedule = NULL;
   uCurrentServiceID = 0;
}

cSchedules::~cSchedules()
{
}
/**  */
bool cSchedules::SetCurrentServiceID(unsigned short servid)
{
   pCurrentSchedule = GetSchedule(servid);
   if (pCurrentSchedule == NULL)
   {
      Add(new cSchedule(servid));
      pCurrentSchedule = GetSchedule(servid);
      if (pCurrentSchedule == NULL)
         return false;
   }

   uCurrentServiceID = servid;

   return true;
}
/**  */
const cSchedule * cSchedules::GetSchedule() const
{
   return pCurrentSchedule;
}
/**  */
const cSchedule * cSchedules::GetSchedule(unsigned short servid) const
{
   cSchedule *p;

   p = First();
   while (p != NULL)
   {
      if (p->GetServiceID() == servid)
         return p;
      p = Next(p);
   }

   return NULL;
}

/**  */
void cSchedules::Cleanup()
{
   cSchedule *p;

   p = First();
   while (p != NULL)
   {
      p->Cleanup(time(NULL));
      p = Next(p);
   }
}

/**  */
void cSchedules::Dump(FILE *f, const char *Prefix) const
{
   for (cSchedule *p = First(); p; p = Next(p))
      p->Dump(f, Prefix);
}

// --- cEIT ------------------------------------------------------------------

#define DEC(N) dec << setw(N) << setfill(int('0'))
#define HEX(N) hex << setw(N) << setfill(int('0'))

#define EIT_STUFFING_DESCRIPTOR                 0x42
#define EIT_LINKAGE_DESCRIPTOR                  0x4a
#define EIT_SHORT_EVENT_DESCRIPTOR              0x4d
#define EIT_EXTENDED_EVENT_DESCRIPTOR           0x4e
#define EIT_TIME_SHIFTED_EVENT_DESCRIPTOR       0x4f
#define EIT_COMPONENT_DESCRIPTOR                0x50
#define EIT_CA_IDENTIFIER_DESCRIPTOR            0x53
#define EIT_CONTENT_DESCRIPTOR                  0x54
#define EIT_PARENTAL_RATING_DESCRIPTOR          0x55
#define EIT_TELEPHONE_DESCRIPTOR                0x57
#define EIT_MULTILINGUAL_COMPONENT_DESCRIPTOR   0x5e
#define EIT_PRIVATE_DATE_SPECIFIER_DESCRIPTOR   0x5f
#define EIT_SHORT_SMOOTHING_BUFFER_DESCRIPTOR   0x61
#define EIT_DATA_BROADCAST_DESCRIPTOR           0x64
#define EIT_PDC_DESCRIPTOR                      0x69

typedef struct eit_struct {
   u_char   table_id                   : 8;

#if BYTE_ORDER == BIG_ENDIAN
   u_char   section_syntax_indicator   : 1;
   u_char                              : 3;
   u_char   section_length_hi          : 4;
#else
   u_char   section_length_hi          : 4;
   u_char                              : 3;
   u_char   section_syntax_indicator   : 1;
#endif

   u_char   section_length_lo          : 8;

   u_char   service_id_hi              : 8;
   u_char   service_id_lo              : 8;

#if BYTE_ORDER == BIG_ENDIAN
   u_char                              : 2;
   u_char   version_number             : 5;
   u_char   current_next_indicator     : 1;
#else
   u_char   current_next_indicator     : 1;
   u_char   version_number             : 5;
   u_char                              : 2;
#endif

   u_char   section_number             : 8;
   u_char   last_section_number        : 8;
   u_char   transport_stream_id_hi     : 8;
   u_char   transport_stream_id_lo     : 8;
   u_char   original_network_id_hi     : 8;
   u_char   original_network_id_lo     : 8;
   u_char   segment_last_section_number   : 8;
   u_char   segment_last_table_id      : 8;
} eit_t;

typedef struct eit_loop_struct {
   u_char   event_id_hi                : 8;
   u_char   event_id_lo                : 8;

   u_char   date_hi                    : 8;
   u_char   date_lo                    : 8;
   u_char   time_hour                  : 4;
   u_char   time_hour_ten              : 4;
   u_char   time_minute                : 4;
   u_char   time_minute_ten            : 4;
   u_char   time_second                : 4;
   u_char   time_second_ten            : 4;

   u_char   dur_hour                   : 4;
   u_char   dur_hour_ten               : 4;
   u_char   dur_minute                 : 4;
   u_char   dur_minute_ten             : 4;
   u_char   dur_second                 : 4;
   u_char   dur_second_ten             : 4;

#if BYTE_ORDER == BIG_ENDIAN
   u_char   running_status             : 3;
   u_char   free_ca_mode               : 1;
   u_char   descriptors_loop_length_hi : 4;
#else
   u_char   descriptors_loop_length_hi : 4;
   u_char   free_ca_mode               : 1;
   u_char   running_status             : 3;
#endif

   u_char   descriptors_loop_length_lo : 8;
} eit_loop_t;

typedef struct eit_short_event_struct {
   u_char   descriptor_tag             : 8;
   u_char   descriptor_length          : 8;

   u_char   language_code_1            : 8;
   u_char   language_code_2            : 8;
   u_char   language_code_3            : 8;

   u_char   event_name_length          : 8;
} eit_short_event_t;

typedef struct eit_extended_event_struct {
   u_char   descriptor_tag             : 8;
   u_char   descriptor_length          : 8;

   u_char   last_descriptor_number     : 4;
   u_char   descriptor_number          : 4;

   u_char   language_code_1            : 8;
   u_char   language_code_2            : 8;
   u_char   language_code_3            : 8;

   u_char   length_of_items            : 8;
} eit_extended_event_t;

typedef struct eit_content_descriptor {
   u_char   descriptor_tag             : 8;
   u_char   descriptor_length          : 8;
} eit_content_descriptor_t;

typedef struct eit_content_loop {
   u_char   content_nibble_level_2     : 4;
   u_char   content_nibble_level_1     : 4;
   u_char   user_nibble_2              : 4;
   u_char   user_nibble_1              : 4;
} eit_content_loop_t;

class cEIT {
private:
   cSchedules *schedules;
public:
   cEIT(void *buf, int length, cSchedules *Schedules);
   ~cEIT();
  /**  */
  int ProcessEIT();

protected: // Protected methods
  /**  */
  int strdvbcpy(unsigned char *dst, unsigned char *src, int max);
  /** returns true if this EIT covers a
present/following information, false if it's
schedule information */
  bool IsPresentFollowing();
  /**  */
  bool WriteShortEventDescriptor(unsigned short service, eit_loop_t *eitloop, u_char *buf);
  /**  */
  bool WriteExtEventDescriptor(unsigned short service, eit_loop_t *eitloop, u_char *buf);
protected: // Protected attributes
  int buflen;
protected: // Protected attributes
  /**  */
  u_char buffer[4097];
  /** Table ID of this EIT struct */
  u_char tid;
  /** EITs service id (program number) */
  u_short pid;
};

cEIT::cEIT(void * buf, int length, cSchedules *Schedules)
{
   buflen = length < int(sizeof(buffer)) ? length : sizeof(buffer);
   memset(buffer, 0, sizeof(buffer));
   memcpy(buffer, buf, buflen);
   tid = buffer[0];
   schedules = Schedules;
}

cEIT::~cEIT()
{
}

/**  */
int cEIT::ProcessEIT()
{
   int bufact = 0;
   eit_t *eit;
   eit_loop_t *eitloop;
   u_char tmp[256];

   if (bufact + (int)sizeof(eit_t) > buflen)
      return 0;
   eit = (eit_t *)buffer;
   bufact += sizeof(eit_t);

   unsigned int service = (eit->service_id_hi << 8) | eit->service_id_lo;

   while(bufact + (int)sizeof(eit_loop_t) <= buflen)
   {
      eitloop = (eit_loop_t *)&buffer[bufact];
      bufact += sizeof(eit_loop_t);

      int descdatalen = (eitloop->descriptors_loop_length_hi << 8) + eitloop->descriptors_loop_length_lo;
      int descdataact = 0;

      while (descdataact < descdatalen && bufact < buflen)
      {
         switch (buffer[bufact])
         {
            eit_content_descriptor_t *cont;
            eit_content_loop_t *contloop;

            case EIT_STUFFING_DESCRIPTOR  :
               //dsyslog(LOG_INFO, "Found EIT_STUFFING_DESCRIPTOR");
               break;

            case EIT_LINKAGE_DESCRIPTOR   :
               //dsyslog(LOG_INFO, "Found EIT_LINKAGE_DESCRIPTOR");
               break;

            case EIT_SHORT_EVENT_DESCRIPTOR:
               WriteShortEventDescriptor(service, eitloop, &buffer[bufact]);
               break;

            case EIT_EXTENDED_EVENT_DESCRIPTOR:
               WriteExtEventDescriptor(service, eitloop, &buffer[bufact]);
               break;

            case EIT_TIME_SHIFTED_EVENT_DESCRIPTOR :
               //dsyslog(LOG_INFO, "Found EIT_TIME_SHIFTED_EVENT_DESCRIPTOR");
               break;

            case EIT_COMPONENT_DESCRIPTOR :
               if (buffer[bufact + 1] > 6) // kls 2001-02-24: otherwise strncpy() causes a segfault in strdvbcpy()
                  strdvbcpy(tmp, &buffer[bufact + 8], buffer[bufact + 1] - 6);
               //dsyslog(LOG_INFO, "Found EIT_COMPONENT_DESCRIPTOR %c%c%c 0x%02x/0x%02x/0x%02x '%s'\n", buffer[bufact + 5], buffer[bufact + 6], buffer[bufact + 7], buffer[2], buffer[3], buffer[4], tmp);
               break;

            case EIT_CA_IDENTIFIER_DESCRIPTOR   :
               //dsyslog(LOG_INFO, "Found EIT_CA_IDENTIFIER_DESCRIPTOR");
               break;

            case EIT_CONTENT_DESCRIPTOR   :
               cont = (eit_content_descriptor_t *)buffer;
               contloop = (eit_content_loop_t *)&buffer[sizeof(eit_content_descriptor_t)];
               //dsyslog(LOG_INFO, "Found EIT_CONTENT_DESCRIPTOR 0x%02x/0x%02x\n", contloop->content_nibble_level_1, contloop->content_nibble_level_2);
               break;

            case EIT_PARENTAL_RATING_DESCRIPTOR :
               //dsyslog(LOG_INFO, "Found EIT_PARENTAL_RATING_DESCRIPTOR");
               break;

            case EIT_TELEPHONE_DESCRIPTOR :
               //dsyslog(LOG_INFO, "Found EIT_TELEPHONE_DESCRIPTOR");
               break;

            case EIT_MULTILINGUAL_COMPONENT_DESCRIPTOR   :
               //dsyslog(LOG_INFO, "Found EIT_MULTILINGUAL_COMPONENT_DESCRIPTOR");
               break;

            case EIT_PRIVATE_DATE_SPECIFIER_DESCRIPTOR   :
               //dsyslog(LOG_INFO, "Found EIT_PRIVATE_DATE_SPECIFIER_DESCRIPTOR");
               break;

            case EIT_SHORT_SMOOTHING_BUFFER_DESCRIPTOR   :
               //dsyslog(LOG_INFO, "Found EIT_SHORT_SMOOTHING_BUFFER_DESCRIPTOR");
               break;

            case EIT_DATA_BROADCAST_DESCRIPTOR  :
               //dsyslog(LOG_INFO, "Found EIT_DATA_BROADCAST_DESCRIPTOR");
               break;

            case EIT_PDC_DESCRIPTOR :
               //dsyslog(LOG_INFO, "Found EIT_PDC_DESCRIPTOR");
               break;

            default:
               //dsyslog(LOG_INFO, "Found unhandled descriptor 0x%02x with length of %04d\n", (int)buffer[bufact], (int)buffer[bufact + 1]);
               break;
         }
         descdataact += (buffer[bufact + 1] + 2);
         bufact += (buffer[bufact + 1] + 2);
      }
   }

   return 0;
}

/**  */
int cEIT::strdvbcpy(unsigned char *dst, unsigned char *src, int max)
{
   int a = 0;

   // kls 2001-02-24: if we come in with negative values, the caller must
   // have done something wrong and the strncpy() below will cause a segfault
   if (max <= 0)
   {
      *dst = 0;
      return 0;
   }

   if (*src == 0x05 || (*src >= 0x20 && *src <= 0xff))
   {
      for (a = 0; a < max; a++)
      {
         if (*src == 0)
            break;

         if ((*src >= ' ' && *src <= '~') || (*src >= 0xa0 && *src <= 0xff))
            *dst++ = *src++;
         else
         {
            // if ((*src > '~' && *src < 0xa0) || *src == 0xff)
            // cerr << "found special character 0x" << HEX(2) << (int)*src << endl;
            src++;
         }
      }
      *dst = 0;
   }
   else
   {
      const char *ret;

      switch (*src)
      {
         case 0x01: ret = "Coding according to character table 1"; break;
         case 0x02: ret = "Coding according to character table 2"; break;
         case 0x03: ret = "Coding according to character table 3"; break;
         case 0x04: ret = "Coding according to character table 4"; break;
         case 0x10: ret = "Coding according to ISO/IEC 8859"; break;
         case 0x11: ret = "Coding according to ISO/IEC 10646"; break;
         case 0x12: ret = "Coding according to KSC 5601"; break;
         default: ret = "Unknown coding"; break;
      }
      strncpy((char *)dst, ret, max);
   }
   return a;
}

/** returns true if this EIT covers a
present/following information, false if it's
schedule information */
bool cEIT::IsPresentFollowing()
{
   if (tid == 0x4e || tid == 0x4f)
      return true;

   return false;
}

/**  */
bool cEIT::WriteShortEventDescriptor(unsigned short service, eit_loop_t *eitloop, u_char *buf)
{
   u_char tmp[256];
   eit_short_event_t *evt = (eit_short_event_t *)buf;
   unsigned short eventid = (unsigned short)((eitloop->event_id_hi << 8) | eitloop->event_id_lo);
   cEventInfo *pEvent;

   //isyslog(LOG_INFO, "Found Short Event Descriptor");

   cSchedule *pSchedule = (cSchedule *)schedules->GetSchedule(service);
   if (pSchedule == NULL)
   {
      schedules->Add(new cSchedule(service));
      pSchedule = (cSchedule *)schedules->GetSchedule(service);
      if (pSchedule == NULL)
         return false;
   }

   /* cSchedule::GetPresentEvent() and cSchedule::GetFollowingEvent() verify
      the temporal sanity of these events, so calling them here appears to
      be a bad idea... (kls 2000-11-01)
   //
   // if we are working on a present/following info, let's see whether
   // we already have present/following info for this service and if yes
   // check whether it's the same eventid, if yes, just return, nothing
   // left to do.
   //
   if (IsPresentFollowing())
   {
      if (eitloop->running_status == 4 || eitloop->running_status == 3)
         pEvent = (cEventInfo *)pSchedule->GetPresentEvent();
      else
         pEvent = (cEventInfo *)pSchedule->GetFollowingEvent();

      if (pEvent != NULL)
         if (pEvent->GetEventID() == eventid)
            return true;
   }
   */

   //
   // let's see whether we have that eventid already
   // in case not, we have to create a new cEventInfo for it
   //
   pEvent = (cEventInfo *)pSchedule->GetEvent(eventid);
   if (pEvent == NULL)
   {
      pSchedule->Events.Add(new cEventInfo(service, eventid));
      pEvent = (cEventInfo *)pSchedule->GetEvent(eventid);
      if (pEvent == NULL)
         return false;

      strdvbcpy(tmp, &buf[sizeof(eit_short_event_t)], evt->event_name_length);
      pEvent->SetTitle((char *)tmp);
      strdvbcpy(tmp, &buf[sizeof(eit_short_event_t) + evt->event_name_length + 1],
                (int)buf[sizeof(eit_short_event_t) + evt->event_name_length]);
      pEvent->SetSubtitle((char *)tmp);
      cMJD mjd(eitloop->date_hi, eitloop->date_lo,
               eitloop->time_hour_ten * 10 + eitloop->time_hour,
               eitloop->time_minute_ten * 10 + eitloop->time_minute,
               eitloop->time_second_ten * 10 + eitloop->time_second);
      pEvent->SetTime(mjd.GetTime_t());
      pEvent->SetDuration((long)((long)((eitloop->dur_hour_ten * 10 + eitloop->dur_hour) * 60l * 60l) +
                                 (long)((eitloop->dur_minute_ten * 10 + eitloop->dur_minute) * 60l) +
                                 (long)(eitloop->dur_second_ten * 10 + eitloop->dur_second)));
   }

   if (IsPresentFollowing())
   {
      if (eitloop->running_status == 4 || eitloop->running_status == 3)
         pSchedule->SetPresentEvent(pEvent);
      else if (eitloop->running_status == 1 || eitloop->running_status == 2 || eitloop->running_status == 0)
         pSchedule->SetFollowingEvent(pEvent);
   }

   return true;
}

/**  */
bool cEIT::WriteExtEventDescriptor(unsigned short service, eit_loop_t *eitloop, u_char *buf)
{
   u_char tmp[256];
   eit_extended_event_t *evt = (eit_extended_event_t *)buf;
   int bufact, buflen;
   unsigned short eventid = (unsigned short)((eitloop->event_id_hi << 8) | eitloop->event_id_lo);
   cEventInfo *pEvent;

   //isyslog(LOG_INFO, "Found Extended Event Descriptor");

   cSchedule *pSchedule = (cSchedule *)schedules->GetSchedule(service);
   if (pSchedule == NULL)
   {
      schedules->Add(new cSchedule(service));
      pSchedule = (cSchedule *)schedules->GetSchedule(service);
      if (pSchedule == NULL)
         return false;
   }

   pEvent = (cEventInfo *)pSchedule->GetEvent(eventid);
   if (pEvent == NULL)
      return false;

   if (evt->descriptor_number != pEvent->GetExtendedDescriptorNumber())
      return false;

   bufact = sizeof(eit_extended_event_t);
   buflen = buf[1] + 2;

   if (evt->length_of_items > 0)
   {
      while (bufact - sizeof(eit_extended_event_t) < evt->length_of_items)
      {
         strdvbcpy(tmp, &buf[bufact + 1], (int)buf[bufact]);
         // could use value in tmp now to do something,
         // haven't seen any items as of yet transmitted from satellite
         bufact += (buf[bufact] + 1);
      }
   }

   strdvbcpy(tmp, &buf[bufact + 1], (int)buf[bufact]);
   if (pEvent->AddExtendedDescription((char *)tmp))
   {
      pEvent->IncreaseExtendedDescriptorNumber();
      return true;
   }

   return false;
}

// --- cSIProcessor ----------------------------------------------------------

#define MAX_FILTERS 20
#define EPGDATAFILENAME "epg.data"

int cSIProcessor::numSIProcessors = 0;
cSchedules *cSIProcessor::schedules = NULL;
cMutex cSIProcessor::schedulesMutex;
const char *cSIProcessor::epgDataFileName = EPGDATAFILENAME;

/**  */
cSIProcessor::cSIProcessor(const char *FileName)
{
   fileName = strdup(FileName);
   masterSIProcessor = numSIProcessors == 0; // the first one becomes the 'master'
   useTStime = false;
   filters = NULL;
   if (!numSIProcessors++) // the first one creates it
      schedules = new cSchedules;
   filters = (SIP_FILTER *)calloc(MAX_FILTERS, sizeof(SIP_FILTER));
   SetStatus(true);
   Start();
}

cSIProcessor::~cSIProcessor()
{
   active = false;
   Cancel(3);
   ShutDownFilters();
   delete filters;
   if (!--numSIProcessors) // the last one deletes it
      delete schedules;
   delete fileName;
}

void cSIProcessor::SetEpgDataFileName(const char *FileName)
{
  epgDataFileName = NULL;
  if (FileName)
     epgDataFileName = strdup(DirectoryOk(FileName) ? AddDirectory(FileName, EPGDATAFILENAME) : FileName);
}

const char *cSIProcessor::GetEpgDataFileName(void)
{
  if (epgDataFileName)
     return *epgDataFileName == '/' ? epgDataFileName : AddDirectory(VideoDirectory, epgDataFileName);
  return NULL;
}

void cSIProcessor::SetStatus(bool On)
{
   LOCK_THREAD;
   schedulesMutex.Lock();
   ShutDownFilters();
   if (On)
   {
      AddFilter(0x14, 0x70);  // TDT
      AddFilter(0x14, 0x73);  // TOT
      AddFilter(0x12, 0x4e);  // event info, actual TS, present/following
      AddFilter(0x12, 0x4f);  // event info, other TS, present/following
      AddFilter(0x12, 0x50);  // event info, actual TS, schedule
      AddFilter(0x12, 0x60);  // event info, other TS, schedule
   }
   schedulesMutex.Unlock();
}

/** use the vbi device to parse all relevant SI
information and let the classes corresponding
to the tables write their information to the disk */
void cSIProcessor::Action()
{
   dsyslog(LOG_INFO, "EIT processing thread started (pid=%d)%s", getpid(), masterSIProcessor ? " - master" : "");

   time_t lastCleanup = time(NULL);
   time_t lastDump = time(NULL);

   active = true;

   while(active)
   {
      if (masterSIProcessor)
      {
         time_t now = time(NULL);
         struct tm *ptm = localtime(&now);
         if (now - lastCleanup > 3600 && ptm->tm_hour == 5)
         {
            LOCK_THREAD;

            schedulesMutex.Lock();
            isyslog(LOG_INFO, "cleaning up schedules data");
            schedules->Cleanup();
            schedulesMutex.Unlock();
            lastCleanup = now;
         }
         if (epgDataFileName && now - lastDump > 600)
         {
            LOCK_THREAD;

            schedulesMutex.Lock();
            FILE *f = fopen(GetEpgDataFileName(), "w");
            if (f) {
               schedules->Dump(f);
               fclose(f);
               }
            else
               LOG_ERROR;
            lastDump = now;
            schedulesMutex.Unlock();
         }
      }

      // set up pfd structures for all active filter
      pollfd pfd[MAX_FILTERS];
      int NumUsedFilters = 0;
      for (int a = 0; a < MAX_FILTERS ; a++)
      {
         if (filters[a].inuse)
         {
            pfd[NumUsedFilters].fd = filters[a].handle;
            pfd[NumUsedFilters].events = POLLIN;
            NumUsedFilters++;
         }
      }

      // wait until data becomes ready from the bitfilter
      if (poll(pfd, NumUsedFilters, 1000) != 0)
      {
         for (int a = 0; a < NumUsedFilters ; a++)
         {
            if (pfd[a].revents & POLLIN)
            {
               /* read section */
               unsigned char buf[4096+1]; // max. allowed size for any EIT section (+1 for safety ;-)
               if (read(filters[a].handle, buf, 3) == 3)
               {
                  int seclen = ((buf[1] & 0x0F) << 8) | (buf[2] & 0xFF);
                  int pid = filters[a].pid;
                  int n = read(filters[a].handle, buf + 3, seclen);
                  if (n == seclen)
                  {
                     seclen += 3;
                     //dsyslog(LOG_INFO, "Received pid 0x%02x with table ID 0x%02x and length of %04d\n", pid, buf[0], seclen);
                     switch (pid)
                     {
                        case 0x14:
                           if (buf[0] == 0x70)
                           {
                              if (useTStime)
                              {
                                 cTDT ctdt((tdt_t *)buf);
                                 ctdt.SetSystemTime();
                              }
                           }
                              /*XXX this comes pretty often:
                           else
                              dsyslog(LOG_INFO, "Time packet was not 0x70 but 0x%02x\n", (int)buf[0]);
                              XXX*/
                           break;

                        case 0x12:
                           if (buf[0] != 0x72)
                           {
                              LOCK_THREAD;

                              schedulesMutex.Lock();
                              cEIT ceit(buf, seclen, schedules);
                              ceit.ProcessEIT();
                              schedulesMutex.Unlock();
                           }
                           else
                              dsyslog(LOG_INFO, "Received stuffing section in EIT\n");
                           break;

                        default:
                           break;
                     }
                  }
                  else
                     dsyslog(LOG_INFO, "read incomplete section - seclen = %d, n = %d", seclen, n);
               }
            }
         }
      }
   }

   dsyslog(LOG_INFO, "EIT processing thread ended (pid=%d)%s", getpid(), masterSIProcessor ? " - master" : "");
}

/** Add a filter with packet identifier pid and
table identifer tid */
bool cSIProcessor::AddFilter(u_char pid, u_char tid)
{
   dmxSctFilterParams sctFilterParams;
   sctFilterParams.pid = pid;
   memset(&sctFilterParams.filter.filter, 0, DMX_FILTER_SIZE);
   memset(&sctFilterParams.filter.mask, 0, DMX_FILTER_SIZE);
   sctFilterParams.timeout = 0;
   sctFilterParams.flags = DMX_IMMEDIATE_START;
   sctFilterParams.filter.filter[0] = tid;
   sctFilterParams.filter.mask[0] = 0xFF;

   for (int a = 0; a < MAX_FILTERS; a++)
   {
      if (!filters[a].inuse)
      {
         filters[a].pid = pid;
         filters[a].tid = tid;
         if ((filters[a].handle = open(fileName, O_RDWR | O_NONBLOCK)) >= 0)
         {
            if (ioctl(filters[a].handle, DMX_SET_FILTER, &sctFilterParams) >= 0)
               filters[a].inuse = true;
            else
            {
               esyslog(LOG_ERR, "ERROR: can't set filter");
               close(filters[a].handle);
               return false;
            }
            // dsyslog(LOG_INFO, "  Registered filter handle %04x, pid = %02d, tid = %02d", filters[a].handle, filters[a].pid, filters[a].tid);
         }
         else
         {
            esyslog(LOG_ERR, "ERROR: can't open filter handle");
            return false;
         }
         return true;
      }
   }
   esyslog(LOG_ERR, "ERROR: too many filters");

   return false;
}

/** set whether local systems time should be
set by the received TDT or TOT packets */
bool cSIProcessor::SetUseTSTime(bool use)
{
   useTStime = use;
   return useTStime;
}

/**  */
bool cSIProcessor::ShutDownFilters(void)
{
   for (int a = 0; a < MAX_FILTERS; a++)
   {
      if (filters[a].inuse)
      {
         close(filters[a].handle);
         // dsyslog(LOG_INFO, "Deregistered filter handle %04x, pid = %02d, tid = %02d", filters[a].handle, filters[a].pid, filters[a].tid);
         filters[a].inuse = false;
      }
   }

   return true; // there's no real 'boolean' to return here...
}

/** */
bool cSIProcessor::SetCurrentServiceID(unsigned short servid)
{
  LOCK_THREAD;
  return schedules ? schedules->SetCurrentServiceID(servid) : false;
}