summaryrefslogtreecommitdiffstats
path: root/third_party/rust/crc32fast/src/lib.rs
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/crc32fast/src/lib.rs')
-rw-r--r--third_party/rust/crc32fast/src/lib.rs178
1 files changed, 178 insertions, 0 deletions
diff --git a/third_party/rust/crc32fast/src/lib.rs b/third_party/rust/crc32fast/src/lib.rs
new file mode 100644
index 0000000000..f147286531
--- /dev/null
+++ b/third_party/rust/crc32fast/src/lib.rs
@@ -0,0 +1,178 @@
+//! ## Example
+//!
+//! ```rust
+//! use crc32fast::Hasher;
+//!
+//! let mut hasher = Hasher::new();
+//! hasher.update(b"foo bar baz");
+//! let checksum = hasher.finalize();
+//! ```
+//!
+//! ## Performance
+//!
+//! This crate contains multiple CRC32 implementations:
+//!
+//! - A fast baseline implementation which processes up to 16 bytes per iteration
+//! - An optimized implementation for modern `x86` using `sse` and `pclmulqdq` instructions
+//!
+//! Calling the `Hasher::new` constructor at runtime will perform a feature detection to select the most
+//! optimal implementation for the current CPU feature set.
+
+#![cfg_attr(not(feature = "std"), no_std)]
+#![cfg_attr(
+ all(feature = "nightly", target_arch = "aarch64"),
+ feature(stdsimd, aarch64_target_feature)
+)]
+
+#[deny(missing_docs)]
+#[cfg(test)]
+#[macro_use]
+extern crate quickcheck;
+
+#[macro_use]
+extern crate cfg_if;
+
+#[cfg(feature = "std")]
+use std as core;
+
+use core::fmt;
+use core::hash;
+
+mod baseline;
+mod combine;
+mod specialized;
+mod table;
+
+#[derive(Clone)]
+enum State {
+ Baseline(baseline::State),
+ Specialized(specialized::State),
+}
+
+#[derive(Clone)]
+/// Represents an in-progress CRC32 computation.
+pub struct Hasher {
+ amount: u64,
+ state: State,
+}
+
+const DEFAULT_INIT_STATE: u32 = 0;
+
+impl Hasher {
+ /// Create a new `Hasher`.
+ ///
+ /// This will perform a CPU feature detection at runtime to select the most
+ /// optimal implementation for the current processor architecture.
+ pub fn new() -> Self {
+ Self::new_with_initial(DEFAULT_INIT_STATE)
+ }
+
+ /// Create a new `Hasher` with an initial CRC32 state.
+ ///
+ /// This works just like `Hasher::new`, except that it allows for an initial
+ /// CRC32 state to be passed in.
+ pub fn new_with_initial(init: u32) -> Self {
+ Self::internal_new_specialized(init).unwrap_or_else(|| Self::internal_new_baseline(init))
+ }
+
+ #[doc(hidden)]
+ // Internal-only API. Don't use.
+ pub fn internal_new_baseline(init: u32) -> Self {
+ Hasher {
+ amount: 0,
+ state: State::Baseline(baseline::State::new(init)),
+ }
+ }
+
+ #[doc(hidden)]
+ // Internal-only API. Don't use.
+ pub fn internal_new_specialized(init: u32) -> Option<Self> {
+ {
+ if let Some(state) = specialized::State::new(init) {
+ return Some(Hasher {
+ amount: 0,
+ state: State::Specialized(state),
+ });
+ }
+ }
+ None
+ }
+
+ /// Process the given byte slice and update the hash state.
+ pub fn update(&mut self, buf: &[u8]) {
+ self.amount += buf.len() as u64;
+ match self.state {
+ State::Baseline(ref mut state) => state.update(buf),
+ State::Specialized(ref mut state) => state.update(buf),
+ }
+ }
+
+ /// Finalize the hash state and return the computed CRC32 value.
+ pub fn finalize(self) -> u32 {
+ match self.state {
+ State::Baseline(state) => state.finalize(),
+ State::Specialized(state) => state.finalize(),
+ }
+ }
+
+ /// Reset the hash state.
+ pub fn reset(&mut self) {
+ self.amount = 0;
+ match self.state {
+ State::Baseline(ref mut state) => state.reset(),
+ State::Specialized(ref mut state) => state.reset(),
+ }
+ }
+
+ /// Combine the hash state with the hash state for the subsequent block of bytes.
+ pub fn combine(&mut self, other: &Self) {
+ self.amount += other.amount;
+ let other_crc = other.clone().finalize();
+ match self.state {
+ State::Baseline(ref mut state) => state.combine(other_crc, other.amount),
+ State::Specialized(ref mut state) => state.combine(other_crc, other.amount),
+ }
+ }
+}
+
+impl fmt::Debug for Hasher {
+ fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
+ f.debug_struct("crc32fast::Hasher").finish()
+ }
+}
+
+impl Default for Hasher {
+ fn default() -> Self {
+ Self::new()
+ }
+}
+
+impl hash::Hasher for Hasher {
+ fn write(&mut self, bytes: &[u8]) {
+ self.update(bytes)
+ }
+
+ fn finish(&self) -> u64 {
+ u64::from(self.clone().finalize())
+ }
+}
+
+#[cfg(test)]
+mod test {
+ use super::Hasher;
+
+ quickcheck! {
+ fn combine(bytes_1: Vec<u8>, bytes_2: Vec<u8>) -> bool {
+ let mut hash_a = Hasher::new();
+ hash_a.update(&bytes_1);
+ hash_a.update(&bytes_2);
+ let mut hash_b = Hasher::new();
+ hash_b.update(&bytes_2);
+ let mut hash_c = Hasher::new();
+ hash_c.update(&bytes_1);
+ hash_c.combine(&hash_b);
+
+ hash_a.finalize() == hash_c.finalize()
+ }
+ }
+}