2008-02-06 12:37:37 +00:00
|
|
|
/*
|
|
|
|
Copyright (C) 2008 Reimar Döffinger
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person
|
|
|
|
obtaining a copy of this software and associated documentation
|
|
|
|
files (the "Software"), to deal in the Software without
|
|
|
|
restriction, including without limitation the rights to use, copy,
|
|
|
|
modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
|
|
of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be
|
|
|
|
included in all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
DEALINGS IN THE SOFTWARE.
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2008-05-09 11:56:36 +00:00
|
|
|
#include "libavutil/bswap.h"
|
|
|
|
#include "libavutil/avstring.h"
|
2009-04-13 16:20:26 +00:00
|
|
|
#include "libavcodec/get_bits.h"
|
2008-05-09 11:56:36 +00:00
|
|
|
#include "libavcodec/bytestream.h"
|
2008-02-06 12:37:37 +00:00
|
|
|
#include "avformat.h"
|
|
|
|
#include "oggdec.h"
|
|
|
|
|
2009-10-12 21:30:03 +00:00
|
|
|
struct speex_params {
|
|
|
|
int final_packet_duration;
|
|
|
|
};
|
|
|
|
|
2008-02-06 12:37:37 +00:00
|
|
|
static int speex_header(AVFormatContext *s, int idx) {
|
2008-11-06 01:50:56 +00:00
|
|
|
struct ogg *ogg = s->priv_data;
|
|
|
|
struct ogg_stream *os = ogg->streams + idx;
|
2008-02-06 12:37:37 +00:00
|
|
|
AVStream *st = s->streams[idx];
|
|
|
|
uint8_t *p = os->buf + os->pstart;
|
|
|
|
|
2009-03-21 08:22:09 +00:00
|
|
|
if (os->seq > 1)
|
|
|
|
return 0;
|
2008-02-06 12:37:37 +00:00
|
|
|
|
2009-03-21 08:22:09 +00:00
|
|
|
if (os->seq == 0) {
|
2009-08-28 00:44:54 +00:00
|
|
|
int frames_per_packet;
|
2009-03-21 08:22:36 +00:00
|
|
|
st->codec->codec_type = CODEC_TYPE_AUDIO;
|
|
|
|
st->codec->codec_id = CODEC_ID_SPEEX;
|
2008-02-06 12:37:37 +00:00
|
|
|
|
2009-03-21 08:22:36 +00:00
|
|
|
st->codec->sample_rate = AV_RL32(p + 36);
|
|
|
|
st->codec->channels = AV_RL32(p + 48);
|
2009-08-28 00:44:54 +00:00
|
|
|
|
|
|
|
/* We treat the whole Speex packet as a single frame everywhere Speex
|
|
|
|
is handled in FFmpeg. This avoids the complexities of splitting
|
|
|
|
and joining individual Speex frames, which are not always
|
|
|
|
byte-aligned. */
|
2009-05-22 18:34:01 +00:00
|
|
|
st->codec->frame_size = AV_RL32(p + 56);
|
2009-08-28 00:44:54 +00:00
|
|
|
frames_per_packet = AV_RL32(p + 64);
|
|
|
|
if (frames_per_packet)
|
|
|
|
st->codec->frame_size *= frames_per_packet;
|
|
|
|
|
2009-03-21 08:22:36 +00:00
|
|
|
st->codec->extradata_size = os->psize;
|
2009-06-05 23:39:11 +00:00
|
|
|
st->codec->extradata = av_malloc(st->codec->extradata_size
|
|
|
|
+ FF_INPUT_BUFFER_PADDING_SIZE);
|
2009-03-21 08:22:36 +00:00
|
|
|
memcpy(st->codec->extradata, p, st->codec->extradata_size);
|
2008-02-06 12:37:37 +00:00
|
|
|
|
2009-03-21 08:22:36 +00:00
|
|
|
st->time_base.num = 1;
|
|
|
|
st->time_base.den = st->codec->sample_rate;
|
2009-03-21 08:22:09 +00:00
|
|
|
} else
|
|
|
|
vorbis_comment(s, p, os->psize);
|
2008-02-06 12:37:37 +00:00
|
|
|
|
2009-03-21 08:22:09 +00:00
|
|
|
return 1;
|
2008-02-06 12:37:37 +00:00
|
|
|
}
|
|
|
|
|
2009-10-12 21:30:03 +00:00
|
|
|
static int ogg_page_packets(struct ogg_stream *os)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int packets = 0;
|
|
|
|
for (i = 0; i < os->nsegs; i++)
|
|
|
|
if (os->segments[i] < 255)
|
|
|
|
packets++;
|
|
|
|
return packets;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int speex_packet(AVFormatContext *s, int idx)
|
|
|
|
{
|
|
|
|
struct ogg *ogg = s->priv_data;
|
|
|
|
struct ogg_stream *os = ogg->streams + idx;
|
|
|
|
struct speex_params *spxp = os->private;
|
|
|
|
int packet_size = s->streams[idx]->codec->frame_size;
|
|
|
|
|
|
|
|
if (!spxp) {
|
|
|
|
spxp = av_mallocz(sizeof(*spxp));
|
|
|
|
os->private = spxp;
|
|
|
|
}
|
|
|
|
|
2009-12-12 20:18:43 +00:00
|
|
|
if (os->flags & OGG_FLAG_EOS && os->lastpts != AV_NOPTS_VALUE &&
|
|
|
|
os->granule > 0) {
|
2009-10-12 21:30:03 +00:00
|
|
|
/* first packet of final page. we have to calculate the final packet
|
|
|
|
duration here because it is the only place we know the next-to-last
|
|
|
|
granule position. */
|
2009-12-12 20:18:43 +00:00
|
|
|
spxp->final_packet_duration = os->granule - os->lastpts -
|
2009-10-12 21:30:03 +00:00
|
|
|
packet_size * (ogg_page_packets(os) - 1);
|
|
|
|
}
|
|
|
|
|
2009-12-12 20:18:43 +00:00
|
|
|
if (!os->lastpts && os->granule > 0)
|
2009-10-12 21:30:03 +00:00
|
|
|
/* first packet */
|
|
|
|
os->pduration = os->granule - packet_size * (ogg_page_packets(os) - 1);
|
|
|
|
else if (os->flags & OGG_FLAG_EOS && os->segp == os->nsegs &&
|
|
|
|
spxp->final_packet_duration)
|
|
|
|
/* final packet */
|
|
|
|
os->pduration = spxp->final_packet_duration;
|
|
|
|
else
|
|
|
|
os->pduration = packet_size;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-11-06 01:50:56 +00:00
|
|
|
const struct ogg_codec ff_speex_codec = {
|
2008-02-06 12:37:37 +00:00
|
|
|
.magic = "Speex ",
|
|
|
|
.magicsize = 8,
|
2009-10-12 21:30:03 +00:00
|
|
|
.header = speex_header,
|
|
|
|
.packet = speex_packet
|
2008-02-06 12:37:37 +00:00
|
|
|
};
|