1use std::cmp;
14use std::fs;
15use std::io;
16use std::marker;
17use std::path;
18use super::{Error, Result, Sample, SampleFormat, WavSpec, WavSpecEx};
19
20pub trait ReadExt: io::Read {
24 fn read_into(&mut self, buf: &mut [u8]) -> io::Result<()>;
30
31 fn read_4_bytes(&mut self) -> io::Result<[u8; 4]>;
33
34 fn skip_bytes(&mut self, n: usize) -> io::Result<()>;
36
37 fn read_i8(&mut self) -> io::Result<i8>;
39
40 fn read_u8(&mut self) -> io::Result<u8>;
42
43 fn read_le_i16(&mut self) -> io::Result<i16>;
45
46 fn read_le_u16(&mut self) -> io::Result<u16>;
48
49 fn read_le_i24(&mut self) -> io::Result<i32>;
53
54 fn read_le_i24_4(&mut self) -> io::Result<i32>;
58
59 fn read_le_u24(&mut self) -> io::Result<u32>;
63
64 fn read_le_i32(&mut self) -> io::Result<i32>;
66
67 fn read_le_u32(&mut self) -> io::Result<u32>;
69
70 fn read_le_f32(&mut self) -> io::Result<f32>;
72}
73
74impl<R> ReadExt for R
75 where R: io::Read
76{
77 #[inline(always)]
78 fn read_into(&mut self, buf: &mut [u8]) -> io::Result<()> {
79 let mut n = 0;
80 while n < buf.len() {
81 let progress = try!(self.read(&mut buf[n..]));
82 if progress > 0 {
83 n += progress;
84 } else {
85 return Err(io::Error::new(io::ErrorKind::Other, "Failed to read enough bytes."));
86 }
87 }
88 Ok(())
89 }
90
91 #[inline(always)]
92 fn skip_bytes(&mut self, n: usize) -> io::Result<()> {
93 let mut n_read = 0;
98 let mut buf = [0u8; 1024];
99 while n_read < n {
100 let end = cmp::min(n - n_read, 1024);
101 let progress = try!(self.read(&mut buf[0..end]));
102 if progress > 0 {
103 n_read += progress;
104 } else {
105 return Err(io::Error::new(io::ErrorKind::Other, "Failed to read enough bytes."));
106 }
107 }
108 Ok(())
109 }
110
111 #[inline(always)]
112 fn read_4_bytes(&mut self) -> io::Result<[u8; 4]> {
113 let mut buf = [0_u8; 4];
114 try!(self.read_into(&mut buf[..]));
115 Ok(buf)
116 }
117
118 #[inline(always)]
119 fn read_i8(&mut self) -> io::Result<i8> {
120 self.read_u8().map(|x| x as i8)
121 }
122
123 #[inline(always)]
124 fn read_u8(&mut self) -> io::Result<u8> {
125 let mut buf = [0u8; 1];
126 try!(self.read_into(&mut buf));
127 Ok(buf[0])
128 }
129
130 #[inline(always)]
131 fn read_le_i16(&mut self) -> io::Result<i16> {
132 self.read_le_u16().map(|x| x as i16)
133 }
134
135 #[inline(always)]
136 fn read_le_u16(&mut self) -> io::Result<u16> {
137 let mut buf = [0u8; 2];
138 try!(self.read_into(&mut buf));
139 Ok((buf[1] as u16) << 8 | (buf[0] as u16))
140 }
141
142 #[inline(always)]
143 fn read_le_i24(&mut self) -> io::Result<i32> {
144 self.read_le_u24().map(|x|
145 if x & (1 << 23) == 0 {
148 x as i32
149 } else {
150 (x | 0xff_00_00_00) as i32
151 }
152 )
153 }
154
155 #[inline(always)]
156 fn read_le_i24_4(&mut self) -> io::Result<i32> {
157 self.read_le_u32().map(|x|
158 if x & (1 << 23) == 0 {
161 (x & 0x00_ff_ff_ff) as i32
162 } else {
163 (x | 0xff_00_00_00) as i32
164 }
165 )
166 }
167
168 #[inline(always)]
169 fn read_le_u24(&mut self) -> io::Result<u32> {
170 let mut buf = [0u8; 3];
171 try!(self.read_into(&mut buf));
172 Ok((buf[2] as u32) << 16 | (buf[1] as u32) << 8 | (buf[0] as u32))
173 }
174
175 #[inline(always)]
176 fn read_le_i32(&mut self) -> io::Result<i32> {
177 self.read_le_u32().map(|x| x as i32)
178 }
179
180 #[inline(always)]
181 fn read_le_u32(&mut self) -> io::Result<u32> {
182 let mut buf = [0u8; 4];
183 try!(self.read_into(&mut buf));
184 Ok((buf[3] as u32) << 24 | (buf[2] as u32) << 16 |
185 (buf[1] as u32) << 8 | (buf[0] as u32) << 0)
186 }
187
188 #[inline(always)]
189 fn read_le_f32(&mut self) -> io::Result<f32> {
190 let mut buf = [0u8; 4];
191 try!(self.read_into(&mut buf));
192 Ok(f32::from_le_bytes(buf))
193 }
194}
195
196enum ChunkKind {
198 Fmt,
199 Fact,
200 Data,
201 Unknown,
202}
203
204struct ChunkHeader {
206 pub kind: ChunkKind,
207 pub len: u32,
208}
209
210pub struct WavReader<R> {
218 spec: WavSpec,
220
221 bytes_per_sample: u16,
223
224 num_samples: u32,
230
231 samples_read: u32,
233
234 reader: R,
236}
237
238pub struct WavSamples<'wr, R, S>
243 where R: 'wr
244{
245 reader: &'wr mut WavReader<R>,
246 phantom_sample: marker::PhantomData<S>,
247}
248
249pub struct WavIntoSamples<R, S> {
254 reader: WavReader<R>,
255 phantom_sample: marker::PhantomData<S>,
256}
257
258pub fn read_wave_header<R: io::Read>(reader: &mut R) -> Result<u64> {
267 if b"RIFF" != &try!(reader.read_4_bytes())[..] {
273 return Err(Error::FormatError("no RIFF tag found"));
274 }
275
276 let file_len = try!(reader.read_le_u32());
277
278 if b"WAVE" != &try!(reader.read_4_bytes())[..] {
280 return Err(Error::FormatError("no WAVE tag found"));
281 }
282
283 Ok(file_len as u64 + 8)
286}
287
288pub fn read_until_data<R: io::Read>(mut reader: R) -> Result<(WavSpecEx, u32)> {
294 let mut spec_opt = None;
295
296 loop {
297 let header = try!(WavReader::read_chunk_header(&mut reader));
298 match header.kind {
299 ChunkKind::Fmt => {
300 let spec = try!(WavReader::read_fmt_chunk(&mut reader, header.len));
301 spec_opt = Some(spec);
302 }
303 ChunkKind::Fact => {
304 let _samples_per_channel = reader.read_le_u32();
315 }
316 ChunkKind::Data => {
317 if let Some(spec) = spec_opt {
320 return Ok((spec, header.len));
321 } else {
322 return Err(Error::FormatError("missing fmt chunk"));
323 }
324 }
325 ChunkKind::Unknown => {
326 try!(reader.skip_bytes(header.len as usize));
328 }
329 }
330 }
333}
334
335impl<R> WavReader<R>
336 where R: io::Read
337{
338 fn read_chunk_header(reader: &mut R) -> Result<ChunkHeader> {
340 let mut kind_str = [0; 4];
341 try!(reader.read_into(&mut kind_str));
342 let len = try!(reader.read_le_u32());
343
344 let kind = match &kind_str[..] {
345 b"fmt " => ChunkKind::Fmt,
346 b"fact" => ChunkKind::Fact,
347 b"data" => ChunkKind::Data,
348 _ => ChunkKind::Unknown,
349 };
350
351 Ok(ChunkHeader { kind: kind, len: len })
352 }
353
354 fn read_fmt_chunk(reader: &mut R, chunk_len: u32) -> Result<WavSpecEx> {
356 if chunk_len < 16 {
361 return Err(Error::FormatError("invalid fmt chunk size"));
362 }
363
364 let format_tag = try!(reader.read_le_u16());
402 let n_channels = try!(reader.read_le_u16());
403 let n_samples_per_sec = try!(reader.read_le_u32());
404 let n_bytes_per_sec = try!(reader.read_le_u32());
405 let block_align = try!(reader.read_le_u16());
406 let bits_per_sample = try!(reader.read_le_u16());
407
408 if n_channels == 0 {
409 return Err(Error::FormatError("file contains zero channels"));
410 }
411
412 let bytes_per_sample = block_align / n_channels;
413 if Some(bits_per_sample) > bytes_per_sample.checked_mul(8) {
416 return Err(Error::FormatError("sample bits exceeds size of sample"));
417 }
418
419 if Some(n_bytes_per_sec) != (block_align as u32).checked_mul(n_samples_per_sec) {
422 return Err(Error::FormatError("inconsistent fmt chunk"));
423 }
424
425 if bits_per_sample % 8 != 0 {
428 return Err(Error::FormatError("bits per sample is not a multiple of 8"));
429 }
430
431 if bits_per_sample == 0 {
432 return Err(Error::FormatError("bits per sample is 0"));
433 }
434
435 let mut spec = WavSpec {
436 channels: n_channels,
437 sample_rate: n_samples_per_sec,
438 bits_per_sample: bits_per_sample,
439 sample_format: SampleFormat::Int,
440 };
441
442 const PCM: u16 = 0x0001;
447 const ADPCM: u16 = 0x0002;
448 const IEEE_FLOAT: u16 = 0x0003;
449 const EXTENSIBLE: u16 = 0xfffe;
450 match format_tag {
452 PCM => try!(WavReader::read_wave_format_pcm(reader, chunk_len, &spec)),
453 ADPCM => return Err(Error::Unsupported),
454 IEEE_FLOAT => try!(WavReader::read_wave_format_ieee_float(reader, chunk_len, &mut spec)),
455 EXTENSIBLE => try!(WavReader::read_wave_format_extensible(reader, chunk_len, &mut spec)),
456 _ => return Err(Error::Unsupported),
457 };
458
459 Ok(WavSpecEx {
460 spec: spec,
461 bytes_per_sample: bytes_per_sample,
462 })
463 }
464
465 fn read_wave_format_pcm(mut reader: R, chunk_len: u32, spec: &WavSpec) -> Result<()> {
466 let is_wave_format_ex = match chunk_len {
469 16 => false,
470 18 => true,
471 40 => true,
474 _ => return Err(Error::FormatError("unexpected fmt chunk size")),
475 };
476
477 if is_wave_format_ex {
478 let _cb_size = try!(reader.read_le_u16());
483
484 match spec.bits_per_sample {
491 8 => {}
492 16 => {}
493 24 => {}
494 _ => return Err(Error::FormatError("bits per sample is not 8 or 16")),
495 }
496 }
497
498 if chunk_len == 40 {
500 try!(reader.skip_bytes(22));
501 }
502 Ok(())
503 }
504
505 fn read_wave_format_ieee_float(mut reader: R, chunk_len: u32, spec: &mut WavSpec) -> Result<()> {
506 let is_wave_format_ex = chunk_len == 18;
509
510 if !is_wave_format_ex && chunk_len != 16 {
511 return Err(Error::FormatError("unexpected fmt chunk size"));
512 }
513
514 if is_wave_format_ex {
515 let cb_size = try!(reader.read_le_u16());
518 if cb_size != 0 {
519 return Err(Error::FormatError("unexpected WAVEFORMATEX size"));
520 }
521 }
522
523 if spec.bits_per_sample != 32 {
530 return Err(Error::FormatError("bits per sample is not 32"));
531 }
532
533 spec.sample_format = SampleFormat::Float;
534 Ok(())
535 }
536
537 fn read_wave_format_extensible(mut reader: R, chunk_len: u32, spec: &mut WavSpec) -> Result<()> {
538 if chunk_len < 40 {
542 return Err(Error::FormatError("unexpected fmt chunk size"));
543 }
544
545 let cb_size = try!(reader.read_le_u16());
547
548 if cb_size != 22 {
551 return Err(Error::FormatError("unexpected WAVEFORMATEXTENSIBLE size"));
552 }
553
554 let valid_bits_per_sample = try!(reader.read_le_u16());
569 let _channel_mask = try!(reader.read_le_u32()); let mut subformat = [0u8; 16];
571 try!(reader.read_into(&mut subformat));
572
573 let sample_format = match subformat {
578 super::KSDATAFORMAT_SUBTYPE_PCM => SampleFormat::Int,
579 super::KSDATAFORMAT_SUBTYPE_IEEE_FLOAT => SampleFormat::Float,
580 _ => return Err(Error::Unsupported),
581 };
582
583 if valid_bits_per_sample > 0 {
585 spec.bits_per_sample = valid_bits_per_sample;
586 }
587
588 spec.sample_format = sample_format;
589 Ok(())
590 }
591
592 pub fn new(mut reader: R) -> Result<WavReader<R>> {
597 try!(read_wave_header(&mut reader));
598 let (spec_ex, data_len) = try!(read_until_data(&mut reader));
599
600 let num_samples = data_len / spec_ex.bytes_per_sample as u32;
601
602 if num_samples * spec_ex.bytes_per_sample as u32 != data_len {
608 let msg = "data chunk length is not a multiple of sample size";
609 return Err(Error::FormatError(msg));
610 }
611
612 if num_samples % spec_ex.spec.channels as u32 != 0 {
616 return Err(Error::FormatError("invalid data chunk length"));
617 }
618
619 let wav_reader = WavReader {
620 spec: spec_ex.spec,
621 bytes_per_sample: spec_ex.bytes_per_sample,
622 num_samples: num_samples,
623 samples_read: 0,
624 reader: reader,
625 };
626
627 Ok(wav_reader)
628 }
629
630 pub fn spec(&self) -> WavSpec {
632 self.spec
633 }
634
635 pub fn samples<'wr, S: Sample>(&'wr mut self) -> WavSamples<'wr, R, S> {
651 WavSamples {
652 reader: self,
653 phantom_sample: marker::PhantomData,
654 }
655 }
656
657 pub fn into_samples<S: Sample>(self) -> WavIntoSamples<R, S> {
661 WavIntoSamples {
662 reader: self,
663 phantom_sample: marker::PhantomData,
664 }
665 }
666
667 pub fn duration(&self) -> u32 {
674 self.num_samples / self.spec.channels as u32
675 }
676
677 pub fn len(&self) -> u32 {
684 self.num_samples
685 }
686
687 pub fn into_inner(self) -> R {
689 self.reader
690 }
691
692 pub fn seek(&mut self, time: u32) -> io::Result<()>
703 where R: io::Seek,
704 {
705 let bytes_per_sample = self.spec.bits_per_sample / 8;
706 let sample_position = time * self.spec.channels as u32;
707 let offset_samples = sample_position as i64 - self.samples_read as i64;
708 let offset_bytes = offset_samples * bytes_per_sample as i64;
709 try!(self.reader.seek(io::SeekFrom::Current(offset_bytes)));
710 self.samples_read = sample_position;
711 Ok(())
712 }
713}
714
715impl WavReader<io::BufReader<fs::File>> {
716 pub fn open<P: AsRef<path::Path>>(filename: P) -> Result<WavReader<io::BufReader<fs::File>>> {
721 let file = try!(fs::File::open(filename));
722 let buf_reader = io::BufReader::new(file);
723 WavReader::new(buf_reader)
724 }
725}
726
727fn iter_next<R, S>(reader: &mut WavReader<R>) -> Option<Result<S>>
728 where R: io::Read,
729 S: Sample
730{
731 if reader.samples_read < reader.num_samples {
732 reader.samples_read += 1;
733 let sample = Sample::read(&mut reader.reader,
734 reader.spec.sample_format,
735 reader.bytes_per_sample,
736 reader.spec.bits_per_sample);
737 Some(sample.map_err(Error::from))
738 } else {
739 None
740 }
741}
742
743fn iter_size_hint<R>(reader: &WavReader<R>) -> (usize, Option<usize>) {
744 let samples_left = reader.num_samples - reader.samples_read;
745 (samples_left as usize, Some(samples_left as usize))
746}
747
748impl<'wr, R, S> Iterator for WavSamples<'wr, R, S>
749 where R: io::Read,
750 S: Sample
751{
752 type Item = Result<S>;
753
754 fn next(&mut self) -> Option<Result<S>> {
755 iter_next(&mut self.reader)
756 }
757
758 fn size_hint(&self) -> (usize, Option<usize>) {
759 iter_size_hint(&self.reader)
760 }
761}
762
763impl<'wr, R, S> ExactSizeIterator for WavSamples<'wr, R, S>
764 where R: io::Read,
765 S: Sample
766{
767}
768
769impl<R, S> Iterator for WavIntoSamples<R, S>
770 where R: io::Read,
771 S: Sample
772{
773 type Item = Result<S>;
774
775 fn next(&mut self) -> Option<Result<S>> {
776 iter_next(&mut self.reader)
777 }
778
779 fn size_hint(&self) -> (usize, Option<usize>) {
780 iter_size_hint(&self.reader)
781 }
782}
783
784impl<R, S> ExactSizeIterator for WavIntoSamples<R, S>
785 where R: io::Read,
786 S: Sample
787{
788}
789
790#[test]
791fn duration_and_len_agree() {
792 let files = &["testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
793 "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
794 "testsamples/waveformatextensible-32bit-48kHz-stereo.wav"];
795
796 for fname in files {
797 let reader = WavReader::open(fname).unwrap();
798 assert_eq!(reader.spec().channels as u32 * reader.duration(),
799 reader.len());
800 }
801}
802
803#[test]
805fn read_wav_pcm_wave_format_pcm() {
806 let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-16bit-44100Hz-mono.wav")
807 .unwrap();
808
809 assert_eq!(wav_reader.spec().channels, 1);
810 assert_eq!(wav_reader.spec().sample_rate, 44100);
811 assert_eq!(wav_reader.spec().bits_per_sample, 16);
812 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
813
814 let samples: Vec<i16> = wav_reader.samples()
815 .map(|r| r.unwrap())
816 .collect();
817
818 assert_eq!(&samples[..], &[2, -3, 5, -7]);
820}
821
822#[test]
823fn read_wav_skips_unknown_chunks() {
824 let files = ["testsamples/pcmwaveformat-16bit-44100Hz-mono-extra.wav",
827 "testsamples/waveformatex-16bit-44100Hz-mono-extra.wav"];
828
829 for file in &files {
830 let mut wav_reader = WavReader::open(file).unwrap();
831
832 assert_eq!(wav_reader.spec().channels, 1);
833 assert_eq!(wav_reader.spec().sample_rate, 44100);
834 assert_eq!(wav_reader.spec().bits_per_sample, 16);
835 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
836
837 let sample = wav_reader.samples::<i16>().next().unwrap().unwrap();
838 assert_eq!(sample, 2);
839 }
840}
841
842#[test]
843fn read_wav_0_valid_bits_fallback() {
844 let mut wav_reader = WavReader::open("testsamples/nonstandard-02.wav")
845 .unwrap();
846
847 assert_eq!(wav_reader.spec().channels, 2);
848 assert_eq!(wav_reader.spec().sample_rate, 48000);
849 assert_eq!(wav_reader.spec().bits_per_sample, 32);
850 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
851
852 let samples: Vec<i32> = wav_reader.samples()
853 .map(|r| r.unwrap())
854 .collect();
855
856 assert_eq!(&samples[..], &[19, -229373, 33587161, -2147483497]);
858}
859
860#[test]
861fn len_and_size_hint_are_correct() {
862 let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-16bit-44100Hz-mono.wav")
863 .unwrap();
864
865 assert_eq!(wav_reader.len(), 4);
866
867 {
868 let mut samples = wav_reader.samples::<i16>();
869
870 assert_eq!(samples.size_hint(), (4, Some(4)));
871 samples.next();
872 assert_eq!(samples.size_hint(), (3, Some(3)));
873 }
874
875 assert_eq!(wav_reader.len(), 4);
877
878 {
880 let mut samples = wav_reader.samples::<i16>();
881
882 assert_eq!(samples.size_hint(), (3, Some(3)));
883 samples.next();
884 assert_eq!(samples.size_hint(), (2, Some(2)));
885 }
886}
887
888#[test]
889fn size_hint_is_exact() {
890 let files = &["testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
891 "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
892 "testsamples/waveformatextensible-32bit-48kHz-stereo.wav"];
893
894 for fname in files {
895 let mut reader = WavReader::open(fname).unwrap();
896 let len = reader.len();
897 let mut iter = reader.samples::<i32>();
898 for i in 0..len {
899 let remaining = (len - i) as usize;
900 assert_eq!(iter.size_hint(), (remaining, Some(remaining)));
901 assert!(iter.next().is_some());
902 }
903 assert!(iter.next().is_none());
904 }
905}
906
907#[test]
908fn samples_equals_into_samples() {
909 let wav_reader_val = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav").unwrap();
910 let mut wav_reader_ref = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav").unwrap();
911
912 let samples_val: Vec<i16> = wav_reader_val.into_samples()
913 .map(|r| r.unwrap())
914 .collect();
915
916 let samples_ref: Vec<i16> = wav_reader_ref.samples()
917 .map(|r| r.unwrap())
918 .collect();
919
920 assert_eq!(samples_val, samples_ref);
921}
922
923#[test]
925fn read_wav_wave_format_ex_pcm() {
926 let mut wav_reader = WavReader::open("testsamples/waveformatex-16bit-44100Hz-mono.wav")
927 .unwrap();
928
929 assert_eq!(wav_reader.spec().channels, 1);
930 assert_eq!(wav_reader.spec().sample_rate, 44100);
931 assert_eq!(wav_reader.spec().bits_per_sample, 16);
932 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
933
934 let samples: Vec<i16> = wav_reader.samples()
935 .map(|r| r.unwrap())
936 .collect();
937
938 assert_eq!(&samples[..], &[2, -3, 5, -7]);
940}
941
942#[test]
943fn read_wav_wave_format_ex_ieee_float() {
944 let mut wav_reader = WavReader::open("testsamples/waveformatex-ieeefloat-44100Hz-mono.wav")
945 .unwrap();
946
947 assert_eq!(wav_reader.spec().channels, 1);
948 assert_eq!(wav_reader.spec().sample_rate, 44100);
949 assert_eq!(wav_reader.spec().bits_per_sample, 32);
950 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Float);
951
952 let samples: Vec<f32> = wav_reader.samples()
953 .map(|r| r.unwrap())
954 .collect();
955
956 assert_eq!(&samples[..], &[2.0, 3.0, -16411.0, 1019.0]);
958}
959
960#[test]
961fn read_wav_stereo() {
962 let mut wav_reader = WavReader::open("testsamples/waveformatex-16bit-44100Hz-stereo.wav")
963 .unwrap();
964
965 assert_eq!(wav_reader.spec().channels, 2);
966 assert_eq!(wav_reader.spec().sample_rate, 44100);
967 assert_eq!(wav_reader.spec().bits_per_sample, 16);
968 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
969
970 let samples: Vec<i16> = wav_reader.samples()
971 .map(|r| r.unwrap())
972 .collect();
973
974 assert_eq!(&samples[..], &[2, -3, 5, -7, 11, -13, 17, -19]);
976
977}
978
979#[test]
980fn read_wav_pcm_wave_format_8bit() {
981 let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav")
982 .unwrap();
983
984 assert_eq!(wav_reader.spec().channels, 1);
985 assert_eq!(wav_reader.spec().bits_per_sample, 8);
986 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
987
988 let samples: Vec<i16> = wav_reader.samples()
989 .map(|r| r.unwrap())
990 .collect();
991
992 assert_eq!(&samples[..], &[19, -53, 89, -127]);
994}
995
996#[test]
1000fn read_wav_pcm_wave_format_24bit_4byte() {
1001 let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-24bit-4byte-48kHz-stereo.wav")
1002 .unwrap();
1003
1004 assert_eq!(wav_reader.spec().channels, 2);
1005 assert_eq!(wav_reader.spec().sample_rate, 48_000);
1006 assert_eq!(wav_reader.spec().bits_per_sample, 24);
1007 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
1008
1009 let samples: Vec<i32> = wav_reader.samples()
1010 .map(|r| r.unwrap())
1011 .collect();
1012
1013 assert_eq!(&samples[..], &[-96, 23_052, 8_388_607, -8_360_672]);
1015}
1016
1017#[test]
1019fn read_wav_wave_format_ex_8bit() {
1020 let mut wav_reader = WavReader::open("testsamples/waveformatex-8bit-11025Hz-mono.wav").unwrap();
1021
1022 assert_eq!(wav_reader.spec().channels, 1);
1023 assert_eq!(wav_reader.spec().bits_per_sample, 8);
1024 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
1025
1026 let samples: Vec<i32> = wav_reader.samples()
1027 .map(|r| r.unwrap())
1028 .collect();
1029
1030 assert_eq!(&samples[..], &[-128, -128, -128, -128]);
1033}
1034
1035#[test]
1037fn read_wav_wave_format_extensible_pcm_24bit() {
1038 let mut wav_reader = WavReader::open("testsamples/waveformatextensible-24bit-192kHz-mono.wav")
1039 .unwrap();
1040
1041 assert_eq!(wav_reader.spec().channels, 1);
1042 assert_eq!(wav_reader.spec().sample_rate, 192_000);
1043 assert_eq!(wav_reader.spec().bits_per_sample, 24);
1044 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
1045
1046 let samples: Vec<i32> = wav_reader.samples()
1047 .map(|r| r.unwrap())
1048 .collect();
1049
1050 assert_eq!(&samples[..], &[-17, 4_194_319, -6_291_437, 8_355_817]);
1052}
1053
1054#[test]
1057fn read_wav_wave_format_extensible_pcm_24bit_4byte() {
1058 let mut wav_reader = WavReader::open("testsamples/waveformatextensible-24bit-4byte-48kHz-stereo.wav")
1059 .unwrap();
1060
1061 assert_eq!(wav_reader.spec().channels, 2);
1062 assert_eq!(wav_reader.spec().sample_rate, 48_000);
1063 assert_eq!(wav_reader.spec().bits_per_sample, 24);
1064 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
1065
1066 let samples: Vec<i32> = wav_reader.samples()
1067 .map(|r| r.unwrap())
1068 .collect();
1069
1070 assert_eq!(&samples[..], &[-96, 23_052, 8_388_607, -8_360_672]);
1072}
1073
1074#[test]
1075fn read_wav_32bit() {
1076 let mut wav_reader = WavReader::open("testsamples/waveformatextensible-32bit-48kHz-stereo.wav")
1077 .unwrap();
1078
1079 assert_eq!(wav_reader.spec().bits_per_sample, 32);
1080 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
1081
1082 let samples: Vec<i32> = wav_reader.samples()
1083 .map(|r| r.unwrap())
1084 .collect();
1085
1086 assert_eq!(&samples[..], &[19, -229_373, 33_587_161, -2_147_483_497]);
1088}
1089
1090#[test]
1091fn read_wav_wave_format_extensible_ieee_float() {
1092 let mut wav_reader =
1093 WavReader::open("testsamples/waveformatextensible-ieeefloat-44100Hz-mono.wav").unwrap();
1094
1095 assert_eq!(wav_reader.spec().channels, 1);
1096 assert_eq!(wav_reader.spec().sample_rate, 44100);
1097 assert_eq!(wav_reader.spec().bits_per_sample, 32);
1098 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Float);
1099
1100 let samples: Vec<f32> = wav_reader.samples()
1101 .map(|r| r.unwrap())
1102 .collect();
1103
1104 assert_eq!(&samples[..], &[2.0, 3.0, -16411.0, 1019.0]);
1106}
1107
1108#[test]
1109fn read_wav_nonstandard_01() {
1110 let mut wav_reader = WavReader::open("testsamples/nonstandard-01.wav").unwrap();
1120
1121 assert_eq!(wav_reader.spec().bits_per_sample, 24);
1122 assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
1123
1124 let samples: Vec<i32> = wav_reader.samples()
1125 .map(|r| r.unwrap())
1126 .collect();
1127
1128 assert_eq!(&samples[..], &[0, 0]);
1129}
1130
1131#[test]
1132fn wide_read_should_signal_error() {
1133 let mut reader24 = WavReader::open("testsamples/waveformatextensible-24bit-192kHz-mono.wav")
1134 .unwrap();
1135
1136 assert!(reader24.samples::<i8>().next().unwrap().is_err());
1140 assert!(reader24.samples::<i16>().next().unwrap().is_err());
1141 assert!(reader24.samples::<i32>().next().unwrap().is_ok());
1142
1143 let mut reader32 = WavReader::open("testsamples/waveformatextensible-32bit-48kHz-stereo.wav")
1144 .unwrap();
1145
1146 assert!(reader32.samples::<i8>().next().unwrap().is_err());
1148 assert!(reader32.samples::<i16>().next().unwrap().is_err());
1149 assert!(reader32.samples::<i32>().next().unwrap().is_ok());
1150}
1151
1152#[test]
1153fn sample_format_mismatch_should_signal_error() {
1154 let mut reader_f32 = WavReader::open("testsamples/waveformatex-ieeefloat-44100Hz-mono.wav")
1155 .unwrap();
1156
1157 assert!(reader_f32.samples::<i8>().next().unwrap().is_err());
1158 assert!(reader_f32.samples::<i16>().next().unwrap().is_err());
1159 assert!(reader_f32.samples::<i32>().next().unwrap().is_err());
1160 assert!(reader_f32.samples::<f32>().next().unwrap().is_ok());
1161
1162 let mut reader_i8 = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav").unwrap();
1163
1164 assert!(reader_i8.samples::<i8>().next().unwrap().is_ok());
1165 assert!(reader_i8.samples::<i16>().next().unwrap().is_ok());
1166 assert!(reader_i8.samples::<i32>().next().unwrap().is_ok());
1167 assert!(reader_i8.samples::<f32>().next().unwrap().is_err());
1168}
1169
1170#[test]
1171fn fuzz_crashes_should_be_fixed() {
1172 use std::fs;
1173 use std::ffi::OsStr;
1174
1175 let dir = fs::read_dir("testsamples/fuzz").ok()
1178 .expect("failed to enumerate fuzz test corpus");
1179 for path in dir {
1180 let path = path.ok().expect("failed to obtain path info").path();
1181 let is_file = fs::metadata(&path).unwrap().file_type().is_file();
1182 if is_file && path.extension() == Some(OsStr::new("wav")) {
1183 println!(" testing {} ...", path.to_str()
1184 .expect("unsupported filename"));
1185 let mut reader = match WavReader::open(path) {
1186 Ok(r) => r,
1187 Err(..) => continue,
1188 };
1189 match reader.spec().sample_format {
1190 SampleFormat::Int => {
1191 for sample in reader.samples::<i32>() {
1192 match sample {
1193 Ok(..) => { }
1194 Err(..) => break,
1195 }
1196 }
1197 }
1198 SampleFormat::Float => {
1199 for sample in reader.samples::<f32>() {
1200 match sample {
1201 Ok(..) => { }
1202 Err(..) => break,
1203 }
1204 }
1205 }
1206 }
1207 }
1208 }
1209}
1210
1211#[test]
1212fn seek_is_consistent() {
1213 let files = &["testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
1214 "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
1215 "testsamples/waveformatextensible-32bit-48kHz-stereo.wav"];
1216 for fname in files {
1217 let mut reader = WavReader::open(fname).unwrap();
1218
1219 let count = reader.samples::<i32>().count();
1221 reader.seek(0).unwrap();
1222 assert_eq!(reader.samples_read, 0);
1223 assert_eq!(count, reader.samples::<i32>().count());
1224
1225 let last_time = reader.duration() - 1;
1227 let channels = reader.spec.channels;
1228 reader.seek(last_time).unwrap();
1229 {
1230 let mut samples = reader.samples::<i32>();
1231 for _ in 0..channels {
1232 assert!(samples.next().is_some());
1233 }
1234 assert!(samples.next().is_none());
1235 }
1236
1237 let num_samples = reader.len();
1239 reader.seek(num_samples).unwrap();
1240 assert!(reader.samples::<i32>().next().is_none());
1241 reader.seek(::std::u32::MAX / channels as u32).unwrap();
1242 assert!(reader.samples::<i32>().next().is_none());
1243 }
1244}