Import 'hound' crate

Request Document: go/android-rust-importing-crates
For CL Reviewers: go/android3p#cl-review
For Build Team: go/ab-third-party-imports
Bug: 355564847
Test: m libhound

Change-Id: Ifb313345cde3fdbd41a2ab3e21bece7510c3307d
diff --git a/.cargo_vcs_info.json b/.cargo_vcs_info.json
new file mode 100644
index 0000000..44c7fcc
--- /dev/null
+++ b/.cargo_vcs_info.json
@@ -0,0 +1,5 @@
+{
+  "git": {
+    "sha1": "2cddb275183a6146c0dff2c758ff936d00147af1"
+  }
+}
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..133c829
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,10 @@
+# Cargo files
+/target
+
+# Editor files
+*.swp
+*.swo
+
+# Example and test output
+/append.wav
+/sine.wav
diff --git a/Android.bp b/Android.bp
new file mode 100644
index 0000000..5be1567
--- /dev/null
+++ b/Android.bp
@@ -0,0 +1,29 @@
+// This file is generated by cargo_embargo.
+// Do not modify this file because the changes will be overridden on upgrade.
+
+package {
+    default_applicable_licenses: ["external_rust_crates_hound_license"],
+}
+
+license {
+    name: "external_rust_crates_hound_license",
+    visibility: [":__subpackages__"],
+    license_kinds: ["SPDX-license-identifier-Apache-2.0"],
+    license_text: ["LICENSE"],
+}
+
+rust_library {
+    name: "libhound",
+    host_supported: true,
+    crate_name: "hound",
+    cargo_env_compat: true,
+    cargo_pkg_version: "3.5.1",
+    crate_root: "src/lib.rs",
+    edition: "2015",
+    apex_available: [
+        "//apex_available:platform",
+        "//apex_available:anyapex",
+    ],
+    product_available: true,
+    vendor_available: true,
+}
diff --git a/Cargo.lock b/Cargo.lock
new file mode 100644
index 0000000..be15e7f
--- /dev/null
+++ b/Cargo.lock
@@ -0,0 +1,100 @@
+# This file is automatically @generated by Cargo.
+# It is not intended for manual editing.
+[[package]]
+name = "alsa-sys"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+dependencies = [
+ "libc 0.2.23 (registry+https://github.com/rust-lang/crates.io-index)",
+ "pkg-config 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)",
+]
+
+[[package]]
+name = "bitflags"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+
+[[package]]
+name = "coreaudio-rs"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+dependencies = [
+ "bitflags 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)",
+ "coreaudio-sys 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
+ "libc 0.2.23 (registry+https://github.com/rust-lang/crates.io-index)",
+]
+
+[[package]]
+name = "coreaudio-sys"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+dependencies = [
+ "libc 0.2.23 (registry+https://github.com/rust-lang/crates.io-index)",
+]
+
+[[package]]
+name = "cpal"
+version = "0.2.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+dependencies = [
+ "alsa-sys 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
+ "coreaudio-rs 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)",
+ "lazy_static 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
+ "libc 0.2.23 (registry+https://github.com/rust-lang/crates.io-index)",
+ "ole32-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
+ "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)",
+]
+
+[[package]]
+name = "hound"
+version = "3.5.1"
+dependencies = [
+ "cpal 0.2.12 (registry+https://github.com/rust-lang/crates.io-index)",
+]
+
+[[package]]
+name = "lazy_static"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+
+[[package]]
+name = "libc"
+version = "0.2.23"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+
+[[package]]
+name = "ole32-sys"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+dependencies = [
+ "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)",
+ "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
+]
+
+[[package]]
+name = "pkg-config"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+
+[[package]]
+name = "winapi"
+version = "0.2.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+
+[[package]]
+name = "winapi-build"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+
+[metadata]
+"checksum alsa-sys 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "9013f855a808ab924a4c08b5c1ec9bd6b04fdb2295b4d570fb723e0ed2802a4f"
+"checksum bitflags 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "32866f4d103c4e438b1db1158aa1b1a80ee078e5d77a59a2f906fd62a577389c"
+"checksum coreaudio-rs 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)" = "7f97dd1cb4381fc1d3a80f2c63ce965c4c936befe7051fd33289a2dc45914bda"
+"checksum coreaudio-sys 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "31231897622a4cd14cb211af6f26d6fcf0c78078fa60c586ce9db8f0b581cd44"
+"checksum cpal 0.2.12 (registry+https://github.com/rust-lang/crates.io-index)" = "989df64d59c65f18776046436c3e50964699ec789e76b1af04f6fc32b6a2b713"
+"checksum lazy_static 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "4f9e1e434b35b61b6422b19998d5c219f6b4a90aae841583444b44077adb953b"
+"checksum libc 0.2.23 (registry+https://github.com/rust-lang/crates.io-index)" = "e7eb6b826bfc1fdea7935d46556250d1799b7fe2d9f7951071f4291710665e3e"
+"checksum ole32-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "5d2c49021782e5233cd243168edfa8037574afed4eba4bbaf538b3d8d1789d8c"
+"checksum pkg-config 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)" = "3a8b4c6b8165cd1a1cd4b9b120978131389f64bdaf456435caa41e630edba903"
+"checksum winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)" = "167dc9d6949a9b857f3451275e911c3f44255842c1f7a76f33c55103a909087a"
+"checksum winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2d315eee3b34aca4797b2da6b13ed88266e6d612562a0c46390af8299fc699bc"
diff --git a/Cargo.toml b/Cargo.toml
new file mode 100644
index 0000000..630af06
--- /dev/null
+++ b/Cargo.toml
@@ -0,0 +1,26 @@
+# THIS FILE IS AUTOMATICALLY GENERATED BY CARGO
+#
+# When uploading crates to the registry Cargo will automatically
+# "normalize" Cargo.toml files for maximal compatibility
+# with all versions of Cargo and also rewrite `path` dependencies
+# to registry (e.g., crates.io) dependencies
+#
+# If you believe there's an error in this file please file an
+# issue against the rust-lang/cargo repository. If you're
+# editing this file be aware that the upstream Cargo.toml
+# will likely look very different (and much more reasonable)
+
+[package]
+name = "hound"
+version = "3.5.1"
+authors = ["Ruud van Asseldonk <[email protected]>"]
+description = "A wav encoding and decoding library"
+homepage = "https://github.com/ruuda/hound"
+documentation = "https://docs.rs/hound"
+readme = "readme.md"
+keywords = ["wav", "wave", "audio", "codec"]
+categories = ["multimedia::encoding", "multimedia::audio"]
+license = "Apache-2.0"
+repository = "https://github.com/ruuda/hound"
+[dev-dependencies.cpal]
+version = "0.2.12"
diff --git a/Cargo.toml.orig b/Cargo.toml.orig
new file mode 100644
index 0000000..a6fa89a
--- /dev/null
+++ b/Cargo.toml.orig
@@ -0,0 +1,19 @@
+[package]
+
+name = "hound"
+version = "3.5.1"
+authors = ["Ruud van Asseldonk <[email protected]>"]
+description = "A wav encoding and decoding library"
+keywords = ["wav", "wave", "audio", "codec"]
+categories = ["multimedia::encoding", "multimedia::audio"]
+license = "Apache-2.0"
+readme = "readme.md"
+homepage = "https://github.com/ruuda/hound"
+repository = "https://github.com/ruuda/hound"
+documentation = "https://docs.rs/hound"
+
+[dev-dependencies]
+# An older version of cpal, but newer versions depend on the futures library,
+# which is incompatible with the version of Rust that Hound guarantees to
+# support.
+cpal = "0.2.12"
diff --git a/LICENSE b/LICENSE
new file mode 120000
index 0000000..0484eba
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1 @@
+license
\ No newline at end of file
diff --git a/METADATA b/METADATA
new file mode 100644
index 0000000..5cbf973
--- /dev/null
+++ b/METADATA
@@ -0,0 +1,20 @@
+name: "hound"
+description: "A wav encoding and decoding library"
+third_party {
+  identifier {
+    type: "crates.io"
+    value: "hound"
+  }
+  identifier {
+    type: "Archive"
+    value: "https://static.crates.io/crates/hound/hound-3.5.1.crate"
+    primary_source: true
+  }
+  version: "3.5.1"
+  license_type: NOTICE
+  last_upgrade_date {
+    year: 2024
+    month: 8
+    day: 9
+  }
+}
diff --git a/MODULE_LICENSE_APACHE2 b/MODULE_LICENSE_APACHE2
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/MODULE_LICENSE_APACHE2
diff --git a/OWNERS b/OWNERS
new file mode 100644
index 0000000..48bea6e
--- /dev/null
+++ b/OWNERS
@@ -0,0 +1,2 @@
+# Bug component: 688011
+include platform/prebuilts/rust:main:/OWNERS
diff --git a/cargo_embargo.json b/cargo_embargo.json
new file mode 100644
index 0000000..e6a2930
--- /dev/null
+++ b/cargo_embargo.json
@@ -0,0 +1,4 @@
+{
+    "run_cargo": false,
+    "tests": false
+}
diff --git a/changelog.md b/changelog.md
new file mode 100644
index 0000000..2c65b22
--- /dev/null
+++ b/changelog.md
@@ -0,0 +1,292 @@
+Changelog
+=========
+
+3.5.1
+-----
+
+Released 2023-09-25.
+
+**Compatibility**:
+
+ * Ensures compatibility with Rust 1.40.0 through 1.72.1. This bumps the minimum
+   supported Rust version from 1.16 to 1.40.
+
+Changes:
+
+ * Soundness: Wrap writes to uninitialized memory in `mem::MaybeUninit`. The
+   unsoundness was present in all versions since 0.2.0. There is no evidence
+   that rustc took advantage of the unsoundness to compile programs in a
+   problematic way.
+
+Thanks to Cam Lloyd for originally contributing these changes, and thanks to
+Maxwell McKinnon for rebasing them on top of 3.5.0.
+
+3.5.0
+-----
+
+Released 2022-09-09.
+
+This is a maintenance release that includes most of the bugfixes and features
+that have been contributed since 3.4.0, which could be cherry-picked on top of
+3.4.0. Some other contributions with more far-reaching changes remain unreleased
+as of yet.
+
+**Compatibility**:
+
+ * Ensures compatibility with Rust 1.16 through 1.63 stable. Previously the
+   minimum supported Rust version was 1.4. Cargo from 1.4 is no longer
+   compatible with the current crates.io registry, and Rustup fails signature
+   verification for these binaries, so it is infeasible to continue to support
+   it.
+
+New features:
+
+ * Add support for `S24_LE` files, which store 24 bits in 4 bytes ([#40][40],
+   [#41][41])
+ * Add `WavWriter::new_with_spec_ex` ([#42][42])
+ * Add `WavSpec::into_header_for_infinite_file` ([#33][33], [#36][36])
+
+Bugfixes and compatibility improvements:
+
+ * Handle files that have the `wValidBitsPerSample` field set to zero
+   ([#50][50], [#51][51])
+ * Avoid overflow in the channel mask when writing file with more than 32
+   channels ([#59][59], [#60][60])
+
+[33]: https://github.com/ruuda/hound/pull/33
+[36]: https://github.com/ruuda/hound/pull/36
+[40]: https://github.com/ruuda/hound/pull/40
+[41]: https://github.com/ruuda/hound/pull/41
+[42]: https://github.com/ruuda/hound/pull/42
+[50]: https://github.com/ruuda/hound/pull/50
+[51]: https://github.com/ruuda/hound/pull/51
+[59]: https://github.com/ruuda/hound/pull/59
+[60]: https://github.com/ruuda/hound/pull/60
+
+Many thanks to Diffuse, Fletcher Woodruff, Matt Wilkinson, Vitaly Vi Shukela,
+and Tuckerrrrrrrrrr for contributing to this release.
+
+3.4.0
+-----
+
+Released 2018-04-07.
+
+**Breaking changes**:
+
+- None.
+
+Release highlights:
+
+- Exposes `read_wave_header()`, to quickly determine whether a file could be
+  a wav file.
+- Adds support for appending to an existing file. See `WavWriter::append()` for
+  constructing a writer that appends to a file, and `WavWriter::new_append()`
+  for the generic case.
+- Adds `WavWriter::flush()` to flush the underlying writer and update the
+  header. This can be used to minimize data loss when writing a large file.
+- Adds `WavWriter::duration()`, `WavWriter::len()`, and `WavWriter::spec()` to
+  obtain the duration and number of samples written so far, and the spec of the
+  file being written. The latter is useful when appending.
+- Hound now fails earlier when requesting to write an unsupported spec:
+  `WavWriter::new()` will already return `Error::Unsupported`. Previously this
+  error was returned when writing a sample.
+- Hound now verifies that the data chunk has no trailing bytes.
+- `WavWriter::finalize()` now performs a flush as its last operation, to be able
+  to observe errors when using a buffered writer.
+- Ensures compatibility with Rust 1.4 through 1.25 stable.
+
+3.3.1
+-----
+
+Released 2018-02-18.
+
+**Breaking changes**:
+
+- None.
+
+Release highlights:
+
+- Hound now reads certain WAVEFORMATEX files that were previously
+  rejected incorrectly.
+- Ensures compatibility with Rust 1.4 through 1.24 stable.
+
+3.3.0
+-----
+
+Released 2017-12-02.
+
+**Breaking changes**:
+
+- None.
+
+Release highlights:
+
+- Hound now supports seeking to a particular time in the file.
+  See `WavReader::seek()`.
+- Ensures compatibility with Rust 1.4 through 1.22 stable.
+
+Many thanks to Mitchell Nordine for contributing to this release.
+
+3.2.0
+-----
+
+Released 2017-10-14.
+
+**Breaking changes**:
+
+- None.
+
+Release highlights:
+
+- Hound will now write the older PCMWAVEFORMAT format whenever possible, rather
+  than the newer WAVEFORMATEXTENSIBLE, to improve compatibility.
+- Certain nonstandard files (produced among others by “Pro Tools”) can now
+  be read.
+- Ensures compatibility with Rust 1.4 through 1.21 stable.
+
+Many thanks to Denis Kolodin for contributing to this release.
+
+3.1.0
+-----
+
+Released 2017-04-09.
+
+**Breaking changes**:
+
+- None.
+
+Release highlights:
+
+- Support for writing IEEE float was added.
+- The cpal example was updated, and it now compiles on OS X.
+- An OS X target was added to the CI configuration.
+- Ensures compatibility with Rust 1.4 through 1.16 stable.
+
+Many thanks to Alex Zywicki for contributing to this release.
+
+3.0.1
+-----
+
+Released 2017-04-01.
+
+This release fixes a few bugs discovered through fuzzing.
+
+**Breaking changes**:
+
+- None.
+
+Release highlights:
+
+- Fixes high memory usage issue that could occur when reading unknown blocks.
+- Resolve various division by zero and arithmetic overflow errors.
+- Ensures compatibility with Rust 1.4 through 1.16 stable.
+
+3.0.0
+-----
+
+Released 2016-11-27.
+
+This release focuses on improving write performance.
+
+**Breaking changes**:
+
+- When a `WavWriter` is constructed, the header is now written immediately,
+  therefore the constructor now returns a `Result`.
+
+Other changes:
+
+- `WavWriter` no longer maintains a buffer internally.
+  `WavWriter::create()` does still wrap the file it opens in a buffered writer.
+- Adds `SampleWriter16` for fast writing of 16-bit samples. Dedicated
+  writers for other bit depths might be added in future releases.
+
+Upgrading requires dealing with the `Result` in `WavWriter::new()`
+and `WavWriter::create()`. In many cases this should be as simple as
+wrapping the call in a `try!()`, or appending a `?` on recent versions
+of Rust.
+
+2.0.0
+-----
+
+Released 2016-07-31.
+
+**Breaking changes**:
+
+- Support for Rust 1.0 through 1.3 has been dropped.
+- The `WavSpec` struct gained a new `sample_format` member. To upgrade,
+  add `sample_format: hound::SampleFormat::Int` to places where a `WavSpec`
+  is constructed.
+
+Release highlights:
+
+- Ensures compatibility with Rust 1.4 through 1.10.
+- Adds support for reading files with 32-bit IEEE float samples.
+
+Many thanks to Mitchell Nordine for his contributions to this release.
+
+1.1.0
+-----
+
+Released 2015-09-14.
+
+Release highlights:
+
+- New `WavReader::into_inner` method for consistency with the standard library.
+- New `WavReader::into_samples` method for ergonomics and consistency.
+- Ensures compatibility with Rust 1.4.
+
+Many thanks to Pierre Krieger for his contributions to this release.
+
+1.0.0
+-----
+
+Released 2015-07-21.
+
+This is the first stable release of Hound. Only small changes have been made
+with respect to v0.4.0. Release highlights:
+
+- `WavWriter::create` now wraps the writer in a `BufWriter`.
+- `WavSamples` now implements `ExactSizeIterator`.
+- `WavReader::spec` now returns the spec by value.
+- Internal cleanups
+
+0.4.0
+-----
+
+Released 2015-05-16.
+
+Release highlights:
+
+- Works with Rust 1.0.0.
+- Hound can now read and write files with 8, 16, 24, or 32 bits per sample.
+- Better error reporting
+- Improved documentation
+- An improved test suite
+
+0.3.0
+-----
+
+Released 2015-05-05.
+
+Release highlights:
+
+- Hound can now read WAVEFORMATEXTENSIBLE, so it can read the files it writes.
+- Hound can now read files with PCMWAVEFORMAT and WAVEFORMATEX header.
+- Hound now uses a custom error type.
+- New convenient filename-based constructors for `WavReader` and `WavWriter`.
+- More examples
+- An improved test suite
+
+0.2.0
+-----
+
+Released 2015-04-09.
+
+This version adds support for decoding wav files in addition to writing them.
+
+0.1.0
+-----
+
+Released 2015-04-01.
+
+Initial release with only write support.
diff --git a/contributing.md b/contributing.md
new file mode 100644
index 0000000..ed01f9f
--- /dev/null
+++ b/contributing.md
@@ -0,0 +1,19 @@
+Contributing
+============
+
+Contributions in the form of bug reports, feature requests, or pull requests are
+welcome. For pull requests, please consider:
+
+ * Write [a proper commit message][proper-commit] and keep the history clean.
+ * Avoid unrelated formatting changes, they make it harder to identify
+   functional changes in the diff.
+ * You agree to license your contribution under the Apache 2.0 license.
+
+Code of conduct
+---------------
+
+ * Be nice.
+ * Please do not discuss politics in the issue tracker,
+   the issue tracker is for technical issues.
+
+[proper-commit]: http://tbaggery.com/2008/04/19/a-note-about-git-commit-messages.html
diff --git a/examples/append.rs b/examples/append.rs
new file mode 100644
index 0000000..b5fbdc4
--- /dev/null
+++ b/examples/append.rs
@@ -0,0 +1,52 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright 2018 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// This example appends one second of a 440 Hz sine wave to the file "sine.wav".
+// If the file does not exist, it is created instead.
+
+use std::f32::consts::PI;
+use std::i16;
+use std::path::Path;
+
+extern crate hound;
+
+fn main() {
+    let spec = hound::WavSpec {
+        channels: 1,
+        sample_rate: 44100,
+        bits_per_sample: 16,
+        sample_format: hound::SampleFormat::Int,
+    };
+
+    let path: &Path = "sine.wav".as_ref();
+
+    let mut writer = match path.is_file() {
+        true => hound::WavWriter::append(path).unwrap(),
+        false => hound::WavWriter::create(path, spec).unwrap(),
+    };
+
+    // We should not append blindly, we should make sure that the existing file
+    // has the right spec, because that is what we assume when writing.
+    assert_eq!(spec, writer.spec());
+
+    println!("Old duration is {} seconds.", writer.duration() / spec.sample_rate);
+
+    for t in (0 .. 44100).map(|x| x as f32 / 44100.0) {
+        let sample = (t * 440.0 * 2.0 * PI).sin();
+        let amplitude = i16::MAX as f32;
+        writer.write_sample((sample * amplitude) as i16).unwrap();
+    }
+
+    println!("New duration is {} seconds.", writer.duration() / spec.sample_rate);
+
+    writer.finalize().unwrap();
+}
diff --git a/examples/cpal.rs b/examples/cpal.rs
new file mode 100644
index 0000000..c434a2a
--- /dev/null
+++ b/examples/cpal.rs
@@ -0,0 +1,106 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright (C) 2015 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// This example shows how to play a wav file using the cpal crate.
+
+extern crate hound;
+extern crate cpal;
+
+use std::env;
+use std::thread;
+
+fn main() {
+    // Make a WavReader that reads the file provided as program argument.
+    let fname = env::args().nth(1).expect("no file given");
+    let mut reader = hound::WavReader::open(fname).unwrap();
+    let spec = reader.spec();
+
+    let endpoint = cpal::get_default_endpoint().unwrap();
+
+    // Pick a playback format supported by the endpoint, which matches the spec
+    // of the wav file.
+    let format = endpoint.get_supported_formats_list().unwrap()
+                         .filter(|f| matches_format(f, &spec))
+                         .next()
+                         .expect("no supported playback format");
+
+    // A voice in cpal is used for playback.
+    let mut voice = cpal::Voice::new(&endpoint, &format).unwrap();
+
+    let mut samples_left = reader.len() as usize;
+
+    let mut append_data = |voice: &mut cpal::Voice| {
+        match voice.append_data(samples_left) {
+            cpal::UnknownTypeBuffer::I16(mut wrapped_buf) => {
+                // We cannot rely on Rust's autoderef here, because we want to
+                // call .len() on the buffer, which would cause a deref() of the
+                // buffer, not a deref_mut(), and cpal's deref() implementation
+                // is to panic.
+                let buf: &mut [i16] = &mut *wrapped_buf;
+                for (dst, src) in buf.iter_mut().zip(reader.samples::<i16>()) {
+                    *dst = src.unwrap();
+                }
+                samples_left -= buf.len();
+            }
+            cpal::UnknownTypeBuffer::F32(mut wrapped_buf) => {
+                let buf: &mut [f32] = &mut *wrapped_buf;
+                for (dst, src) in buf.iter_mut().zip(reader.samples::<f32>()) {
+                    *dst = src.unwrap();
+                }
+                samples_left -= buf.len();
+            }
+            _ => unreachable!()
+        }
+
+        // Loop again if there are samples left.
+        samples_left > 0
+    };
+
+    // The voice must have some data before playing for the first time.
+    let mut has_more = append_data(&mut voice);
+    voice.play();
+
+    // Then we keep providing new data until the end of the audio.
+    while has_more {
+        has_more = append_data(&mut voice);
+    }
+
+    // Wait for playback to complete.
+    while voice.underflowed() {
+        thread::yield_now();
+    }
+}
+
+fn matches_format(format: &cpal::Format, spec: &hound::WavSpec) -> bool {
+    let cpal::SamplesRate(sample_rate) = format.samples_rate;
+    if sample_rate != spec.sample_rate {
+        return false
+    }
+
+    if format.channels.len() != spec.channels as usize {
+        return false
+    }
+
+    let data_type = match (spec.bits_per_sample, spec.sample_format) {
+        (16, hound::SampleFormat::Int) => Some(cpal::SampleFormat::I16),
+        (32, hound::SampleFormat::Float) => Some(cpal::SampleFormat::F32),
+        _ => None
+    };
+
+    if Some(format.data_type) != data_type {
+        return false
+    }
+
+    // If the sample rate, channel count, and sample format match, then we can
+    // play back the file in this format.
+    true
+}
diff --git a/examples/mean.rs b/examples/mean.rs
new file mode 100644
index 0000000..5c32d4b
--- /dev/null
+++ b/examples/mean.rs
@@ -0,0 +1,53 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright (C) 2015 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// This example computes the mean value and rms of a file, where samples are
+// first interpreted as 16-bit signed integer, and then as a 16-bit unsigned
+// integer. This should allow us to determine whether the samples stored are
+// signed or unsigned: for signed the average value is expected to be 0, and
+// for unsigned the value is expected to be 2^16 - 1.
+//   Note that this example is not a particularly good example of proper coding
+// style; it does not handle failure properly, and it assumes that the provided
+// file has 16 bits per sample.
+
+// TODO: This example should probably be removed, it is just here for verifying
+// and assumption at this point.
+
+extern crate hound;
+
+use std::env;
+
+fn main() {
+    let fname = env::args().nth(1).expect("no file given");
+    let mut reader = hound::WavReader::open(&fname).unwrap();
+    let samples: Vec<i16> = reader.samples().map(|s| s.unwrap()).collect();
+
+    let (ts, tu, n) = samples.iter().fold((0.0, 0.0, 0.0), |(ts, tu, n), &s| {
+        let signed = s as f64;
+        let unsigned = (s as u16) as f64;
+        (ts + signed, tu + unsigned, n + 1.0)
+    });
+    let ms = ts / n;
+    let mu = tu / n;
+    println!("mean signed:   {} (should be 0, deviation is {})", ms, ms.abs());
+    println!("mean unsigned: {} (should be 2^16 - 1, deviation is {})", mu, (mu - 32767.0).abs());
+
+    let (ts, tu) = samples.iter().fold((0.0, 0.0), |(ts, tu), &s| {
+        let ds = s as f64 - ms;
+        let du = (s as u16) as f64 - mu;
+        (ts + ds * ds, tu + du * du)
+    });
+    let rmss = (ts / n).sqrt();
+    let rmsu = (tu / n).sqrt();
+    println!("rms signed:    {}", rmss);
+    println!("rms unsigned:  {}", rmsu);
+}
diff --git a/examples/rms.rs b/examples/rms.rs
new file mode 100644
index 0000000..32e2b47
--- /dev/null
+++ b/examples/rms.rs
@@ -0,0 +1,47 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright (C) 2017 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// This example computes the root mean square (rms) of an audio file with
+// integer or float samples, of at most 32 bits per sample. It is a slightly
+// more elaborate version of the example found in the readme, mostly useful for
+// checking whether Hound can read a specific file.
+
+extern crate hound;
+
+use std::env;
+use std::io;
+
+/// Compute the RMS of either integers or float samples.
+fn compute_rms<S, R>(reader: &mut hound::WavReader<R>) -> f64
+where
+    f64: From<S>,
+    S: hound::Sample,
+    R: io::Read,
+{
+    let sqr_sum = reader.samples::<S>().fold(0.0, |sqr_sum, s| {
+        let sample = f64::from(s.unwrap());
+        sqr_sum + sample * sample
+    });
+    (sqr_sum / reader.len() as f64).sqrt()
+}
+
+fn main() {
+    // Compute the RMS for all files given on the command line.
+    for fname in env::args().skip(1) {
+        let mut reader = hound::WavReader::open(&fname).unwrap();
+        let rms = match reader.spec().sample_format {
+            hound::SampleFormat::Float => compute_rms::<f32, _>(&mut reader),
+            hound::SampleFormat::Int => compute_rms::<i32, _>(&mut reader),
+        };
+        println!("{}: {:0.1} ({} samples)", fname, rms, reader.len());
+    }
+}
diff --git a/examples/wavstdout.rs b/examples/wavstdout.rs
new file mode 100644
index 0000000..475a44b
--- /dev/null
+++ b/examples/wavstdout.rs
@@ -0,0 +1,31 @@
+// Generate endless screeching noise to stdout
+
+// Usage: cargo run --example wavstdout | mpv -
+
+extern crate hound;
+use std::io::Write;
+
+fn main() {
+    let spec = hound::WavSpec {
+        bits_per_sample: 16,
+        channels: 1,
+        sample_format: hound::SampleFormat::Int,
+        sample_rate: 16000,
+    };
+
+    let v = spec.into_header_for_infinite_file();
+
+    let so = std::io::stdout();
+    let mut so = so.lock();
+    so.write_all(&v[..]).unwrap();
+
+    loop {
+        for i in 0..126 {
+            use hound::Sample;
+            let x : i16 = (i * 256) as i16;
+            if x.write(&mut so, 16).is_err() {
+                return;
+            }
+        }
+    }
+}
diff --git a/license b/license
new file mode 100644
index 0000000..d645695
--- /dev/null
+++ b/license
@@ -0,0 +1,202 @@
+
+                                 Apache License
+                           Version 2.0, January 2004
+                        http://www.apache.org/licenses/
+
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+   1. Definitions.
+
+      "License" shall mean the terms and conditions for use, reproduction,
+      and distribution as defined by Sections 1 through 9 of this document.
+
+      "Licensor" shall mean the copyright owner or entity authorized by
+      the copyright owner that is granting the License.
+
+      "Legal Entity" shall mean the union of the acting entity and all
+      other entities that control, are controlled by, or are under common
+      control with that entity. For the purposes of this definition,
+      "control" means (i) the power, direct or indirect, to cause the
+      direction or management of such entity, whether by contract or
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
+      outstanding shares, or (iii) beneficial ownership of such entity.
+
+      "You" (or "Your") shall mean an individual or Legal Entity
+      exercising permissions granted by this License.
+
+      "Source" form shall mean the preferred form for making modifications,
+      including but not limited to software source code, documentation
+      source, and configuration files.
+
+      "Object" form shall mean any form resulting from mechanical
+      transformation or translation of a Source form, including but
+      not limited to compiled object code, generated documentation,
+      and conversions to other media types.
+
+      "Work" shall mean the work of authorship, whether in Source or
+      Object form, made available under the License, as indicated by a
+      copyright notice that is included in or attached to the work
+      (an example is provided in the Appendix below).
+
+      "Derivative Works" shall mean any work, whether in Source or Object
+      form, that is based on (or derived from) the Work and for which the
+      editorial revisions, annotations, elaborations, or other modifications
+      represent, as a whole, an original work of authorship. For the purposes
+      of this License, Derivative Works shall not include works that remain
+      separable from, or merely link (or bind by name) to the interfaces of,
+      the Work and Derivative Works thereof.
+
+      "Contribution" shall mean any work of authorship, including
+      the original version of the Work and any modifications or additions
+      to that Work or Derivative Works thereof, that is intentionally
+      submitted to Licensor for inclusion in the Work by the copyright owner
+      or by an individual or Legal Entity authorized to submit on behalf of
+      the copyright owner. For the purposes of this definition, "submitted"
+      means any form of electronic, verbal, or written communication sent
+      to the Licensor or its representatives, including but not limited to
+      communication on electronic mailing lists, source code control systems,
+      and issue tracking systems that are managed by, or on behalf of, the
+      Licensor for the purpose of discussing and improving the Work, but
+      excluding communication that is conspicuously marked or otherwise
+      designated in writing by the copyright owner as "Not a Contribution."
+
+      "Contributor" shall mean Licensor and any individual or Legal Entity
+      on behalf of whom a Contribution has been received by Licensor and
+      subsequently incorporated within the Work.
+
+   2. Grant of Copyright License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      copyright license to reproduce, prepare Derivative Works of,
+      publicly display, publicly perform, sublicense, and distribute the
+      Work and such Derivative Works in Source or Object form.
+
+   3. Grant of Patent License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      (except as stated in this section) patent license to make, have made,
+      use, offer to sell, sell, import, and otherwise transfer the Work,
+      where such license applies only to those patent claims licensable
+      by such Contributor that are necessarily infringed by their
+      Contribution(s) alone or by combination of their Contribution(s)
+      with the Work to which such Contribution(s) was submitted. If You
+      institute patent litigation against any entity (including a
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
+      or a Contribution incorporated within the Work constitutes direct
+      or contributory patent infringement, then any patent licenses
+      granted to You under this License for that Work shall terminate
+      as of the date such litigation is filed.
+
+   4. Redistribution. You may reproduce and distribute copies of the
+      Work or Derivative Works thereof in any medium, with or without
+      modifications, and in Source or Object form, provided that You
+      meet the following conditions:
+
+      (a) You must give any other recipients of the Work or
+          Derivative Works a copy of this License; and
+
+      (b) You must cause any modified files to carry prominent notices
+          stating that You changed the files; and
+
+      (c) You must retain, in the Source form of any Derivative Works
+          that You distribute, all copyright, patent, trademark, and
+          attribution notices from the Source form of the Work,
+          excluding those notices that do not pertain to any part of
+          the Derivative Works; and
+
+      (d) If the Work includes a "NOTICE" text file as part of its
+          distribution, then any Derivative Works that You distribute must
+          include a readable copy of the attribution notices contained
+          within such NOTICE file, excluding those notices that do not
+          pertain to any part of the Derivative Works, in at least one
+          of the following places: within a NOTICE text file distributed
+          as part of the Derivative Works; within the Source form or
+          documentation, if provided along with the Derivative Works; or,
+          within a display generated by the Derivative Works, if and
+          wherever such third-party notices normally appear. The contents
+          of the NOTICE file are for informational purposes only and
+          do not modify the License. You may add Your own attribution
+          notices within Derivative Works that You distribute, alongside
+          or as an addendum to the NOTICE text from the Work, provided
+          that such additional attribution notices cannot be construed
+          as modifying the License.
+
+      You may add Your own copyright statement to Your modifications and
+      may provide additional or different license terms and conditions
+      for use, reproduction, or distribution of Your modifications, or
+      for any such Derivative Works as a whole, provided Your use,
+      reproduction, and distribution of the Work otherwise complies with
+      the conditions stated in this License.
+
+   5. Submission of Contributions. Unless You explicitly state otherwise,
+      any Contribution intentionally submitted for inclusion in the Work
+      by You to the Licensor shall be under the terms and conditions of
+      this License, without any additional terms or conditions.
+      Notwithstanding the above, nothing herein shall supersede or modify
+      the terms of any separate license agreement you may have executed
+      with Licensor regarding such Contributions.
+
+   6. Trademarks. This License does not grant permission to use the trade
+      names, trademarks, service marks, or product names of the Licensor,
+      except as required for reasonable and customary use in describing the
+      origin of the Work and reproducing the content of the NOTICE file.
+
+   7. Disclaimer of Warranty. Unless required by applicable law or
+      agreed to in writing, Licensor provides the Work (and each
+      Contributor provides its Contributions) on an "AS IS" BASIS,
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+      implied, including, without limitation, any warranties or conditions
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+      PARTICULAR PURPOSE. You are solely responsible for determining the
+      appropriateness of using or redistributing the Work and assume any
+      risks associated with Your exercise of permissions under this License.
+
+   8. Limitation of Liability. In no event and under no legal theory,
+      whether in tort (including negligence), contract, or otherwise,
+      unless required by applicable law (such as deliberate and grossly
+      negligent acts) or agreed to in writing, shall any Contributor be
+      liable to You for damages, including any direct, indirect, special,
+      incidental, or consequential damages of any character arising as a
+      result of this License or out of the use or inability to use the
+      Work (including but not limited to damages for loss of goodwill,
+      work stoppage, computer failure or malfunction, or any and all
+      other commercial damages or losses), even if such Contributor
+      has been advised of the possibility of such damages.
+
+   9. Accepting Warranty or Additional Liability. While redistributing
+      the Work or Derivative Works thereof, You may choose to offer,
+      and charge a fee for, acceptance of support, warranty, indemnity,
+      or other liability obligations and/or rights consistent with this
+      License. However, in accepting such obligations, You may act only
+      on Your own behalf and on Your sole responsibility, not on behalf
+      of any other Contributor, and only if You agree to indemnify,
+      defend, and hold each Contributor harmless for any liability
+      incurred by, or claims asserted against, such Contributor by reason
+      of your accepting any such warranty or additional liability.
+
+   END OF TERMS AND CONDITIONS
+
+   APPENDIX: How to apply the Apache License to your work.
+
+      To apply the Apache License to your work, attach the following
+      boilerplate notice, with the fields enclosed by brackets "[]"
+      replaced with your own identifying information. (Don't include
+      the brackets!)  The text should be enclosed in the appropriate
+      comment syntax for the file format. We also recommend that a
+      file or class name and description of purpose be included on the
+      same "printed page" as the copyright notice for easier
+      identification within third-party archives.
+
+   Copyright [yyyy] [name of copyright owner]
+
+   Licensed under the Apache License, Version 2.0 (the "License");
+   you may not use this file except in compliance with the License.
+   You may obtain a copy of the License at
+
+       http://www.apache.org/licenses/LICENSE-2.0
+
+   Unless required by applicable law or agreed to in writing, software
+   distributed under the License is distributed on an "AS IS" BASIS,
+   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+   See the License for the specific language governing permissions and
+   limitations under the License.
diff --git a/readme.md b/readme.md
new file mode 100644
index 0000000..9b85909
--- /dev/null
+++ b/readme.md
@@ -0,0 +1,85 @@
+Hound
+=====
+A wav encoding and decoding library in Rust.
+
+[![Crates.io version][crate-img]][crate]
+[![Changelog][changelog-img]](changelog.md)
+[![Documentation][docs-img]][docs]
+
+Hound can read and write the WAVE audio format, an ubiquitous format for raw,
+uncompressed audio. The main motivation to write it was to test
+[Claxon][claxon], a FLAC decoding library written in Rust.
+
+Examples
+--------
+The following example renders a 440 Hz sine wave, and stores it as a mono wav
+file with a sample rate of 44.1 kHz and 16 bits per sample.
+
+```rust
+use std::f32::consts::PI;
+use std::i16;
+use hound;
+
+let spec = hound::WavSpec {
+    channels: 1,
+    sample_rate: 44100,
+    bits_per_sample: 16,
+    sample_format: hound::SampleFormat::Int,
+};
+let mut writer = hound::WavWriter::create("sine.wav", spec).unwrap();
+for t in (0 .. 44100).map(|x| x as f32 / 44100.0) {
+    let sample = (t * 440.0 * 2.0 * PI).sin();
+    let amplitude = i16::MAX as f32;
+    writer.write_sample((sample * amplitude) as i16).unwrap();
+}
+```
+
+The file is finalized implicitly when the writer is dropped, call
+`writer.finalize()` to observe errors.
+
+The following example computes the root mean square (RMS) of an audio file with
+at most 16 bits per sample.
+
+```rust
+use hound;
+
+let mut reader = hound::WavReader::open("testsamples/pop.wav").unwrap();
+let sqr_sum = reader.samples::<i16>()
+                    .fold(0.0, |sqr_sum, s| {
+    let sample = s.unwrap() as f64;
+    sqr_sum + sample * sample
+});
+println!("RMS is {}", (sqr_sum / reader.len() as f64).sqrt());
+```
+
+Features
+--------
+
+|                 | Read                                                    | Write                                   |
+|-----------------|---------------------------------------------------------|-----------------------------------------|
+| Format          | `PCMWAVEFORMAT`, `WAVEFORMATEX`, `WAVEFORMATEXTENSIBLE` | `PCMWAVEFORMAT`, `WAVEFORMATEXTENSIBLE` |
+| Encoding        | Integer PCM, IEEE Float                                 | Integer PCM, IEEE Float                 |
+| Bits per sample | 8, 16, 24, 32 (integer), 32 (float)                     | 8, 16, 24, 32 (integer), 32 (float)     |
+
+Contributing
+------------
+Contributions in the form of bug reports, feature requests, or pull requests are
+welcome. See [contributing.md](contributing.md).
+
+License
+-------
+Hound is licensed under the [Apache 2.0][apache2] license. It may be used in
+free software as well as closed-source applications, both for commercial and
+non-commercial use under the conditions given in the license. If you want to
+use Hound in your GPLv2-licensed software, you can add an [exception][exception]
+to your copyright notice. Please do not open an issue if you disagree with the
+choice of license.
+
+[crate-img]:     https://img.shields.io/crates/v/hound.svg
+[crate]:         https://crates.io/crates/hound
+[changelog-img]: https://img.shields.io/badge/changelog-online-blue.svg
+[docs-img]:      https://img.shields.io/badge/docs-online-blue.svg
+[docs]:          https://docs.rs/hound
+[claxon]:        https://github.com/ruuda/claxon
+[apache2]:       https://www.apache.org/licenses/LICENSE-2.0
+[exception]:     https://www.gnu.org/licenses/gpl-faq.html#GPLIncompatibleLibs
diff --git a/rust-toolchain.toml b/rust-toolchain.toml
new file mode 100644
index 0000000..072ab04
--- /dev/null
+++ b/rust-toolchain.toml
@@ -0,0 +1,5 @@
+[toolchain]
+# Note, this is the Minimum Supported Rust Version. We build with this by
+# default to ensure we don't accidentally break it, but of course you should be
+# able to build with more recent versions.
+channel = "1.40.0"
diff --git a/src/lib.rs b/src/lib.rs
new file mode 100644
index 0000000..fe435cb
--- /dev/null
+++ b/src/lib.rs
@@ -0,0 +1,925 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright (C) 2015 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+//! Hound, a wav encoding and decoding library.
+//!
+//! Examples
+//! ========
+//!
+//! The following example renders a 440 Hz sine wave, and stores it as as a
+//! mono wav file with a sample rate of 44.1 kHz and 16 bits per sample.
+//!
+//! ```
+//! use std::f32::consts::PI;
+//! use std::i16;
+//! use hound;
+//!
+//! let spec = hound::WavSpec {
+//!     channels: 1,
+//!     sample_rate: 44100,
+//!     bits_per_sample: 16,
+//!     sample_format: hound::SampleFormat::Int,
+//! };
+//! let mut writer = hound::WavWriter::create("sine.wav", spec).unwrap();
+//! for t in (0 .. 44100).map(|x| x as f32 / 44100.0) {
+//!     let sample = (t * 440.0 * 2.0 * PI).sin();
+//!     let amplitude = i16::MAX as f32;
+//!     writer.write_sample((sample * amplitude) as i16).unwrap();
+//! }
+//! writer.finalize().unwrap();
+//! ```
+//!
+//! The following example computes the root mean square (RMS) of an audio file
+//! with at most 16 bits per sample.
+//!
+//! ```
+//! use hound;
+//!
+//! let mut reader = hound::WavReader::open("testsamples/pop.wav").unwrap();
+//! let sqr_sum = reader.samples::<i16>()
+//!                     .fold(0.0, |sqr_sum, s| {
+//!     let sample = s.unwrap() as f64;
+//!     sqr_sum + sample * sample
+//! });
+//! println!("RMS is {}", (sqr_sum / reader.len() as f64).sqrt());
+//! ```
+
+#![warn(missing_docs)]
+
+use std::error;
+use std::fmt;
+use std::io;
+use std::result;
+use read::ReadExt;
+use write::WriteExt;
+
+mod read;
+mod write;
+
+pub use read::{WavReader, WavIntoSamples, WavSamples, read_wave_header};
+pub use write::{SampleWriter16, WavWriter};
+
+/// A type that can be used to represent audio samples.
+///
+/// Via this trait, decoding can be generic over `i8`, `i16`, `i32` and `f32`.
+///
+/// All integer formats with bit depths up to 32 bits per sample can be decoded
+/// into `i32`, but it takes up more memory. If you know beforehand that you
+/// will be reading a file with 16 bits per sample, then decoding into an `i16`
+/// will be sufficient.
+pub trait Sample: Sized {
+    /// Writes the audio sample to the WAVE data chunk.
+    fn write<W: io::Write>(self, writer: &mut W, bits: u16) -> Result<()>;
+
+    /// Writes the audio sample to the WAVE data chunk, zero padding the size of
+    /// the written sample out to `byte_width`.
+    fn write_padded<W: io::Write>(self, writer: &mut W, bits: u16, byte_width: u16) -> Result<()>;
+
+    /// Reads the audio sample from the WAVE data chunk.
+    fn read<R: io::Read>(reader: &mut R, SampleFormat, bytes: u16, bits: u16) -> Result<Self>;
+
+    /// Cast the sample to a 16-bit sample.
+    ///
+    /// This does not change the value of the sample, it only casts it. The
+    /// value is assumed to fit within the range. This is not verified,
+    /// truncation may occur.
+    fn as_i16(self) -> i16;
+}
+
+/// Converts an unsigned integer in the range 0-255 to a signed one in the range -128-127.
+///
+/// Presumably, the designers of the WAVE format did not like consistency. For
+/// all bit depths except 8, samples are stored as little-endian _signed_
+/// integers. However, an 8-bit sample is instead stored as an _unsigned_
+/// integer. Hound abstracts away this idiosyncrasy by providing only signed
+/// sample types.
+fn signed_from_u8(x: u8) -> i8 {
+    (x as i16 - 128) as i8
+}
+
+/// Converts a signed integer in the range -128-127 to an unsigned one in the range 0-255.
+fn u8_from_signed(x: i8) -> u8 {
+    (x as i16 + 128) as u8
+}
+
+#[test]
+fn u8_sign_conversion_is_bijective() {
+    for x in 0..255 {
+        assert_eq!(x, u8_from_signed(signed_from_u8(x)));
+    }
+    for x in -128..127 {
+        assert_eq!(x, signed_from_u8(u8_from_signed(x)));
+    }
+}
+
+/// Tries to cast the sample to an 8-bit signed integer, returning an error on overflow.
+#[inline(always)]
+fn narrow_to_i8(x: i32) -> Result<i8> {
+    use std::i8;
+    if x < i8::MIN as i32 || x > i8::MAX as i32 {
+        Err(Error::TooWide)
+    } else {
+        Ok(x as i8)
+    }
+}
+
+#[test]
+fn verify_narrow_to_i8() {
+    assert!(narrow_to_i8(127).is_ok());
+    assert!(narrow_to_i8(128).is_err());
+    assert!(narrow_to_i8(-128).is_ok());
+    assert!(narrow_to_i8(-129).is_err());
+}
+
+/// Tries to cast the sample to a 16-bit signed integer, returning an error on overflow.
+#[inline(always)]
+fn narrow_to_i16(x: i32) -> Result<i16> {
+    use std::i16;
+    if x < i16::MIN as i32 || x > i16::MAX as i32 {
+        Err(Error::TooWide)
+    } else {
+        Ok(x as i16)
+    }
+}
+
+#[test]
+fn verify_narrow_to_i16() {
+    assert!(narrow_to_i16(32767).is_ok());
+    assert!(narrow_to_i16(32768).is_err());
+    assert!(narrow_to_i16(-32768).is_ok());
+    assert!(narrow_to_i16(-32769).is_err());
+}
+
+/// Tries to cast the sample to a 24-bit signed integer, returning an error on overflow.
+#[inline(always)]
+fn narrow_to_i24(x: i32) -> Result<i32> {
+    if x < -(1 << 23) || x > (1 << 23) - 1 {
+        Err(Error::TooWide)
+    } else {
+        Ok(x)
+    }
+}
+
+#[test]
+fn verify_narrow_to_i24() {
+    assert!(narrow_to_i24(8_388_607).is_ok());
+    assert!(narrow_to_i24(8_388_608).is_err());
+    assert!(narrow_to_i24(-8_388_608).is_ok());
+    assert!(narrow_to_i24(-8_388_609).is_err());
+}
+
+impl Sample for i8 {
+    fn write<W: io::Write>(self, writer: &mut W, bits: u16) -> Result<()> {
+        self.write_padded(writer, bits, bits / 8)
+    }
+
+    fn write_padded<W: io::Write>(self, writer: &mut W, bits: u16, byte_width: u16) -> Result<()> {
+        match (bits, byte_width) {
+            (8, 1) => Ok(try!(writer.write_u8(u8_from_signed(self)))),
+            (16, 2) => Ok(try!(writer.write_le_i16(self as i16))),
+            (24, 3) => Ok(try!(writer.write_le_i24(self as i32))),
+            (24, 4) => Ok(try!(writer.write_le_i24_4(self as i32))),
+            (32, 4) => Ok(try!(writer.write_le_i32(self as i32))),
+            _ => Err(Error::Unsupported),
+        }
+    }
+
+    #[inline(always)]
+    fn as_i16(self) -> i16 {
+        self as i16
+    }
+
+    fn read<R: io::Read>(reader: &mut R, fmt: SampleFormat, bytes: u16, bits: u16) -> Result<i8> {
+        if fmt != SampleFormat::Int {
+            return Err(Error::InvalidSampleFormat);
+        }
+        match (bytes, bits) {
+            (1, 8) => Ok(try!(reader.read_u8().map(signed_from_u8))),
+            (n, _) if n > 1 => Err(Error::TooWide),
+            // TODO: add a genric decoder for any bit depth.
+            _ => Err(Error::Unsupported),
+        }
+    }
+}
+
+impl Sample for i16 {
+    fn write<W: io::Write>(self, writer: &mut W, bits: u16) -> Result<()> {
+        self.write_padded(writer, bits, bits / 8)
+    }
+
+    fn write_padded<W: io::Write>(self, writer: &mut W, bits: u16, byte_width: u16) -> Result<()> {
+        match (bits, byte_width) {
+            (8, 1) => Ok(try!(
+                writer.write_u8(u8_from_signed(try!(narrow_to_i8(self as i32))))
+            )),
+            (16, 2) => Ok(try!(writer.write_le_i16(self))),
+            (24, 3) => Ok(try!(writer.write_le_i24(self as i32))),
+            (24, 4) => Ok(try!(writer.write_le_i24_4(self as i32))),
+            (32, 4) => Ok(try!(writer.write_le_i32(self as i32))),
+            _ => Err(Error::Unsupported),
+        }
+    }
+
+    #[inline(always)]
+    fn as_i16(self) -> i16 {
+        self
+    }
+
+    fn read<R: io::Read>(reader: &mut R, fmt: SampleFormat, bytes: u16, bits: u16) -> Result<i16> {
+        if fmt != SampleFormat::Int {
+            return Err(Error::InvalidSampleFormat);
+        }
+        match (bytes, bits) {
+            (1, 8) => Ok(try!(reader.read_u8().map(signed_from_u8).map(|x| x as i16))),
+            (2, 16) => Ok(try!(reader.read_le_i16())),
+            (n, _) if n > 2 => Err(Error::TooWide),
+            // TODO: add a generic decoder for any bit depth.
+            _ => Err(Error::Unsupported),
+        }
+    }
+}
+
+impl Sample for i32 {
+    fn write<W: io::Write>(self, writer: &mut W, bits: u16) -> Result<()> {
+        self.write_padded(writer, bits, bits / 8)
+    }
+
+    fn write_padded<W: io::Write>(self, writer: &mut W, bits: u16, byte_width: u16) -> Result<()> {
+        match (bits, byte_width) {
+            (8, 1) => Ok(try!(
+                writer.write_u8(u8_from_signed(try!(narrow_to_i8(self))))
+            )),
+            (16, 2) => Ok(try!(writer.write_le_i16(try!(narrow_to_i16(self))))),
+            (24, 3) => Ok(try!(writer.write_le_i24(try!(narrow_to_i24(self))))),
+            (24, 4) => Ok(try!(writer.write_le_i24_4(try!(narrow_to_i24(self))))),
+            (32, 4) => Ok(try!(writer.write_le_i32(self))),
+            _ => Err(Error::Unsupported),
+        }
+    }
+
+    #[inline(always)]
+    fn as_i16(self) -> i16 {
+        self as i16
+    }
+
+    fn read<R: io::Read>(reader: &mut R, fmt: SampleFormat, bytes: u16, bits: u16) -> Result<i32> {
+        if fmt != SampleFormat::Int {
+            return Err(Error::InvalidSampleFormat);
+        }
+        match (bytes, bits) {
+            (1, 8) => Ok(try!(reader.read_u8().map(signed_from_u8).map(|x| x as i32))),
+            (2, 16) => Ok(try!(reader.read_le_i16().map(|x| x as i32))),
+            (3, 24) => Ok(try!(reader.read_le_i24())),
+            (4, 24) => Ok(try!(reader.read_le_i24_4())),
+            (4, 32) => Ok(try!(reader.read_le_i32())),
+            (n, _) if n > 4 => Err(Error::TooWide),
+            // TODO: add a generic decoder for any bit depth.
+            _ => Err(Error::Unsupported),
+        }
+    }
+}
+
+impl Sample for f32 {
+    fn write<W: io::Write>(self, writer: &mut W, bits: u16) -> Result<()> {
+        self.write_padded(writer, bits, bits / 8)
+    }
+
+    fn write_padded<W: io::Write>(self, writer: &mut W, bits: u16, byte_width: u16) -> Result<()> {
+        match (bits, byte_width) {
+            (32, 4) => Ok(try!(writer.write_le_f32(self))),
+            _ => Err(Error::Unsupported),
+        }
+    }
+
+    fn as_i16(self) -> i16 {
+        panic!("Calling as_i16 with an f32 is invalid.");
+    }
+
+    fn read<R: io::Read>(reader: &mut R, fmt: SampleFormat, bytes: u16, bits: u16) -> Result<Self> {
+        if fmt != SampleFormat::Float {
+            return Err(Error::InvalidSampleFormat);
+        }
+        match (bytes, bits) {
+            (4, 32) => Ok(try!(reader.read_le_f32())),
+            (n, _) if n > 4 => Err(Error::TooWide),
+            _ => Err(Error::Unsupported),
+        }
+    }
+}
+
+/// Specifies whether a sample is stored as an "IEEE Float" or an integer.
+#[derive(Clone, Copy, Debug, PartialEq, Eq)]
+pub enum SampleFormat {
+    /// Wave files with the `WAVE_FORMAT_IEEE_FLOAT` format tag store samples as floating point
+    /// values.
+    ///
+    /// Values are normally in the range [-1.0, 1.0].
+    Float,
+    /// Wave files with the `WAVE_FORMAT_PCM` format tag store samples as integer values.
+    Int,
+}
+
+/// Specifies properties of the audio data.
+#[derive(Clone, Copy, Debug, PartialEq, Eq)]
+pub struct WavSpec {
+    /// The number of channels.
+    pub channels: u16,
+
+    /// The number of samples per second.
+    ///
+    /// A common value is 44100, this is 44.1 kHz which is used for CD audio.
+    pub sample_rate: u32,
+
+    /// The number of bits per sample.
+    ///
+    /// A common value is 16 bits per sample, which is used for CD audio.
+    pub bits_per_sample: u16,
+
+    /// Whether the wav's samples are float or integer values.
+    pub sample_format: SampleFormat,
+}
+
+/// Specifies properties of the audio data, as well as the layout of the stream.
+#[derive(Clone, Copy)]
+pub struct WavSpecEx {
+    /// The normal information about the audio data.
+    ///
+    /// Bits per sample here is the number of _used_ bits per sample, not the
+    /// number of bits used to _store_ a sample.
+    pub spec: WavSpec,
+
+    /// The number of bytes used to store a sample.
+    pub bytes_per_sample: u16,
+}
+
+/// The error type for operations on `WavReader` and `WavWriter`.
+#[derive(Debug)]
+pub enum Error {
+    /// An IO error occured in the underlying reader or writer.
+    IoError(io::Error),
+    /// Ill-formed WAVE data was encountered.
+    FormatError(&'static str),
+    /// The sample has more bits than the destination type.
+    ///
+    /// When iterating using the `samples` iterator, this means that the
+    /// destination type (produced by the iterator) is not wide enough to hold
+    /// the sample. When writing, this means that the sample cannot be written,
+    /// because it requires more bits than the bits per sample specified.
+    TooWide,
+    /// The number of samples written is not a multiple of the number of channels.
+    UnfinishedSample,
+    /// The format is not supported.
+    Unsupported,
+    /// The sample format is different than the destination format.
+    ///
+    /// When iterating using the `samples` iterator, this means the destination
+    /// type (produced by the iterator) has a different sample format than the
+    /// samples in the wav file.
+    ///
+    /// For example, this will occur if the user attempts to produce `i32`
+    /// samples (which have a `SampleFormat::Int`) from a wav file that
+    /// contains floating point data (`SampleFormat::Float`).
+    InvalidSampleFormat,
+}
+
+impl fmt::Display for Error {
+    fn fmt(&self, formatter: &mut fmt::Formatter) -> result::Result<(), fmt::Error> {
+        match *self {
+            Error::IoError(ref err) => err.fmt(formatter),
+            Error::FormatError(reason) => {
+                try!(formatter.write_str("Ill-formed WAVE file: "));
+                formatter.write_str(reason)
+            }
+            Error::TooWide => {
+                formatter.write_str("The sample has more bits than the destination type.")
+            }
+            Error::UnfinishedSample => {
+                formatter.write_str(
+                    "The number of samples written is not a multiple of the number of channels.")
+            }
+            Error::Unsupported => {
+                formatter.write_str("The wave format of the file is not supported.")
+            }
+            Error::InvalidSampleFormat => {
+                formatter.write_str("The sample format differs from the destination format.")
+            }
+        }
+    }
+}
+
+impl error::Error for Error {
+    fn description(&self) -> &str {
+        match *self {
+            Error::IoError(ref err) => err.description(),
+            Error::FormatError(reason) => reason,
+            Error::TooWide => "the sample has more bits than the destination type",
+            Error::UnfinishedSample => "the number of samples written is not a multiple of the number of channels",
+            Error::Unsupported => "the wave format of the file is not supported",
+            Error::InvalidSampleFormat => "the sample format differs from the destination format",
+        }
+    }
+
+    fn cause(&self) -> Option<&error::Error> {
+        match *self {
+            Error::IoError(ref err) => Some(err),
+            Error::FormatError(_) => None,
+            Error::TooWide => None,
+            Error::UnfinishedSample => None,
+            Error::Unsupported => None,
+            Error::InvalidSampleFormat => None,
+        }
+    }
+}
+
+impl From<io::Error> for Error {
+    fn from(err: io::Error) -> Error {
+        Error::IoError(err)
+    }
+}
+
+/// A type for results generated by Hound where the error type is hard-wired.
+pub type Result<T> = result::Result<T, Error>;
+
+// The WAVEFORMATEXTENSIBLE struct can contain several subformats.
+// These are identified by a GUID. The various GUIDS can be found in the file
+// mmreg.h that is part of the Windows SDK. The following GUIDS are defined:
+// - PCM:        00000001-0000-0010-8000-00aa00389b71
+// - IEEE_FLOAT: 00000003-0000-0010-8000-00aa00389b71
+// When written to a wav file, the byte order of a GUID is native for the first
+// three sections, which is assumed to be little endian, and big endian for the
+// last 8-byte section (which does contain a hyphen, for reasons unknown to me).
+
+/// Subformat type for PCM audio with integer samples.
+const KSDATAFORMAT_SUBTYPE_PCM: [u8; 16] = [0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x80,
+                                            0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71];
+
+/// Subformat type for IEEE_FLOAT audio with float samples.
+const KSDATAFORMAT_SUBTYPE_IEEE_FLOAT: [u8; 16] = [0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00,
+                                                   0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71];
+
+
+impl WavSpec {
+    /// Get "stand-alone" wav header representing infinite or unknown size wav file.
+    /// Use this if you need to write audio data to non-seekable sinks (like stdout).
+    ///
+    /// Actual samples are supposed to be written using low-level [`Sample::write`] call.
+    ///
+    /// Such wav files are produced e.g. by FFmpeg and have `0xFFFFFFFF` instead of chunk sizes.
+    ///
+    /// Note that such files may be non-standard. Consider using [`WavWriter`] for better API.
+    ///
+    /// Example:
+    ///
+    /// ```no_run
+    /// extern crate hound;
+    /// use std::io::Write;
+    /// 
+    /// let spec = hound::WavSpec {
+    ///     bits_per_sample: 16,
+    ///     channels: 1,
+    ///     sample_format: hound::SampleFormat::Int,
+    ///     sample_rate: 16000,
+    /// };
+    /// 
+    /// let v = spec.into_header_for_infinite_file();
+    /// 
+    /// let so = std::io::stdout();
+    /// let mut so = so.lock();
+    /// so.write_all(&v[..]).unwrap();
+    /// 
+    /// loop {
+    ///    for i in 0..126 {
+    ///       let x : i16 = (i * 256) as i16;
+    ///       hound::Sample::write(x, &mut so, 16).unwrap();
+    ///    }
+    /// }
+    /// ```
+    pub fn into_header_for_infinite_file(self) -> Vec<u8> {
+        let mut c = std::io::Cursor::new(Vec::with_capacity(0x44));
+        {
+            let w = WavWriter::new(&mut c, self);
+            drop(w);
+        }
+        let mut v = c.into_inner();
+
+        // Set WAVE chunk size to a special signal value
+        v[4] = 0xFF; v[5] = 0xFF; v[6] = 0xFF; v[7] = 0xFF;
+
+        // Detect fmt size, get offset of data chunk's size and set it to signal value
+        if v[16] == 0x10 {
+            // pcm wave
+            v[0x28] = 0xFF; v[0x29] = 0xFF; v[0x2A] = 0xFF; v[0x2B] = 0xFF; 
+        } else if v[16] == 0x28 {
+            // extensible
+            v[0x40] = 0xFF; v[0x41] = 0xFF; v[0x42] = 0xFF; v[0x43] = 0xFF; 
+        } else {
+            unreachable!()
+        }
+
+        v
+    }
+}
+
+#[test]
+fn write_read_i16_is_lossless() {
+    let mut buffer = io::Cursor::new(Vec::new());
+    let write_spec = WavSpec {
+        channels: 2,
+        sample_rate: 44100,
+        bits_per_sample: 16,
+        sample_format: SampleFormat::Int,
+    };
+
+    {
+        let mut writer = WavWriter::new(&mut buffer, write_spec).unwrap();
+        for s in -1024_i16..1024 {
+            writer.write_sample(s).unwrap();
+        }
+        writer.finalize().unwrap();
+    }
+
+    {
+        buffer.set_position(0);
+        let mut reader = WavReader::new(&mut buffer).unwrap();
+        assert_eq!(write_spec, reader.spec());
+        assert_eq!(reader.len(), 2048);
+        for (expected, read) in (-1024_i16..1024).zip(reader.samples()) {
+            assert_eq!(expected, read.unwrap());
+        }
+    }
+}
+
+#[test]
+fn write_read_i16_via_sample_writer_is_lossless() {
+    let mut buffer = io::Cursor::new(Vec::new());
+    let write_spec = WavSpec {
+        channels: 2,
+        sample_rate: 44100,
+        bits_per_sample: 16,
+        sample_format: SampleFormat::Int,
+    };
+
+    {
+        let mut writer = WavWriter::new(&mut buffer, write_spec).unwrap();
+        {
+            {
+                let mut sample_writer = writer.get_i16_writer(1024);
+                for s in -1024_i16..0 {
+                    sample_writer.write_sample(s);
+                }
+                sample_writer.flush().unwrap();
+            }
+
+            {
+                let mut sample_writer = writer.get_i16_writer(1024);
+                for s in 0i16..1024 {
+                    unsafe { sample_writer.write_sample_unchecked(s); }
+                }
+                sample_writer.flush().unwrap();
+            }
+        }
+        writer.finalize().unwrap();
+    }
+
+    {
+        buffer.set_position(0);
+        let mut reader = WavReader::new(&mut buffer).unwrap();
+        assert_eq!(write_spec, reader.spec());
+        assert_eq!(reader.len(), 2048);
+        for (expected, read) in (-1024_i16..1024).zip(reader.samples()) {
+            assert_eq!(expected, read.unwrap());
+        }
+    }
+}
+
+#[test]
+fn write_read_i8_is_lossless() {
+    let mut buffer = io::Cursor::new(Vec::new());
+    let write_spec = WavSpec {
+        channels: 16,
+        sample_rate: 48000,
+        bits_per_sample: 8,
+        sample_format: SampleFormat::Int,
+    };
+
+    // Write `i8` samples.
+    {
+        let mut writer = WavWriter::new(&mut buffer, write_spec).unwrap();
+        // Iterate over i16 because we cannot specify the upper bound otherwise.
+        for s in -128_i16..127 + 1 {
+            writer.write_sample(s as i8).unwrap();
+        }
+        writer.finalize().unwrap();
+    }
+
+    // Then read them into `i16`.
+    {
+        buffer.set_position(0);
+        let mut reader = WavReader::new(&mut buffer).unwrap();
+        assert_eq!(write_spec, reader.spec());
+        assert_eq!(reader.len(), 256);
+        for (expected, read) in (-128_i16..127 + 1).zip(reader.samples()) {
+            assert_eq!(expected, read.unwrap());
+        }
+    }
+}
+
+#[test]
+fn write_read_i24_is_lossless() {
+    let mut buffer = io::Cursor::new(Vec::new());
+    let write_spec = WavSpec {
+        channels: 16,
+        sample_rate: 96000,
+        bits_per_sample: 24,
+        sample_format: SampleFormat::Int,
+    };
+
+    // Write `i32` samples, but with at most 24 bits per sample.
+    {
+        let mut writer = WavWriter::new(&mut buffer, write_spec).unwrap();
+        for s in -128_i32..127 + 1 {
+            writer.write_sample(s * 256 * 256).unwrap();
+        }
+        writer.finalize().unwrap();
+    }
+
+    // Then read them into `i32`. This should extend the sign in the correct
+    // manner.
+    {
+        buffer.set_position(0);
+        let mut reader = WavReader::new(&mut buffer).unwrap();
+        assert_eq!(write_spec, reader.spec());
+        assert_eq!(reader.len(), 256);
+        for (expected, read) in (-128_i32..127 + 1)
+            .map(|x| x * 256 * 256)
+            .zip(reader.samples()) {
+            assert_eq!(expected, read.unwrap());
+        }
+    }
+}
+#[test]
+fn write_read_f32_is_lossless() {
+    let mut buffer = io::Cursor::new(Vec::new());
+    let write_spec = WavSpec {
+        channels: 2,
+        sample_rate: 44100,
+        bits_per_sample: 32,
+        sample_format: SampleFormat::Float,
+    };
+
+    {
+        let mut writer = WavWriter::new(&mut buffer, write_spec).unwrap();
+        for s in 1_u32..257 {
+            writer.write_sample(1.0f32 / s as f32).unwrap();
+        }
+        writer.finalize().unwrap();
+    }
+
+    {
+        buffer.set_position(0);
+        let mut reader = WavReader::new(&mut buffer).unwrap();
+        assert_eq!(write_spec, reader.spec());
+        assert_eq!(reader.len(), 256);
+        for (expected, read) in (1..257)
+            .map(|x| 1.0_f32 / x as f32)
+            .zip(reader.samples()) {
+            assert_eq!(expected, read.unwrap());
+        }
+    }
+}
+
+#[test]
+#[should_panic]
+fn no_32_bps_for_float_sample_format_panics() {
+    let mut buffer = io::Cursor::new(Vec::new());
+    let write_spec = WavSpec {
+        channels: 2,
+        sample_rate: 44100,
+        bits_per_sample: 16, // will panic, because value must be 32 for floating point
+        sample_format: SampleFormat::Float,
+    };
+
+    WavWriter::new(&mut buffer, write_spec).unwrap();
+}
+
+#[test]
+fn flush_should_produce_valid_file() {
+    use std::mem;
+    use std::io::Seek;
+
+    let mut buffer = io::Cursor::new(Vec::new());
+    let samples = &[2, 4, 5, 7, 11, 13];
+
+    {
+        let spec = WavSpec {
+            channels: 2,
+            sample_rate: 44100,
+            bits_per_sample: 16,
+            sample_format: SampleFormat::Int,
+        };
+        let mut writer = WavWriter::new(&mut buffer, spec).unwrap();
+
+        for &x in samples {
+            writer.write_sample(x).unwrap();
+        }
+
+        // We should be able to see everything up to the flush later.
+        writer.flush().unwrap();
+
+        // Write more samples. These should be in the buffer, but not read by the
+        // reader if we don't finalize the writer.
+        writer.write_sample(17).unwrap();
+        writer.write_sample(19).unwrap();
+
+        mem::forget(writer);
+    }
+
+    buffer.seek(io::SeekFrom::Start(0)).unwrap();
+
+    let mut reader = WavReader::new(&mut buffer).unwrap();
+    let read_samples: Vec<i16> = reader.samples()
+        .map(|r| r.unwrap())
+        .collect();
+
+    // We expect to see all samples up to the flush, but not the later ones.
+    assert_eq!(&read_samples[..], &samples[..]);
+}
+
+#[test]
+fn new_append_should_append() {
+    use std::io::Seek;
+
+    let mut buffer = io::Cursor::new(Vec::new());
+    let samples = &[2, 5, 7, 11];
+    let spec = WavSpec {
+        channels: 2,
+        sample_rate: 44100,
+        bits_per_sample: 16,
+        sample_format: SampleFormat::Int,
+    };
+
+    // Write initial file.
+    {
+        let mut writer = WavWriter::new(&mut buffer, spec).unwrap();
+        for s in samples { writer.write_sample(*s).unwrap(); }
+    }
+
+    buffer.seek(io::SeekFrom::Start(0)).unwrap();
+
+    // Append samples (the same ones a second time).
+    {
+        let mut writer = WavWriter::new_append(&mut buffer).unwrap();
+        assert_eq!(writer.spec(), spec);
+        for s in samples { writer.write_sample(*s).unwrap(); }
+    }
+
+    buffer.seek(io::SeekFrom::Start(0)).unwrap();
+
+    let mut reader = WavReader::new(&mut buffer).unwrap();
+    let read_samples: Vec<i16> = reader.samples()
+        .map(|r| r.unwrap())
+        .collect();
+
+    // We expect to see all samples up to the flush, but not the later ones.
+    assert_eq!(&read_samples[..], &[2, 5, 7, 11, 2, 5, 7, 11]);
+}
+
+#[test]
+fn new_append_does_not_corrupt_files() {
+    use std::io::Read;
+    use std::fs;
+
+    let sample_files = [
+        "testsamples/pcmwaveformat-16bit-44100Hz-mono-extra.wav",
+        "testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
+        "testsamples/pcmwaveformat-8bit-44100Hz-mono.wav",
+        "testsamples/pop.wav",
+        "testsamples/waveformatex-16bit-44100Hz-mono-extra.wav",
+        "testsamples/waveformatex-16bit-44100Hz-mono.wav",
+        "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
+        "testsamples/waveformatextensible-24bit-192kHz-mono.wav",
+        "testsamples/waveformatextensible-32bit-48kHz-stereo.wav",
+        "testsamples/nonstandard-01.wav",
+        "testsamples/nonstandard-02.wav",
+        "testsamples/waveformatex-8bit-11025Hz-mono.wav",
+    ];
+
+    for fname in &sample_files {
+        print!("testing {} ... ", fname);
+
+        let mut buffer = Vec::new();
+        let mut f = fs::File::open(fname).unwrap();
+        f.read_to_end(&mut buffer).unwrap();
+
+        let samples_orig: Vec<i32>;
+        let samples_after: Vec<i32>;
+
+        // Read samples first.
+        let mut cursor = io::Cursor::new(buffer);
+        {
+            let mut reader = WavReader::new(&mut cursor).unwrap();
+            samples_orig = reader.samples().map(|r| r.unwrap()).collect();
+        }
+        buffer = cursor.into_inner();
+
+        // Open in append mode and append one sample.
+        let mut cursor = io::Cursor::new(buffer);
+        {
+            let mut writer = WavWriter::new_append(&mut cursor).unwrap();
+            writer.write_sample(41_i8).unwrap();
+            writer.write_sample(43_i8).unwrap();
+        }
+        buffer = cursor.into_inner();
+
+        {
+            let cursor = io::Cursor::new(buffer);
+            let mut reader = WavReader::new(cursor)
+                .expect("Reading wav failed after append.");
+            samples_after = reader.samples().map(|r| r.unwrap()).collect();
+        }
+
+        assert_eq!(&samples_orig[..], &samples_after[..samples_orig.len()]);
+        assert_eq!(samples_after[samples_after.len() - 2], 41_i32);
+        assert_eq!(samples_after[samples_after.len() - 1], 43_i32);
+
+        println!("ok");
+    }
+}
+
+#[cfg(test)]
+fn assert_contents(fname: &str, expected: &[i16]) {
+    let mut reader = WavReader::open(fname).unwrap();
+    let samples: Vec<i16> = reader.samples().map(|s| s.unwrap()).collect();
+    assert_eq!(&samples[..], expected);
+}
+
+#[test]
+fn append_works_on_files() {
+    use std::fs;
+
+    let spec = WavSpec {
+        channels: 1,
+        sample_rate: 44100,
+        bits_per_sample: 16,
+        sample_format: SampleFormat::Int,
+    };
+
+    let mut writer = WavWriter::create("append.wav", spec).unwrap();
+    writer.write_sample(11_i16).unwrap();
+    writer.write_sample(13_i16).unwrap();
+    writer.write_sample(17_i16).unwrap();
+    writer.finalize().unwrap();
+
+    assert_contents("append.wav", &[11, 13, 17]);
+
+    let len = fs::metadata("append.wav").unwrap().len();
+
+    let mut appender = WavWriter::append("append.wav").unwrap();
+
+    appender.write_sample(19_i16).unwrap();
+    appender.write_sample(23_i16).unwrap();
+    appender.finalize().unwrap();
+
+    // We appended four bytes of audio data (2 16-bit samples), so the file
+    // should have grown by 4 bytes.
+    assert_eq!(fs::metadata("append.wav").unwrap().len(), len + 4);
+
+    assert_contents("append.wav", &[11, 13, 17, 19, 23]);
+}
+
+#[cfg(test)]
+#[test]
+fn test_into_header_for_infinite_file() {
+    let spec = WavSpec {
+        bits_per_sample: 16,
+        channels: 1,
+        sample_format: SampleFormat::Int,
+        sample_rate: 16000,
+    };
+    let v = spec.into_header_for_infinite_file();
+    assert_eq!(&v[..], &b"RIFF\xFF\xFF\xFF\xFFWAVE\
+fmt \x10\x00\x00\x00\x01\x00\x01\x00\x80\x3e\x00\x00\x00\x7d\x00\x00\x02\x00\x10\x00\
+data\xFF\xFF\xFF\xFF"[..]);
+
+    let spec = WavSpec {
+        bits_per_sample: 16,
+        channels: 10,
+        sample_format: SampleFormat::Int,
+        sample_rate: 16000,
+    };
+    let v = spec.into_header_for_infinite_file();
+    assert_eq!(&v[..], &b"RIFF\xFF\xFF\xFF\xFFWAVE\
+fmt \x28\x00\x00\x00\xfe\xff\x0a\x00\x80\x3e\x00\x00\x00\xe2\x04\x00\
+\x14\x00\x10\x00\x16\x00\x10\x00\xff\x03\x00\x00\x01\x00\x00\x00\
+\x00\x00\x10\x00\x80\x00\x00\xaa\x00\x38\x9b\x71\
+data\xFF\xFF\xFF\xFF"[..]);
+}
diff --git a/src/read.rs b/src/read.rs
new file mode 100644
index 0000000..9c9a9a4
--- /dev/null
+++ b/src/read.rs
@@ -0,0 +1,1244 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright (C) 2015 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+use std::cmp;
+use std::fs;
+use std::io;
+use std::marker;
+use std::path;
+use super::{Error, Result, Sample, SampleFormat, WavSpec, WavSpecEx};
+
+/// Extends the functionality of `io::Read` with additional methods.
+///
+/// The methods may be used on any type that implements `io::Read`.
+pub trait ReadExt: io::Read {
+    /// Reads as many bytes as `buf` is long.
+    ///
+    /// This may issue multiple `read` calls internally. An error is returned
+    /// if `read` read 0 bytes before the buffer is full.
+    //  TODO: There is an RFC proposing a method like this for the standard library.
+    fn read_into(&mut self, buf: &mut [u8]) -> io::Result<()>;
+
+    /// Reads 4 bytes and returns them in an array.
+    fn read_4_bytes(&mut self) -> io::Result<[u8; 4]>;
+
+    /// Skip over `n` bytes.
+    fn skip_bytes(&mut self, n: usize) -> io::Result<()>;
+
+    /// Reads a single byte and interprets it as an 8-bit signed integer.
+    fn read_i8(&mut self) -> io::Result<i8>;
+
+    /// Reads a single byte and interprets it as an 8-bit unsigned integer.
+    fn read_u8(&mut self) -> io::Result<u8>;
+
+    /// Reads two bytes and interprets them as a little-endian 16-bit signed integer.
+    fn read_le_i16(&mut self) -> io::Result<i16>;
+
+    /// Reads two bytes and interprets them as a little-endian 16-bit unsigned integer.
+    fn read_le_u16(&mut self) -> io::Result<u16>;
+
+    /// Reads three bytes and interprets them as a little-endian 24-bit signed integer.
+    ///
+    /// The sign bit will be extended into the most significant byte.
+    fn read_le_i24(&mut self) -> io::Result<i32>;
+
+    /// Reads four bytes and interprets them as a little-endian 24-bit signed integer.
+    ///
+    /// The sign bit will be extended into the most significant byte.
+    fn read_le_i24_4(&mut self) -> io::Result<i32>;
+
+    /// Reads three bytes and interprets them as a little-endian 24-bit unsigned integer.
+    ///
+    /// The most significant byte will be 0.
+    fn read_le_u24(&mut self) -> io::Result<u32>;
+
+    /// Reads four bytes and interprets them as a little-endian 32-bit signed integer.
+    fn read_le_i32(&mut self) -> io::Result<i32>;
+
+    /// Reads four bytes and interprets them as a little-endian 32-bit unsigned integer.
+    fn read_le_u32(&mut self) -> io::Result<u32>;
+
+    /// Reads four bytes and interprets them as a little-endian 32-bit IEEE float.
+    fn read_le_f32(&mut self) -> io::Result<f32>;
+}
+
+impl<R> ReadExt for R
+    where R: io::Read
+{
+    #[inline(always)]
+    fn read_into(&mut self, buf: &mut [u8]) -> io::Result<()> {
+        let mut n = 0;
+        while n < buf.len() {
+            let progress = try!(self.read(&mut buf[n..]));
+            if progress > 0 {
+                n += progress;
+            } else {
+                return Err(io::Error::new(io::ErrorKind::Other, "Failed to read enough bytes."));
+            }
+        }
+        Ok(())
+    }
+
+    #[inline(always)]
+    fn skip_bytes(&mut self, n: usize) -> io::Result<()> {
+        // Read from the input in chunks of 1024 bytes at a time, and discard
+        // the result. 1024 is a tradeoff between doing a lot of calls, and
+        // using too much stack space. This method is not in a hot path, so it
+        // can afford to do this.
+        let mut n_read = 0;
+        let mut buf = [0u8; 1024];
+        while n_read < n {
+            let end = cmp::min(n - n_read, 1024);
+            let progress = try!(self.read(&mut buf[0..end]));
+            if progress > 0 {
+                n_read += progress;
+            } else {
+                return Err(io::Error::new(io::ErrorKind::Other, "Failed to read enough bytes."));
+            }
+        }
+        Ok(())
+    }
+
+    #[inline(always)]
+    fn read_4_bytes(&mut self) -> io::Result<[u8; 4]> {
+        let mut buf = [0_u8; 4];
+        try!(self.read_into(&mut buf[..]));
+        Ok(buf)
+    }
+
+    #[inline(always)]
+    fn read_i8(&mut self) -> io::Result<i8> {
+        self.read_u8().map(|x| x as i8)
+    }
+
+    #[inline(always)]
+    fn read_u8(&mut self) -> io::Result<u8> {
+        let mut buf = [0u8; 1];
+        try!(self.read_into(&mut buf));
+        Ok(buf[0])
+    }
+
+    #[inline(always)]
+    fn read_le_i16(&mut self) -> io::Result<i16> {
+        self.read_le_u16().map(|x| x as i16)
+    }
+
+    #[inline(always)]
+    fn read_le_u16(&mut self) -> io::Result<u16> {
+        let mut buf = [0u8; 2];
+        try!(self.read_into(&mut buf));
+        Ok((buf[1] as u16) << 8 | (buf[0] as u16))
+    }
+
+    #[inline(always)]
+    fn read_le_i24(&mut self) -> io::Result<i32> {
+        self.read_le_u24().map(|x|
+            // Test the sign bit, if it is set, extend the sign bit into the
+            // most significant byte.
+            if x & (1 << 23) == 0 {
+                x as i32
+            } else {
+                (x | 0xff_00_00_00) as i32
+            }
+        )
+    }
+
+    #[inline(always)]
+    fn read_le_i24_4(&mut self) -> io::Result<i32> {
+        self.read_le_u32().map(|x|
+            // Test the sign bit, if it is set, extend the sign bit into the
+            // most significant byte. Otherwise, mask out the top byte.
+            if x & (1 << 23) == 0 {
+                (x & 0x00_ff_ff_ff) as i32
+            } else {
+                (x | 0xff_00_00_00) as i32
+            }
+        )
+    }
+
+    #[inline(always)]
+    fn read_le_u24(&mut self) -> io::Result<u32> {
+        let mut buf = [0u8; 3];
+        try!(self.read_into(&mut buf));
+        Ok((buf[2] as u32) << 16 | (buf[1] as u32) << 8 | (buf[0] as u32))
+    }
+
+    #[inline(always)]
+    fn read_le_i32(&mut self) -> io::Result<i32> {
+        self.read_le_u32().map(|x| x as i32)
+    }
+
+    #[inline(always)]
+    fn read_le_u32(&mut self) -> io::Result<u32> {
+        let mut buf = [0u8; 4];
+        try!(self.read_into(&mut buf));
+        Ok((buf[3] as u32) << 24 | (buf[2] as u32) << 16 |
+           (buf[1] as u32) << 8  | (buf[0] as u32) << 0)
+    }
+
+    #[inline(always)]
+    fn read_le_f32(&mut self) -> io::Result<f32> {
+        let mut buf = [0u8; 4];
+        try!(self.read_into(&mut buf));
+        Ok(f32::from_le_bytes(buf))
+    }
+}
+
+/// The different chunks that a WAVE file can contain.
+enum ChunkKind {
+    Fmt,
+    Fact,
+    Data,
+    Unknown,
+}
+
+/// Describes the structure of a chunk in the WAVE file.
+struct ChunkHeader {
+    pub kind: ChunkKind,
+    pub len: u32,
+}
+
+/// A reader that reads the WAVE format from the underlying reader.
+///
+/// A `WavReader` is a streaming reader. It reads data from the underlying
+/// reader on demand, and it reads no more than strictly necessary. No internal
+/// buffering is performed on the underlying reader, but this can easily be
+/// added by wrapping the reader in an `io::BufReader`. The `open` constructor
+/// takes care of this for you.
+pub struct WavReader<R> {
+    /// Specification of the file as found in the fmt chunk.
+    spec: WavSpec,
+
+    /// The number of bytes used to store a sample in the stream.
+    bytes_per_sample: u16,
+
+    /// The number of samples in the data chunk.
+    ///
+    /// The data chunk is limited to a 4 GiB length because its header has a
+    /// 32-bit length field. A sample takes at least one byte to store, so the
+    /// number of samples is always less than 2^32.
+    num_samples: u32,
+
+    /// The number of samples read so far.
+    samples_read: u32,
+
+    /// The reader from which the WAVE format is read.
+    reader: R,
+}
+
+/// An iterator that yields samples of type `S` read from a `WavReader`.
+///
+/// The type `S` must have at least as many bits as the bits per sample of the
+/// file, otherwise every iteration will return an error.
+pub struct WavSamples<'wr, R, S>
+    where R: 'wr
+{
+    reader: &'wr mut WavReader<R>,
+    phantom_sample: marker::PhantomData<S>,
+}
+
+/// An iterator that yields samples of type `S` read from a `WavReader`.
+///
+/// The type `S` must have at least as many bits as the bits per sample of the
+/// file, otherwise every iteration will return an error.
+pub struct WavIntoSamples<R, S> {
+    reader: WavReader<R>,
+    phantom_sample: marker::PhantomData<S>,
+}
+
+/// Reads the RIFF WAVE header, returns the supposed file size.
+///
+/// This function can be used to quickly check if the file could be a wav file
+/// by reading 12 bytes of the header. If an `Ok` is returned, the file is
+/// likely a wav file. If an `Err` is returned, it is definitely not a wav
+/// file.
+///
+/// The returned file size cannot be larger than 2<sup>32</sup> + 7 bytes.
+pub fn read_wave_header<R: io::Read>(reader: &mut R) -> Result<u64> {
+    // Every WAVE file starts with the four bytes 'RIFF' and a file length.
+    // TODO: the old approach of having a slice on the stack and reading
+    // into it is more cumbersome, but also avoids a heap allocation. Is
+    // the compiler smart enough to avoid the heap allocation anyway? I
+    // would not expect it to be.
+    if b"RIFF" != &try!(reader.read_4_bytes())[..] {
+        return Err(Error::FormatError("no RIFF tag found"));
+    }
+
+    let file_len = try!(reader.read_le_u32());
+
+    // Next four bytes indicate the file type, which should be WAVE.
+    if b"WAVE" != &try!(reader.read_4_bytes())[..] {
+        return Err(Error::FormatError("no WAVE tag found"));
+    }
+
+    // The stored file length does not include the "RIFF" magic and 4-byte
+    // length field, so the total size is 8 bytes more than what is stored.
+    Ok(file_len as u64 + 8)
+}
+
+/// Reads chunks until a data chunk is encountered.
+///
+/// Returns the information from the fmt chunk and the length of the data
+/// chunk in bytes. Afterwards, the reader will be positioned at the first
+/// content byte of the data chunk.
+pub fn read_until_data<R: io::Read>(mut reader: R) -> Result<(WavSpecEx, u32)> {
+    let mut spec_opt = None;
+
+    loop {
+        let header = try!(WavReader::read_chunk_header(&mut reader));
+        match header.kind {
+            ChunkKind::Fmt => {
+                let spec = try!(WavReader::read_fmt_chunk(&mut reader, header.len));
+                spec_opt = Some(spec);
+            }
+            ChunkKind::Fact => {
+                // All (compressed) non-PCM formats must have a fact chunk
+                // (Rev. 3 documentation). The chunk contains at least one
+                // value, the number of samples in the file.
+                //
+                // The number of samples field is redundant for sampled
+                // data, since the Data chunk indicates the length of the
+                // data. The number of samples can be determined from the
+                // length of the data and the container size as determined
+                // from the Format chunk.
+                // http://www-mmsp.ece.mcgill.ca/documents/audioformats/wave/wave.html
+                let _samples_per_channel = reader.read_le_u32();
+            }
+            ChunkKind::Data => {
+                // The "fmt" chunk must precede the "data" chunk. Any
+                // chunks that come after the data chunk will be ignored.
+                if let Some(spec) = spec_opt {
+                    return Ok((spec, header.len));
+                } else {
+                    return Err(Error::FormatError("missing fmt chunk"));
+                }
+            }
+            ChunkKind::Unknown => {
+                // Ignore the chunk; skip all of its bytes.
+                try!(reader.skip_bytes(header.len as usize));
+            }
+        }
+        // If no data chunk is ever encountered, the function will return
+        // via one of the try! macros that return an Err on end of file.
+    }
+}
+
+impl<R> WavReader<R>
+    where R: io::Read
+{
+    /// Attempts to read an 8-byte chunk header.
+    fn read_chunk_header(reader: &mut R) -> Result<ChunkHeader> {
+        let mut kind_str = [0; 4];
+        try!(reader.read_into(&mut kind_str));
+        let len = try!(reader.read_le_u32());
+
+        let kind = match &kind_str[..] {
+            b"fmt " => ChunkKind::Fmt,
+            b"fact" => ChunkKind::Fact,
+            b"data" => ChunkKind::Data,
+            _ => ChunkKind::Unknown,
+        };
+
+        Ok(ChunkHeader { kind: kind, len: len })
+    }
+
+    /// Reads the fmt chunk of the file, returns the information it provides.
+    fn read_fmt_chunk(reader: &mut R, chunk_len: u32) -> Result<WavSpecEx> {
+        // A minimum chunk length of at least 16 is assumed. Note: actually,
+        // the first 14 bytes contain enough information to fully specify the
+        // file. I have not encountered a file with a 14-byte fmt section
+        // though. If you ever encounter such file, please contact me.
+        if chunk_len < 16 {
+            return Err(Error::FormatError("invalid fmt chunk size"));
+        }
+
+        // Read the WAVEFORMAT struct, as defined at
+        // https://msdn.microsoft.com/en-us/library/ms713498.aspx.
+        // ```
+        // typedef struct {
+        //     WORD  wFormatTag;
+        //     WORD  nChannels;
+        //     DWORD nSamplesPerSec;
+        //     DWORD nAvgBytesPerSec;
+        //     WORD  nBlockAlign;
+        // } WAVEFORMAT;
+        // ```
+        // The WAVEFORMATEX struct has two more members, as defined at
+        // https://msdn.microsoft.com/en-us/library/ms713497.aspx
+        // ```
+        // typedef struct {
+        //     WORD  wFormatTag;
+        //     WORD  nChannels;
+        //     DWORD nSamplesPerSec;
+        //     DWORD nAvgBytesPerSec;
+        //     WORD  nBlockAlign;
+        //     WORD  wBitsPerSample;
+        //     WORD  cbSize;
+        // } WAVEFORMATEX;
+        // ```
+        // There is also PCMWAVEFORMAT as defined at
+        // https://msdn.microsoft.com/en-us/library/dd743663.aspx.
+        // ```
+        // typedef struct {
+        //   WAVEFORMAT wf;
+        //   WORD       wBitsPerSample;
+        // } PCMWAVEFORMAT;
+        // ```
+        // In either case, the minimal length of the fmt section is 16 bytes,
+        // meaning that it does include the `wBitsPerSample` field. (The name
+        // is misleading though, because it is the number of bits used to store
+        // a sample, not all of the bits need to be valid for all versions of
+        // the WAVE format.)
+        let format_tag = try!(reader.read_le_u16());
+        let n_channels = try!(reader.read_le_u16());
+        let n_samples_per_sec = try!(reader.read_le_u32());
+        let n_bytes_per_sec = try!(reader.read_le_u32());
+        let block_align = try!(reader.read_le_u16());
+        let bits_per_sample = try!(reader.read_le_u16());
+
+        if n_channels == 0 {
+            return Err(Error::FormatError("file contains zero channels"));
+        }
+
+        let bytes_per_sample = block_align / n_channels;
+        // We allow bits_per_sample to be less than bytes_per_sample so that
+        // we can support things such as 24 bit samples in 4 byte containers.
+        if Some(bits_per_sample) > bytes_per_sample.checked_mul(8) {
+            return Err(Error::FormatError("sample bits exceeds size of sample"));
+        }
+
+        // This field is redundant, and may be ignored. We do validate it to
+        // fail early for ill-formed files.
+        if Some(n_bytes_per_sec) != (block_align as u32).checked_mul(n_samples_per_sec) {
+            return Err(Error::FormatError("inconsistent fmt chunk"));
+        }
+
+        // The bits per sample for a WAVEFORMAT struct is the number of bits
+        // used to store a sample. Therefore, it must be a multiple of 8.
+        if bits_per_sample % 8 != 0 {
+            return Err(Error::FormatError("bits per sample is not a multiple of 8"));
+        }
+
+        if bits_per_sample == 0 {
+            return Err(Error::FormatError("bits per sample is 0"));
+        }
+
+        let mut spec = WavSpec {
+            channels: n_channels,
+            sample_rate: n_samples_per_sec,
+            bits_per_sample: bits_per_sample,
+            sample_format: SampleFormat::Int,
+        };
+
+        // The different format tag definitions can be found in mmreg.h that is
+        // part of the Windows SDK. The vast majority are esoteric vendor-
+        // specific formats. We handle only a few. The following values could
+        // be of interest:
+        const PCM: u16 = 0x0001;
+        const ADPCM: u16 = 0x0002;
+        const IEEE_FLOAT: u16 = 0x0003;
+        const EXTENSIBLE: u16 = 0xfffe;
+        // We may update our WavSpec based on more data we read from the header.
+        match format_tag {
+            PCM => try!(WavReader::read_wave_format_pcm(reader, chunk_len, &spec)),
+            ADPCM => return Err(Error::Unsupported),
+            IEEE_FLOAT => try!(WavReader::read_wave_format_ieee_float(reader, chunk_len, &mut spec)),
+            EXTENSIBLE => try!(WavReader::read_wave_format_extensible(reader, chunk_len, &mut spec)),
+            _ => return Err(Error::Unsupported),
+        };
+
+        Ok(WavSpecEx {
+            spec: spec,
+            bytes_per_sample: bytes_per_sample,
+        })
+    }
+
+    fn read_wave_format_pcm(mut reader: R, chunk_len: u32, spec: &WavSpec) -> Result<()> {
+        // When there is a PCMWAVEFORMAT struct, the chunk is 16 bytes long.
+        // The WAVEFORMATEX structs includes two extra bytes, `cbSize`.
+        let is_wave_format_ex = match chunk_len {
+            16 => false,
+            18 => true,
+            // Other sizes are unexpected, but such files do occur in the wild,
+            // and reading these files is still possible, so we allow this.
+            40 => true,
+            _ => return Err(Error::FormatError("unexpected fmt chunk size")),
+        };
+
+        if is_wave_format_ex {
+            // `cbSize` can be used for non-PCM formats to specify the size of
+            // additional data. However, for WAVE_FORMAT_PCM, the member should
+            // be ignored, see https://msdn.microsoft.com/en-us/library/ms713497.aspx.
+            // Nonzero values do in fact occur in practice.
+            let _cb_size = try!(reader.read_le_u16());
+
+            // For WAVE_FORMAT_PCM in WAVEFORMATEX, only 8 or 16 bits per
+            // sample are valid according to
+            // https://msdn.microsoft.com/en-us/library/ms713497.aspx.
+            // 24 bits per sample is explicitly not valid inside a WAVEFORMATEX
+            // structure, but such files do occur in the wild nonetheless, and
+            // there is no good reason why we couldn't read them.
+            match spec.bits_per_sample {
+                8 => {}
+                16 => {}
+                24 => {}
+                _ => return Err(Error::FormatError("bits per sample is not 8 or 16")),
+            }
+        }
+
+        // If the chunk len was longer than expected, ignore the additional bytes.
+        if chunk_len == 40 {
+            try!(reader.skip_bytes(22));
+        }
+        Ok(())
+    }
+
+    fn read_wave_format_ieee_float(mut reader: R, chunk_len: u32, spec: &mut WavSpec) -> Result<()> {
+        // When there is a PCMWAVEFORMAT struct, the chunk is 16 bytes long.
+        // The WAVEFORMATEX structs includes two extra bytes, `cbSize`.
+        let is_wave_format_ex = chunk_len == 18;
+
+        if !is_wave_format_ex && chunk_len != 16 {
+            return Err(Error::FormatError("unexpected fmt chunk size"));
+        }
+
+        if is_wave_format_ex {
+            // For WAVE_FORMAT_IEEE_FLOAT which we are reading, there should
+            // be no extra data, so `cbSize` should be 0.
+            let cb_size = try!(reader.read_le_u16());
+            if cb_size != 0 {
+                return Err(Error::FormatError("unexpected WAVEFORMATEX size"));
+            }
+        }
+
+        // For WAVE_FORMAT_IEEE_FLOAT, the bits_per_sample field should be
+        // set to `32` according to
+        // https://msdn.microsoft.com/en-us/library/windows/hardware/ff538799(v=vs.85).aspx.
+        //
+        // Note that some applications support 64 bits per sample. This is
+        // not yet supported by hound.
+        if spec.bits_per_sample != 32 {
+            return Err(Error::FormatError("bits per sample is not 32"));
+        }
+
+        spec.sample_format = SampleFormat::Float;
+        Ok(())
+    }
+
+    fn read_wave_format_extensible(mut reader: R, chunk_len: u32, spec: &mut WavSpec) -> Result<()> {
+        // 16 bytes were read already, there must be two more for the `cbSize`
+        // field, and `cbSize` itself must be at least 22, so the chunk length
+        // must be at least 40.
+        if chunk_len < 40 {
+            return Err(Error::FormatError("unexpected fmt chunk size"));
+        }
+
+        // `cbSize` is the last field of the WAVEFORMATEX struct.
+        let cb_size = try!(reader.read_le_u16());
+
+        // `cbSize` must be at least 22, but in this case we assume that it is
+        // 22, because we would not know how to handle extra data anyway.
+        if cb_size != 22 {
+            return Err(Error::FormatError("unexpected WAVEFORMATEXTENSIBLE size"));
+        }
+
+        // What follows is the rest of the `WAVEFORMATEXTENSIBLE` struct, as
+        // defined at https://msdn.microsoft.com/en-us/library/ms713496.aspx.
+        // ```
+        // typedef struct {
+        //   WAVEFORMATEX  Format;
+        //   union {
+        //     WORD  wValidBitsPerSample;
+        //     WORD  wSamplesPerBlock;
+        //     WORD  wReserved;
+        //   } Samples;
+        //   DWORD   dwChannelMask;
+        //   GUID    SubFormat;
+        // } WAVEFORMATEXTENSIBLE, *PWAVEFORMATEXTENSIBLE;
+        // ```
+        let valid_bits_per_sample = try!(reader.read_le_u16());
+        let _channel_mask = try!(reader.read_le_u32()); // Not used for now.
+        let mut subformat = [0u8; 16];
+        try!(reader.read_into(&mut subformat));
+
+        // Several GUIDS are defined. At the moment, only the following are supported:
+        //
+        // * KSDATAFORMAT_SUBTYPE_PCM (PCM audio with integer samples).
+        // * KSDATAFORMAT_SUBTYPE_IEEE_FLOAT (PCM audio with floating point samples).
+        let sample_format = match subformat {
+            super::KSDATAFORMAT_SUBTYPE_PCM => SampleFormat::Int,
+            super::KSDATAFORMAT_SUBTYPE_IEEE_FLOAT => SampleFormat::Float,
+            _ => return Err(Error::Unsupported),
+        };
+
+        // Fallback to bits_per_sample if the valid_bits_per_sample is obviously wrong to support non standard headers found in the wild.
+        if valid_bits_per_sample > 0 {
+            spec.bits_per_sample = valid_bits_per_sample;
+        }
+
+        spec.sample_format = sample_format;
+        Ok(())
+    }
+
+    /// Attempts to create a reader that reads the WAVE format.
+    ///
+    /// The header is read immediately. Reading the data will be done on
+    /// demand.
+    pub fn new(mut reader: R) -> Result<WavReader<R>> {
+        try!(read_wave_header(&mut reader));
+        let (spec_ex, data_len) = try!(read_until_data(&mut reader));
+
+        let num_samples = data_len / spec_ex.bytes_per_sample as u32;
+
+        // It could be that num_samples * bytes_per_sample < data_len.
+        // If data_len is not a multiple of bytes_per_sample, there is some
+        // trailing data. Either somebody is playing some steganography game,
+        // but more likely something is very wrong, and we should refuse to
+        // decode the file, as it is invalid.
+        if num_samples * spec_ex.bytes_per_sample as u32 != data_len {
+            let msg = "data chunk length is not a multiple of sample size";
+            return Err(Error::FormatError(msg));
+        }
+
+        // The number of samples must be a multiple of the number of channels,
+        // otherwise the last inter-channel sample would not have data for all
+        // channels.
+        if num_samples % spec_ex.spec.channels as u32 != 0 {
+            return Err(Error::FormatError("invalid data chunk length"));
+        }
+
+        let wav_reader = WavReader {
+            spec: spec_ex.spec,
+            bytes_per_sample: spec_ex.bytes_per_sample,
+            num_samples: num_samples,
+            samples_read: 0,
+            reader: reader,
+        };
+
+        Ok(wav_reader)
+    }
+
+    /// Returns information about the WAVE file.
+    pub fn spec(&self) -> WavSpec {
+        self.spec
+    }
+
+    /// Returns an iterator over all samples.
+    ///
+    /// The channel data is interleaved. The iterator is streaming. That is,
+    /// if you call this method once, read a few samples, and call this method
+    /// again, the second iterator will not start again from the beginning of
+    /// the file, it will continue where the first iterator stopped.
+    ///
+    /// The type `S` must have at least `spec().bits_per_sample` bits,
+    /// otherwise every iteration will return an error. All bit depths up to
+    /// 32 bits per sample can be decoded into an `i32`, but if you know
+    /// beforehand that you will be reading a file with 16 bits per sample, you
+    /// can save memory by decoding into an `i16`.
+    ///
+    /// The type of `S` (int or float) must match `spec().sample_format`,
+    /// otherwise every iteration will return an error.
+    pub fn samples<'wr, S: Sample>(&'wr mut self) -> WavSamples<'wr, R, S> {
+        WavSamples {
+            reader: self,
+            phantom_sample: marker::PhantomData,
+        }
+    }
+
+    /// Same as `samples`, but takes ownership of the `WavReader`.
+    ///
+    /// See `samples()` for more info.
+    pub fn into_samples<S: Sample>(self) -> WavIntoSamples<R, S> {
+        WavIntoSamples {
+            reader: self,
+            phantom_sample: marker::PhantomData,
+        }
+    }
+
+    /// Returns the duration of the file in samples.
+    ///
+    /// The duration is independent of the number of channels. It is expressed
+    /// in units of samples. The duration in seconds can be obtained by
+    /// dividing this number by the sample rate. The duration is independent of
+    /// how many samples have been read already.
+    pub fn duration(&self) -> u32 {
+        self.num_samples / self.spec.channels as u32
+    }
+
+    /// Returns the number of values that the sample iterator will yield.
+    ///
+    /// The length of the file is its duration (in samples) times the number of
+    /// channels. The length is independent of how many samples have been read
+    /// already. To get the number of samples left, use `len()` on the
+    /// `samples()` iterator.
+    pub fn len(&self) -> u32 {
+        self.num_samples
+    }
+
+    /// Destroys the `WavReader` and returns the underlying reader.
+    pub fn into_inner(self) -> R {
+        self.reader
+    }
+
+    /// Seek to the given time within the file.
+    ///
+    /// The given time is measured in number of samples (independent of the
+    /// number of channels) since the beginning of the audio data. To seek to
+    /// a particular time in seconds, multiply the number of seconds with
+    /// `WavSpec::sample_rate`. The given time should not exceed the duration of
+    /// the file (returned by `duration()`). The behavior when seeking beyond
+    /// `duration()` depends on the reader's `Seek` implementation.
+    ///
+    /// This method requires that the inner reader `R` implements `Seek`.
+    pub fn seek(&mut self, time: u32) -> io::Result<()>
+        where R: io::Seek,
+    {
+        let bytes_per_sample = self.spec.bits_per_sample / 8;
+        let sample_position = time * self.spec.channels as u32;
+        let offset_samples = sample_position as i64 - self.samples_read as i64;
+        let offset_bytes = offset_samples * bytes_per_sample as i64;
+        try!(self.reader.seek(io::SeekFrom::Current(offset_bytes)));
+        self.samples_read = sample_position;
+        Ok(())
+    }
+}
+
+impl WavReader<io::BufReader<fs::File>> {
+    /// Attempts to create a reader that reads from the specified file.
+    ///
+    /// This is a convenience constructor that opens a `File`, wraps it in a
+    /// `BufReader` and then constructs a `WavReader` from it.
+    pub fn open<P: AsRef<path::Path>>(filename: P) -> Result<WavReader<io::BufReader<fs::File>>> {
+        let file = try!(fs::File::open(filename));
+        let buf_reader = io::BufReader::new(file);
+        WavReader::new(buf_reader)
+    }
+}
+
+fn iter_next<R, S>(reader: &mut WavReader<R>) -> Option<Result<S>>
+    where R: io::Read,
+          S: Sample
+{
+    if reader.samples_read < reader.num_samples {
+        reader.samples_read += 1;
+        let sample = Sample::read(&mut reader.reader,
+                                  reader.spec.sample_format,
+                                  reader.bytes_per_sample,
+                                  reader.spec.bits_per_sample);
+        Some(sample.map_err(Error::from))
+    } else {
+        None
+    }
+}
+
+fn iter_size_hint<R>(reader: &WavReader<R>) -> (usize, Option<usize>) {
+    let samples_left = reader.num_samples - reader.samples_read;
+    (samples_left as usize, Some(samples_left as usize))
+}
+
+impl<'wr, R, S> Iterator for WavSamples<'wr, R, S>
+    where R: io::Read,
+          S: Sample
+{
+    type Item = Result<S>;
+
+    fn next(&mut self) -> Option<Result<S>> {
+        iter_next(&mut self.reader)
+    }
+
+    fn size_hint(&self) -> (usize, Option<usize>) {
+        iter_size_hint(&self.reader)
+    }
+}
+
+impl<'wr, R, S> ExactSizeIterator for WavSamples<'wr, R, S>
+    where R: io::Read,
+          S: Sample
+{
+}
+
+impl<R, S> Iterator for WavIntoSamples<R, S>
+    where R: io::Read,
+          S: Sample
+{
+    type Item = Result<S>;
+
+    fn next(&mut self) -> Option<Result<S>> {
+        iter_next(&mut self.reader)
+    }
+
+    fn size_hint(&self) -> (usize, Option<usize>) {
+        iter_size_hint(&self.reader)
+    }
+}
+
+impl<R, S> ExactSizeIterator for WavIntoSamples<R, S>
+    where R: io::Read,
+          S: Sample
+{
+}
+
+#[test]
+fn duration_and_len_agree() {
+    let files = &["testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
+                  "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
+                  "testsamples/waveformatextensible-32bit-48kHz-stereo.wav"];
+
+    for fname in files {
+        let reader = WavReader::open(fname).unwrap();
+        assert_eq!(reader.spec().channels as u32 * reader.duration(),
+                   reader.len());
+    }
+}
+
+/// Tests reading a wave file with the PCMWAVEFORMAT struct.
+#[test]
+fn read_wav_pcm_wave_format_pcm() {
+    let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-16bit-44100Hz-mono.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().sample_rate, 44100);
+    assert_eq!(wav_reader.spec().bits_per_sample, 16);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i16> = wav_reader.samples()
+        .map(|r| r.unwrap())
+        .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[2, -3, 5, -7]);
+}
+
+#[test]
+fn read_wav_skips_unknown_chunks() {
+    // The test samples are the same as without the -extra suffix, but ffmpeg
+    // has kindly added some useless chunks in between the fmt and data chunk.
+    let files = ["testsamples/pcmwaveformat-16bit-44100Hz-mono-extra.wav",
+                 "testsamples/waveformatex-16bit-44100Hz-mono-extra.wav"];
+
+    for file in &files {
+        let mut wav_reader = WavReader::open(file).unwrap();
+
+        assert_eq!(wav_reader.spec().channels, 1);
+        assert_eq!(wav_reader.spec().sample_rate, 44100);
+        assert_eq!(wav_reader.spec().bits_per_sample, 16);
+        assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+        let sample = wav_reader.samples::<i16>().next().unwrap().unwrap();
+        assert_eq!(sample, 2);
+    }
+}
+
+#[test]
+fn read_wav_0_valid_bits_fallback() {
+    let mut wav_reader = WavReader::open("testsamples/nonstandard-02.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 2);
+    assert_eq!(wav_reader.spec().sample_rate, 48000);
+    assert_eq!(wav_reader.spec().bits_per_sample, 32);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+        .map(|r| r.unwrap())
+        .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[19, -229373, 33587161, -2147483497]);
+}
+
+#[test]
+fn len_and_size_hint_are_correct() {
+    let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-16bit-44100Hz-mono.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.len(), 4);
+
+    {
+        let mut samples = wav_reader.samples::<i16>();
+
+        assert_eq!(samples.size_hint(), (4, Some(4)));
+        samples.next();
+        assert_eq!(samples.size_hint(), (3, Some(3)));
+    }
+
+    // Reading should not affect the initial length.
+    assert_eq!(wav_reader.len(), 4);
+
+    // Creating a new iterator resumes where the previous iterator stopped.
+    {
+        let mut samples = wav_reader.samples::<i16>();
+
+        assert_eq!(samples.size_hint(), (3, Some(3)));
+        samples.next();
+        assert_eq!(samples.size_hint(), (2, Some(2)));
+    }
+}
+
+#[test]
+fn size_hint_is_exact() {
+    let files = &["testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
+                  "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
+                  "testsamples/waveformatextensible-32bit-48kHz-stereo.wav"];
+
+    for fname in files {
+        let mut reader = WavReader::open(fname).unwrap();
+        let len = reader.len();
+        let mut iter = reader.samples::<i32>();
+        for i in 0..len {
+            let remaining = (len - i) as usize;
+            assert_eq!(iter.size_hint(), (remaining, Some(remaining)));
+            assert!(iter.next().is_some());
+        }
+        assert!(iter.next().is_none());
+    }
+}
+
+#[test]
+fn samples_equals_into_samples() {
+    let wav_reader_val = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav").unwrap();
+    let mut wav_reader_ref = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav").unwrap();
+
+    let samples_val: Vec<i16> = wav_reader_val.into_samples()
+                                              .map(|r| r.unwrap())
+                                              .collect();
+
+    let samples_ref: Vec<i16> = wav_reader_ref.samples()
+                                              .map(|r| r.unwrap())
+                                              .collect();
+
+    assert_eq!(samples_val, samples_ref);
+}
+
+/// Tests reading a wave file with the WAVEFORMATEX struct.
+#[test]
+fn read_wav_wave_format_ex_pcm() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatex-16bit-44100Hz-mono.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().sample_rate, 44100);
+    assert_eq!(wav_reader.spec().bits_per_sample, 16);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i16> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[2, -3, 5, -7]);
+}
+
+#[test]
+fn read_wav_wave_format_ex_ieee_float() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatex-ieeefloat-44100Hz-mono.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().sample_rate, 44100);
+    assert_eq!(wav_reader.spec().bits_per_sample, 32);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Float);
+
+    let samples: Vec<f32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[2.0, 3.0, -16411.0, 1019.0]);
+}
+
+#[test]
+fn read_wav_stereo() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatex-16bit-44100Hz-stereo.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 2);
+    assert_eq!(wav_reader.spec().sample_rate, 44100);
+    assert_eq!(wav_reader.spec().bits_per_sample, 16);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i16> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact eight samples.
+    assert_eq!(&samples[..], &[2, -3, 5, -7, 11, -13, 17, -19]);
+
+}
+
+#[test]
+fn read_wav_pcm_wave_format_8bit() {
+    let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav")
+                                   .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().bits_per_sample, 8);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i16> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[19, -53, 89, -127]);
+}
+
+/// Test reading 24 bit samples in a 4 byte container using the pcmwaveformat header. This is
+/// technically a non-compliant wave file, but it is the sort of file generated by
+/// 'arecord -f S24_LE -r 48000 -c 2 input.wav' so it should be supported.
+#[test]
+fn read_wav_pcm_wave_format_24bit_4byte() {
+    let mut wav_reader = WavReader::open("testsamples/pcmwaveformat-24bit-4byte-48kHz-stereo.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 2);
+    assert_eq!(wav_reader.spec().sample_rate, 48_000);
+    assert_eq!(wav_reader.spec().bits_per_sample, 24);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[-96, 23_052, 8_388_607, -8_360_672]);
+}
+
+/// Regression test for a real-world wav file encountered in Quake.
+#[test]
+fn read_wav_wave_format_ex_8bit() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatex-8bit-11025Hz-mono.wav").unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().bits_per_sample, 8);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The audio data has been zeroed out, but for 8-bit files, a zero means a
+    // sample value of 128.
+    assert_eq!(&samples[..], &[-128, -128, -128, -128]);
+}
+
+/// This test sample tests both reading the WAVEFORMATEXTENSIBLE header, and 24-bit samples.
+#[test]
+fn read_wav_wave_format_extensible_pcm_24bit() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatextensible-24bit-192kHz-mono.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().sample_rate, 192_000);
+    assert_eq!(wav_reader.spec().bits_per_sample, 24);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[-17, 4_194_319, -6_291_437, 8_355_817]);
+}
+
+/// This test sample tests both reading the WAVEFORMATEXTENSIBLE header, and 24-bit samples with a
+/// 4 byte container size.
+#[test]
+fn read_wav_wave_format_extensible_pcm_24bit_4byte() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatextensible-24bit-4byte-48kHz-stereo.wav")
+        .unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 2);
+    assert_eq!(wav_reader.spec().sample_rate, 48_000);
+    assert_eq!(wav_reader.spec().bits_per_sample, 24);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[-96, 23_052, 8_388_607, -8_360_672]);
+}
+
+#[test]
+fn read_wav_32bit() {
+    let mut wav_reader = WavReader::open("testsamples/waveformatextensible-32bit-48kHz-stereo.wav")
+                                   .unwrap();
+
+    assert_eq!(wav_reader.spec().bits_per_sample, 32);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[19, -229_373, 33_587_161, -2_147_483_497]);
+}
+
+#[test]
+fn read_wav_wave_format_extensible_ieee_float() {
+    let mut wav_reader =
+        WavReader::open("testsamples/waveformatextensible-ieeefloat-44100Hz-mono.wav").unwrap();
+
+    assert_eq!(wav_reader.spec().channels, 1);
+    assert_eq!(wav_reader.spec().sample_rate, 44100);
+    assert_eq!(wav_reader.spec().bits_per_sample, 32);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Float);
+
+    let samples: Vec<f32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    // The test file has been prepared with these exact four samples.
+    assert_eq!(&samples[..], &[2.0, 3.0, -16411.0, 1019.0]);
+}
+
+#[test]
+fn read_wav_nonstandard_01() {
+    // The test sample here is adapted from a file encountered in the wild (data
+    // chunk replaced with two zero samples, some metadata dropped, and the file
+    // length in the header fixed). It is not a valid file according to the
+    // standard, but many players can deal with it nonetheless. (The file even
+    // contains some metadata; open it in a hex editor if you would like to know
+    // which program created it.) The file contains a regular PCM format tag,
+    // but the size of the fmt chunk is one that would be expected of a
+    // WAVEFORMATEXTENSIBLE chunk. The bits per sample is 24, which is invalid
+    // for WAVEFORMATEX, but we can read it nonetheless.
+    let mut wav_reader = WavReader::open("testsamples/nonstandard-01.wav").unwrap();
+
+    assert_eq!(wav_reader.spec().bits_per_sample, 24);
+    assert_eq!(wav_reader.spec().sample_format, SampleFormat::Int);
+
+    let samples: Vec<i32> = wav_reader.samples()
+                                      .map(|r| r.unwrap())
+                                      .collect();
+
+    assert_eq!(&samples[..], &[0, 0]);
+}
+
+#[test]
+fn wide_read_should_signal_error() {
+    let mut reader24 = WavReader::open("testsamples/waveformatextensible-24bit-192kHz-mono.wav")
+        .unwrap();
+
+    // Even though we know the first value is 17, and it should fit in an `i8`,
+    // a general 24-bit sample will not fit in an `i8`, so this should fail.
+    // 16-bit is still not wide enough, but 32-bit should do the trick.
+    assert!(reader24.samples::<i8>().next().unwrap().is_err());
+    assert!(reader24.samples::<i16>().next().unwrap().is_err());
+    assert!(reader24.samples::<i32>().next().unwrap().is_ok());
+
+    let mut reader32 = WavReader::open("testsamples/waveformatextensible-32bit-48kHz-stereo.wav")
+        .unwrap();
+
+    // In general, 32-bit samples will not fit in anything but an `i32`.
+    assert!(reader32.samples::<i8>().next().unwrap().is_err());
+    assert!(reader32.samples::<i16>().next().unwrap().is_err());
+    assert!(reader32.samples::<i32>().next().unwrap().is_ok());
+}
+
+#[test]
+fn sample_format_mismatch_should_signal_error() {
+    let mut reader_f32 = WavReader::open("testsamples/waveformatex-ieeefloat-44100Hz-mono.wav")
+        .unwrap();
+
+    assert!(reader_f32.samples::<i8>().next().unwrap().is_err());
+    assert!(reader_f32.samples::<i16>().next().unwrap().is_err());
+    assert!(reader_f32.samples::<i32>().next().unwrap().is_err());
+    assert!(reader_f32.samples::<f32>().next().unwrap().is_ok());
+
+    let mut reader_i8 = WavReader::open("testsamples/pcmwaveformat-8bit-44100Hz-mono.wav").unwrap();
+
+    assert!(reader_i8.samples::<i8>().next().unwrap().is_ok());
+    assert!(reader_i8.samples::<i16>().next().unwrap().is_ok());
+    assert!(reader_i8.samples::<i32>().next().unwrap().is_ok());
+    assert!(reader_i8.samples::<f32>().next().unwrap().is_err());
+}
+
+#[test]
+fn fuzz_crashes_should_be_fixed() {
+    use std::fs;
+    use std::ffi::OsStr;
+
+    // This is a regression test: all crashes and other issues found through
+    // fuzzing should not cause a crash.
+    let dir = fs::read_dir("testsamples/fuzz").ok()
+                 .expect("failed to enumerate fuzz test corpus");
+    for path in dir {
+        let path = path.ok().expect("failed to obtain path info").path();
+        let is_file = fs::metadata(&path).unwrap().file_type().is_file();
+        if is_file && path.extension() == Some(OsStr::new("wav")) {
+            println!("    testing {} ...", path.to_str()
+                                               .expect("unsupported filename"));
+            let mut reader = match WavReader::open(path) {
+                Ok(r) => r,
+                Err(..) => continue,
+            };
+            match reader.spec().sample_format {
+                SampleFormat::Int => {
+                    for sample in reader.samples::<i32>() {
+                        match sample {
+                            Ok(..) => { }
+                            Err(..) => break,
+                        }
+                    }
+                }
+                SampleFormat::Float => {
+                    for sample in reader.samples::<f32>() {
+                        match sample {
+                            Ok(..) => { }
+                            Err(..) => break,
+                        }
+                    }
+                }
+            }
+        }
+    }
+}
+
+#[test]
+fn seek_is_consistent() {
+    let files = &["testsamples/pcmwaveformat-16bit-44100Hz-mono.wav",
+                  "testsamples/waveformatex-16bit-44100Hz-stereo.wav",
+                  "testsamples/waveformatextensible-32bit-48kHz-stereo.wav"];
+    for fname in files {
+        let mut reader = WavReader::open(fname).unwrap();
+
+        // Seeking back to the start should "reset" the reader.
+        let count = reader.samples::<i32>().count();
+        reader.seek(0).unwrap();
+        assert_eq!(reader.samples_read, 0);
+        assert_eq!(count, reader.samples::<i32>().count());
+
+        // Seek to the last sample.
+        let last_time = reader.duration() - 1;
+        let channels = reader.spec.channels;
+        reader.seek(last_time).unwrap();
+        {
+            let mut samples = reader.samples::<i32>();
+            for _ in 0..channels {
+                assert!(samples.next().is_some());
+            }
+            assert!(samples.next().is_none());
+        }
+
+        // Seeking beyond the audio data produces no samples.
+        let num_samples = reader.len();
+        reader.seek(num_samples).unwrap();
+        assert!(reader.samples::<i32>().next().is_none());
+        reader.seek(::std::u32::MAX / channels as u32).unwrap();
+        assert!(reader.samples::<i32>().next().is_none());
+    }
+}
diff --git a/src/write.rs b/src/write.rs
new file mode 100644
index 0000000..585206a
--- /dev/null
+++ b/src/write.rs
@@ -0,0 +1,911 @@
+// Hound -- A wav encoding and decoding library in Rust
+// Copyright (C) 2015 Ruud van Asseldonk
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// A copy of the License has been included in the root of the repository.
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+use std::fs;
+use std::io;
+use std::mem;
+use std::io::{Seek, Write};
+use std::mem::MaybeUninit;
+use std::path;
+use super::{Error, Result, Sample, SampleFormat, WavSpec, WavSpecEx};
+use ::read;
+
+/// Extends the functionality of `io::Write` with additional methods.
+///
+/// The methods may be used on any type that implements `io::Write`.
+pub trait WriteExt: io::Write {
+    /// Writes an unsigned 8-bit integer.
+    fn write_u8(&mut self, x: u8) -> io::Result<()>;
+
+    /// Writes a signed 16-bit integer in little endian format.
+    fn write_le_i16(&mut self, x: i16) -> io::Result<()>;
+
+    /// Writes an unsigned 16-bit integer in little endian format.
+    fn write_le_u16(&mut self, x: u16) -> io::Result<()>;
+
+    /// Writes a signed 24-bit integer in little endian format.
+    ///
+    /// The most significant byte of the `i32` is ignored.
+    fn write_le_i24(&mut self, x: i32) -> io::Result<()>;
+
+    /// Writes a signed 24-bit integer in 4-byte little endian format.
+    ///
+    /// The most significant byte of the `i32` is replaced with zeroes.
+    fn write_le_i24_4(&mut self, x: i32) -> io::Result<()>;
+
+    /// Writes an unsigned 24-bit integer in little endian format.
+    ///
+    /// The most significant byte of the `u32` is ignored.
+    fn write_le_u24(&mut self, x: u32) -> io::Result<()>;
+
+    /// Writes a signed 32-bit integer in little endian format.
+    fn write_le_i32(&mut self, x: i32) -> io::Result<()>;
+
+    /// Writes an unsigned 32-bit integer in little endian format.
+    fn write_le_u32(&mut self, x: u32) -> io::Result<()>;
+
+    /// Writes an IEEE float in little endian format.
+    fn write_le_f32(&mut self, x: f32) -> io::Result<()>;
+}
+
+impl<W> WriteExt for W
+    where W: io::Write
+{
+    #[inline(always)]
+    fn write_u8(&mut self, x: u8) -> io::Result<()> {
+        let buf = [x];
+        self.write_all(&buf)
+    }
+
+    #[inline(always)]
+    fn write_le_i16(&mut self, x: i16) -> io::Result<()> {
+        self.write_le_u16(x as u16)
+    }
+
+    #[inline(always)]
+    fn write_le_u16(&mut self, x: u16) -> io::Result<()> {
+        let mut buf = [0u8; 2];
+        buf[0] = (x & 0xff) as u8;
+        buf[1] = (x >> 8) as u8;
+        self.write_all(&buf)
+    }
+
+    #[inline(always)]
+    fn write_le_i24(&mut self, x: i32) -> io::Result<()> {
+        self.write_le_u24(x as u32)
+    }
+
+    #[inline(always)]
+    fn write_le_i24_4(&mut self, x: i32) -> io::Result<()> {
+        self.write_le_u32((x as u32) & 0x00_ff_ff_ff)
+    }
+
+    #[inline(always)]
+    fn write_le_u24(&mut self, x: u32) -> io::Result<()> {
+        let mut buf = [0u8; 3];
+        buf[0] = ((x >> 00) & 0xff) as u8;
+        buf[1] = ((x >> 08) & 0xff) as u8;
+        buf[2] = ((x >> 16) & 0xff) as u8;
+        self.write_all(&buf)
+    }
+
+    #[inline(always)]
+    fn write_le_i32(&mut self, x: i32) -> io::Result<()> {
+        self.write_le_u32(x as u32)
+    }
+
+    #[inline(always)]
+    fn write_le_u32(&mut self, x: u32) -> io::Result<()> {
+        let mut buf = [0u8; 4];
+        buf[0] = ((x >> 00) & 0xff) as u8;
+        buf[1] = ((x >> 08) & 0xff) as u8;
+        buf[2] = ((x >> 16) & 0xff) as u8;
+        buf[3] = ((x >> 24) & 0xff) as u8;
+        self.write_all(&buf)
+    }
+
+    #[inline(always)]
+    fn write_le_f32(&mut self, x: f32) -> io::Result<()> {
+        let u = unsafe { mem::transmute::<f32, u32>(x) };
+        self.write_le_u32(u)
+    }
+}
+
+/// Generates a bitmask with `channels` ones in the least significant bits.
+///
+/// According to the [spec](https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/ksmedia/ns-ksmedia-waveformatextensible#remarks),
+/// if `channels` is greater than the number of bits in the channel mask, 18 non-reserved bits,
+/// extra channels are not assigned to any physical speaker location.  In this scenario, this
+/// function will return a filled channel mask.
+fn channel_mask(channels: u16) -> u32 {
+    // Clamp to 0-18 to stay within reserved bits.
+    let channels = if channels > 18 { 18 } else { channels };
+    (0..channels as u32).map(|c| 1 << c).fold(0, |a, c| a | c)
+}
+
+#[test]
+fn verify_channel_mask() {
+    assert_eq!(channel_mask(0), 0);
+    assert_eq!(channel_mask(1), 1);
+    assert_eq!(channel_mask(2), 3);
+    assert_eq!(channel_mask(3), 7);
+    assert_eq!(channel_mask(4), 0xF);
+    assert_eq!(channel_mask(8), 0xFF);
+    assert_eq!(channel_mask(16), 0xFFFF);
+    // expect channels >= 18 to yield the same mask
+    assert_eq!(channel_mask(18), 0x3FFFF);
+    assert_eq!(channel_mask(32), 0x3FFFF);
+    assert_eq!(channel_mask(64), 0x3FFFF);
+    assert_eq!(channel_mask(129), 0x3FFFF);
+}
+
+/// A writer that accepts samples and writes the WAVE format.
+///
+/// The writer needs a `WavSpec` or `WavSpecEx` that describes the audio
+/// properties. Then samples can be written with `write_sample`. Channel data is
+/// interleaved. The number of samples written must be a multiple of the number
+/// of channels. After all samples have been written, the file must be
+/// finalized. This can be done by calling `finalize`. If `finalize` is not
+/// called, the file will be finalized upon drop. However, finalization may
+/// fail, and without calling `finalize`, such a failure cannot be observed.
+pub struct WavWriter<W>
+    where W: io::Write + io::Seek
+{
+    /// Specifies properties of the audio data.
+    spec: WavSpec,
+
+    /// The (container) bytes per sample. This is the bit rate / 8 rounded up.
+    bytes_per_sample: u16,
+
+    /// The writer that will be written to.
+    writer: W,
+
+    /// The number of bytes written to the data section.
+    ///
+    /// This is an `u32` because WAVE cannot accomodate more data.
+    data_bytes_written: u32,
+
+    /// Whether the header has been finalized.
+    finalized: bool,
+
+    /// The buffer for the sample writer, which is recycled throughout calls to
+    /// avoid allocating frequently.
+    sample_writer_buffer: Vec<MaybeUninit<u8>>,
+
+    /// The offset of the length field of the data chunk.
+    ///
+    /// This field needs to be overwritten after all data has been written. To
+    /// support different size fmt chunks, and other chunks interspersed, the
+    /// offset is flexible.
+    data_len_offset: u32,
+}
+
+enum FmtKind {
+    PcmWaveFormat,
+    WaveFormatExtensible,
+}
+
+impl<W> WavWriter<W>
+    where W: io::Write + io::Seek
+{
+    /// Creates a writer that writes the WAVE format to the underlying writer.
+    ///
+    /// The underlying writer is assumed to be at offset 0. `WavWriter` employs
+    /// *no* buffering internally. It is recommended to wrap the writer in a
+    /// `BufWriter` to avoid too many `write` calls. The `create()` constructor
+    /// does this automatically.
+    ///
+    /// This writes parts of the header immediately, hence a `Result` is
+    /// returned.
+    pub fn new(writer: W, spec: WavSpec) -> Result<WavWriter<W>> {
+        let spec_ex = WavSpecEx {
+            spec: spec,
+            bytes_per_sample: (spec.bits_per_sample + 7) / 8,
+        };
+        WavWriter::new_with_spec_ex(writer, spec_ex)
+    }
+
+
+    /// Creates a writer that writes the WAVE format to the underlying writer.
+    ///
+    /// The underlying writer is assumed to be at offset 0. `WavWriter` employs
+    /// *no* buffering internally. It is recommended to wrap the writer in a
+    /// `BufWriter` to avoid too many `write` calls. The `create()` constructor
+    /// does this automatically.
+    ///
+    /// This writes parts of the header immediately, hence a `Result` is
+    /// returned.
+    pub fn new_with_spec_ex(writer: W, spec_ex: WavSpecEx) -> Result<WavWriter<W>> {
+        let spec = spec_ex.spec;
+
+        // Write the older PCMWAVEFORMAT structure if possible, because it is
+        // more widely supported. For more than two channels or more than 16
+        // bits per sample, the newer WAVEFORMATEXTENSIBLE is required. See also
+        // https://msdn.microsoft.com/en-us/library/ms713497.aspx.
+        let fmt_kind = if spec.channels > 2 || spec.bits_per_sample > 16 {
+            FmtKind::WaveFormatExtensible
+        } else {
+            FmtKind::PcmWaveFormat
+        };
+
+        let mut writer = WavWriter {
+            spec: spec,
+            bytes_per_sample: spec_ex.bytes_per_sample,
+            writer: writer,
+            data_bytes_written: 0,
+            sample_writer_buffer: Vec::new(),
+            finalized: false,
+            data_len_offset: match fmt_kind {
+                FmtKind::WaveFormatExtensible => 64,
+                FmtKind::PcmWaveFormat => 40,
+            },
+        };
+
+        // Hound can only write those bit depths. If something else was
+        // requested, fail early, rather than writing a header but then failing
+        // at the first sample.
+        let supported = match spec.bits_per_sample {
+            8 => true,
+            16 => true,
+            24 => true,
+            32 => true,
+            _ => false,
+        };
+
+        if !supported {
+            return Err(Error::Unsupported)
+        }
+
+        // Write headers, up to the point where data should be written.
+        try!(writer.write_headers(fmt_kind));
+
+        Ok(writer)
+    }
+
+    /// Writes the RIFF WAVE header, fmt chunk, and data chunk header.
+    fn write_headers(&mut self, fmt_kind: FmtKind) -> io::Result<()> {
+        // Write to an in-memory buffer before writing to the underlying writer.
+        let mut header = [0u8; 68];
+
+        {
+            let mut buffer = io::Cursor::new(&mut header[..]);
+
+            // Write the headers for the RIFF WAVE format.
+            try!(buffer.write_all("RIFF".as_bytes()));
+
+            // Skip 4 bytes that will be filled with the file size afterwards.
+            try!(buffer.write_le_u32(0));
+
+            try!(buffer.write_all("WAVE".as_bytes()));
+            try!(buffer.write_all("fmt ".as_bytes()));
+
+            match fmt_kind {
+                FmtKind::PcmWaveFormat => {
+                    try!(self.write_pcmwaveformat(&mut buffer));
+                }
+                FmtKind::WaveFormatExtensible => {
+                    try!(self.write_waveformatextensible(&mut buffer));
+                }
+            }
+
+            // Finally the header of the "data" chunk. The number of bytes
+            // that this will take is not known at this point. The 0 will
+            // be overwritten later.
+            try!(buffer.write_all("data".as_bytes()));
+            try!(buffer.write_le_u32(0));
+        }
+
+        // The data length field are the last 4 bytes of the header.
+        let header_len = self.data_len_offset as usize + 4;
+
+        self.writer.write_all(&header[..header_len])
+    }
+
+    /// Writes the spec as a WAVEFORMAT structure.
+    ///
+    /// The `WAVEFORMAT` struct is a subset of both `WAVEFORMATEX` and
+    /// `WAVEFORMATEXTENSIBLE`. This does not write the `wFormatTag` member.
+    fn write_waveformat(&self, buffer: &mut io::Cursor<&mut [u8]>) -> io::Result<()> {
+        let spec = &self.spec;
+        // The field nChannels.
+        try!(buffer.write_le_u16(spec.channels));
+
+        // The field nSamplesPerSec.
+        try!(buffer.write_le_u32(spec.sample_rate));
+        let bytes_per_sec = spec.sample_rate
+                          * self.bytes_per_sample as u32
+                          * spec.channels as u32;
+
+        // The field nAvgBytesPerSec;
+        try!(buffer.write_le_u32(bytes_per_sec));
+
+        // The field nBlockAlign. Block align * sample rate = bytes per sec.
+        try!(buffer.write_le_u16((bytes_per_sec / spec.sample_rate) as u16));
+
+        Ok(())
+    }
+
+    /// Writes the content of the fmt chunk as PCMWAVEFORMAT struct.
+    fn write_pcmwaveformat(&mut self, buffer: &mut io::Cursor<&mut [u8]>) -> io::Result<()> {
+        // Write the size of the WAVE header chunk.
+        try!(buffer.write_le_u32(16));
+
+        // The following is based on the PCMWAVEFORMAT struct as documented at
+        // https://msdn.microsoft.com/en-us/library/ms712832.aspx. See also
+        // http://soundfile.sapp.org/doc/WaveFormat/.
+
+        // The field wFormatTag
+        match self.spec.sample_format {
+            // WAVE_FORMAT_PCM
+            SampleFormat::Int => {
+                try!(buffer.write_le_u16(1));
+            },
+            // WAVE_FORMAT_IEEE_FLOAT
+            SampleFormat::Float => {
+                if self.spec.bits_per_sample == 32 {
+                    try!(buffer.write_le_u16(3));
+                } else {
+                    panic!("Invalid number of bits per sample. \
+                           When writing SampleFormat::Float, \
+                           bits_per_sample must be 32.");
+                }
+            },
+        };
+
+        try!(self.write_waveformat(buffer));
+
+        // The field wBitsPerSample, the real number of bits per sample.
+        try!(buffer.write_le_u16(self.spec.bits_per_sample));
+
+        // Note: for WAVEFORMATEX, there would be another 16-byte field `cbSize`
+        // here that should be set to zero. And the header size would be 18
+        // rather than 16.
+
+        Ok(())
+    }
+
+    /// Writes the contents of the fmt chunk as WAVEFORMATEXTENSIBLE struct.
+    fn write_waveformatextensible(&mut self, buffer: &mut io::Cursor<&mut [u8]>) -> io::Result<()> {
+        // Write the size of the WAVE header chunk.
+        try!(buffer.write_le_u32(40));
+
+        // The following is based on the WAVEFORMATEXTENSIBLE struct, documented
+        // at https://msdn.microsoft.com/en-us/library/ms713496.aspx and
+        // https://msdn.microsoft.com/en-us/library/ms713462.aspx.
+
+        // The field wFormatTag, value 1 means WAVE_FORMAT_PCM, but we use
+        // the slightly more sophisticated WAVE_FORMAT_EXTENSIBLE.
+        try!(buffer.write_le_u16(0xfffe));
+
+        try!(self.write_waveformat(buffer));
+
+        // The field wBitsPerSample. This is actually the size of the
+        // container, so this is a multiple of 8.
+        try!(buffer.write_le_u16(self.bytes_per_sample as u16 * 8));
+        // The field cbSize, the number of remaining bytes in the struct.
+        try!(buffer.write_le_u16(22));
+        // The field wValidBitsPerSample, the real number of bits per sample.
+        try!(buffer.write_le_u16(self.spec.bits_per_sample));
+        // The field dwChannelMask.
+        // TODO: add the option to specify the channel mask. For now, use
+        // the default assignment.
+        try!(buffer.write_le_u32(channel_mask(self.spec.channels)));
+
+        // The field SubFormat.
+        let subformat_guid = match self.spec.sample_format {
+            // PCM audio with integer samples.
+            SampleFormat::Int => super::KSDATAFORMAT_SUBTYPE_PCM,
+            // PCM audio with 32-bit IEEE float samples.
+            SampleFormat::Float => {
+                if self.spec.bits_per_sample == 32 {
+                    super::KSDATAFORMAT_SUBTYPE_IEEE_FLOAT
+                } else {
+                    panic!("Invalid number of bits per sample. \
+                           When writing SampleFormat::Float, \
+                           bits_per_sample must be 32.");
+                }
+            }
+        };
+        try!(buffer.write_all(&subformat_guid));
+
+        Ok(())
+    }
+
+    /// Writes a single sample for one channel.
+    ///
+    /// WAVE interleaves channel data, so the channel that this writes the
+    /// sample to depends on previous writes. This will return an error if the
+    /// sample does not fit in the number of bits specified in the `WavSpec`.
+    #[inline]
+    pub fn write_sample<S: Sample>(&mut self, sample: S) -> Result<()> {
+        try!(sample.write_padded(
+            &mut self.writer,
+            self.spec.bits_per_sample,
+            self.bytes_per_sample,
+        ));
+        self.data_bytes_written += self.bytes_per_sample as u32;
+        Ok(())
+    }
+
+    /// Create an efficient writer that writes 16-bit integer samples only.
+    ///
+    /// When it is known what the kind of samples will be, many dynamic checks
+    /// can be omitted. Furthermore, this writer employs buffering internally,
+    /// which allows omitting return value checks except on flush. The internal
+    /// buffer will be sized such that exactly `num_samples` samples can be
+    /// written to it, and the buffer is recycled across calls to
+    /// `get_i16_writer()` if the previous buffer was sufficiently large.
+    ///
+    /// # Panics
+    ///
+    /// Panics if the spec does not match a 16 bits per sample integer format.
+    ///
+    /// Attempting to write more than `num_samples` samples to the writer will
+    /// panic too.
+    pub fn get_i16_writer<'s>(&'s mut self,
+                              num_samples: u32)
+                              -> SampleWriter16<'s, W> {
+        if self.spec.sample_format != SampleFormat::Int {
+            panic!("When calling get_i16_writer, the sample format must be int.");
+        }
+        if self.spec.bits_per_sample != 16 {
+            panic!("When calling get_i16_writer, the number of bits per sample must be 16.");
+        }
+
+        let num_bytes = num_samples as usize * 2;
+
+        if self.sample_writer_buffer.len() < num_bytes {
+            // We need a bigger buffer. There is no point in growing the old
+            // one, as we are going to overwrite the samples anyway, so just
+            // allocate a new one.
+            let mut new_buffer = Vec::<MaybeUninit<u8>>::with_capacity(num_bytes);
+
+            // The potentially garbage memory here will not be exposed: the
+            // buffer is only exposed when flushing, but `flush()` asserts that
+            // all samples have been written.
+            unsafe { new_buffer.set_len(num_bytes); }
+
+            self.sample_writer_buffer = new_buffer;
+        }
+
+        SampleWriter16 {
+            writer: &mut self.writer,
+            buffer: &mut self.sample_writer_buffer[..num_bytes],
+            data_bytes_written: &mut self.data_bytes_written,
+            index: 0,
+        }
+    }
+
+    fn update_header(&mut self) -> Result<()> {
+        // The header size minus magic and 32-bit filesize (8 bytes).
+        // The data chunk length (4 bytes) is the last part of the header.
+        let header_size = self.data_len_offset + 4 - 8;
+        let file_size = self.data_bytes_written + header_size;
+
+        try!(self.writer.seek(io::SeekFrom::Start(4)));
+        try!(self.writer.write_le_u32(file_size));
+        try!(self.writer.seek(io::SeekFrom::Start(self.data_len_offset as u64)));
+        try!(self.writer.write_le_u32(self.data_bytes_written));
+
+        // Signal error if the last sample was not finished, but do so after
+        // everything has been written, so that no data is lost, even though
+        // the file is now ill-formed.
+        if (self.data_bytes_written / self.bytes_per_sample as u32)
+            % self.spec.channels as u32 != 0 {
+            Err(Error::UnfinishedSample)
+        } else {
+            Ok(())
+        }
+    }
+
+    /// Updates the WAVE header and flushes the underlying writer.
+    ///
+    /// Flush writes the WAVE header to the underlying writer to make the
+    /// written bytes a valid wav file, and then flushes the writer. It is still
+    /// possible to write more samples after flushing.
+    ///
+    /// Flush can be used for “checkpointing”. Even if after the flush there is
+    /// an IO error or the writing process dies, the file can still be read by a
+    /// compliant decoder up to the last flush.
+    ///
+    /// Note that if the number of samples written is not a multiple of the
+    /// channel count, the intermediate wav file will not be valid. In that case
+    /// `flush()` will still flush the data and write the (invalid) wav file,
+    /// but `Error::UnfinishedSample` will be returned afterwards.
+    ///
+    /// It is not necessary to call `finalize()` directly after `flush()`, if no
+    /// samples have been written after flushing.
+    pub fn flush(&mut self) -> Result<()> {
+        let current_pos = try!(self.writer.seek(io::SeekFrom::Current(0)));
+        try!(self.update_header());
+        try!(self.writer.flush());
+        try!(self.writer.seek(io::SeekFrom::Start(current_pos)));
+        Ok(())
+    }
+
+    /// Updates the WAVE header (which requires knowing all samples).
+    ///
+    /// This method must be called after all samples have been written. If it
+    /// is not called, the destructor will finalize the file, but any errors
+    /// that occur in the process cannot be observed in that manner.
+    pub fn finalize(mut self) -> Result<()> {
+        self.finalized = true;
+        try!(self.update_header());
+        // We need to perform a flush here to truly capture all errors before
+        // the writer is dropped: for a buffered writer, the write to the buffer
+        // may succeed, but the write to the underlying writer may fail. So
+        // flush explicitly.
+        try!(self.writer.flush());
+        Ok(())
+    }
+
+    /// Returns information about the WAVE file being written.
+    ///
+    /// This is the same spec that was passed to `WavWriter::new()`. For a
+    /// writer constructed with `WavWriter::new_append()` or
+    /// `WavWriter::append()`, this method returns the spec of the file being
+    /// appended to.
+    pub fn spec(&self) -> WavSpec {
+        self.spec
+    }
+
+    /// Returns the duration of the file written so far, in samples.
+    ///
+    /// The duration is independent of the number of channels. It is expressed
+    /// in units of samples. The duration in seconds can be obtained by
+    /// dividing this number by the sample rate.
+    pub fn duration(&self) -> u32 {
+        self.data_bytes_written / (self.bytes_per_sample as u32 * self.spec.channels as u32)
+    }
+
+    /// Returns the number of samples in the file written so far.
+    ///
+    /// The length of the file is its duration (in samples) times the number of
+    /// channels.
+    pub fn len(&self) -> u32 {
+        self.data_bytes_written / self.bytes_per_sample as u32
+    }
+}
+
+impl<W> Drop for WavWriter<W>
+    where W: io::Write + io::Seek
+{
+    fn drop(&mut self) {
+        // If the file was not explicitly finalized (to update the headers), do
+        // it in the drop. This can fail, but drop should not panic, so a
+        // failure is ignored silently here.
+        if !self.finalized {
+            let _r = self.update_header();
+        }
+    }
+}
+
+/// Reads the relevant parts of the header required to support append.
+///
+/// Returns (spec_ex, data_len, data_len_offset).
+fn read_append<W: io::Read + io::Seek>(mut reader: &mut W) -> Result<(WavSpecEx, u32, u32)> {
+    let (spec_ex, data_len) = {
+        try!(read::read_wave_header(&mut reader));
+        try!(read::read_until_data(&mut reader))
+    };
+
+    // Record the position of the data chunk length, so we can overwrite it
+    // later.
+    let data_len_offset = try!(reader.seek(io::SeekFrom::Current(0))) as u32 - 4;
+
+    let spec = spec_ex.spec;
+    let num_samples = data_len / spec_ex.bytes_per_sample as u32;
+
+    // There must not be trailing bytes in the data chunk, otherwise the
+    // bytes we write will be off.
+    if num_samples * spec_ex.bytes_per_sample as u32 != data_len {
+        let msg = "data chunk length is not a multiple of sample size";
+        return Err(Error::FormatError(msg));
+    }
+
+    // Hound cannot read or write other bit depths than those, so rather
+    // than refusing to write later, fail early.
+    let supported = match (spec_ex.bytes_per_sample, spec.bits_per_sample) {
+        (1, 8) => true,
+        (2, 16) => true,
+        (3, 24) => true,
+        (4, 32) => true,
+        _ => false,
+    };
+
+    if !supported {
+        return Err(Error::Unsupported);
+    }
+
+    // The number of samples must be a multiple of the number of channels,
+    // otherwise the last inter-channel sample would not have data for all
+    // channels.
+    if num_samples % spec_ex.spec.channels as u32 != 0 {
+        return Err(Error::FormatError("invalid data chunk length"));
+    }
+
+    Ok((spec_ex, data_len, data_len_offset))
+}
+
+impl WavWriter<io::BufWriter<fs::File>> {
+    /// Creates a writer that writes the WAVE format to a file.
+    ///
+    /// This is a convenience constructor that creates the file, wraps it in a
+    /// `BufWriter`, and then constructs a `WavWriter` from it. The file will
+    /// be overwritten if it exists.
+    pub fn create<P: AsRef<path::Path>>(filename: P,
+                                        spec: WavSpec)
+                                        -> Result<WavWriter<io::BufWriter<fs::File>>> {
+        let file = try!(fs::File::create(filename));
+        let buf_writer = io::BufWriter::new(file);
+        WavWriter::new(buf_writer, spec)
+    }
+
+    /// Creates a writer that appends samples to an existing file.
+    ///
+    /// This is a convenience constructor that opens the file in append mode,
+    /// reads its header using a buffered reader, and then constructs an
+    /// appending `WavWriter` that writes to the file using a `BufWriter`.
+    ///
+    /// See `WavWriter::new_append()` for more details about append behavior.
+    pub fn append<P: AsRef<path::Path>>(filename: P) -> Result<WavWriter<io::BufWriter<fs::File>>> {
+        // Open the file in append mode, start reading from the start.
+        let mut file = try!(fs::OpenOptions::new().read(true).write(true).open(filename));
+        try!(file.seek(io::SeekFrom::Start(0)));
+
+        // Read the header using a buffered reader.
+        let mut buf_reader = io::BufReader::new(file);
+        let (spec_ex, data_len, data_len_offset) = try!(read_append(&mut buf_reader));
+        let mut file = buf_reader.into_inner();
+
+        // Seek to the data position, and from now on, write using a buffered
+        // writer.
+        try!(file.seek(io::SeekFrom::Current(data_len as i64)));
+        let buf_writer = io::BufWriter::new(file);
+
+        let writer = WavWriter {
+            spec: spec_ex.spec,
+            bytes_per_sample: spec_ex.bytes_per_sample,
+            writer: buf_writer,
+            data_bytes_written: data_len,
+            sample_writer_buffer: Vec::new(),
+            finalized: false,
+            data_len_offset: data_len_offset,
+        };
+
+        Ok(writer)
+    }
+}
+
+impl<W> WavWriter<W> where W: io::Read + io::Write + io::Seek {
+    /// Creates a writer that appends samples to an existing file stream.
+    ///
+    /// This first reads the existing header to obtain the spec, then seeks to
+    /// the end of the writer. The writer then appends new samples to the end of
+    /// the stream.
+    ///
+    /// The underlying writer is assumed to be at offset 0.
+    ///
+    /// If the existing file includes a fact chunk, it will not be updated after
+    /// appending, and hence become outdated. For files produced by Hound this
+    /// is not an issue, because Hound never writes a fact chunk. For all the
+    /// formats that Hound can write, the fact chunk is redundant.
+    pub fn new_append(mut writer: W) -> Result<WavWriter<W>> {
+        let (spec_ex, data_len, data_len_offset) = try!(read_append(&mut writer));
+        try!(writer.seek(io::SeekFrom::Current(data_len as i64)));
+        let writer = WavWriter {
+            spec: spec_ex.spec,
+            bytes_per_sample: spec_ex.bytes_per_sample,
+            writer: writer,
+            data_bytes_written: data_len,
+            sample_writer_buffer: Vec::new(),
+            finalized: false,
+            data_len_offset: data_len_offset,
+        };
+
+        Ok(writer)
+    }
+}
+
+
+/// A writer that specifically only writes integer samples of 16 bits per sample.
+///
+/// The writer buffers written samples internally so they can be written in a
+/// single batch later on. This has two advantages when performance is
+/// important:
+///
+///  * There is no need for error handling during writing, only on flush. This
+///    eliminates a lot of branches.
+///  * The buffer can be written once, which reduces the overhead of the write
+///    call. Because writing to an `io::BufWriter` is implemented with a
+///    `memcpy` (even for single bytes), there is a large overhead to writing
+///    small amounts of data such as a 16-bit sample. By writing large blocks
+///    (or by not using `BufWriter`) this overhead can be avoided.
+///
+/// A `SampleWriter16` can be obtained by calling [`WavWriter::get_i16_writer`](
+/// struct.WavWriter.html#method.get_i16_writer).
+pub struct SampleWriter16<'parent, W> where W: io::Write + io::Seek + 'parent {
+    /// The writer borrowed from the wrapped WavWriter.
+    writer: &'parent mut W,
+
+    /// The internal buffer that samples are written to before they are flushed.
+    buffer: &'parent mut [MaybeUninit<u8>],
+
+    /// Reference to the `data_bytes_written` field of the writer.
+    data_bytes_written: &'parent mut u32,
+
+    /// The index into the buffer where the next bytes will be written.
+    index: u32,
+}
+
+impl<'parent, W: io::Write + io::Seek> SampleWriter16<'parent, W> {
+    /// Writes a single sample for one channel.
+    ///
+    /// WAVE interleaves channel data, so the channel that this writes the
+    /// sample to depends on previous writes.
+    ///
+    /// Unlike `WavWriter::write_sample()`, no range check is performed. Only
+    /// the least significant 16 bits are considered, everything else is
+    /// discarded.  Apart from that check, this method is more efficient than
+    /// `WavWriter::write_sample()`, because it can avoid dispatching on the
+    /// number of bits. That was done already when the `SampleWriter16` was
+    /// constructed.
+    ///
+    /// Note that nothing is actually written until `flush()` is called.
+    #[inline(always)]
+    pub fn write_sample<S: Sample>(&mut self, sample: S) {
+        assert!((self.index as usize) + 2 <= self.buffer.len(),
+          "Trying to write more samples than reserved for the sample writer.");
+
+        // SAFETY: We performed the bounds check in the above assertion.
+        unsafe { self.write_sample_unchecked(sample) };
+    }
+
+    unsafe fn write_u16_le_unchecked(&mut self, value: u16) {
+        // On little endian machines the compiler produces assembly code
+        // that merges the following two lines into a single instruction.
+        *self.buffer.get_unchecked_mut(self.index as usize) = MaybeUninit::new(value as u8);
+        self.buffer.get_unchecked_mut(self.index as usize).assume_init();
+        *self.buffer.get_unchecked_mut(self.index as usize + 1) = MaybeUninit::new((value >> 8) as u8);
+        self.buffer.get_unchecked_mut(self.index as usize + 1).assume_init();
+    }
+
+    /// Like `write_sample()`, but does not perform a bounds check when writing
+    /// to the internal buffer.
+    ///
+    /// It is the responsibility of the programmer to ensure that no more
+    /// samples are written than allocated when the writer was created.
+    #[inline(always)]
+    pub unsafe fn write_sample_unchecked<S: Sample>(&mut self, sample: S) {
+        self.write_u16_le_unchecked(sample.as_i16() as u16);
+        self.index += 2;
+    }
+
+    /// Flush the internal buffer to the underlying writer.
+    ///
+    /// # Panics
+    ///
+    /// Panics if insufficient samples (less than specified when the writer was
+    /// constructed) have been written with `write_sample()`.
+    pub fn flush(self) -> Result<()> {
+        if self.index as usize != self.buffer.len() {
+            panic!("Insufficient samples written to the sample writer.");
+        }
+
+        // SAFETY: casting `self.buffer` to a `*const [MaybeUninit<u8>]` is safe
+        // since the caller guarantees that `self.buffer` is initialized, and
+        // `MaybeUninit<u8>` is guaranteed to have the same layout as `u8`. The
+        // pointer obtained is valid since it refers to memory owned by
+        // `self.buffer` which is a reference and thus guaranteed to be valid
+        // for reads. This is copied from the nightly implementation for
+        // slice_assume_init_ref.
+        let slice = unsafe { &*(self.buffer as *const [MaybeUninit<u8>] as *const [u8]) };
+
+        try!(self.writer.write_all(slice));
+
+        *self.data_bytes_written += self.buffer.len() as u32;
+        Ok(())
+    }
+}
+
+#[test]
+fn short_write_should_signal_error() {
+    use SampleFormat;
+
+    let mut buffer = io::Cursor::new(Vec::new());
+
+    let write_spec = WavSpec {
+        channels: 17,
+        sample_rate: 48000,
+        bits_per_sample: 8,
+        sample_format: SampleFormat::Int,
+    };
+
+    // Deliberately write one sample less than 17 * 5.
+    let mut writer = WavWriter::new(&mut buffer, write_spec).unwrap();
+    for s in 0..17 * 5 - 1 {
+        writer.write_sample(s as i16).unwrap();
+    }
+    let error = writer.finalize().err().unwrap();
+
+    match error {
+        Error::UnfinishedSample => {}
+        _ => panic!("UnfinishedSample error should have been returned."),
+    }
+}
+
+#[test]
+fn wide_write_should_signal_error() {
+    let mut buffer = io::Cursor::new(Vec::new());
+
+    let spec8 = WavSpec {
+        channels: 1,
+        sample_rate: 44100,
+        bits_per_sample: 8,
+        sample_format: SampleFormat::Int,
+    };
+    {
+        let mut writer = WavWriter::new(&mut buffer, spec8).unwrap();
+        assert!(writer.write_sample(127_i8).is_ok());
+        assert!(writer.write_sample(127_i16).is_ok());
+        assert!(writer.write_sample(127_i32).is_ok());
+        assert!(writer.write_sample(128_i16).is_err());
+        assert!(writer.write_sample(128_i32).is_err());
+    }
+
+    let spec16 = WavSpec { bits_per_sample: 16, ..spec8 };
+    {
+        let mut writer = WavWriter::new(&mut buffer, spec16).unwrap();
+        assert!(writer.write_sample(32767_i16).is_ok());
+        assert!(writer.write_sample(32767_i32).is_ok());
+        assert!(writer.write_sample(32768_i32).is_err());
+    }
+
+    let spec24 = WavSpec { bits_per_sample: 24, ..spec8 };
+    {
+        let mut writer = WavWriter::new(&mut buffer, spec24).unwrap();
+        assert!(writer.write_sample(8_388_607_i32).is_ok());
+        assert!(writer.write_sample(8_388_608_i32).is_err());
+    }
+}
+
+#[test]
+fn s24_wav_write() {
+    use std::fs::File;
+    use std::io::Read;
+    let mut buffer = io::Cursor::new(Vec::new());
+
+    let spec = WavSpecEx {
+        spec: WavSpec {
+            channels: 2,
+            sample_rate: 48000,
+            bits_per_sample: 24,
+            sample_format: SampleFormat::Int,
+        },
+        bytes_per_sample: 4,
+    };
+    {
+        let mut writer = WavWriter::new_with_spec_ex(&mut buffer, spec).unwrap();
+        assert!(writer.write_sample(-96_i32).is_ok());
+        assert!(writer.write_sample(23_052_i32).is_ok());
+        assert!(writer.write_sample(8_388_607_i32).is_ok());
+        assert!(writer.write_sample(-8_360_672_i32).is_ok());
+    }
+
+    let mut expected = Vec::new();
+    File::open("testsamples/waveformatextensible-24bit-4byte-48kHz-stereo.wav")
+        .unwrap()
+        .read_to_end(&mut expected)
+        .unwrap();
+
+    assert_eq!(buffer.into_inner(), expected);
+}
diff --git a/testsamples/fuzz/crash-24728523ef4be15c838293b676f6853e73723bf4.wav b/testsamples/fuzz/crash-24728523ef4be15c838293b676f6853e73723bf4.wav
new file mode 100644
index 0000000..b092015
--- /dev/null
+++ b/testsamples/fuzz/crash-24728523ef4be15c838293b676f6853e73723bf4.wav
Binary files differ
diff --git a/testsamples/fuzz/crash-b8447179832529c48f9c6bf17feab6337bbc78ea.wav b/testsamples/fuzz/crash-b8447179832529c48f9c6bf17feab6337bbc78ea.wav
new file mode 100644
index 0000000..3c9517d
--- /dev/null
+++ b/testsamples/fuzz/crash-b8447179832529c48f9c6bf17feab6337bbc78ea.wav
Binary files differ
diff --git a/testsamples/fuzz/crash-cbd757427cea12bd8a21f86cd8cf74d98ce56bee.wav b/testsamples/fuzz/crash-cbd757427cea12bd8a21f86cd8cf74d98ce56bee.wav
new file mode 100644
index 0000000..4c0ee74
--- /dev/null
+++ b/testsamples/fuzz/crash-cbd757427cea12bd8a21f86cd8cf74d98ce56bee.wav
Binary files differ
diff --git a/testsamples/fuzz/crash-e5471f5b58397287b509db7d026e95f1724454f5.wav b/testsamples/fuzz/crash-e5471f5b58397287b509db7d026e95f1724454f5.wav
new file mode 100644
index 0000000..44fddd2
--- /dev/null
+++ b/testsamples/fuzz/crash-e5471f5b58397287b509db7d026e95f1724454f5.wav
Binary files differ
diff --git a/testsamples/fuzz/crash-e879de4eb4d206c59e21f0e01def16457af80fdc.wav b/testsamples/fuzz/crash-e879de4eb4d206c59e21f0e01def16457af80fdc.wav
new file mode 100644
index 0000000..b460b72
--- /dev/null
+++ b/testsamples/fuzz/crash-e879de4eb4d206c59e21f0e01def16457af80fdc.wav
Binary files differ
diff --git a/testsamples/fuzz/oom-48ae4cd061ff8578ad3f23dc87624bd365cf5216.wav b/testsamples/fuzz/oom-48ae4cd061ff8578ad3f23dc87624bd365cf5216.wav
new file mode 100644
index 0000000..fd06dca
--- /dev/null
+++ b/testsamples/fuzz/oom-48ae4cd061ff8578ad3f23dc87624bd365cf5216.wav
@@ -0,0 +1 @@
+RIFFPIFFWAVEooooRIF±FF
\ No newline at end of file
diff --git a/testsamples/nonstandard-01.wav b/testsamples/nonstandard-01.wav
new file mode 100644
index 0000000..95afc56
--- /dev/null
+++ b/testsamples/nonstandard-01.wav
Binary files differ
diff --git a/testsamples/nonstandard-02.wav b/testsamples/nonstandard-02.wav
new file mode 100644
index 0000000..8cd1df4
--- /dev/null
+++ b/testsamples/nonstandard-02.wav
Binary files differ
diff --git a/testsamples/pcmwaveformat-16bit-44100Hz-mono-extra.wav b/testsamples/pcmwaveformat-16bit-44100Hz-mono-extra.wav
new file mode 100644
index 0000000..9680aba
--- /dev/null
+++ b/testsamples/pcmwaveformat-16bit-44100Hz-mono-extra.wav
Binary files differ
diff --git a/testsamples/pcmwaveformat-16bit-44100Hz-mono.wav b/testsamples/pcmwaveformat-16bit-44100Hz-mono.wav
new file mode 100644
index 0000000..4a673d3
--- /dev/null
+++ b/testsamples/pcmwaveformat-16bit-44100Hz-mono.wav
Binary files differ
diff --git a/testsamples/pcmwaveformat-24bit-4byte-48kHz-stereo.wav b/testsamples/pcmwaveformat-24bit-4byte-48kHz-stereo.wav
new file mode 100644
index 0000000..1f0d6f6
--- /dev/null
+++ b/testsamples/pcmwaveformat-24bit-4byte-48kHz-stereo.wav
Binary files differ
diff --git a/testsamples/pcmwaveformat-8bit-44100Hz-mono.wav b/testsamples/pcmwaveformat-8bit-44100Hz-mono.wav
new file mode 100644
index 0000000..d51e53a
--- /dev/null
+++ b/testsamples/pcmwaveformat-8bit-44100Hz-mono.wav
Binary files differ
diff --git a/testsamples/pop.wav b/testsamples/pop.wav
new file mode 100644
index 0000000..9e61802
--- /dev/null
+++ b/testsamples/pop.wav
Binary files differ
diff --git a/testsamples/waveformatex-16bit-44100Hz-mono-extra.wav b/testsamples/waveformatex-16bit-44100Hz-mono-extra.wav
new file mode 100644
index 0000000..d3159a9
--- /dev/null
+++ b/testsamples/waveformatex-16bit-44100Hz-mono-extra.wav
Binary files differ
diff --git a/testsamples/waveformatex-16bit-44100Hz-mono.wav b/testsamples/waveformatex-16bit-44100Hz-mono.wav
new file mode 100644
index 0000000..5de7062
--- /dev/null
+++ b/testsamples/waveformatex-16bit-44100Hz-mono.wav
Binary files differ
diff --git a/testsamples/waveformatex-16bit-44100Hz-stereo.wav b/testsamples/waveformatex-16bit-44100Hz-stereo.wav
new file mode 100644
index 0000000..475de3a
--- /dev/null
+++ b/testsamples/waveformatex-16bit-44100Hz-stereo.wav
Binary files differ
diff --git a/testsamples/waveformatex-8bit-11025Hz-mono.wav b/testsamples/waveformatex-8bit-11025Hz-mono.wav
new file mode 100644
index 0000000..95ab441
--- /dev/null
+++ b/testsamples/waveformatex-8bit-11025Hz-mono.wav
Binary files differ
diff --git a/testsamples/waveformatex-ieeefloat-44100Hz-mono.wav b/testsamples/waveformatex-ieeefloat-44100Hz-mono.wav
new file mode 100644
index 0000000..73e51ba
--- /dev/null
+++ b/testsamples/waveformatex-ieeefloat-44100Hz-mono.wav
Binary files differ
diff --git a/testsamples/waveformatextensible-24bit-192kHz-mono.wav b/testsamples/waveformatextensible-24bit-192kHz-mono.wav
new file mode 100644
index 0000000..71cd6ae
--- /dev/null
+++ b/testsamples/waveformatextensible-24bit-192kHz-mono.wav
Binary files differ
diff --git a/testsamples/waveformatextensible-24bit-4byte-48kHz-stereo.wav b/testsamples/waveformatextensible-24bit-4byte-48kHz-stereo.wav
new file mode 100644
index 0000000..f19c243
--- /dev/null
+++ b/testsamples/waveformatextensible-24bit-4byte-48kHz-stereo.wav
Binary files differ
diff --git a/testsamples/waveformatextensible-32bit-48kHz-stereo.wav b/testsamples/waveformatextensible-32bit-48kHz-stereo.wav
new file mode 100644
index 0000000..3984a2e
--- /dev/null
+++ b/testsamples/waveformatextensible-32bit-48kHz-stereo.wav
Binary files differ
diff --git a/testsamples/waveformatextensible-ieeefloat-44100Hz-mono.wav b/testsamples/waveformatextensible-ieeefloat-44100Hz-mono.wav
new file mode 100644
index 0000000..06306ca
--- /dev/null
+++ b/testsamples/waveformatextensible-ieeefloat-44100Hz-mono.wav
Binary files differ