Browse code

VQF demuxer

Originally committed as revision 17866 to svn://svn.ffmpeg.org/ffmpeg/trunk

Vitor Sessak authored on 2009/03/08 07:42:09
Showing 6 changed files
... ...
@@ -3,6 +3,7 @@ version <next>:
3 3
 - PB-frame decoding for H.263
4 4
 - deprecated vhook subsystem removed
5 5
 - deprecated old scaler removed
6
+- VQF demuxer
6 7
 
7 8
 
8 9
 
... ...
@@ -211,6 +211,7 @@ library:
211 211
     @tab Tiertex .seq files used in the DOS CD-ROM version of the game Flashback.
212 212
 @item True Audio                @tab   @tab X
213 213
 @item VC-1 test bitstream       @tab X @tab X
214
+@item VQF                       @tab   @tab X
214 215
 @item WAV                       @tab X @tab X
215 216
 @item WavPack                   @tab   @tab X
216 217
 @item Wing Commander III movie  @tab   @tab X
... ...
@@ -309,6 +309,7 @@ enum CodecID {
309 309
     CODEC_ID_EAC3,
310 310
     CODEC_ID_SIPR,
311 311
     CODEC_ID_MP1,
312
+    CODEC_ID_TWINVQ,
312 313
 
313 314
     /* subtitle codecs */
314 315
     CODEC_ID_DVD_SUBTITLE= 0x17000,
... ...
@@ -214,6 +214,7 @@ OBJS-$(CONFIG_VC1T_MUXER)                += vc1testenc.o
214 214
 OBJS-$(CONFIG_VMD_DEMUXER)               += sierravmd.o
215 215
 OBJS-$(CONFIG_VOC_DEMUXER)               += vocdec.o voc.o
216 216
 OBJS-$(CONFIG_VOC_MUXER)                 += vocenc.o voc.o
217
+OBJS-$(CONFIG_VQF_DEMUXER)               += vqf.o
217 218
 OBJS-$(CONFIG_WAV_DEMUXER)               += wav.o riff.o raw.o
218 219
 OBJS-$(CONFIG_WAV_MUXER)                 += wav.o riff.o
219 220
 OBJS-$(CONFIG_WC3_DEMUXER)               += wc3movie.o
... ...
@@ -188,6 +188,7 @@ void av_register_all(void)
188 188
     REGISTER_MUXDEMUX (VC1T, vc1t);
189 189
     REGISTER_DEMUXER  (VMD, vmd);
190 190
     REGISTER_MUXDEMUX (VOC, voc);
191
+    REGISTER_DEMUXER  (VQF, vqf);
191 192
     REGISTER_MUXDEMUX (WAV, wav);
192 193
     REGISTER_DEMUXER  (WC3, wc3);
193 194
     REGISTER_DEMUXER  (WSAUD, wsaud);
194 195
new file mode 100644
... ...
@@ -0,0 +1,256 @@
0
+/*
1
+ * VQF demuxer
2
+ * Copyright (c) 2009 Vitor Sessak
3
+ *
4
+ * This file is part of FFmpeg.
5
+ *
6
+ * FFmpeg is free software; you can redistribute it and/or
7
+ * modify it under the terms of the GNU Lesser General Public
8
+ * License as published by the Free Software Foundation; either
9
+ * version 2.1 of the License, or (at your option) any later version.
10
+ *
11
+ * FFmpeg is distributed in the hope that it will be useful,
12
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
13
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14
+ * Lesser General Public License for more details.
15
+ *
16
+ * You should have received a copy of the GNU Lesser General Public
17
+ * License along with FFmpeg; if not, write to the Free Software
18
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19
+ */
20
+
21
+#include "avformat.h"
22
+#include "libavutil/intreadwrite.h"
23
+
24
+typedef struct VqfContext {
25
+    int frame_bit_len;
26
+    uint8_t last_frame_bits;
27
+    int remaining_bits;
28
+} VqfContext;
29
+
30
+static int vqf_probe(AVProbeData *probe_packet)
31
+{
32
+    if (AV_RL32(probe_packet->buf) != MKTAG('T','W','I','N'))
33
+        return 0;
34
+
35
+    if (!memcmp(probe_packet->buf + 4, "97012000", 8))
36
+        return AVPROBE_SCORE_MAX;
37
+
38
+    if (!memcmp(probe_packet->buf + 4, "00052200", 8))
39
+        return AVPROBE_SCORE_MAX;
40
+
41
+    return AVPROBE_SCORE_MAX/2;
42
+}
43
+
44
+static void add_metadata(AVFormatContext *s, const char *tag,
45
+                         unsigned int tag_len, unsigned int remaining)
46
+{
47
+    char buf[2048];
48
+    int len = FFMIN3(tag_len, remaining, sizeof(buf) - 1);
49
+
50
+    if (len != tag_len)
51
+        av_log(s, AV_LOG_ERROR, "Warning: truncating metadata!\n");
52
+
53
+    get_buffer(s->pb, buf, len);
54
+    buf[len] = 0;
55
+    av_metadata_set(&s->metadata, tag, buf);
56
+}
57
+
58
+static int vqf_read_header(AVFormatContext *s, AVFormatParameters *ap)
59
+{
60
+    VqfContext *c = s->priv_data;
61
+    AVStream *st  = av_new_stream(s, 0);
62
+    int chunk_tag;
63
+    int rate_flag = -1;
64
+    int header_size;
65
+    int read_bitrate = 0;
66
+    int size;
67
+
68
+    if (!st)
69
+        return AVERROR(ENOMEM);
70
+
71
+    url_fskip(s->pb, 12);
72
+
73
+    header_size = get_be32(s->pb);
74
+
75
+    st->codec->codec_type = CODEC_TYPE_AUDIO;
76
+    st->codec->codec_id   = CODEC_ID_TWINVQ;
77
+    st->start_time = 0;
78
+
79
+    do {
80
+        int len;
81
+        chunk_tag = get_le32(s->pb);
82
+
83
+        if (chunk_tag == MKTAG('D','A','T','A'))
84
+            break;
85
+
86
+        len = get_be32(s->pb);
87
+
88
+        if ((unsigned) len > INT_MAX/2) {
89
+            av_log(s, AV_LOG_ERROR, "Malformed header\n");
90
+            return -1;
91
+        }
92
+
93
+        header_size -= 8;
94
+
95
+        switch(chunk_tag){
96
+        case MKTAG('C','O','M','M'):
97
+            st->codec->channels = get_be32(s->pb) + 1;
98
+            read_bitrate        = get_be32(s->pb);
99
+            rate_flag           = get_be32(s->pb);
100
+            url_fskip(s->pb, len-12);
101
+
102
+            st->codec->bit_rate              = read_bitrate*1000;
103
+            st->codec->bits_per_coded_sample = 16;
104
+            break;
105
+        case MKTAG('N','A','M','E'):
106
+            add_metadata(s, "title"    , len, header_size);
107
+            break;
108
+        case MKTAG('(','c',')',' '):
109
+            add_metadata(s, "copyright", len, header_size);
110
+            break;
111
+        case MKTAG('A','U','T','H'):
112
+            add_metadata(s, "author"   , len, header_size);
113
+            break;
114
+        case MKTAG('A','L','B','M'):
115
+            add_metadata(s, "album"    , len, header_size);
116
+            break;
117
+        case MKTAG('T','R','C','K'):
118
+            add_metadata(s, "track"    , len, header_size);
119
+            break;
120
+        case MKTAG('C','O','M','T'):
121
+            add_metadata(s, "comment"  , len, header_size);
122
+            break;
123
+        case MKTAG('F','I','L','E'):
124
+            add_metadata(s, "filename" , len, header_size);
125
+            break;
126
+        case MKTAG('D','S','I','Z'):
127
+            add_metadata(s, "size"     , len, header_size);
128
+            break;
129
+        case MKTAG('D','A','T','E'):
130
+            add_metadata(s, "date"     , len, header_size);
131
+            break;
132
+        case MKTAG('G','E','N','R'):
133
+            add_metadata(s, "genre"    , len, header_size);
134
+            break;
135
+        default:
136
+            av_log(s, AV_LOG_ERROR, "Unknown chunk: %c%c%c%c\n",
137
+                   ((char*)&chunk_tag)[0], ((char*)&chunk_tag)[1],
138
+                   ((char*)&chunk_tag)[2], ((char*)&chunk_tag)[3]);
139
+            url_fskip(s->pb, FFMIN(len, header_size));
140
+            break;
141
+        }
142
+
143
+        header_size -= len;
144
+
145
+    } while (header_size >= 0);
146
+
147
+    switch (rate_flag) {
148
+    case -1:
149
+        av_log(s, AV_LOG_ERROR, "COMM tag not found!\n");
150
+        return -1;
151
+    case 44:
152
+        st->codec->sample_rate = 44100;
153
+        break;
154
+    case 22:
155
+        st->codec->sample_rate = 22050;
156
+        break;
157
+    case 11:
158
+        st->codec->sample_rate = 11025;
159
+        break;
160
+    default:
161
+        st->codec->sample_rate = rate_flag*1000;
162
+        break;
163
+    }
164
+
165
+    switch (((st->codec->sample_rate/1000) << 8) +
166
+            read_bitrate/st->codec->channels) {
167
+    case (11<<8) + 8 :
168
+    case (8 <<8) + 8 :
169
+    case (11<<8) + 10:
170
+    case (22<<8) + 32:
171
+        size = 512;
172
+        break;
173
+    case (16<<8) + 16:
174
+    case (22<<8) + 20:
175
+    case (22<<8) + 24:
176
+        size = 1024;
177
+        break;
178
+    case (44<<8) + 40:
179
+    case (44<<8) + 48:
180
+        size = 2048;
181
+        break;
182
+    default:
183
+        av_log(s, AV_LOG_ERROR, "Mode not suported: %d Hz, %d kb/s.\n",
184
+               st->codec->sample_rate, st->codec->bit_rate);
185
+        return -1;
186
+    }
187
+    c->frame_bit_len = st->codec->bit_rate*size/st->codec->sample_rate;
188
+
189
+    return 0;
190
+}
191
+
192
+static int vqf_read_packet(AVFormatContext *s, AVPacket *pkt)
193
+{
194
+    VqfContext *c = s->priv_data;
195
+    int ret;
196
+    int size = (c->frame_bit_len - c->remaining_bits + 7)>>3;
197
+
198
+    pkt->pos          = url_ftell(s->pb);
199
+    pkt->stream_index = 0;
200
+
201
+    if (av_new_packet(pkt, size+2) < 0)
202
+        return AVERROR(EIO);
203
+
204
+    pkt->data[0] = 8 - c->remaining_bits; // Number of bits to skip
205
+    pkt->data[1] = c->last_frame_bits;
206
+    ret = get_buffer(s->pb, pkt->data+2, size);
207
+
208
+    if (ret<=0) {
209
+        av_free_packet(pkt);
210
+        return AVERROR(EIO);
211
+    }
212
+
213
+    c->last_frame_bits = pkt->data[size+1];
214
+    c->remaining_bits  = (size << 3) - c->frame_bit_len + c->remaining_bits;
215
+
216
+    return size+2;
217
+}
218
+
219
+static int vqf_read_seek(AVFormatContext *s,
220
+                         int stream_index, int64_t timestamp, int flags)
221
+{
222
+    VqfContext *c = s->priv_data;
223
+    AVStream *st;
224
+    int ret;
225
+    int64_t pos;
226
+
227
+    st = s->streams[stream_index];
228
+    pos = av_rescale_rnd(timestamp * st->codec->bit_rate,
229
+                         st->time_base.num,
230
+                         st->time_base.den * (int64_t)c->frame_bit_len,
231
+                         (flags & AVSEEK_FLAG_BACKWARD) ?
232
+                                                   AV_ROUND_DOWN : AV_ROUND_UP);
233
+    pos *= c->frame_bit_len;
234
+
235
+    st->cur_dts = av_rescale(pos, st->time_base.den,
236
+                             st->codec->bit_rate * (int64_t)st->time_base.num);
237
+
238
+    if ((ret = url_fseek(s->pb, ((pos-7) >> 3) + s->data_offset, SEEK_SET)) < 0)
239
+        return ret;
240
+
241
+    c->remaining_bits = -7 - ((pos-7)&7);
242
+    return 0;
243
+}
244
+
245
+AVInputFormat vqf_demuxer = {
246
+    "vqf",
247
+    NULL_IF_CONFIG_SMALL("Nippon Telegraph and Telephone Corporation (NTT) TwinVQ"),
248
+    sizeof(VqfContext),
249
+    vqf_probe,
250
+    vqf_read_header,
251
+    vqf_read_packet,
252
+    NULL,
253
+    vqf_read_seek,
254
+    .extensions = "vqf",
255
+};