tvheadend/src/epg.h

550 lines
20 KiB
C

/*
* Electronic Program Guide - Common functions
* Copyright (C) 2012 Adam Sutton
*
* 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 3 of 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 of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef EPG_H
#define EPG_H
#include "settings.h"
#include "lang_str.h"
/*
* External forward decls
*/
struct channel;
struct channel_tag;
struct epggrab_module;
/*
* Map/List types
*/
typedef LIST_HEAD(,epg_object) epg_object_list_t;
typedef RB_HEAD (,epg_object) epg_object_tree_t;
typedef LIST_HEAD(,epg_brand) epg_brand_list_t;
typedef LIST_HEAD(,epg_season) epg_season_list_t;
typedef LIST_HEAD(,epg_episode) epg_episode_list_t;
typedef LIST_HEAD(,epg_broadcast) epg_broadcast_list_t;
typedef RB_HEAD (,epg_broadcast) epg_broadcast_tree_t;
typedef LIST_HEAD(,epg_genre) epg_genre_list_t;
/*
* Typedefs (most are redundant!)
*/
typedef struct epg_genre epg_genre_t;
typedef struct epg_object epg_object_t;
typedef struct epg_brand epg_brand_t;
typedef struct epg_season epg_season_t;
typedef struct epg_episode epg_episode_t;
typedef struct epg_broadcast epg_broadcast_t;
typedef struct epg_serieslink epg_serieslink_t;
/* ************************************************************************
* Genres
* ***********************************************************************/
/* Genre object */
struct epg_genre
{
LIST_ENTRY(epg_genre) link;
uint8_t code;
};
/* Accessors */
uint8_t epg_genre_get_eit ( const epg_genre_t *genre );
size_t epg_genre_get_str ( const epg_genre_t *genre, int major_only,
int major_prefix, char *buf, size_t len );
/* Delete */
void epg_genre_list_destroy ( epg_genre_list_t *list );
/* Add to list */
int epg_genre_list_add ( epg_genre_list_t *list, epg_genre_t *genre );
int epg_genre_list_add_by_eit ( epg_genre_list_t *list, uint8_t eit );
int epg_genre_list_add_by_str ( epg_genre_list_t *list, const char *str );
/* Search */
int epg_genre_list_contains
( epg_genre_list_t *list, epg_genre_t *genre, int partial );
/* List all available genres */
htsmsg_t *epg_genres_list_all ( int major_only, int major_prefix );
/* ************************************************************************
* Generic Object
* ***********************************************************************/
/* Object type */
typedef enum epg_object_type
{
EPG_UNDEF,
EPG_BRAND,
EPG_SEASON,
EPG_EPISODE,
EPG_BROADCAST,
EPG_SERIESLINK
} epg_object_type_t;
/* Object */
struct epg_object
{
RB_ENTRY(epg_object) uri_link; ///< Global URI link
LIST_ENTRY(epg_object) id_link; ///< Global (ID) link
LIST_ENTRY(epg_object) un_link; ///< Global unref'd link
LIST_ENTRY(epg_object) up_link; ///< Global updated link
epg_object_type_t type; ///< Specific object type
uint64_t id; ///< Internal ID
char *uri; ///< Unique ID (from grabber)
int _updated; ///< Flag to indicate updated
int refcount; ///< Reference counting
// Note: could use LIST_ENTRY field to determine this!
struct epggrab_module *grabber; ///< Originating grabber
void (*getref) ( void *o ); ///< Get a reference
void (*putref) ( void *o ); ///< Release a reference
void (*destroy) ( void *o ); ///< Delete the object
void (*updated) ( void *o ); ///< Updated
};
/* ************************************************************************
* Brand - Represents a specific show
* e.g. The Simpsons, 24, Eastenders, etc...
* ***********************************************************************/
/* Object */
struct epg_brand
{
epg_object_t; ///< Base object
lang_str_t *title; ///< Brand name
lang_str_t *summary; ///< Brand summary
uint16_t season_count; ///< Total number of seasons
char *image; ///< Brand image
epg_season_list_t seasons; ///< Season list
epg_episode_list_t episodes; ///< Episode list
};
/* Lookup */
epg_brand_t *epg_brand_find_by_uri
( const char *uri, int create, int *save );
epg_brand_t *epg_brand_find_by_id ( uint64_t id );
/* Accessors */
const char *epg_brand_get_title
( const epg_brand_t *b, const char *lang );
const char *epg_brand_get_summary
( const epg_brand_t *b, const char *lang );
/* Mutators */
int epg_brand_set_title
( epg_brand_t *b, const char *title, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_brand_set_summary
( epg_brand_t *b, const char *summary, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_brand_set_season_count
( epg_brand_t *b, uint16_t season_count, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_brand_set_image
( epg_brand_t *b, const char *i, struct epggrab_module *src )
__attribute__((warn_unused_result));
/* Serialization */
htsmsg_t *epg_brand_serialize ( epg_brand_t *b );
epg_brand_t *epg_brand_deserialize ( htsmsg_t *m, int create, int *save );
/* List all brands (serialized) */
htsmsg_t *epg_brand_list ( void );
/* ************************************************************************
* Season
* ***********************************************************************/
/* Object */
struct epg_season
{
epg_object_t; ///< Parent object
lang_str_t *summary; ///< Season summary
uint16_t number; ///< The season number
uint16_t episode_count; ///< Total number of episodes
char *image; ///< Season image
LIST_ENTRY(epg_season) blink; ///< Brand list link
epg_brand_t *brand; ///< Parent brand
epg_episode_list_t episodes; ///< Episode list
};
/* Lookup */
epg_season_t *epg_season_find_by_uri
( const char *uri, int create, int *save );
epg_season_t *epg_season_find_by_id ( uint64_t id );
/* Accessors */
const char *epg_season_get_summary
( const epg_season_t *s, const char *lang );
/* Mutators */
int epg_season_set_summary
( epg_season_t *s, const char *summary, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_season_set_number
( epg_season_t *s, uint16_t number, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_season_set_episode_count
( epg_season_t *s, uint16_t episode_count, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_season_set_brand
( epg_season_t *s, epg_brand_t *b, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_season_set_image
( epg_season_t *s, const char *image, struct epggrab_module *src )
__attribute__((warn_unused_result));
/* Serialization */
htsmsg_t *epg_season_serialize ( epg_season_t *b );
epg_season_t *epg_season_deserialize ( htsmsg_t *m, int create, int *save );
/* ************************************************************************
* Episode
* ***********************************************************************/
/* Episode numbering object - this is for some back-compat and also
* to allow episode information to be "collated" into easy to use object
*/
typedef struct epg_episode_num
{
uint16_t s_num; ///< Series number
uint16_t s_cnt; ///< Series count
uint16_t e_num; ///< Episode number
uint16_t e_cnt; ///< Episode count
uint16_t p_num; ///< Part number
uint16_t p_cnt; ///< Part count
char *text; ///< Arbitary text description of episode num
} epg_episode_num_t;
/* Object */
struct epg_episode
{
epg_object_t; ///< Parent object
lang_str_t *title; ///< Title
lang_str_t *subtitle; ///< Sub-title
lang_str_t *summary; ///< Summary
lang_str_t *description; ///< An extended description
char *image; ///< Episode image
epg_genre_list_t genre; ///< Episode genre(s)
epg_episode_num_t epnum; ///< Episode numbering
// Note: do not use epnum directly! use the accessor routine
uint8_t is_bw; ///< Is black and white
// TODO: certification and rating
// TODO: film/year
LIST_ENTRY(epg_episode) blink; ///< Brand link
LIST_ENTRY(epg_episode) slink; ///< Season link
epg_brand_t *brand; ///< (Grand-)Parent brand
epg_season_t *season; ///< Parent season
epg_broadcast_list_t broadcasts; ///< Broadcast list
};
/* Lookup */
epg_episode_t *epg_episode_find_by_uri
( const char *uri, int create, int *save );
epg_episode_t *epg_episode_find_by_id ( uint64_t id );
/* Accessors */
const char *epg_episode_get_title
( const epg_episode_t *e, const char *lang );
const char *epg_episode_get_subtitle
( const epg_episode_t *e, const char *lang );
const char *epg_episode_get_summary
( const epg_episode_t *e, const char *lang );
const char *epg_episode_get_description
( const epg_episode_t *e, const char *lang );
/* Mutators */
int epg_episode_set_title
( epg_episode_t *e, const char *title, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_subtitle
( epg_episode_t *e, const char *subtitle, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_summary
( epg_episode_t *e, const char *summary, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_description
( epg_episode_t *e, const char *description, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_number
( epg_episode_t *e, uint16_t number, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_part
( epg_episode_t *e, uint16_t number, uint16_t count,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_epnum
( epg_episode_t *e, epg_episode_num_t *num, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_brand
( epg_episode_t *e, epg_brand_t *b, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_season
( epg_episode_t *e, epg_season_t *s, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_genre
( epg_episode_t *e, epg_genre_list_t *g, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_image
( epg_episode_t *e, const char *i, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_is_bw
( epg_episode_t *e, uint8_t bw, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_title2
( epg_episode_t *e, const lang_str_t *str, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_episode_set_subtitle2
( epg_episode_t *e, const lang_str_t *str, struct epggrab_module *src )
__attribute__((warn_unused_result));
// Note: this does NOT strdup the text field
void epg_episode_get_epnum
( epg_episode_t *e, epg_episode_num_t *epnum );
/* EpNum format helper */
// output string will be:
// if (episode_num)
// ret = pre
// if (season_num) ret += sprintf(sfmt, season_num)
// if (season_cnt && cnt) ret += sprintf(cnt, season_cnt)
// ret += sep
// ret += sprintf(efmt, episode_num)
// if (episode_cnt) ret += sprintf(cfmt, episode_cnt)
// and will return num chars written
size_t epg_episode_number_format
( epg_episode_t *e, char *buf, size_t len,
const char *pre, const char *sfmt,
const char *sep, const char *efmt,
const char *cfmt );
int epg_episode_number_cmp
( epg_episode_num_t *a, epg_episode_num_t *b );
/* Matching */
int epg_episode_fuzzy_match
( epg_episode_t *ee, const char *uri, const char *title,
const char *summary, const char *description );
/* Serialization */
htsmsg_t *epg_episode_serialize ( epg_episode_t *b );
epg_episode_t *epg_episode_deserialize ( htsmsg_t *m, int create, int *save );
/* ************************************************************************
* Series Link - broadcast level linkage
* ***********************************************************************/
/* Object */
struct epg_serieslink
{
epg_object_t;
epg_broadcast_list_t broadcasts; ///< Episode list
};
/* Lookup */
epg_serieslink_t *epg_serieslink_find_by_uri
( const char *uri, int create, int *save );
epg_serieslink_t *epg_serieslink_find_by_id
( uint64_t id );
/* Serialization */
htsmsg_t *epg_serieslink_serialize ( epg_serieslink_t *s );
epg_serieslink_t *epg_serieslink_deserialize
( htsmsg_t *m, int create, int *save );
/* ************************************************************************
* Broadcast - specific airing (channel & time) of an episode
* ***********************************************************************/
/* Object */
struct epg_broadcast
{
epg_object_t; ///< Parent object
uint16_t dvb_eid; ///< DVB Event ID
time_t start; ///< Start time
time_t stop; ///< End time
/* Some quality info */
uint8_t is_widescreen; ///< Is widescreen
uint8_t is_hd; ///< Is HD
uint16_t lines; ///< Lines in image (quality)
uint16_t aspect; ///< Aspect ratio (*100)
/* Some accessibility support */
uint8_t is_deafsigned; ///< In screen signing
uint8_t is_subtitled; ///< Teletext subtitles
uint8_t is_audio_desc; ///< Audio description
/* Misc flags */
uint8_t is_new; ///< New series / file premiere
uint8_t is_repeat; ///< Repeat screening
/* Broadcast level text */
lang_str_t *summary; ///< Summary
lang_str_t *description; ///< Description
RB_ENTRY(epg_broadcast) sched_link; ///< Schedule link
LIST_ENTRY(epg_broadcast) ep_link; ///< Episode link
epg_episode_t *episode; ///< Episode shown
LIST_ENTRY(epg_broadcast) sl_link; ///< SeriesLink link
epg_serieslink_t *serieslink; ///< SeriesLink;
struct channel *channel; ///< Channel being broadcast on
};
/* Lookup */
epg_broadcast_t *epg_broadcast_find_by_time
( struct channel *ch, time_t start, time_t stop,
uint16_t eid, int create, int *save );
epg_broadcast_t *epg_broadcast_find_by_eid ( struct channel *ch, uint16_t eid );
epg_broadcast_t *epg_broadcast_find_by_id ( uint64_t id, struct channel *ch );
/* Mutators */
int epg_broadcast_set_episode
( epg_broadcast_t *b, epg_episode_t *e, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_widescreen
( epg_broadcast_t *b, uint8_t ws, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_hd
( epg_broadcast_t *b, uint8_t hd, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_lines
( epg_broadcast_t *b, uint16_t lines, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_aspect
( epg_broadcast_t *b, uint16_t aspect, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_deafsigned
( epg_broadcast_t *b, uint8_t ds, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_subtitled
( epg_broadcast_t *b, uint8_t st, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_audio_desc
( epg_broadcast_t *b, uint8_t ad, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_new
( epg_broadcast_t *b, uint8_t n, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_is_repeat
( epg_broadcast_t *b, uint8_t r, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_summary
( epg_broadcast_t *b, const char *str, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_description
( epg_broadcast_t *b, const char *str, const char *lang,
struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_summary2
( epg_broadcast_t *b, const lang_str_t *str, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_description2
( epg_broadcast_t *b, const lang_str_t *str, struct epggrab_module *src )
__attribute__((warn_unused_result));
int epg_broadcast_set_serieslink
( epg_broadcast_t *b, epg_serieslink_t *sl, struct epggrab_module *src )
__attribute__((warn_unused_result));
/* Accessors */
epg_broadcast_t *epg_broadcast_get_next ( epg_broadcast_t *b );
epg_episode_t *epg_broadcast_get_episode
( epg_broadcast_t *b, int create, int *save );
const char *epg_broadcast_get_title
( epg_broadcast_t *b, const char *lang );
const char *epg_broadcast_get_summary
( epg_broadcast_t *b, const char *lang );
const char *epg_broadcast_get_description
( epg_broadcast_t *b, const char *lang );
/* Serialization */
htsmsg_t *epg_broadcast_serialize ( epg_broadcast_t *b );
epg_broadcast_t *epg_broadcast_deserialize
( htsmsg_t *m, int create, int *save );
/* ************************************************************************
* Channel - provides mapping from EPG channels to real channels
* ***********************************************************************/
/* Unlink */
void epg_channel_unlink ( struct channel *ch );
/* ************************************************************************
* Querying
* ***********************************************************************/
/*
* Query result
*/
typedef struct epg_query_result {
epg_broadcast_t **eqr_array;
int eqr_entries;
int eqr_alloced;
} epg_query_result_t;
void epg_query_free(epg_query_result_t *eqr);
/* Sorting */
// WIBNI: might be useful to have a user defined comparator function
void epg_query_sort(epg_query_result_t *eqr);
/* Query routines */
void epg_query0(epg_query_result_t *eqr, struct channel *ch,
struct channel_tag *ct, epg_genre_t *genre, const char *title,
const char *lang);
void epg_query(epg_query_result_t *eqr, const char *channel, const char *tag,
epg_genre_t *genre, const char *title, const char *lang);
/* ************************************************************************
* Setup/Shutdown
* ***********************************************************************/
void epg_init (void);
void epg_save (void);
void epg_updated (void);
/* ************************************************************************
* Miscellaneous
* ***********************************************************************/
char *epg_hash ( const char *t, const char *s, const char *d );
#endif /* EPG_H */