tvheadend/src/subscriptions.c

619 lines
14 KiB
C

/*
* tvheadend, transport and subscription 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/>.
*/
#include <assert.h>
#include <pthread.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/time.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include "tvheadend.h"
#include "subscriptions.h"
#include "streaming.h"
#include "channels.h"
#include "service.h"
#include "htsmsg.h"
#include "notify.h"
#include "atomic.h"
#include "dvb/dvb.h"
struct th_subscription_list subscriptions;
static gtimer_t subscription_reschedule_timer;
/**
*
*/
int
subscriptions_active(void)
{
return LIST_FIRST(&subscriptions) != NULL;
}
/**
*
*/
static int
subscription_sort(th_subscription_t *a, th_subscription_t *b)
{
return b->ths_weight - a->ths_weight;
}
/**
* The service is producing output.
*/
static void
subscription_link_service(th_subscription_t *s, service_t *t)
{
streaming_message_t *sm;
s->ths_state = SUBSCRIPTION_TESTING_SERVICE;
s->ths_service = t;
LIST_INSERT_HEAD(&t->s_subscriptions, s, ths_service_link);
pthread_mutex_lock(&t->s_stream_mutex);
if(TAILQ_FIRST(&t->s_components) != NULL)
s->ths_start_message =
streaming_msg_create_data(SMT_START, service_build_stream_start(t));
// Link to service output
streaming_target_connect(&t->s_streaming_pad, &s->ths_input);
if(s->ths_start_message != NULL && t->s_streaming_status & TSS_PACKETS) {
s->ths_state = SUBSCRIPTION_GOT_SERVICE;
// Send a START message to the subscription client
streaming_target_deliver(s->ths_output, s->ths_start_message);
s->ths_start_message = NULL;
// Send status report
sm = streaming_msg_create_code(SMT_SERVICE_STATUS,
t->s_streaming_status);
streaming_target_deliver(s->ths_output, sm);
}
pthread_mutex_unlock(&t->s_stream_mutex);
}
/**
* Called from service code
*/
void
subscription_unlink_service(th_subscription_t *s, int reason)
{
streaming_message_t *sm;
service_t *t = s->ths_service;
pthread_mutex_lock(&t->s_stream_mutex);
// Unlink from service output
streaming_target_disconnect(&t->s_streaming_pad, &s->ths_input);
if(TAILQ_FIRST(&t->s_components) != NULL &&
s->ths_state == SUBSCRIPTION_GOT_SERVICE) {
// Send a STOP message to the subscription client
sm = streaming_msg_create_code(SMT_STOP, reason);
streaming_target_deliver(s->ths_output, sm);
}
pthread_mutex_unlock(&t->s_stream_mutex);
LIST_REMOVE(s, ths_service_link);
s->ths_service = NULL;
}
/**
*
*/
static void
subscription_reschedule_cb(void *aux)
{
subscription_reschedule();
}
/**
*
*/
void
subscription_reschedule(void)
{
th_subscription_t *s;
service_t *t, *skip;
streaming_message_t *sm;
char buf[128];
int error;
lock_assert(&global_lock);
gtimer_arm(&subscription_reschedule_timer,
subscription_reschedule_cb, NULL, 2);
LIST_FOREACH(s, &subscriptions, ths_global_link) {
if(s->ths_channel == NULL)
continue; /* stale entry, channel has been destroyed */
if(s->ths_service != NULL) {
/* Already got a service */
if(s->ths_state != SUBSCRIPTION_BAD_SERVICE)
continue; /* And it seems to work ok, so we're happy */
skip = s->ths_service;
error = s->ths_testing_error;
service_remove_subscriber(s->ths_service, s, s->ths_testing_error);
} else {
error = 0;
skip = NULL;
}
snprintf(buf, sizeof(buf), "Subscription \"%s\"", s->ths_title);
t = service_find(s->ths_channel, s->ths_weight, buf, &error, skip);
if(t == NULL) {
/* No service available */
sm = streaming_msg_create_code(SMT_NOSTART, error);
streaming_target_deliver(s->ths_output, sm);
continue;
}
subscription_link_service(s, t);
}
}
/**
*
*/
void
subscription_unsubscribe(th_subscription_t *s)
{
service_t *t = s->ths_service;
lock_assert(&global_lock);
LIST_REMOVE(s, ths_global_link);
if(s->ths_channel != NULL) {
LIST_REMOVE(s, ths_channel_link);
tvhlog(LOG_INFO, "subscription", "\"%s\" unsubscribing from \"%s\"",
s->ths_title, s->ths_channel->ch_name);
} else {
tvhlog(LOG_INFO, "subscription", "\"%s\" unsubscribing",
s->ths_title);
}
if(t != NULL)
service_remove_subscriber(t, s, SM_CODE_OK);
if(s->ths_tdmi != NULL) {
LIST_REMOVE(s, ths_tdmi_link);
th_dvb_adapter_t *tda = s->ths_tdmi->tdmi_adapter;
pthread_mutex_lock(&tda->tda_delivery_mutex);
streaming_target_disconnect(&tda->tda_streaming_pad, &s->ths_input);
pthread_mutex_unlock(&tda->tda_delivery_mutex);
}
if(s->ths_start_message != NULL)
streaming_msg_free(s->ths_start_message);
free(s->ths_title);
free(s->ths_hostname);
free(s->ths_username);
free(s->ths_client);
free(s);
subscription_reschedule();
notify_reload("subscriptions");
}
/**
* This callback is invoked when we receive data and status updates from
* the currently bound service
*/
static void
subscription_input(void *opauqe, streaming_message_t *sm)
{
th_subscription_t *s = opauqe;
if(s->ths_state == SUBSCRIPTION_TESTING_SERVICE) {
// We are just testing if this service is good
if(sm->sm_type == SMT_START) {
if(s->ths_start_message != NULL)
streaming_msg_free(s->ths_start_message);
s->ths_start_message = sm;
return;
}
if(sm->sm_type == SMT_SERVICE_STATUS &&
sm->sm_code & (TSS_GRACEPERIOD | TSS_ERRORS)) {
// No, mark our subscription as bad_service
// the scheduler will take care of things
s->ths_testing_error = tss2errcode(sm->sm_code);
s->ths_state = SUBSCRIPTION_BAD_SERVICE;
streaming_msg_free(sm);
return;
}
if(sm->sm_type == SMT_SERVICE_STATUS &&
sm->sm_code & TSS_PACKETS) {
if(s->ths_start_message != NULL) {
streaming_target_deliver(s->ths_output, s->ths_start_message);
s->ths_start_message = NULL;
}
s->ths_state = SUBSCRIPTION_GOT_SERVICE;
}
}
if(s->ths_state != SUBSCRIPTION_GOT_SERVICE) {
streaming_msg_free(sm);
return;
}
if(sm->sm_type == SMT_PACKET) {
th_pkt_t *pkt = sm->sm_data;
if(pkt->pkt_err)
s->ths_total_err++;
s->ths_bytes += pkt->pkt_payload->pb_size;
} else if(sm->sm_type == SMT_MPEGTS) {
pktbuf_t *pb = sm->sm_data;
s->ths_bytes += pb->pb_size;
}
streaming_target_deliver(s->ths_output, sm);
}
/**
*
*/
static void
subscription_input_direct(void *opauqe, streaming_message_t *sm)
{
th_subscription_t *s = opauqe;
streaming_target_deliver(s->ths_output, sm);
}
/**
*
*/
th_subscription_t *
subscription_create(int weight, const char *name, streaming_target_t *st,
int flags, st_callback_t *cb, const char *hostname,
const char *username, const char *client)
{
th_subscription_t *s = calloc(1, sizeof(th_subscription_t));
int reject = 0;
static int tally;
if(flags & SUBSCRIPTION_RAW_MPEGTS)
reject |= SMT_TO_MASK(SMT_PACKET); // Reject parsed frames
else
reject |= SMT_TO_MASK(SMT_MPEGTS); // Reject raw mpegts
streaming_target_init(&s->ths_input,
cb ?: subscription_input_direct, s, reject);
s->ths_weight = weight;
s->ths_title = strdup(name);
s->ths_hostname = hostname ? strdup(hostname) : NULL;
s->ths_username = username ? strdup(username) : NULL;
s->ths_client = client ? strdup(client) : NULL;
s->ths_total_err = 0;
s->ths_output = st;
s->ths_flags = flags;
time(&s->ths_start);
s->ths_id = ++tally;
LIST_INSERT_SORTED(&subscriptions, s, ths_global_link, subscription_sort);
return s;
}
/**
*
*/
th_subscription_t *
subscription_create_from_channel(channel_t *ch, unsigned int weight,
const char *name, streaming_target_t *st,
int flags, const char *hostname,
const char *username, const char *client)
{
th_subscription_t *s;
s = subscription_create(weight, name, st, flags, subscription_input,
hostname, username, client);
s->ths_channel = ch;
LIST_INSERT_HEAD(&ch->ch_subscriptions, s, ths_channel_link);
s->ths_service = NULL;
subscription_reschedule();
if(s->ths_service == NULL) {
tvhlog(LOG_NOTICE, "subscription",
"No transponder available for subscription \"%s\" "
"to channel \"%s\"",
s->ths_title, ch->ch_name);
} else {
source_info_t si;
s->ths_service->s_setsourceinfo(s->ths_service, &si);
tvhlog(LOG_INFO, "subscription",
"\"%s\" subscribing on \"%s\", weight: %d, adapter: \"%s\", "
"network: \"%s\", mux: \"%s\", provider: \"%s\", "
"service: \"%s\", quality: %d",
s->ths_title, ch->ch_name, weight,
si.si_adapter ?: "<N/A>",
si.si_network ?: "<N/A>",
si.si_mux ?: "<N/A>",
si.si_provider ?: "<N/A>",
si.si_service ?: "<N/A>",
s->ths_service->s_quality_index(s->ths_service));
service_source_info_free(&si);
}
notify_reload("subscriptions");
return s;
}
/**
*
*/
th_subscription_t *
subscription_create_from_service(service_t *t, const char *name,
streaming_target_t *st, int flags)
{
th_subscription_t *s;
source_info_t si;
int r;
s = subscription_create(INT32_MAX, name, st, flags,
subscription_input_direct,
NULL, NULL, NULL);
if(t->s_status != SERVICE_RUNNING) {
if((r = service_start(t, INT32_MAX, 1)) != 0) {
subscription_unsubscribe(s);
tvhlog(LOG_INFO, "subscription",
"\"%s\" direct subscription failed -- %s", name,
streaming_code2txt(r));
return NULL;
}
}
t->s_setsourceinfo(t, &si);
tvhlog(LOG_INFO, "subscription",
"\"%s\" direct subscription to adapter: \"%s\", "
"network: \"%s\", mux: \"%s\", provider: \"%s\", "
"service: \"%s\", quality: %d",
s->ths_title,
si.si_adapter ?: "<N/A>",
si.si_network ?: "<N/A>",
si.si_mux ?: "<N/A>",
si.si_provider ?: "<N/A>",
si.si_service ?: "<N/A>",
t->s_quality_index(t));
service_source_info_free(&si);
subscription_link_service(s, t);
notify_reload("subscriptions");
return s;
}
/**
*
*/
void
subscription_change_weight(th_subscription_t *s, int weight)
{
if(s->ths_weight == weight)
return;
LIST_REMOVE(s, ths_global_link);
s->ths_weight = weight;
LIST_INSERT_SORTED(&subscriptions, s, ths_global_link, subscription_sort);
subscription_reschedule();
}
/**
*
*/
static void
dummy_callback(void *opauqe, streaming_message_t *sm)
{
switch(sm->sm_type) {
case SMT_START:
fprintf(stderr, "dummysubscription START\n");
break;
case SMT_STOP:
fprintf(stderr, "dummysubscription STOP\n");
break;
case SMT_SERVICE_STATUS:
fprintf(stderr, "dummsubscription: %x\n", sm->sm_code);
break;
default:
break;
}
streaming_msg_free(sm);
}
static gtimer_t dummy_sub_timer;
/**
*
*/
static void
dummy_retry(void *opaque)
{
subscription_dummy_join(opaque, 0);
free(opaque);
}
/**
*
*/
void
subscription_dummy_join(const char *id, int first)
{
service_t *t = service_find_by_identifier(id);
streaming_target_t *st;
if(first) {
gtimer_arm(&dummy_sub_timer, dummy_retry, strdup(id), 2);
return;
}
if(t == NULL) {
tvhlog(LOG_ERR, "subscription",
"Unable to dummy join %s, service not found, retrying...", id);
gtimer_arm(&dummy_sub_timer, dummy_retry, strdup(id), 1);
return;
}
st = calloc(1, sizeof(streaming_target_t));
streaming_target_init(st, dummy_callback, NULL, 0);
subscription_create_from_service(t, "dummy", st, 0);
tvhlog(LOG_NOTICE, "subscription",
"Dummy join %s ok", id);
}
/**
*
*/
htsmsg_t *
subscription_create_msg(th_subscription_t *s)
{
htsmsg_t *m = htsmsg_create_map();
htsmsg_add_u32(m, "id", s->ths_id);
htsmsg_add_u32(m, "start", s->ths_start);
htsmsg_add_u32(m, "errors", s->ths_total_err);
const char *state;
switch(s->ths_state) {
default:
state = "Idle";
break;
case SUBSCRIPTION_TESTING_SERVICE:
state = "Testing";
break;
case SUBSCRIPTION_GOT_SERVICE:
state = "Running";
break;
case SUBSCRIPTION_BAD_SERVICE:
state = "Bad";
break;
}
htsmsg_add_str(m, "state", state);
if(s->ths_hostname != NULL)
htsmsg_add_str(m, "hostname", s->ths_hostname);
if(s->ths_username != NULL)
htsmsg_add_str(m, "username", s->ths_username);
if(s->ths_client != NULL)
htsmsg_add_str(m, "title", s->ths_client);
else if(s->ths_title != NULL)
htsmsg_add_str(m, "title", s->ths_title);
if(s->ths_channel != NULL)
htsmsg_add_str(m, "channel", s->ths_channel->ch_name);
if(s->ths_service != NULL)
htsmsg_add_str(m, "service", s->ths_service->s_nicename);
return m;
}
static gtimer_t every_sec;
/**
*
*/
static void
every_sec_cb(void *aux)
{
th_subscription_t *s;
gtimer_arm(&every_sec, every_sec_cb, NULL, 1);
LIST_FOREACH(s, &subscriptions, ths_global_link) {
int errors = s->ths_total_err;
int bw = atomic_exchange(&s->ths_bytes, 0);
htsmsg_t *m = subscription_create_msg(s);
htsmsg_delete_field(m, "errors");
htsmsg_add_u32(m, "errors", errors);
htsmsg_add_u32(m, "bw", bw);
htsmsg_add_u32(m, "updateEntry", 1);
notify_by_msg("subscriptions", m);
}
}
/**
*
*/
void
subscription_init(void)
{
gtimer_arm(&every_sec, every_sec_cb, NULL, 1);
}