434 lines
16 KiB
C
434 lines
16 KiB
C
/*
|
|
* Electronic Program Guide - Common functions
|
|
* Copyright (C) 2007 Andreas Öman
|
|
*
|
|
* 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"
|
|
|
|
/*
|
|
* Forward decls
|
|
*/
|
|
struct channel;
|
|
struct channel_tag;
|
|
|
|
/*
|
|
* Map/List types
|
|
*/
|
|
LIST_HEAD(epg_object_list, epg_object);
|
|
RB_HEAD (epg_object_tree, epg_object);
|
|
LIST_HEAD(epg_brand_list, epg_brand);
|
|
LIST_HEAD(epg_season_list, epg_season);
|
|
LIST_HEAD(epg_episode_list, epg_episode);
|
|
LIST_HEAD(epg_broadcast_list, epg_broadcast);
|
|
RB_HEAD (epg_broadcast_tree, epg_broadcast);
|
|
LIST_HEAD(epg_genre_list, epg_genre);
|
|
|
|
/*
|
|
* 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_season_list epg_season_list_t;
|
|
typedef struct epg_episode_list epg_episode_list_t;
|
|
typedef struct epg_broadcast_list epg_broadcast_list_t;
|
|
typedef struct epg_broadcast_tree epg_broadcast_tree_t;
|
|
typedef struct epg_object_list epg_object_list_t;
|
|
typedef struct epg_object_tree epg_object_tree_t;
|
|
typedef struct epg_genre_list epg_genre_list_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_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!
|
|
|
|
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
|
|
|
|
char *title; ///< Brand name
|
|
char *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 );
|
|
|
|
/* Mutators */
|
|
int epg_brand_set_title ( epg_brand_t *b, const char *title )
|
|
__attribute__((warn_unused_result));
|
|
int epg_brand_set_summary ( epg_brand_t *b, const char *summary )
|
|
__attribute__((warn_unused_result));
|
|
int epg_brand_set_season_count ( epg_brand_t *b, uint16_t season_count )
|
|
__attribute__((warn_unused_result));
|
|
int epg_brand_set_image ( epg_brand_t *b, const char *i )
|
|
__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
|
|
|
|
char *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 );
|
|
|
|
/* Mutators */
|
|
int epg_season_set_summary ( epg_season_t *s, const char *summary )
|
|
__attribute__((warn_unused_result));
|
|
int epg_season_set_number ( epg_season_t *s, uint16_t number )
|
|
__attribute__((warn_unused_result));
|
|
int epg_season_set_episode_count ( epg_season_t *s, uint16_t episode_count )
|
|
__attribute__((warn_unused_result));
|
|
int epg_season_set_brand ( epg_season_t *s, epg_brand_t *b, int u )
|
|
__attribute__((warn_unused_result));
|
|
int epg_season_set_image ( epg_season_t *s, const char *image )
|
|
__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
|
|
|
|
char *title; ///< Title
|
|
char *subtitle; ///< Sub-title
|
|
char *summary; ///< Summary
|
|
char *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 );
|
|
|
|
/* Mutators */
|
|
int epg_episode_set_title ( epg_episode_t *e, const char *title )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_subtitle ( epg_episode_t *e, const char *subtitle )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_summary ( epg_episode_t *e, const char *summary )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_description ( epg_episode_t *e, const char *description )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_number ( epg_episode_t *e, uint16_t number )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_part ( epg_episode_t *e,
|
|
uint16_t number, uint16_t count )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_epnum ( epg_episode_t *e, epg_episode_num_t *num )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_brand ( epg_episode_t *e, epg_brand_t *b )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_season ( epg_episode_t *e, epg_season_t *s )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_genre ( epg_episode_t *e, epg_genre_list_t *g )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_genre_str ( epg_episode_t *e, const char **s )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_image ( epg_episode_t *e, const char *i )
|
|
__attribute__((warn_unused_result));
|
|
int epg_episode_set_is_bw ( epg_episode_t *b, uint8_t bw )
|
|
__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 );
|
|
|
|
/* ************************************************************************
|
|
* Broadcast - specific airing (channel & time) of an episode
|
|
* ***********************************************************************/
|
|
|
|
/* Object */
|
|
struct epg_broadcast
|
|
{
|
|
epg_object_t; ///< Parent object
|
|
|
|
int 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
|
|
|
|
RB_ENTRY(epg_broadcast) sched_link; ///< Schedule link
|
|
LIST_ENTRY(epg_broadcast) ep_link; ///< Episode link
|
|
epg_episode_t *episode; ///< Episode shown
|
|
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, int 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 )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_widescreen ( epg_broadcast_t *b, uint8_t ws )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_hd ( epg_broadcast_t *b, uint8_t hd )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_lines ( epg_broadcast_t *b, uint16_t lines )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_aspect ( epg_broadcast_t *b, uint16_t aspect )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_deafsigned ( epg_broadcast_t *b, uint8_t ds )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_subtitled ( epg_broadcast_t *b, uint8_t st )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_audio_desc ( epg_broadcast_t *b, uint8_t ad )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_new ( epg_broadcast_t *b, uint8_t n )
|
|
__attribute__((warn_unused_result));
|
|
int epg_broadcast_set_is_repeat ( epg_broadcast_t *b, uint8_t r )
|
|
__attribute__((warn_unused_result));
|
|
|
|
/* Accessors */
|
|
epg_broadcast_t *epg_broadcast_get_next ( epg_broadcast_t *b );
|
|
|
|
/* 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);
|
|
void epg_query(epg_query_result_t *eqr, const char *channel, const char *tag,
|
|
epg_genre_t *genre, const char *title);
|
|
|
|
|
|
/* ************************************************************************
|
|
* 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 */
|