source.c 46.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10
/* Icecast
 *
 * This program is distributed under the GNU General Public License, version 2.
 * A copy of this license is included with this source.
 *
 * Copyright 2000-2004, Jack Moffitt <jack@xiph.org, 
 *                      Michael Smith <msmith@xiph.org>,
 *                      oddsock <oddsock@xiph.org>,
 *                      Karl Heyes <karl@xiph.org>
 *                      and others (see AUTHORS for details).
Philipp Schafft's avatar
Philipp Schafft committed
11
 * Copyright 2012-2014, Philipp "ph3-der-loewe" Schafft <lion@lion.leolix.org>,
12 13
 */

14
/* -*- c-basic-offset: 4; indent-tabs-mode: nil; -*- */
15 16 17 18
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

Jack Moffitt's avatar
Jack Moffitt committed
19 20 21 22
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
23
#include <ogg/ogg.h>
Michael Smith's avatar
Michael Smith committed
24
#include <errno.h>
25 26

#ifndef _WIN32
27
#include <unistd.h>
Jack Moffitt's avatar
Jack Moffitt committed
28
#include <sys/time.h>
29
#include <sys/socket.h>
30
#include <sys/wait.h>
31
#include <limits.h>
32 33 34
#ifndef PATH_MAX
#define PATH_MAX 4096
#endif
35
#else
36 37
#include <winsock2.h>
#include <windows.h>
38
#define snprintf _snprintf
39
#endif
Jack Moffitt's avatar
Jack Moffitt committed
40

41 42 43 44 45 46
#ifndef _WIN32
/* for __setup_empty_script_environment() */
#include <sys/stat.h>
#include <fcntl.h>
#endif

Karl Heyes's avatar
Karl Heyes committed
47 48 49 50
#include "thread/thread.h"
#include "avl/avl.h"
#include "httpp/httpp.h"
#include "net/sock.h"
Jack Moffitt's avatar
Jack Moffitt committed
51 52 53 54 55 56

#include "connection.h"
#include "global.h"
#include "refbuf.h"
#include "client.h"
#include "stats.h"
57
#include "logging.h"
58
#include "cfgfile.h"
59
#include "util.h"
Jack Moffitt's avatar
Jack Moffitt committed
60
#include "source.h"
Michael Smith's avatar
Michael Smith committed
61
#include "format.h"
62
#include "fserve.h"
Michael Smith's avatar
Michael Smith committed
63
#include "auth.h"
64
#include "compat.h"
Jack Moffitt's avatar
Jack Moffitt committed
65

66 67 68
#undef CATMODULE
#define CATMODULE "source"

Michael Smith's avatar
Michael Smith committed
69 70
#define MAX_FALLBACK_DEPTH 10

71 72
mutex_t move_clients_mutex;

Jack Moffitt's avatar
Jack Moffitt committed
73 74 75
/* avl tree helper */
static int _compare_clients(void *compare_arg, void *a, void *b);
static int _free_client(void *key);
76
static void _parse_audio_info (source_t *source, const char *s);
Karl Heyes's avatar
Karl Heyes committed
77
static void source_shutdown (source_t *source);
78
#ifdef _WIN32
79
#define source_run_script(w,x,y,z)  ICECAST_LOG_WARN("on [dis]connect scripts disabled");
80
#else
81
static void source_run_script (const char *command, source_t *source, mount_proxy *mountinfo, const char *action);
82
#endif
Jack Moffitt's avatar
Jack Moffitt committed
83

84 85 86 87 88 89 90 91
/* Allocate a new source with the stated mountpoint, if one already
 * exists with that mountpoint in the global source tree then return
 * NULL.
 */
source_t *source_reserve (const char *mount)
{
    source_t *src = NULL;

92
    if(mount[0] != '/')
93
        ICECAST_LOG_WARN("Source at \"%s\" does not start with '/', clients will be "
94 95
                "unable to connect", mount);

96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
    do
    {
        avl_tree_wlock (global.source_tree);
        src = source_find_mount_raw (mount);
        if (src)
        {
            src = NULL;
            break;
        }

        src = calloc (1, sizeof(source_t));
        if (src == NULL)
            break;

        src->client_tree = avl_tree_new(_compare_clients, NULL);
        src->pending_tree = avl_tree_new(_compare_clients, NULL);

        /* make duplicates for strings or similar */
114
        src->mount = strdup(mount);
115
        src->max_listeners = -1;
116
        thread_mutex_create(&src->lock);
117

118
        avl_insert(global.source_tree, src);
119 120 121

    } while (0);

122
    avl_tree_unlock(global.source_tree);
123 124 125 126
    return src;
}


