00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00032 #include "libavutil/intreadwrite.h"
00033 #include "avformat.h"
00034 #include "raw.h"
00035 #include "sox.h"
00036
00037 static int sox_probe(AVProbeData *p)
00038 {
00039 if (AV_RL32(p->buf) == SOX_TAG || AV_RB32(p->buf) == SOX_TAG)
00040 return AVPROBE_SCORE_MAX;
00041 return 0;
00042 }
00043
00044 static int sox_read_header(AVFormatContext *s,
00045 AVFormatParameters *ap)
00046 {
00047 ByteIOContext *pb = s->pb;
00048 unsigned header_size, comment_size;
00049 double sample_rate, sample_rate_frac;
00050 AVStream *st;
00051
00052 st = av_new_stream(s, 0);
00053 if (!st)
00054 return AVERROR(ENOMEM);
00055
00056 st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
00057
00058 if (get_le32(pb) == SOX_TAG) {
00059 st->codec->codec_id = CODEC_ID_PCM_S32LE;
00060 header_size = get_le32(pb);
00061 url_fskip(pb, 8);
00062 sample_rate = av_int2dbl(get_le64(pb));
00063 st->codec->channels = get_le32(pb);
00064 comment_size = get_le32(pb);
00065 } else {
00066 st->codec->codec_id = CODEC_ID_PCM_S32BE;
00067 header_size = get_be32(pb);
00068 url_fskip(pb, 8);
00069 sample_rate = av_int2dbl(get_be64(pb));
00070 st->codec->channels = get_be32(pb);
00071 comment_size = get_be32(pb);
00072 }
00073
00074 if (comment_size > 0xFFFFFFFFU - SOX_FIXED_HDR - 4U) {
00075 av_log(s, AV_LOG_ERROR, "invalid comment size (%u)\n", comment_size);
00076 return -1;
00077 }
00078
00079 if (sample_rate <= 0 || sample_rate > INT_MAX) {
00080 av_log(s, AV_LOG_ERROR, "invalid sample rate (%f)\n", sample_rate);
00081 return -1;
00082 }
00083
00084 sample_rate_frac = sample_rate - floor(sample_rate);
00085 if (sample_rate_frac)
00086 av_log(s, AV_LOG_WARNING,
00087 "truncating fractional part of sample rate (%f)\n",
00088 sample_rate_frac);
00089
00090 if ((header_size + 4) & 7 || header_size < SOX_FIXED_HDR + comment_size
00091 || st->codec->channels > 65535) {
00092 av_log(s, AV_LOG_ERROR, "invalid header\n");
00093 return -1;
00094 }
00095
00096 if (comment_size && comment_size < UINT_MAX) {
00097 char *comment = av_malloc(comment_size+1);
00098 if (get_buffer(pb, comment, comment_size) != comment_size) {
00099 av_freep(&comment);
00100 return AVERROR(EIO);
00101 }
00102 comment[comment_size] = 0;
00103
00104 av_metadata_set2(&s->metadata, "comment", comment,
00105 AV_METADATA_DONT_STRDUP_VAL);
00106 }
00107
00108 url_fskip(pb, header_size - SOX_FIXED_HDR - comment_size);
00109
00110 st->codec->sample_rate = sample_rate;
00111 st->codec->bits_per_coded_sample = 32;
00112 st->codec->bit_rate = st->codec->sample_rate *
00113 st->codec->bits_per_coded_sample *
00114 st->codec->channels;
00115 st->codec->block_align = st->codec->bits_per_coded_sample *
00116 st->codec->channels / 8;
00117
00118 av_set_pts_info(st, 64, 1, st->codec->sample_rate);
00119
00120 return 0;
00121 }
00122
00123 #define SOX_SAMPLES 1024
00124
00125 static int sox_read_packet(AVFormatContext *s,
00126 AVPacket *pkt)
00127 {
00128 int ret, size;
00129
00130 if (url_feof(s->pb))
00131 return AVERROR_EOF;
00132
00133 size = SOX_SAMPLES*s->streams[0]->codec->block_align;
00134 ret = av_get_packet(s->pb, pkt, size);
00135 if (ret < 0)
00136 return AVERROR(EIO);
00137 pkt->stream_index = 0;
00138 pkt->size = ret;
00139
00140 return 0;
00141 }
00142
00143 AVInputFormat sox_demuxer = {
00144 "sox",
00145 NULL_IF_CONFIG_SMALL("SoX native format"),
00146 0,
00147 sox_probe,
00148 sox_read_header,
00149 sox_read_packet,
00150 NULL,
00151 pcm_read_seek,
00152 };