/* 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 , * oddsock , * Karl Heyes * and others (see AUTHORS for details). */ /* Ogg codec handler for kate logical streams */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #ifdef HAVE_KATE #include #endif typedef struct source_tag source_t; #include "refbuf.h" #include "format_ogg.h" #include "format_kate.h" #include "client.h" #include "stats.h" #define CATMODULE "format-kate" #include "logging.h" typedef struct _kate_codec_tag { unsigned int headers_done; #ifdef HAVE_KATE kate_info ki; kate_comment kc; #endif unsigned int num_headers; int granule_shift; ogg_int64_t last_iframe; ogg_int64_t prev_granulepos; } kate_codec_t; static void kate_codec_free (ogg_state_t *ogg_info, ogg_codec_t *codec) { kate_codec_t *kate = codec->specific; DEBUG0 ("freeing kate codec"); /* TODO: should i replace with something or just remove stats_event (ogg_info->mount, "video_bitrate", NULL); stats_event (ogg_info->mount, "video_quality", NULL); stats_event (ogg_info->mount, "frame_rate", NULL); stats_event (ogg_info->mount, "frame_size", NULL); */ #ifdef HAVE_KATE kate_info_clear (&kate->ki); kate_comment_clear (&kate->kc); #endif ogg_stream_clear (&codec->os); free (kate); free (codec); } /* kate pages are not rebuilt, so here we just for headers and then * pass them straight through to the the queue */ static refbuf_t *process_kate_page (ogg_state_t *ogg_info, ogg_codec_t *codec, ogg_page *page) { kate_codec_t *kate = codec->specific; ogg_packet packet; int header_page = 0; refbuf_t *refbuf = NULL; ogg_int64_t granulepos; if (ogg_stream_pagein (&codec->os, page) < 0) { ogg_info->error = 1; return NULL; } granulepos = ogg_page_granulepos (page); while (ogg_stream_packetout (&codec->os, &packet) > 0) { if (!kate->headers_done) { #ifdef HAVE_KATE int ret = kate_ogg_decode_headerin (&kate->ki, &kate->kc, &packet); if (ret < 0) { ogg_info->error = 1; WARN0 ("problem with kate header"); return NULL; } header_page = 1; kate->num_headers = kate->ki.num_headers; codec->headers++; if (ret > 0) { kate->headers_done = 1; /* TODO: what to replace this with ? ogg_info->bitrate += theora->ti.target_bitrate; stats_event_args (ogg_info->mount, "video_bitrate", "%ld", (long)theora->ti.target_bitrate); stats_event_args (ogg_info->mount, "video_quality", "%ld", (long)theora->ti.quality); stats_event_args (ogg_info->mount, "frame_size", "%ld x %ld", (long)theora->ti.frame_width, (long)theora->ti.frame_height); stats_event_args (ogg_info->mount, "frame_rate", "%.2f", (float)theora->ti.fps_numerator/theora->ti.fps_denominator); */ } continue; #else header_page = (packet.bytes>0 && (packet.packet[0]&0x80)); if (!header_page) break; codec->headers++; if (packet.packet[0]==0x80) { if (packet.bytes<64) return NULL; /* we peek for the number of headers to expect */ kate->num_headers = packet.packet[11]; } continue; #endif } if (codec->headers < kate->num_headers) { ogg_info->error = 1; ERROR0 ("Not enough header packets"); return NULL; } } if (header_page) { format_ogg_attach_header (codec, page); return NULL; } refbuf = make_refbuf_with_page (codec, page); /* DEBUG3 ("refbuf %p has pageno %ld, %llu", refbuf, ogg_page_pageno (page), (uint64_t)granulepos); */ return refbuf; } /* Check if specified BOS page is the start of a kate stream and * if so, create a codec structure for handling it */ ogg_codec_t *initial_kate_page (format_plugin_t *plugin, ogg_page *page) { ogg_codec_t *codec = calloc (1, sizeof (ogg_codec_t)); ogg_packet packet; kate_codec_t *kate_codec = calloc (1, sizeof (kate_codec_t)); ogg_stream_init (&codec->os, ogg_page_serialno (page)); ogg_stream_pagein (&codec->os, page); #ifdef HAVE_KATE kate_info_init (&kate_codec->ki); kate_comment_init (&kate_codec->kc); #endif ogg_stream_packetout (&codec->os, &packet); DEBUG0("checking for kate codec"); #ifdef HAVE_KATE if (kate_ogg_decode_headerin (&kate_codec->ki, &kate_codec->kc, &packet) < 0) { kate_info_clear (&kate_codec->ki); kate_comment_clear (&kate_codec->kc); ogg_stream_clear (&codec->os); free (kate_codec); free (codec); return NULL; } #else /* we don't have libkate, so we examine the packet magic by hand */ if ((packet.bytes<8) || memcmp(packet.packet, "\x80kate\0\0\0", 8)) { ogg_stream_clear (&codec->os); free (kate_codec); free (codec); return NULL; } #endif INFO0 ("seen initial kate header"); codec->specific = kate_codec; codec->process_page = process_kate_page; codec->codec_free = kate_codec_free; codec->headers = 1; codec->name = "Kate"; format_ogg_attach_header (codec, page); return codec; }