tvheadend/src/muxer/muxer_tvh.c

249 lines
4.8 KiB
C

/*
* tvheadend, wrapper for the builtin dvr muxer
* Copyright (C) 2012 John Törnblom
*
* 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 <htmlui://www.gnu.org/licenses/>.
*/
#include <assert.h>
#include "tvheadend.h"
#include "streaming.h"
#include "epg.h"
#include "channels.h"
#include "muxer_tvh.h"
#include "tvh/mkmux.h"
typedef struct tvh_muxer {
muxer_t;
mk_mux_t *tm_ref;
} tvh_muxer_t;
/**
* Figure out the mimetype
*/
static const char*
tvh_muxer_mime(muxer_t* m, const struct streaming_start *ss)
{
int i;
int has_audio;
int has_video;
const streaming_start_component_t *ssc;
has_audio = 0;
has_video = 0;
for(i=0; i < ss->ss_num_components; i++) {
ssc = &ss->ss_components[i];
if(ssc->ssc_disabled)
continue;
has_video |= SCT_ISVIDEO(ssc->ssc_type);
has_audio |= SCT_ISAUDIO(ssc->ssc_type);
}
if(has_video)
return muxer_container_type2mime(m->m_config.m_type, 1);
else if(has_audio)
return muxer_container_type2mime(m->m_config.m_type, 0);
else
return muxer_container_type2mime(MC_UNKNOWN, 0);
}
/**
* Init the builtin mkv muxer with streams
*/
static int
tvh_muxer_init(muxer_t* m, const struct streaming_start *ss, const char *name)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(mk_mux_init(tm->tm_ref, name, ss)) {
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Insert a new chapter at the current location
*/
static int
tvh_muxer_add_marker(muxer_t* m)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(mk_mux_insert_chapter(tm->tm_ref)) {
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Multisegment matroska files do exist but I am not sure if they are supported
* by many media players. For now, we'll treat it as an error.
*/
static int
tvh_muxer_reconfigure(muxer_t* m, const struct streaming_start *ss)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
tm->m_errors++;
return -1;
}
/**
* Open the muxer as a stream muxer (using a non-seekable socket)
*/
static int
tvh_muxer_open_stream(muxer_t *m, int fd)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(mk_mux_open_stream(tm->tm_ref, fd)) {
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Open a file
*/
static int
tvh_muxer_open_file(muxer_t *m, const char *filename)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(mk_mux_open_file(tm->tm_ref, filename, tm->m_config.m_file_permissions)) {
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Write a packet to the muxer
*/
static int
tvh_muxer_write_pkt(muxer_t *m, streaming_message_type_t smt, void *data)
{
th_pkt_t *pkt = (th_pkt_t*)data;
tvh_muxer_t *tm = (tvh_muxer_t*)m;
int r;
assert(smt == SMT_PACKET);
if((r = mk_mux_write_pkt(tm->tm_ref, pkt)) != 0) {
if (MC_IS_EOS_ERROR(r))
tm->m_eos = 1;
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Append meta data when a channel changes its programme
*/
static int
tvh_muxer_write_meta(muxer_t *m, struct epg_broadcast *eb,
const char *comment)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(mk_mux_write_meta(tm->tm_ref, NULL, eb, comment)) {
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Close the muxer and append trailer to output
*/
static int
tvh_muxer_close(muxer_t *m)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(mk_mux_close(tm->tm_ref)) {
tm->m_errors++;
return -1;
}
return 0;
}
/**
* Free all memory associated with the muxer
*/
static void
tvh_muxer_destroy(muxer_t *m)
{
tvh_muxer_t *tm = (tvh_muxer_t*)m;
if(tm->tm_ref)
mk_mux_destroy(tm->tm_ref);
free(tm);
}
/**
* Create a new builtin muxer
*/
muxer_t*
tvh_muxer_create(const muxer_config_t *m_cfg)
{
tvh_muxer_t *tm;
if(m_cfg->m_type != MC_MATROSKA && m_cfg->m_type != MC_WEBM)
return NULL;
tm = calloc(1, sizeof(tvh_muxer_t));
tm->m_open_stream = tvh_muxer_open_stream;
tm->m_open_file = tvh_muxer_open_file;
tm->m_mime = tvh_muxer_mime;
tm->m_init = tvh_muxer_init;
tm->m_reconfigure = tvh_muxer_reconfigure;
tm->m_add_marker = tvh_muxer_add_marker;
tm->m_write_meta = tvh_muxer_write_meta;
tm->m_write_pkt = tvh_muxer_write_pkt;
tm->m_close = tvh_muxer_close;
tm->m_destroy = tvh_muxer_destroy;
tm->tm_ref = mk_mux_create((muxer_t *)tm, m_cfg->m_type == MC_WEBM);
return (muxer_t*)tm;
}