Mercurial > libavformat.hg
annotate wav.c @ 3925:dfd3033e7de7 libavformat
add error message for unsupported codecs in WAVE.
Resolves issue 623.
| author | jbr |
|---|---|
| date | Sun, 14 Sep 2008 17:29:34 +0000 |
| parents | e6aeb2733e34 |
| children | 1aa5e41ceaef |
| rev | line source |
|---|---|
| 885 | 1 /* |
|
1415
3b00fb8ef8e4
replace coder/decoder file description in libavformat by muxer/demuxer
aurel
parents:
1358
diff
changeset
|
2 * WAV muxer and demuxer |
| 0 | 3 * Copyright (c) 2001, 2002 Fabrice Bellard. |
| 4 * | |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
1172
diff
changeset
|
5 * This file is part of FFmpeg. |
|
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
1172
diff
changeset
|
6 * |
|
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
1172
diff
changeset
|
7 * FFmpeg is free software; you can redistribute it and/or |
| 0 | 8 * modify it under the terms of the GNU Lesser General Public |
| 9 * License as published by the Free Software Foundation; either | |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
1172
diff
changeset
|
10 * version 2.1 of the License, or (at your option) any later version. |
| 0 | 11 * |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
1172
diff
changeset
|
12 * FFmpeg is distributed in the hope that it will be useful, |
| 0 | 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
| 15 * Lesser General Public License for more details. | |
| 16 * | |
| 17 * You should have received a copy of the GNU Lesser General Public | |
|
1358
0899bfe4105c
Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents:
1172
diff
changeset
|
18 * License along with FFmpeg; if not, write to the Free Software |
|
896
edbe5c3717f9
Update licensing information: The FSF changed postal address.
diego
parents:
893
diff
changeset
|
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| 0 | 20 */ |
| 21 #include "avformat.h" | |
|
2545
213268d7594e
move unrelated functions declarations out of allformats.h
aurel
parents:
2274
diff
changeset
|
22 #include "raw.h" |
|
1172
6a5e58d2114b
move common stuff from avienc.c and wav.c to new file riff.c
mru
parents:
1169
diff
changeset
|
23 #include "riff.h" |
| 0 | 24 |
| 25 typedef struct { | |
| 26 offset_t data; | |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
27 offset_t data_end; |
| 1781 | 28 int64_t minpts; |
| 29 int64_t maxpts; | |
| 30 int last_duration; | |
| 0 | 31 } WAVContext; |
| 32 | |
|
3871
e6aeb2733e34
Replace generic CONFIG_MUXERS preprocessor conditionals by more specific
diego
parents:
3766
diff
changeset
|
33 #ifdef CONFIG_WAV_MUXER |
| 0 | 34 static int wav_write_header(AVFormatContext *s) |
| 35 { | |
| 36 WAVContext *wav = s->priv_data; | |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
37 ByteIOContext *pb = s->pb; |
| 1781 | 38 offset_t fmt, fact; |
| 0 | 39 |
| 40 put_tag(pb, "RIFF"); | |
| 41 put_le32(pb, 0); /* file length */ | |
| 42 put_tag(pb, "WAVE"); | |
| 43 | |
| 44 /* format header */ | |
| 45 fmt = start_tag(pb, "fmt "); | |
|
820
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
46 if (put_wav_header(pb, s->streams[0]->codec) < 0) { |
| 3925 | 47 av_log(s, AV_LOG_ERROR, "%s codec not supported in WAVE format\n", |
| 48 s->streams[0]->codec->codec->name); | |
| 0 | 49 av_free(wav); |
| 50 return -1; | |
| 51 } | |
| 52 end_tag(pb, fmt); | |
| 53 | |
| 1781 | 54 if(s->streams[0]->codec->codec_tag != 0x01 /* hence for all other than PCM */ |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
55 && !url_is_streamed(s->pb)) { |
| 1781 | 56 fact = start_tag(pb, "fact"); |
| 57 put_le32(pb, 0); | |
| 58 end_tag(pb, fact); | |
| 59 } | |
| 60 | |
|
820
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
61 av_set_pts_info(s->streams[0], 64, 1, s->streams[0]->codec->sample_rate); |
| 1781 | 62 wav->maxpts = wav->last_duration = 0; |
| 63 wav->minpts = INT64_MAX; | |
|
645
9fc2d2cc4608
wav timestamp truncation fix by (Wolfram Gloger <wmglo dent.med.uni-muenchen de>)
michael
parents:
587
diff
changeset
|
64 |
| 0 | 65 /* data header */ |
| 66 wav->data = start_tag(pb, "data"); | |
| 885 | 67 |
| 0 | 68 put_flush_packet(pb); |
| 69 | |
| 70 return 0; | |
| 71 } | |
| 72 | |
| 468 | 73 static int wav_write_packet(AVFormatContext *s, AVPacket *pkt) |
| 0 | 74 { |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
75 ByteIOContext *pb = s->pb; |
| 1781 | 76 WAVContext *wav = s->priv_data; |
| 468 | 77 put_buffer(pb, pkt->data, pkt->size); |
| 1781 | 78 if(pkt->pts != AV_NOPTS_VALUE) { |
| 79 wav->minpts = FFMIN(wav->minpts, pkt->pts); | |
| 80 wav->maxpts = FFMAX(wav->maxpts, pkt->pts); | |
| 81 wav->last_duration = pkt->duration; | |
| 82 } else | |
| 83 av_log(s, AV_LOG_ERROR, "wav_write_packet: NOPTS\n"); | |
| 0 | 84 return 0; |
| 85 } | |
| 86 | |
| 87 static int wav_write_trailer(AVFormatContext *s) | |
| 88 { | |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
89 ByteIOContext *pb = s->pb; |
| 0 | 90 WAVContext *wav = s->priv_data; |
| 91 offset_t file_size; | |
| 92 | |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
93 if (!url_is_streamed(s->pb)) { |
| 0 | 94 end_tag(pb, wav->data); |
| 95 | |
| 96 /* update file size */ | |
| 97 file_size = url_ftell(pb); | |
| 98 url_fseek(pb, 4, SEEK_SET); | |
| 65 | 99 put_le32(pb, (uint32_t)(file_size - 8)); |
| 0 | 100 url_fseek(pb, file_size, SEEK_SET); |
| 101 | |
| 102 put_flush_packet(pb); | |
| 1781 | 103 |
| 104 if(s->streams[0]->codec->codec_tag != 0x01) { | |
| 105 /* Update num_samps in fact chunk */ | |
| 106 int number_of_samples; | |
| 107 number_of_samples = av_rescale(wav->maxpts - wav->minpts + wav->last_duration, | |
| 108 s->streams[0]->codec->sample_rate * (int64_t)s->streams[0]->time_base.num, | |
| 109 s->streams[0]->time_base.den); | |
| 110 url_fseek(pb, wav->data-12, SEEK_SET); | |
| 111 put_le32(pb, number_of_samples); | |
| 112 url_fseek(pb, file_size, SEEK_SET); | |
| 113 put_flush_packet(pb); | |
| 114 } | |
| 0 | 115 } |
| 116 return 0; | |
| 117 } | |
|
3871
e6aeb2733e34
Replace generic CONFIG_MUXERS preprocessor conditionals by more specific
diego
parents:
3766
diff
changeset
|
118 #endif /* CONFIG_WAV_MUXER */ |
| 0 | 119 |
| 120 /* return the size of the found tag */ | |
| 121 /* XXX: > 2GB ? */ | |
| 65 | 122 static int find_tag(ByteIOContext *pb, uint32_t tag1) |
| 0 | 123 { |
| 124 unsigned int tag; | |
| 125 int size; | |
| 126 | |
| 127 for(;;) { | |
| 128 if (url_feof(pb)) | |
| 129 return -1; | |
| 130 tag = get_le32(pb); | |
| 131 size = get_le32(pb); | |
| 132 if (tag == tag1) | |
| 133 break; | |
| 134 url_fseek(pb, size, SEEK_CUR); | |
| 135 } | |
| 136 if (size < 0) | |
| 137 size = 0x7fffffff; | |
| 138 return size; | |
| 139 } | |
| 140 | |
| 141 static int wav_probe(AVProbeData *p) | |
| 142 { | |
| 143 /* check file header */ | |
| 144 if (p->buf_size <= 32) | |
| 145 return 0; | |
| 146 if (p->buf[0] == 'R' && p->buf[1] == 'I' && | |
| 147 p->buf[2] == 'F' && p->buf[3] == 'F' && | |
| 148 p->buf[8] == 'W' && p->buf[9] == 'A' && | |
| 149 p->buf[10] == 'V' && p->buf[11] == 'E') | |
|
3233
e2bdb989f7da
Decrease returning probe score for WAV demuxer to avoid
voroshil
parents:
2771
diff
changeset
|
150 /* |
|
e2bdb989f7da
Decrease returning probe score for WAV demuxer to avoid
voroshil
parents:
2771
diff
changeset
|
151 Since ACT demuxer has standard WAV header at top of it's own, |
|
e2bdb989f7da
Decrease returning probe score for WAV demuxer to avoid
voroshil
parents:
2771
diff
changeset
|
152 returning score is decreased to avoid probe conflict |
|
e2bdb989f7da
Decrease returning probe score for WAV demuxer to avoid
voroshil
parents:
2771
diff
changeset
|
153 between ACT and WAV. |
|
e2bdb989f7da
Decrease returning probe score for WAV demuxer to avoid
voroshil
parents:
2771
diff
changeset
|
154 */ |
|
e2bdb989f7da
Decrease returning probe score for WAV demuxer to avoid
voroshil
parents:
2771
diff
changeset
|
155 return AVPROBE_SCORE_MAX - 1; |
| 0 | 156 else |
| 157 return 0; | |
| 158 } | |
| 159 | |
| 160 /* wav input */ | |
| 161 static int wav_read_header(AVFormatContext *s, | |
| 162 AVFormatParameters *ap) | |
| 163 { | |
| 164 int size; | |
| 165 unsigned int tag; | |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
166 ByteIOContext *pb = s->pb; |
| 0 | 167 AVStream *st; |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
168 WAVContext *wav = s->priv_data; |
| 0 | 169 |
| 170 /* check RIFF header */ | |
| 171 tag = get_le32(pb); | |
| 172 | |
| 173 if (tag != MKTAG('R', 'I', 'F', 'F')) | |
| 174 return -1; | |
| 175 get_le32(pb); /* file size */ | |
| 176 tag = get_le32(pb); | |
| 177 if (tag != MKTAG('W', 'A', 'V', 'E')) | |
| 178 return -1; | |
| 885 | 179 |
| 0 | 180 /* parse fmt header */ |
| 181 size = find_tag(pb, MKTAG('f', 'm', 't', ' ')); | |
| 182 if (size < 0) | |
| 183 return -1; | |
| 184 st = av_new_stream(s, 0); | |
| 185 if (!st) | |
|
2273
7eb456c4ed8a
Replace all occurrences of AVERROR_NOMEM with AVERROR(ENOMEM).
takis
parents:
2023
diff
changeset
|
186 return AVERROR(ENOMEM); |
| 0 | 187 |
|
820
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
188 get_wav_header(pb, st->codec, size); |
| 2023 | 189 st->need_parsing = AVSTREAM_PARSE_FULL; |
| 567 | 190 |
|
820
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
191 av_set_pts_info(st, 64, 1, st->codec->sample_rate); |
| 567 | 192 |
| 0 | 193 size = find_tag(pb, MKTAG('d', 'a', 't', 'a')); |
| 194 if (size < 0) | |
| 195 return -1; | |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
196 wav->data_end= url_ftell(pb) + size; |
| 0 | 197 return 0; |
| 198 } | |
| 199 | |
| 200 #define MAX_SIZE 4096 | |
| 201 | |
| 202 static int wav_read_packet(AVFormatContext *s, | |
| 203 AVPacket *pkt) | |
| 204 { | |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
205 int ret, size, left; |
| 309 | 206 AVStream *st; |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
207 WAVContext *wav = s->priv_data; |
| 0 | 208 |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
209 if (url_feof(s->pb)) |
|
2274
b21c2af60bc9
Replace all occurrences of AVERROR_IO with AVERROR(EIO).
takis
parents:
2273
diff
changeset
|
210 return AVERROR(EIO); |
| 309 | 211 st = s->streams[0]; |
| 212 | |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
213 left= wav->data_end - url_ftell(s->pb); |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
214 if(left <= 0){ |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
215 left = find_tag(s->pb, MKTAG('d', 'a', 't', 'a')); |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
216 if (left < 0) { |
|
2274
b21c2af60bc9
Replace all occurrences of AVERROR_IO with AVERROR(EIO).
takis
parents:
2273
diff
changeset
|
217 return AVERROR(EIO); |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
218 } |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
219 wav->data_end= url_ftell(s->pb) + left; |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
220 } |
|
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
221 |
| 309 | 222 size = MAX_SIZE; |
|
820
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
223 if (st->codec->block_align > 1) { |
|
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
224 if (size < st->codec->block_align) |
|
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
225 size = st->codec->block_align; |
|
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
226 size = (size / st->codec->block_align) * st->codec->block_align; |
| 309 | 227 } |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
228 size= FFMIN(size, left); |
|
2771
d52c718e83f9
Use dynamically allocated ByteIOContext in AVFormatContext
andoma
parents:
2545
diff
changeset
|
229 ret= av_get_packet(s->pb, pkt, size); |
| 1756 | 230 if (ret <= 0) |
|
2274
b21c2af60bc9
Replace all occurrences of AVERROR_IO with AVERROR(EIO).
takis
parents:
2273
diff
changeset
|
231 return AVERROR(EIO); |
| 0 | 232 pkt->stream_index = 0; |
| 233 | |
| 234 /* note: we need to modify the packet size here to handle the last | |
| 235 packet */ | |
| 236 pkt->size = ret; | |
| 237 return ret; | |
| 238 } | |
| 239 | |
| 885 | 240 static int wav_read_seek(AVFormatContext *s, |
| 558 | 241 int stream_index, int64_t timestamp, int flags) |
| 309 | 242 { |
| 243 AVStream *st; | |
| 244 | |
| 245 st = s->streams[0]; | |
|
820
feca73904e67
changing AVCodecContext codec -> *codec in AVStream so additions to AVCodecContext dont randomize AVStream and break binary compatibility
michael
parents:
817
diff
changeset
|
246 switch(st->codec->codec_id) { |
| 309 | 247 case CODEC_ID_MP2: |
| 248 case CODEC_ID_MP3: | |
| 249 case CODEC_ID_AC3: | |
|
496
112057e05179
libdts support by (Benjamin Zores <ben at geexbox dot org>)
michael
parents:
482
diff
changeset
|
250 case CODEC_ID_DTS: |
| 309 | 251 /* use generic seeking with dynamically generated indexes */ |
| 252 return -1; | |
| 253 default: | |
| 254 break; | |
| 255 } | |
| 558 | 256 return pcm_read_seek(s, stream_index, timestamp, flags); |
| 309 | 257 } |
| 258 | |
| 1169 | 259 #ifdef CONFIG_WAV_DEMUXER |
| 260 AVInputFormat wav_demuxer = { | |
| 0 | 261 "wav", |
|
3424
7a0230981402
Make long_names in lavf/lavdev optional depending on CONFIG_SMALL.
diego
parents:
3233
diff
changeset
|
262 NULL_IF_CONFIG_SMALL("WAV format"), |
|
1136
d65cd7c3573e
dont read over the end of a data chunk and at the end search for the next
michael
parents:
1122
diff
changeset
|
263 sizeof(WAVContext), |
| 0 | 264 wav_probe, |
| 265 wav_read_header, | |
| 266 wav_read_packet, | |
| 3484 | 267 NULL, |
| 309 | 268 wav_read_seek, |
| 1756 | 269 .flags= AVFMT_GENERIC_INDEX, |
|
3766
f062deeedb8d
Change codec_tag type from const struct AVCodecTag ** to const struct AVCodecTag * const *
reimar
parents:
3484
diff
changeset
|
270 .codec_tag= (const AVCodecTag* const []){codec_wav_tags, 0}, |
| 0 | 271 }; |
| 1169 | 272 #endif |
| 273 #ifdef CONFIG_WAV_MUXER | |
| 274 AVOutputFormat wav_muxer = { | |
| 0 | 275 "wav", |
|
3424
7a0230981402
Make long_names in lavf/lavdev optional depending on CONFIG_SMALL.
diego
parents:
3233
diff
changeset
|
276 NULL_IF_CONFIG_SMALL("WAV format"), |
| 0 | 277 "audio/x-wav", |
| 278 "wav", | |
| 279 sizeof(WAVContext), | |
| 280 CODEC_ID_PCM_S16LE, | |
| 281 CODEC_ID_NONE, | |
| 282 wav_write_header, | |
| 283 wav_write_packet, | |
| 284 wav_write_trailer, | |
|
3766
f062deeedb8d
Change codec_tag type from const struct AVCodecTag ** to const struct AVCodecTag * const *
reimar
parents:
3484
diff
changeset
|
285 .codec_tag= (const AVCodecTag* const []){codec_wav_tags, 0}, |
| 0 | 286 }; |
| 1169 | 287 #endif |
