mirror of
https://gitlab.xiph.org/xiph/icecast-server.git
synced 2024-12-04 14:46:30 -05:00
555 lines
17 KiB
C
555 lines
17 KiB
C
/* 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).
|
|
* Copyright 2014-2018, Philipp "ph3-der-loewe" Schafft <lion@lion.leolix.org>,
|
|
*/
|
|
|
|
|
|
/* Ogg codec handler for vorbis streams */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <ogg/ogg.h>
|
|
#include <vorbis/codec.h>
|
|
#include <memory.h>
|
|
#include <string.h>
|
|
|
|
#include "refbuf.h"
|
|
#include "source.h"
|
|
#include "client.h"
|
|
|
|
#include "format_ogg.h"
|
|
#include "stats.h"
|
|
#include "format.h"
|
|
|
|
#define CATMODULE "format-vorbis"
|
|
#include "logging.h"
|
|
|
|
|
|
typedef struct vorbis_codec_tag
|
|
{
|
|
vorbis_info vi;
|
|
|
|
int rebuild_comment;
|
|
int stream_notify;
|
|
int initial_audio_page;
|
|
|
|
ogg_stream_state new_os;
|
|
int page_samples_trigger;
|
|
ogg_int64_t prev_granulepos;
|
|
ogg_packet *prev_packet;
|
|
ogg_int64_t granulepos;
|
|
ogg_int64_t initial_page_granulepos;
|
|
ogg_int64_t samples_in_page;
|
|
int prev_window;
|
|
int initial_audio_packet;
|
|
|
|
ogg_page bos_page;
|
|
ogg_packet *header[3];
|
|
ogg_int64_t prev_page_samples;
|
|
|
|
int (*process_packet)(ogg_state_t *ogg_info, ogg_codec_t *codec, format_plugin_t *plugin);
|
|
refbuf_t *(*get_buffer_page)(ogg_state_t *ogg_info, ogg_codec_t *codec);
|
|
|
|
} vorbis_codec_t;
|
|
|
|
static int process_vorbis_headers (ogg_state_t *ogg_info, ogg_codec_t *codec, format_plugin_t *plugin);
|
|
static refbuf_t *process_vorbis_page (ogg_state_t *ogg_info,
|
|
ogg_codec_t *codec, ogg_page *page, format_plugin_t *plugin);
|
|
static refbuf_t *process_vorbis (ogg_state_t *ogg_info, ogg_codec_t *codec, format_plugin_t *plugin);
|
|
static void vorbis_set_tag (format_plugin_t *plugin, const char *tag, const char *value, const char *charset);
|
|
|
|
|
|
static void free_ogg_packet (ogg_packet *packet)
|
|
{
|
|
if (packet)
|
|
{
|
|
free (packet->packet);
|
|
free (packet);
|
|
}
|
|
}
|
|
|
|
|
|
static void vorbis_codec_free (ogg_state_t *ogg_info, ogg_codec_t *codec)
|
|
{
|
|
vorbis_codec_t *vorbis = codec->specific;
|
|
|
|
ICECAST_LOG_DEBUG("freeing vorbis codec");
|
|
stats_event (ogg_info->mount, "audio_bitrate", NULL);
|
|
stats_event (ogg_info->mount, "audio_channels", NULL);
|
|
stats_event (ogg_info->mount, "audio_samplerate", NULL);
|
|
vorbis_info_clear (&vorbis->vi);
|
|
ogg_stream_clear (&codec->os);
|
|
ogg_stream_clear (&vorbis->new_os);
|
|
free_ogg_packet (vorbis->header[0]);
|
|
free_ogg_packet (vorbis->header[1]);
|
|
free_ogg_packet (vorbis->header[2]);
|
|
free_ogg_packet (vorbis->prev_packet);
|
|
free (vorbis->bos_page.header);
|
|
free (vorbis);
|
|
free (codec);
|
|
}
|
|
|
|
|
|
static ogg_packet *copy_ogg_packet (ogg_packet *packet)
|
|
{
|
|
ogg_packet *next;
|
|
do
|
|
{
|
|
next = malloc (sizeof (ogg_packet));
|
|
if (next == NULL)
|
|
break;
|
|
memcpy (next, packet, sizeof (ogg_packet));
|
|
next->packet = malloc (next->bytes);
|
|
if (next->packet == NULL)
|
|
break;
|
|
memcpy (next->packet, packet->packet, next->bytes);
|
|
return next;
|
|
} while (0);
|
|
|
|
if (next)
|
|
free (next);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static void add_audio_packet (vorbis_codec_t *source_vorbis, ogg_packet *packet)
|
|
{
|
|
if (source_vorbis->initial_audio_packet)
|
|
{
|
|
packet->granulepos = 0;
|
|
source_vorbis->initial_audio_packet = 0;
|
|
}
|
|
else
|
|
{
|
|
source_vorbis->samples_in_page +=
|
|
(packet->granulepos - source_vorbis->prev_granulepos);
|
|
source_vorbis->prev_granulepos = packet->granulepos;
|
|
source_vorbis->granulepos += source_vorbis->prev_window;
|
|
}
|
|
ogg_stream_packetin (&source_vorbis->new_os, packet);
|
|
}
|
|
|
|
|
|
static refbuf_t *get_buffer_audio (ogg_state_t *ogg_info, ogg_codec_t *codec)
|
|
{
|
|
refbuf_t *refbuf = NULL;
|
|
ogg_page page;
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
int (*get_ogg_page)(ogg_stream_state*, ogg_page *) = ogg_stream_pageout;
|
|
|
|
if (source_vorbis->samples_in_page > source_vorbis->page_samples_trigger)
|
|
get_ogg_page = ogg_stream_flush;
|
|
|
|
if (get_ogg_page (&source_vorbis->new_os, &page) > 0)
|
|
{
|
|
/* squeeze a page copy into a buffer */
|
|
source_vorbis->samples_in_page -= (ogg_page_granulepos (&page) - source_vorbis->prev_page_samples);
|
|
source_vorbis->prev_page_samples = ogg_page_granulepos (&page);
|
|
|
|
refbuf = make_refbuf_with_page (&page);
|
|
}
|
|
return refbuf;
|
|
}
|
|
|
|
|
|
static refbuf_t *get_buffer_header (ogg_state_t *ogg_info, ogg_codec_t *codec)
|
|
{
|
|
int headers_flushed = 0;
|
|
ogg_page page;
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
|
|
while (ogg_stream_flush (&source_vorbis->new_os, &page) > 0)
|
|
{
|
|
format_ogg_attach_header (ogg_info, &page);
|
|
headers_flushed = 1;
|
|
}
|
|
if (headers_flushed)
|
|
{
|
|
source_vorbis->get_buffer_page = get_buffer_audio;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static refbuf_t *get_buffer_finished(ogg_state_t *ogg_info, ogg_codec_t *codec)
|
|
{
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
ogg_page page;
|
|
refbuf_t *refbuf;
|
|
|
|
if (ogg_stream_flush (&source_vorbis->new_os, &page) > 0)
|
|
{
|
|
source_vorbis->samples_in_page -= (ogg_page_granulepos (&page) - source_vorbis->prev_page_samples);
|
|
source_vorbis->prev_page_samples = ogg_page_granulepos (&page);
|
|
|
|
refbuf = make_refbuf_with_page (&page);
|
|
ICECAST_LOG_DEBUG("flushing page");
|
|
return refbuf;
|
|
}
|
|
ogg_stream_clear (&source_vorbis->new_os);
|
|
ogg_stream_init (&source_vorbis->new_os, rand());
|
|
|
|
format_ogg_free_headers (ogg_info);
|
|
source_vorbis->get_buffer_page = NULL;
|
|
if (source_vorbis->prev_packet)
|
|
source_vorbis->process_packet = process_vorbis_headers;
|
|
else
|
|
source_vorbis->process_packet = NULL;
|
|
|
|
if (source_vorbis->initial_audio_packet == 0)
|
|
source_vorbis->prev_window = 0;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* push last packet into stream marked with eos */
|
|
static void initiate_flush (vorbis_codec_t *source_vorbis)
|
|
{
|
|
if (source_vorbis->prev_packet)
|
|
{
|
|
/* insert prev_packet with eos */
|
|
ICECAST_LOG_DEBUG("adding EOS packet");
|
|
source_vorbis->prev_packet->e_o_s = 1;
|
|
add_audio_packet (source_vorbis, source_vorbis->prev_packet);
|
|
source_vorbis->prev_packet->e_o_s = 0;
|
|
}
|
|
source_vorbis->get_buffer_page = get_buffer_finished;
|
|
source_vorbis->initial_audio_packet = 1;
|
|
}
|
|
|
|
|
|
/* process the vorbis audio packets. Here we just take each packet out
|
|
* and add them into the new stream, flushing after so many samples. We
|
|
* also check if an new headers are requested after each processed page
|
|
*/
|
|
static int process_vorbis_audio(ogg_state_t *ogg_info, ogg_codec_t *codec, format_plugin_t *plugin)
|
|
{
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
|
|
while (1)
|
|
{
|
|
int window;
|
|
ogg_packet packet;
|
|
|
|
/* now, lets extract what packets we can */
|
|
if (ogg_stream_packetout (&codec->os, &packet) <= 0)
|
|
break;
|
|
|
|
/* calculate granulepos for the packet */
|
|
window = vorbis_packet_blocksize (&source_vorbis->vi, &packet) / 4;
|
|
|
|
source_vorbis->granulepos += window;
|
|
if (source_vorbis->prev_packet)
|
|
{
|
|
ogg_packet *prev_packet = source_vorbis->prev_packet;
|
|
|
|
add_audio_packet (source_vorbis, prev_packet);
|
|
free_ogg_packet (prev_packet);
|
|
|
|
/* check for short values on first initial page */
|
|
if (packet . packetno == 4)
|
|
{
|
|
if (source_vorbis->initial_page_granulepos < source_vorbis->granulepos)
|
|
{
|
|
source_vorbis->granulepos -= source_vorbis->initial_page_granulepos;
|
|
source_vorbis->samples_in_page = source_vorbis->page_samples_trigger;
|
|
}
|
|
}
|
|
/* check for long values on first page */
|
|
if (packet.granulepos == source_vorbis->initial_page_granulepos)
|
|
{
|
|
if (source_vorbis->initial_page_granulepos > source_vorbis->granulepos)
|
|
source_vorbis->granulepos = source_vorbis->initial_page_granulepos;
|
|
}
|
|
|
|
if (packet.e_o_s == 0)
|
|
packet . granulepos = source_vorbis->granulepos;
|
|
}
|
|
else
|
|
{
|
|
packet . granulepos = 0;
|
|
}
|
|
|
|
/* store the current packet details */
|
|
source_vorbis->prev_window = window;
|
|
source_vorbis->prev_packet = copy_ogg_packet (&packet);
|
|
if (packet.e_o_s)
|
|
{
|
|
initiate_flush (source_vorbis);
|
|
free_ogg_packet (source_vorbis->prev_packet);
|
|
source_vorbis->prev_packet = NULL;
|
|
return 1;
|
|
}
|
|
|
|
/* allow for pages to be flushed if there's over a certain number of samples */
|
|
if (source_vorbis->samples_in_page > source_vorbis->page_samples_trigger)
|
|
return 1;
|
|
}
|
|
if (source_vorbis->stream_notify)
|
|
{
|
|
initiate_flush (source_vorbis);
|
|
source_vorbis->stream_notify = 0;
|
|
return 1;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
|
|
/* This handles the headers at the backend, here we insert the header packets
|
|
* we want for the queue.
|
|
*/
|
|
static int process_vorbis_headers (ogg_state_t *ogg_info, ogg_codec_t *codec, format_plugin_t *plugin)
|
|
{
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
|
|
if (source_vorbis->header [0] == NULL)
|
|
return 0;
|
|
|
|
ICECAST_LOG_DEBUG("Adding the 3 header packets");
|
|
ogg_stream_packetin (&source_vorbis->new_os, source_vorbis->header [0]);
|
|
/* NOTE: we could build a separate comment packet each time */
|
|
if (source_vorbis->rebuild_comment)
|
|
{
|
|
ogg_packet header;
|
|
ice_config_t *config;
|
|
|
|
config = config_get_config();
|
|
format_set_vorbiscomment(plugin, "server", config->server_id);
|
|
config_release_config();
|
|
vorbis_commentheader_out (&plugin->vc, &header);
|
|
|
|
ogg_stream_packetin (&source_vorbis->new_os, &header);
|
|
ogg_packet_clear (&header);
|
|
}
|
|
else
|
|
ogg_stream_packetin (&source_vorbis->new_os, source_vorbis->header [1]);
|
|
ogg_stream_packetin (&source_vorbis->new_os, source_vorbis->header [2]);
|
|
source_vorbis->rebuild_comment = 0;
|
|
|
|
ogg_info->log_metadata = 1;
|
|
source_vorbis->get_buffer_page = get_buffer_header;
|
|
source_vorbis->process_packet = process_vorbis_audio;
|
|
source_vorbis->granulepos = source_vorbis->prev_window;
|
|
source_vorbis->initial_audio_packet = 1;
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* check if the provided BOS page is the start of a vorbis stream. If so
|
|
* then setup a structure so it can be used
|
|
*/
|
|
ogg_codec_t *initial_vorbis_page (format_plugin_t *plugin, ogg_page *page)
|
|
{
|
|
ogg_codec_t *codec = calloc (1, sizeof (ogg_codec_t));
|
|
ogg_packet packet;
|
|
|
|
vorbis_codec_t *vorbis = calloc (1, sizeof (vorbis_codec_t));
|
|
|
|
ogg_stream_init (&codec->os, ogg_page_serialno (page));
|
|
ogg_stream_pagein (&codec->os, page);
|
|
|
|
vorbis_info_init (&vorbis->vi);
|
|
|
|
ogg_stream_packetout (&codec->os, &packet);
|
|
|
|
vorbis_comment_clear(&plugin->vc);
|
|
vorbis_comment_init(&plugin->vc);
|
|
|
|
ICECAST_LOG_DEBUG("checking for vorbis codec");
|
|
if (vorbis_synthesis_headerin (&vorbis->vi, &plugin->vc, &packet) < 0)
|
|
{
|
|
ogg_stream_clear (&codec->os);
|
|
vorbis_info_clear (&vorbis->vi);
|
|
vorbis_comment_clear (&plugin->vc);
|
|
free (vorbis);
|
|
free (codec);
|
|
return NULL;
|
|
}
|
|
ICECAST_LOG_INFO("seen initial vorbis header");
|
|
codec->specific = vorbis;
|
|
codec->codec_free = vorbis_codec_free;
|
|
codec->headers = 1;
|
|
codec->name = "Vorbis";
|
|
|
|
free_ogg_packet(vorbis->header[0]);
|
|
free_ogg_packet(vorbis->header[1]);
|
|
free_ogg_packet(vorbis->header[2]);
|
|
memset(vorbis->header, 0, sizeof(vorbis->header));
|
|
vorbis->header[0] = copy_ogg_packet(&packet);
|
|
ogg_stream_init(&vorbis->new_os, rand());
|
|
|
|
codec->process_page = process_vorbis_page;
|
|
codec->process = process_vorbis;
|
|
plugin->set_tag = vorbis_set_tag;
|
|
|
|
vorbis->bos_page.header = malloc(page->header_len + page->body_len);
|
|
|
|
memcpy(vorbis->bos_page.header, page->header, page->header_len);
|
|
vorbis->bos_page.header_len = page->header_len;
|
|
|
|
vorbis->bos_page.body = vorbis->bos_page.header + page->header_len;
|
|
memcpy(vorbis->bos_page.body, page->body, page->body_len);
|
|
vorbis->bos_page.body_len = page->body_len;
|
|
|
|
return codec;
|
|
}
|
|
|
|
|
|
/* called from the admin interface, here we update the artist/title info
|
|
* and schedule a new set of header pages
|
|
*/
|
|
static void vorbis_set_tag (format_plugin_t *plugin, const char *tag, const char *in_value, const char *charset)
|
|
{
|
|
ogg_state_t *ogg_info = plugin->_state;
|
|
ogg_codec_t *codec = ogg_info->codecs;
|
|
vorbis_codec_t *source_vorbis;
|
|
char *value;
|
|
|
|
ICECAST_LOG_WARN("Not officially supported metadata update detected, please inform the source client software vendor that they should fix their software!");
|
|
|
|
/* avoid updating if multiple codecs in use */
|
|
if (codec && codec->next == NULL)
|
|
source_vorbis = codec->specific;
|
|
else
|
|
return;
|
|
|
|
if (tag == NULL)
|
|
{
|
|
source_vorbis->stream_notify = 1;
|
|
source_vorbis->rebuild_comment = 1;
|
|
return;
|
|
}
|
|
|
|
value = util_conv_string (in_value, charset, "UTF-8");
|
|
if (value == NULL)
|
|
value = strdup (in_value);
|
|
|
|
if (strcmp(tag, "song") == 0)
|
|
tag = "title";
|
|
|
|
format_set_vorbiscomment(plugin, tag, value);
|
|
free (value);
|
|
}
|
|
|
|
|
|
/* main backend routine when rebuilding streams. Here we loop until we either
|
|
* have a refbuf to add onto the queue, or we want more data to process.
|
|
*/
|
|
static refbuf_t *process_vorbis (ogg_state_t *ogg_info, ogg_codec_t *codec, format_plugin_t *plugin)
|
|
{
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
refbuf_t *refbuf;
|
|
|
|
while (1)
|
|
{
|
|
if (source_vorbis->get_buffer_page)
|
|
{
|
|
refbuf = source_vorbis->get_buffer_page (ogg_info, codec);
|
|
if (refbuf)
|
|
return refbuf;
|
|
}
|
|
|
|
if (source_vorbis->process_packet &&
|
|
source_vorbis->process_packet (ogg_info, codec, plugin) > 0)
|
|
continue;
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/* no processing of pages, just wrap them up in a refbuf and pass
|
|
* back for adding to the queue
|
|
*/
|
|
static refbuf_t *process_vorbis_passthru_page (ogg_state_t *ogg_info,
|
|
ogg_codec_t *codec, ogg_page *page, format_plugin_t *plugin)
|
|
{
|
|
return make_refbuf_with_page (page);
|
|
}
|
|
|
|
|
|
/* handle incoming page. as the stream is being rebuilt, we need to
|
|
* add all pages from the stream before processing packets
|
|
*/
|
|
static refbuf_t *process_vorbis_page (ogg_state_t *ogg_info,
|
|
ogg_codec_t *codec, ogg_page *page, format_plugin_t *plugin)
|
|
{
|
|
ogg_packet header;
|
|
vorbis_codec_t *source_vorbis = codec->specific;
|
|
|
|
if (ogg_stream_pagein (&codec->os, page) < 0)
|
|
{
|
|
ogg_info->error = 1;
|
|
return NULL;
|
|
}
|
|
if (codec->headers == 3)
|
|
{
|
|
if (source_vorbis->initial_audio_page)
|
|
{
|
|
source_vorbis->initial_page_granulepos = ogg_page_granulepos (page);
|
|
source_vorbis->initial_audio_page = 0;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
while (codec->headers < 3)
|
|
{
|
|
/* now, lets extract the packets */
|
|
ICECAST_LOG_DEBUG("processing incoming header packet (%d)", codec->headers);
|
|
|
|
if (ogg_stream_packetout (&codec->os, &header) <= 0)
|
|
{
|
|
if (ogg_info->codecs->next)
|
|
format_ogg_attach_header(ogg_info, page);
|
|
return NULL;
|
|
}
|
|
|
|
if (vorbis_synthesis_headerin (&source_vorbis->vi, &plugin->vc, &header) < 0)
|
|
{
|
|
ogg_info->error = 1;
|
|
ICECAST_LOG_WARN("Problem parsing ogg vorbis header");
|
|
return NULL;
|
|
}
|
|
header.granulepos = 0;
|
|
source_vorbis->header[codec->headers] = copy_ogg_packet(&header);
|
|
codec->headers++;
|
|
}
|
|
ICECAST_LOG_DEBUG("we have the header packets now");
|
|
|
|
/* if vorbis is the only codec then allow rebuilding of the streams */
|
|
if (ogg_info->codecs->next == NULL)
|
|
{
|
|
/* set queued vorbis pages to contain about 1/2 of a second worth of samples */
|
|
source_vorbis->page_samples_trigger = source_vorbis->vi.rate / 2;
|
|
source_vorbis->process_packet = process_vorbis_headers;
|
|
source_vorbis->initial_audio_page = 1;
|
|
}
|
|
else
|
|
{
|
|
format_ogg_attach_header (ogg_info, &source_vorbis->bos_page);
|
|
format_ogg_attach_header (ogg_info, page);
|
|
codec->process_page = process_vorbis_passthru_page;
|
|
}
|
|
|
|
ogg_info->log_metadata = 1;
|
|
|
|
stats_event_args (ogg_info->mount, "audio_samplerate", "%ld", (long)source_vorbis->vi.rate);
|
|
stats_event_args (ogg_info->mount, "audio_channels", "%ld", (long)source_vorbis->vi.channels);
|
|
stats_event_args (ogg_info->mount, "audio_bitrate", "%ld", (long)source_vorbis->vi.bitrate_nominal);
|
|
stats_event_args (ogg_info->mount, "ice-bitrate", "%ld", (long)source_vorbis->vi.bitrate_nominal/1000);
|
|
|
|
return NULL;
|
|
}
|
|
|