Michael Smith's avatar
Michael Smith committed
127 128 129 130
/* Find a mount with this raw name - ignoring fallbacks. You should have the
 * global source tree locked to call this.
 */
source_t *source_find_mount_raw(const char *mount)
Jack Moffitt's avatar
Jack Moffitt committed
131
{
132 133 134 135 136 137 138 139 140
    source_t *source;
    avl_node *node;
    int cmp;

    if (!mount) {
        return NULL;
    }
    /* get the root node */
    node = global.source_tree->root->right;
141

142
    while (node) {
143
        source = (source_t *) node->key;
144 145 146 147 148 149 150 151 152 153 154
        cmp = strcmp(mount, source->mount);
        if (cmp < 0) 
            node = node->left;
        else if (cmp > 0)
            node = node->right;
        else
            return source;
    }
    
    /* didn't find it */
    return NULL;
Jack Moffitt's avatar
Jack Moffitt committed
155 156
}

157 158

/* Search for mount, if the mount is there but not currently running then
Karl Heyes's avatar
Karl Heyes committed
159
 * check the fallback, and so on.  Must have a global source lock to call
160 161
 * this function.
 */
162
source_t *source_find_mount(const char *mount)
Michael Smith's avatar
Michael Smith committed
163
{
164
    source_t *source = NULL;
Michael Smith's avatar
Michael Smith committed
165
    ice_config_t *config;
166 167 168 169
    mount_proxy *mountinfo;
    int depth = 0;

    config = config_get_config();
170
    while (mount && depth < MAX_FALLBACK_DEPTH)
171 172
    {
        source = source_find_mount_raw(mount);
Michael Smith's avatar
Michael Smith committed
173

174 175 176 177 178
        if (source)
        {
            if (source->running || source->on_demand)
                break;
        }
Michael Smith's avatar
Michael Smith committed
179

180 181 182
        /* we either have a source which is not active (relay) or no source
         * at all. Check the mounts list for fallback settings
         */
183
        mountinfo = config_find_mount(config, mount, MOUNT_TYPE_NORMAL);
184
        source = NULL;
185 186 187 188

        if (mountinfo == NULL)
            break;
        mount = mountinfo->fallback_mount;
189
        depth++;
Michael Smith's avatar
Michael Smith committed
190 191
    }

192
    config_release_config();
Michael Smith's avatar
Michael Smith committed
193 194 195 196
    return source;
}


Jack Moffitt's avatar
Jack Moffitt committed
197 198
int source_compare_sources(void *arg, void *a, void *b)
{
199 200
    source_t *srca = (source_t *)a;
    source_t *srcb = (source_t *)b;
Jack Moffitt's avatar
Jack Moffitt committed
201

202
    return strcmp(srca->mount, srcb->mount);
Jack Moffitt's avatar
Jack Moffitt committed
203 204
}

205 206 207

void source_clear_source (source_t *source)
{
208 209
    int c;

210
    ICECAST_LOG_DEBUG("clearing source \"%s\"", source->mount);
211

212
    avl_tree_wlock (source->pending_tree);
213 214
    client_destroy(source->client);
    source->client = NULL;
215 216
    source->parser = NULL;
    source->con = NULL;
217

218 219 220 221
    /* log bytes read in access log */
    if (source->client && source->format)
        source->client->con->sent_bytes = source->format->read_bytes;

222 223
    if (source->dumpfile)
    {
224
        ICECAST_LOG_INFO("Closing dumpfile for %s", source->mount);
225 226 227 228
        fclose (source->dumpfile);
        source->dumpfile = NULL;
    }

229
    /* lets kick off any clients that are left on here */
230
    avl_tree_wlock (source->client_tree);
231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
    c=0;
    while (1)
    {
        avl_node *node = avl_get_first (source->client_tree);
        if (node)
        {
            client_t *client = node->key;
            if (client->respcode == 200)
                c++; /* only count clients that have had some processing */
            avl_delete (source->client_tree, client, _free_client);
            continue;
        }
        break;
    }
    if (c)
246
    {
247
        stats_event_sub (NULL, "listeners", source->listeners);
248
        ICECAST_LOG_INFO("%d active listeners on %s released", c, source->mount);
249
    }
250
    avl_tree_unlock (source->client_tree);
251 252 253 254 255 256 257 258 259 260

    while (avl_get_first (source->pending_tree))
    {
        avl_delete (source->pending_tree,
                avl_get_first(source->pending_tree)->key, _free_client);
    }

    if (source->format && source->format->free_plugin)
        source->format->free_plugin (source->format);
    source->format = NULL;
261 262 263 264 265 266

    /* Lets clear out the source queue too */
    while (source->stream_data)
    {
        refbuf_t *p = source->stream_data;
        source->stream_data = p->next;
267
        p->next = NULL;
268 269 270 271 272 273 274
        /* can be referenced by burst handler as well */
        while (p->_count > 1)
            refbuf_release (p);
        refbuf_release (p);
    }
    source->stream_data_tail = NULL;

Karl Heyes's avatar
Karl Heyes committed
275 276 277 278
    source->burst_point = NULL;
    source->burst_size = 0;
    source->burst_offset = 0;
    source->queue_size = 0;
279
    source->queue_size_limit = 0;
280 281
    source->listeners = 0;
    source->max_listeners = -1;
282
    source->prev_listeners = 0;
283
    source->hidden = 0;
284
    source->shoutcast_compat = 0;
285
    source->client_stats_update = 0;
286
    util_dict_free(source->audio_info);
287
    source->audio_info = NULL;
288 289 290 291 292 293

    free(source->fallback_mount);
    source->fallback_mount = NULL;

    free(source->dumpfilename);
    source->dumpfilename = NULL;
Karl Heyes's avatar
Karl Heyes committed
294 295 296 297 298 299

    if (source->intro_file)
    {
        fclose (source->intro_file);
        source->intro_file = NULL;
    }
300 301

    source->on_demand_req = 0;
302
    avl_tree_unlock (source->pending_tree);
303 304 305
}


