Mercurial > libavformat.hg
annotate wav.c @ 4200:898c3d1d9e4e libavformat
Use setmode() if it exists in <io.h>, and not based on O_BINARY.
author | ramiro |
---|---|
date | Sun, 11 Jan 2009 22:05:43 +0000 |
parents | 549a09cf23fe |
children | c3102b189cb6 |
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 { | |
3973
549a09cf23fe
Remove offset_t typedef and use int64_t directly instead.
diego
parents:
3934
diff
changeset
|
26 int64_t data; |
549a09cf23fe
Remove offset_t typedef and use int64_t directly instead.
diego
parents:
3934
diff
changeset
|
27 int64_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; |
3973
549a09cf23fe
Remove offset_t typedef and use int64_t directly instead.
diego
parents:
3934
diff
changeset
|
38 int64_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", |
3934 | 48 s->streams[0]->codec->codec ? s->streams[0]->codec->codec->name : "NONE"); |
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; |
3973
549a09cf23fe
Remove offset_t typedef and use int64_t directly instead.
diego
parents:
3934
diff
changeset
|
91 int64_t file_size; |
0 | 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 |