249 lines
4.8 KiB
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;
|
|
}
|
|
|