306
/* Remove the provided source from the global tree and free it */
307
void source_free_source (source_t *source)
Jack Moffitt's avatar
Jack Moffitt committed
308
{
309
    ICECAST_LOG_DEBUG("freeing source \"%s\"", source->mount);
310 311 312 313
    avl_tree_wlock (global.source_tree);
    avl_delete (global.source_tree, source, NULL);
    avl_tree_unlock (global.source_tree);

314 315
    avl_tree_free(source->pending_tree, _free_client);
    avl_tree_free(source->client_tree, _free_client);
316

317 318 319
    /* make sure all YP entries have gone */
    yp_remove (source->mount);

320 321
    free (source->mount);
    free (source);
Jack Moffitt's avatar
Jack Moffitt committed
322

323
    return;
Jack Moffitt's avatar
Jack Moffitt committed
324
}
325

326

327 328 329
client_t *source_find_client(source_t *source, int id)
{
    client_t fakeclient;
330
    void *result;
331 332 333 334 335 336
    connection_t fakecon;

    fakeclient.con = &fakecon;
    fakeclient.con->id = id;

    avl_tree_rlock(source->client_tree);
337
    if(avl_get_by_key(source->client_tree, &fakeclient, &result) == 0)
338 339 340 341 342 343 344 345
    {
        avl_tree_unlock(source->client_tree);
        return result;
    }

    avl_tree_unlock(source->client_tree);
    return NULL;
}
346

347

348 349 350 351 352
/* Move clients from source to dest provided dest is running
 * and that the stream format is the same.
 * The only lock that should be held when this is called is the
 * source tree lock
 */
