1
0
mirror of https://gitlab.xiph.org/xiph/icecast-server.git synced 2024-06-30 06:35:23 +00:00
icecast-server/src/format_flac.c

226 lines
5.5 KiB
C
Raw Normal View History

/* Icecast
*
* This program is distributed under the GNU General Public License, version 2.
* A copy of this license is included with this source.
*
2015-01-10 18:53:44 +00:00
* 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).
2018-11-26 07:42:05 +00:00
* Copyright 2014-2018, Philipp "ph3-der-loewe" Schafft <lion@lion.leolix.org>,
*/
/* Ogg codec handler for FLAC logical streams */
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdlib.h>
#include <stdint.h>
#include <stdbool.h>
#include <ogg/ogg.h>
#include <string.h>
#include "refbuf.h"
#include "format_ogg.h"
#include "client.h"
#include "stats.h"
#define CATMODULE "format-flac"
#include "logging.h"
typedef enum {
FLAC_BLOCK_TYPE__ERROR = -1,
FLAC_BLOCK_TYPE_STREAMINFO = 0,
FLAC_BLOCK_TYPE_PADDING = 1,
FLAC_BLOCK_TYPE_APPLICATION = 2,
FLAC_BLOCK_TYPE_SEEKTABLE = 3,
FLAC_BLOCK_TYPE_VORBIS_COMMENT = 4,
FLAC_BLOCK_TYPE_CUESHEET = 5,
FLAC_BLOCK_TYPE_PICTURE = 6
} flac_block_type_t;
typedef struct {
flac_block_type_t type;
bool last;
size_t len;
const void *data;
} flac_block_t;
/* returns true if parse was ok, false on error */
static bool flac_parse_block(flac_block_t *block, const ogg_packet * packet)
{
uint8_t type;
uint32_t len;
/* check header length */
if (packet->bytes <= 4)
return false;
type = packet->packet[0];
/* 0xFF is the sync code for a FRAME not a block */
if (type == 0xFF)
return false;
memset(block, 0, sizeof(*block));
if (type & 0x80) {
block->last = true;
type &= 0x7F;
}
if (type > FLAC_BLOCK_TYPE_PICTURE)
return false;
block->type = type;
len = (unsigned char)packet->packet[1];
len <<= 8;
len |= (unsigned char)packet->packet[2];
len <<= 8;
len |= (unsigned char)packet->packet[3];
/* check Ogg packet size vs. self-sync size */
if (packet->bytes != (len + 4))
return false;
block->len = len;
block->data = packet->packet + 4;
return true;
}
static const char * flac_block_type_to_name(flac_block_type_t type)
{
switch (type) {
case FLAC_BLOCK_TYPE__ERROR:
return "<error>";
break;
case FLAC_BLOCK_TYPE_STREAMINFO:
return "STREAMINFO";
break;
case FLAC_BLOCK_TYPE_PADDING:
return "PADDING";
break;
case FLAC_BLOCK_TYPE_APPLICATION:
return "APPLICATION";
break;
case FLAC_BLOCK_TYPE_SEEKTABLE:
return "SEEKTABLE";
break;
case FLAC_BLOCK_TYPE_VORBIS_COMMENT:
return "VORBIS_COMMENT";
break;
case FLAC_BLOCK_TYPE_CUESHEET:
return "CUESHEET";
break;
case FLAC_BLOCK_TYPE_PICTURE:
return "PICTURE";
break;
}
return "<unknown>";
}
static void flac_codec_free (ogg_state_t *ogg_info, ogg_codec_t *codec)
{
ICECAST_LOG_DEBUG("freeing FLAC codec");
2022-03-15 22:12:18 +00:00
stats_event(ogg_info->mount, "FLAC_version", NULL);
ogg_stream_clear(&codec->os);
free(codec);
}
/* Here, we just verify the page is ok and then add it to the queue */
static refbuf_t *process_flac_page (ogg_state_t *ogg_info, ogg_codec_t *codec, ogg_page *page, format_plugin_t *plugin)
{
refbuf_t * refbuf;
2022-03-15 22:12:18 +00:00
if (codec->headers) {
ogg_packet packet;
2022-03-15 22:12:18 +00:00
if (ogg_stream_pagein(&codec->os, page) < 0) {
ogg_info->error = 1;
return NULL;
}
2022-03-15 22:12:18 +00:00
while (ogg_stream_packetout(&codec->os, &packet)) {
flac_block_t block;
2022-03-15 22:14:12 +00:00
if (flac_parse_block(&block, &packet)) {
ICECAST_LOG_DEBUG("Found header of type %s%s with %zu bytes of data", flac_block_type_to_name(block.type), block.last ? "(last)" : "", block.len);
if (block.last) {
2022-03-15 22:14:12 +00:00
codec->headers = 0;
break;
}
continue;
}
2022-03-15 22:12:18 +00:00
ogg_info->error = 1;
2022-03-15 22:12:18 +00:00
return NULL;
}
2022-03-15 22:12:18 +00:00
if (codec->headers) {
format_ogg_attach_header(ogg_info, page);
return NULL;
}
}
2022-03-15 22:12:18 +00:00
refbuf = make_refbuf_with_page(page);
return refbuf;
}
/* Check for flac header in logical stream */
ogg_codec_t *initial_flac_page (format_plugin_t *plugin, ogg_page *page)
{
ogg_state_t *ogg_info = plugin->_state;
2022-03-15 22:12:18 +00:00
ogg_codec_t *codec = calloc(1, sizeof(ogg_codec_t));
ogg_packet packet;
2022-03-15 22:12:18 +00:00
ogg_stream_init(&codec->os, ogg_page_serialno(page));
ogg_stream_pagein(&codec->os, page);
2022-03-15 22:12:18 +00:00
ogg_stream_packetout(&codec->os, &packet);
ICECAST_LOG_DEBUG("checking for FLAC codec");
do
{
unsigned char *parse = packet.packet;
if (page->header_len + page->body_len != 79)
break;
if (*parse != 0x7F)
break;
parse++;
if (memcmp(parse, "FLAC", 4) != 0)
break;
ICECAST_LOG_INFO("seen initial FLAC header");
parse += 4;
2022-03-15 22:12:18 +00:00
stats_event_args(ogg_info->mount, "FLAC_version", "%d.%d", parse[0], parse[1]);
codec->process_page = process_flac_page;
codec->codec_free = flac_codec_free;
codec->headers = 1;
codec->name = "FLAC";
format_ogg_attach_header(ogg_info, page);
return codec;
} while (0);
ogg_stream_clear(&codec->os);
free(codec);
return NULL;
}