353
void source_move_clients(source_t *source, source_t *dest)
354
{
355
    unsigned long count = 0;
356 357
    if (strcmp (source->mount, dest->mount) == 0)
    {
358
        ICECAST_LOG_WARN("src and dst are the same \"%s\", skipping", source->mount);
359 360
        return;
    }
361 362 363 364
    /* we don't want the two write locks to deadlock in here */
    thread_mutex_lock (&move_clients_mutex);

    /* if the destination is not running then we can't move clients */
365

366
    avl_tree_wlock (dest->pending_tree);
367
    if (dest->running == 0 && dest->on_demand == 0)
368
    {
369
        ICECAST_LOG_WARN("destination mount %s not running, unable to move clients ", dest->mount);
370
        avl_tree_unlock (dest->pending_tree);
371
        thread_mutex_unlock (&move_clients_mutex);
372 373 374
        return;
    }

375
    do
376
    {
377
        client_t *client;
378

379 380
        /* we need to move the client and pending trees - we must take the
         * locks in this order to avoid deadlocks */
381 382
        avl_tree_wlock(source->pending_tree);
        avl_tree_wlock(source->client_tree);
383

384
        if (source->on_demand == 0 && source->format == NULL)
385
        {
386
            ICECAST_LOG_INFO("source mount %s is not available", source->mount);
387
            break;
388
        }
389
        if (source->format && dest->format)
390
        {
391 392
            if (source->format->type != dest->format->type)
            {
393
                ICECAST_LOG_WARN("stream %s and %s are of different types, ignored", source->mount, dest->mount);
394 395
                break;
            }
396
        }
397

398 399 400 401 402 403 404
        while (1)
        {
            avl_node *node = avl_get_first (source->pending_tree);
            if (node == NULL)
                break;
            client = (client_t *)(node->key);
            avl_delete (source->pending_tree, client, NULL);
405

406 407 408 409 410 411 412 413
            /* when switching a client to a different queue, be wary of the 
             * refbuf it's referring to, if it's http headers then we need
             * to write them so don't release it.
             */
            if (client->check_buffer != format_check_http_buffer)
            {
                client_set_queue (client, NULL);
                client->check_buffer = format_check_file_buffer;
414 415
                if (source->con == NULL)
                    client->intro_offset = -1;
416 417
            }

418
            avl_insert (dest->pending_tree, (void *)client);
419
            count++;
420 421 422 423 424 425 426 427 428 429 430
        }

        while (1)
        {
            avl_node *node = avl_get_first (source->client_tree);
            if (node == NULL)
                break;

            client = (client_t *)(node->key);
            avl_delete (source->client_tree, client, NULL);

431 432 433 434 435 436 437 438
            /* when switching a client to a different queue, be wary of the 
             * refbuf it's referring to, if it's http headers then we need
             * to write them so don't release it.
             */
            if (client->check_buffer != format_check_http_buffer)
            {
                client_set_queue (client, NULL);
                client->check_buffer = format_check_file_buffer;
439 440
                if (source->con == NULL)
                    client->intro_offset = -1;
441
            }
442
            avl_insert (dest->pending_tree, (void *)client);
443
            count++;
444
        }
445
        ICECAST_LOG_INFO("passing %lu listeners to \"%s\"", count, dest->mount);
446

447 448 449 450 451
        source->listeners = 0;
        stats_event (source->mount, "listeners", "0");

    } while (0);

452 453 454
    avl_tree_unlock (source->pending_tree);
    avl_tree_unlock (source->client_tree);

455 456 457 458
    /* see if we need to wake up an on-demand relay */
    if (dest->running == 0 && dest->on_demand && count)
        dest->on_demand_req = 1;

459 460 461 462
    avl_tree_unlock (dest->pending_tree);
    thread_mutex_unlock (&move_clients_mutex);
}

463

Karl Heyes's avatar
Karl Heyes committed
464 465 466 467 468 469 470 471 472 473 474
/* get some data from the source. The stream data is placed in a refbuf
 * and sent back, however NULL is also valid as in the case of a short
 * timeout and there's no data pending.
 */
static refbuf_t *get_next_buffer (source_t *source)
{
    refbuf_t *refbuf = NULL;
    int delay = 250;

    if (source->short_delay)
        delay = 0;
475
    while (global.running == ICECAST_RUNNING && source->running)
Karl Heyes's avatar
Karl Heyes committed
476
    {
477
        int fds = 0;
Karl Heyes's avatar
Karl Heyes committed
478 479
        time_t current = time (NULL);

480
        if (source->client)
481 482 483 484 485 486
            fds = util_timed_wait_for_fd (source->con->sock, delay);
        else
        {
            thread_sleep (delay*1000);
            source->last_read = current;
        }
Karl Heyes's avatar
Karl Heyes committed
487

488 489 490
        if (current >= source->client_stats_update)
        {
            stats_event_args (source->mount, "total_bytes_read",
491
                    "%"PRIu64, source->format->read_bytes);
492
            stats_event_args (source->mount, "total_bytes_sent",
493
                    "%"PRIu64, source->format->sent_bytes);
494 495
            source->client_stats_update = current + 5;
        }
Karl Heyes's avatar
Karl Heyes committed
496 497 498 499
        if (fds < 0)
        {
            if (! sock_recoverable (sock_error()))
            {
500
                ICECAST_LOG_WARN("Error while waiting on socket, Disconnecting source");
Karl Heyes's avatar
Karl Heyes committed
501 502 503 504 505 506
                source->running = 0;
            }
            break;
        }
        if (fds == 0)
        {
507 508
            thread_mutex_lock(&source->lock);
            if ((source->last_read + (time_t)source->timeout) < current)
Karl Heyes's avatar
Karl Heyes committed
509
            {
510
                ICECAST_LOG_DEBUG("last %ld, timeout %d, now %ld", (long)source->last_read,
511
                        source->timeout, (long)current);
512
                ICECAST_LOG_WARN("Disconnecting source due to socket timeout");
Karl Heyes's avatar
Karl Heyes committed
513 514
                source->running = 0;
            }
515
            thread_mutex_unlock(&source->lock);
Karl Heyes's avatar
Karl Heyes committed
516 517 518 519
            break;
        }
        source->last_read = current;
        refbuf = source->format->get_buffer (source);
520
        if (source->client->con && source->client->con->error)
521
        {
522
            ICECAST_LOG_INFO("End of Stream %s", source->mount);
523 524 525
            source->running = 0;
            continue;
        }
Karl Heyes's avatar
Karl Heyes committed
526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546
        if (refbuf)
            break;
    }

    return refbuf;
}


/* general send routine per listener.  The deletion_expected tells us whether
 * the last in the queue is about to disappear, so if this client is still
 * referring to it after writing then drop the client as it's fallen too far
 * behind 
 */ 
static void send_to_listener (source_t *source, client_t *client, int deletion_expected)
{
    int bytes;
    int loop = 10;   /* max number of iterations in one go */
    int total_written = 0;

    while (1)
    {
547 548 549 550
        /* check for limited listener time */
        if (client->con->discon_time)
            if (time(NULL) >= client->con->discon_time)
            {
551
                ICECAST_LOG_INFO("time limit reached for client #%lu", client->con->id);
552 553 554
                client->con->error = 1;
            }

Karl Heyes's avatar
Karl Heyes committed
555 556 557 558 559 560 561 562
        /* jump out if client connection has died */
        if (client->con->error)
            break;

        /* lets not send too much to one client in one go, but don't
           sleep for too long if more data can be sent */
        if (total_written > 20000 || loop == 0)
        {
563 564
            if (client->check_buffer != format_check_file_buffer)
                source->short_delay = 1;
Karl Heyes's avatar
Karl Heyes committed
565 566 567 568 569
            break;
        }

        loop--;

570
        if (client->check_buffer(source, client) < 0)
Karl Heyes's avatar
Karl Heyes committed
571 572
            break;

573
        bytes = client->write_to_client(client);
Karl Heyes's avatar
Karl Heyes committed
574
        if (bytes <= 0)
575
            break; /* can't write any more */
Karl Heyes's avatar
Karl Heyes committed
576 577 578

        total_written += bytes;
    }
579
    source->format->sent_bytes += total_written;
Karl Heyes's avatar
Karl Heyes committed
580 581 582

    /* the refbuf referenced at head (last in queue) may be marked for deletion
     * if so, check to see if this client is still referring to it */
Karl Heyes's avatar
Karl Heyes committed
583
    if (deletion_expected && client->refbuf && client->refbuf == source->stream_data)
Karl Heyes's avatar
Karl Heyes committed
584
    {
585
        ICECAST_LOG_INFO("Client %lu (%s) has fallen too far behind, removing",
586 587
                client->con->id, client->con->ip);
        stats_event_inc (source->mount, "slow_listeners");
Karl Heyes's avatar
Karl Heyes committed
588 589 590 591
        client->con->error = 1;
    }
}

Jack Moffitt's avatar
Jack Moffitt committed
592

593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615
/* Open the file for stream dumping.
 * This function should do all processing of the filename.
 */
static FILE * source_open_dumpfile(const char * filename) {
#ifndef _WIN32
    /* some of the below functions seems not to be standard winapi functions */
    char buffer[PATH_MAX];
    time_t curtime;
    struct tm *loctime;

    /* Get the current time. */
    curtime = time (NULL);

    /* Convert it to local time representation. */
    loctime = localtime (&curtime);

    strftime (buffer, sizeof(buffer), filename, loctime);
    filename = buffer;
#endif

    return fopen (filename, "ab");
}

Karl Heyes's avatar
Karl Heyes committed
616 617 618
/* Perform any initialisation just before the stream data is processed, the header
 * info is processed by now and the format details are setup
 */
619
static void source_init (source_t *source)
Jack Moffitt's avatar
Jack Moffitt committed
620
{
621
    ice_config_t *config = config_get_config();
622 623
    char *listenurl;
    const char *str;
624
    int listen_url_size;
625
    mount_proxy *mountinfo;
626

627
    /* 6 for max size of port */
628
    listen_url_size = strlen("http://") + strlen(config->hostname) +
629
        strlen(":") + 6 + strlen(source->mount) + 1;
630 631 632

    listenurl = malloc (listen_url_size);
    memset (listenurl, '\000', listen_url_size);
633 634
    snprintf (listenurl, listen_url_size, "http://%s:%d%s",
            config->hostname, config->port, source->mount);
635 636
    config_release_config();

637 638 639 640 641 642 643 644
    str = httpp_getvar(source->parser, "ice-audio-info");
    source->audio_info = util_dict_new();
    if (str)
    {
        _parse_audio_info (source, str);
        stats_event (source->mount, "audio_info", str);
    }

645 646
    stats_event (source->mount, "listenurl", listenurl);

Michael Smith's avatar
Michael Smith committed
647
    free(listenurl);
648

649 650
    if (source->dumpfilename != NULL)
    {
651
        source->dumpfile = source_open_dumpfile (source->dumpfilename);
652 653
        if (source->dumpfile == NULL)
        {
654
            ICECAST_LOG_WARN("Cannot open dump file \"%s\" for appending: %s, disabling.",
655 656 657 658
                    source->dumpfilename, strerror(errno));
        }
    }

659 660 661 662 663 664
    /* grab a read lock, to make sure we get a chance to cleanup */
    thread_rwlock_rlock (source->shutdown_rwlock);

    /* start off the statistics */
    source->listeners = 0;
    stats_event_inc (NULL, "source_total_connections");
665
    stats_event (source->mount, "slow_listeners", "0");
666
    stats_event_args (source->mount, "listeners", "%lu", source->listeners);
667
    stats_event_args (source->mount, "listener_peak", "%lu", source->peak_listeners);
Karl Heyes's avatar
Karl Heyes committed
668
    stats_event_time (source->mount, "stream_start");
669
    stats_event_time_iso8601 (source->mount, "stream_start_iso8601");
670

671
    ICECAST_LOG_DEBUG("Source creation complete");
Karl Heyes's avatar
Karl Heyes committed
672
    source->last_read = time (NULL);
673
    source->prev_listeners = -1;
674
    source->running = 1;
675

676
    mountinfo = config_find_mount (config_get_config(), source->mount, MOUNT_TYPE_NORMAL);
677 678 679
    if (mountinfo)
    {
        if (mountinfo->on_connect)
680
            source_run_script (mountinfo->on_connect, source, mountinfo, "source-connect");
Philipp Schafft's avatar
Philipp Schafft committed
681 682
        /* TODO: replace with <event> */
        /* auth_stream_start (mountinfo, source->mount); */
683
    }
684 685
    config_release_config();

Michael Smith's avatar
Michael Smith committed
686 687
    /*
    ** Now, if we have a fallback source and override is on, we want
688
    ** to steal its clients, because it means we've come back online
Michael Smith's avatar
Michael Smith committed
689 690 691 692
    ** after a failure and they should be gotten back from the waiting
    ** loop or jingle track or whatever the fallback is used for
    */

693 694 695 696
    if (source->fallback_override && source->fallback_mount)
    {
        source_t *fallback_source;

Michael Smith's avatar
Michael Smith committed
697 698 699
        avl_tree_rlock(global.source_tree);
        fallback_source = source_find_mount(source->fallback_mount);

700 701
        if (fallback_source)
            source_move_clients (fallback_source, source);
Michael Smith's avatar
Michael Smith committed
702

703
        avl_tree_unlock(global.source_tree);
Michael Smith's avatar
Michael Smith committed
704
    }
705 706 707 708 709
}


void source_main (source_t *source)
{
Karl Heyes's avatar
Karl Heyes committed
710
    refbuf_t *refbuf;
711 712 713 714
    client_t *client;
    avl_node *client_node;

    source_init (source);
Michael Smith's avatar
Michael Smith committed
715

716
    while (global.running == ICECAST_RUNNING && source->running) {
Karl Heyes's avatar
Karl Heyes committed
717
        int remove_from_q;
Jack Moffitt's avatar
Jack Moffitt committed
718

Karl Heyes's avatar
Karl Heyes committed
719
        refbuf = get_next_buffer (source);
720

Karl Heyes's avatar
Karl Heyes committed
721 722
        remove_from_q = 0;
        source->short_delay = 0;
723

Karl Heyes's avatar
Karl Heyes committed
724 725 726 727
        if (refbuf)
        {
            /* append buffer to the in-flight data queue,  */
            if (source->stream_data == NULL)
Michael Smith's avatar
Michael Smith committed
728
            {
Karl Heyes's avatar
Karl Heyes committed
729 730
                source->stream_data = refbuf;
                source->burst_point = refbuf;
731
            }
Karl Heyes's avatar
Karl Heyes committed
732 733 734 735 736
            if (source->stream_data_tail)
                source->stream_data_tail->next = refbuf;
            source->stream_data_tail = refbuf;
            source->queue_size += refbuf->len;
            /* new buffer is referenced for burst */
737
            refbuf_addref(refbuf);
Karl Heyes's avatar
Karl Heyes committed
738 739 740

            /* new data on queue, so check the burst point */
            source->burst_offset += refbuf->len;
741
            while (source->burst_offset > source->burst_size)
Karl Heyes's avatar
Karl Heyes committed
742
            {
743 744 745
                refbuf_t *to_release = source->burst_point;

                if (to_release->next)
Karl Heyes's avatar
Karl Heyes committed
746
                {
747 748
                    source->burst_point = to_release->next;
                    source->burst_offset -= to_release->len;
749
                    refbuf_release(to_release);
750
                    continue;
751
                }
752
                break;
753 754
            }

Karl Heyes's avatar
Karl Heyes committed
755 756
            /* save stream to file */
            if (source->dumpfile && source->format->write_buf_to_file)
757
                source->format->write_buf_to_file(source, refbuf);
758
        }
Karl Heyes's avatar
Karl Heyes committed
759
        /* lets see if we have too much data in the queue, but don't remove it until later */
760
        thread_mutex_lock(&source->lock);
Karl Heyes's avatar
Karl Heyes committed
761 762
        if (source->queue_size > source->queue_size_limit)
            remove_from_q = 1;
763
        thread_mutex_unlock(&source->lock);
764

765 766 767
        /* acquire write lock on pending_tree */
        avl_tree_wlock(source->pending_tree);

768 769 770 771 772
        /* acquire write lock on client_tree */
        avl_tree_wlock(source->client_tree);

        client_node = avl_get_first(source->client_tree);
        while (client_node) {
773
            client = (client_t *) client_node->key;
Karl Heyes's avatar
Karl Heyes committed
774

775
            send_to_listener(source, client, remove_from_q);
Karl Heyes's avatar
Karl Heyes committed
776

777 778
            if (client->con->error) {
                client_node = avl_get_next(client_node);
779
                if (client->respcode == 200)
780 781
                    stats_event_dec(NULL, "listeners");
                avl_delete(source->client_tree, (void *) client, _free_client);
Michael Smith's avatar
Michael Smith committed
782
                source->listeners--;
783
                ICECAST_LOG_DEBUG("Client removed");
784 785 786 787 788 789 790 791
                continue;
            }
            client_node = avl_get_next(client_node);
        }

        /** add pending clients **/
        client_node = avl_get_first(source->pending_tree);
        while (client_node) {
792

Michael Smith's avatar
Michael Smith committed
793
            if(source->max_listeners != -1 && 
794
                    source->listeners >= (unsigned long)source->max_listeners) 
Michael Smith's avatar
Michael Smith committed
795 796 797 798 799 800 801 802 803 804
            {
                /* The common case is caught in the main connection handler,
                 * this deals with rarer cases (mostly concerning fallbacks)
                 * and doesn't give the listening client any information about
                 * why they were disconnected
                 */
                client = (client_t *)client_node->key;
                client_node = avl_get_next(client_node);
                avl_delete(source->pending_tree, (void *)client, _free_client);

805
                ICECAST_LOG_INFO("Client deleted, exceeding maximum listeners for this "
ePirat's avatar
ePirat committed
806
                        "mountpoint (%s).", source->mount);
Michael Smith's avatar
Michael Smith committed
807 808 809 810
                continue;
            }
            
            /* Otherwise, the client is accepted, add it */
811
            avl_insert(source->client_tree, client_node->key);
Michael Smith's avatar
Michael Smith committed
812 813

            source->listeners++;
814
            ICECAST_LOG_DEBUG("Client added for mountpoint (%s)", source->mount);
815 816 817 818 819 820 821
            stats_event_inc(source->mount, "connections");

            client_node = avl_get_next(client_node);
        }

        /** clear pending tree **/
        while (avl_get_first(source->pending_tree)) {
Michael Smith's avatar
Michael Smith committed
822 823 824
            avl_delete(source->pending_tree, 
                    avl_get_first(source->pending_tree)->key, 
                    source_remove_client);
825 826 827 828 829
        }

        /* release write lock on pending_tree */
        avl_tree_unlock(source->pending_tree);

Karl Heyes's avatar
Karl Heyes committed
830
        /* update the stats if need be */
831
        if (source->listeners != source->prev_listeners)
Karl Heyes's avatar
Karl Heyes committed
832
        {
833
            source->prev_listeners = source->listeners;
834
            ICECAST_LOG_INFO("listener count on %s now %lu", source->mount, source->listeners);
Karl Heyes's avatar
Karl Heyes committed
835 836 837 838 839
            if (source->listeners > source->peak_listeners)
            {
                source->peak_listeners = source->listeners;
                stats_event_args (source->mount, "listener_peak", "%lu", source->peak_listeners);
            }
840
            stats_event_args (source->mount, "listeners", "%lu", source->listeners);
841 842
            if (source->listeners == 0 && source->on_demand)
                source->running = 0;
Karl Heyes's avatar
Karl Heyes committed
843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859
        }

        /* lets reduce the queue, any lagging clients should of been
         * terminated by now
         */
        if (source->stream_data)
        {
            /* normal unreferenced queue data will have a refcount 1, but
             * burst queue data will be at least 2, active clients will also
             * increase refcount */
            while (source->stream_data->_count == 1)
            {
                refbuf_t *to_go = source->stream_data;

                if (to_go->next == NULL || source->burst_point == to_go)
                {
                    /* this should not happen */
860
                    ICECAST_LOG_ERROR("queue state is unexpected");
Karl Heyes's avatar
Karl Heyes committed
861 862 863 864 865
                    source->running = 0;
                    break;
                }
                source->stream_data = to_go->next;
                source->queue_size -= to_go->len;
866
                to_go->next = NULL;
Karl Heyes's avatar
Karl Heyes committed
867 868 869 870
                refbuf_release (to_go);
            }
        }

871 872 873
        /* release write lock on client_tree */
        avl_tree_unlock(source->client_tree);
    }
Karl Heyes's avatar
Karl Heyes committed
874 875
    source_shutdown (source);
}
Jack Moffitt's avatar
Jack Moffitt committed
876

Michael Smith's avatar
Michael Smith committed
877

Karl Heyes's avatar
Karl Heyes committed
878 879
static void source_shutdown (source_t *source)
{
880 881
    mount_proxy *mountinfo;

882
    source->running = 0;
883
    ICECAST_LOG_INFO("Source from %s at \"%s\" exiting", source->con->ip, source->mount);
884

885
    mountinfo = config_find_mount (config_get_config(), source->mount, MOUNT_TYPE_NORMAL);
886 887 888
    if (mountinfo)
    {
        if (mountinfo->on_disconnect)
889
            source_run_script (mountinfo->on_disconnect, source, mountinfo, "source-disconnect");
Philipp Schafft's avatar
Philipp Schafft committed
890 891
        /* TODO: replace with <event> */
        /* auth_stream_end (mountinfo, source->mount); */
892
    }
893 894
    config_release_config();

895 896
    /* we have de-activated the source now, so no more clients will be
     * added, now move the listeners we have to the fallback (if any)
897
     */
898 899 900
    if (source->fallback_mount)
    {
        source_t *fallback_source;
Michael Smith's avatar
Michael Smith committed
901

902
        avl_tree_rlock(global.source_tree);
903
        fallback_source = source_find_mount(source->fallback_mount);
Michael Smith's avatar
Michael Smith committed
904

905
        if (fallback_source != NULL)
906
            source_move_clients(source, fallback_source);
907

908
        avl_tree_unlock(global.source_tree);
909
    }
Jack Moffitt's avatar
Jack Moffitt committed
910

911
    /* delete this sources stats */
912
    stats_event(source->mount, NULL, NULL);
Jack Moffitt's avatar
Jack Moffitt committed
913

914
    /* we don't remove the source from the tree here, it may be a relay and
915 916
     therefore reserved */
    source_clear_source(source);
917

918 919
    global_lock();
    global.sources--;
920
    stats_event_args(NULL, "sources", "%d", global.sources);
921
    global_unlock();
Jack Moffitt's avatar
Jack Moffitt committed
922

923 924
    /* release our hold on the lock so the main thread can continue cleaning up */
    thread_rwlock_unlock(source->shutdown_rwlock);
Jack Moffitt's avatar
Jack Moffitt committed
925 926
}

927

Jack Moffitt's avatar
Jack Moffitt committed
928 929
static int _compare_clients(void *compare_arg, void *a, void *b)
{
930 931
    client_t *clienta = (client_t *) a;
    client_t *clientb = (client_t *) b;
932 933 934

    connection_t *cona = clienta->con;
    connection_t *conb = clientb->con;
Jack Moffitt's avatar
Jack Moffitt committed
935

936 937
    if (cona->id < conb->id) return -1;
    if (cona->id > conb->id) return 1;
Jack Moffitt's avatar
Jack Moffitt committed
938

939
    return 0;
Jack Moffitt's avatar
Jack Moffitt committed
940 941
}