mirror of
https://github.com/rust-lang/rust.git
synced 2025-04-28 02:57:37 +00:00
Auto merge of #44015 - kennytm:hasher, r=alexcrichton
impl Hasher for {&mut Hasher, Box<Hasher>} **Rationale:** The `Hash` trait has `fn hash<H: Hasher>(&self, state: &mut H)`, which can only accept a `Sized` hasher, even if the `Hasher` trait is object-safe. We cannot retroactively add the `?Sized` bound without breaking stability, thus implementing `Hasher` to a trait object reference is the next best solution. **Warning:** These `impl` are insta-stable, and should need an FCP. I don't think a full RFC is necessary.
This commit is contained in:
commit
2fdccaffe6
@ -62,7 +62,7 @@ use core::any::Any;
|
||||
use core::borrow;
|
||||
use core::cmp::Ordering;
|
||||
use core::fmt;
|
||||
use core::hash::{self, Hash};
|
||||
use core::hash::{self, Hash, Hasher};
|
||||
use core::iter::FusedIterator;
|
||||
use core::marker::{self, Unsize};
|
||||
use core::mem;
|
||||
@ -456,6 +456,52 @@ impl<T: ?Sized + Hash> Hash for Box<T> {
|
||||
}
|
||||
}
|
||||
|
||||
#[stable(feature = "indirect_hasher_impl", since = "1.22.0")]
|
||||
impl<T: ?Sized + Hasher> Hasher for Box<T> {
|
||||
fn finish(&self) -> u64 {
|
||||
(**self).finish()
|
||||
}
|
||||
fn write(&mut self, bytes: &[u8]) {
|
||||
(**self).write(bytes)
|
||||
}
|
||||
fn write_u8(&mut self, i: u8) {
|
||||
(**self).write_u8(i)
|
||||
}
|
||||
fn write_u16(&mut self, i: u16) {
|
||||
(**self).write_u16(i)
|
||||
}
|
||||
fn write_u32(&mut self, i: u32) {
|
||||
(**self).write_u32(i)
|
||||
}
|
||||
fn write_u64(&mut self, i: u64) {
|
||||
(**self).write_u64(i)
|
||||
}
|
||||
fn write_u128(&mut self, i: u128) {
|
||||
(**self).write_u128(i)
|
||||
}
|
||||
fn write_usize(&mut self, i: usize) {
|
||||
(**self).write_usize(i)
|
||||
}
|
||||
fn write_i8(&mut self, i: i8) {
|
||||
(**self).write_i8(i)
|
||||
}
|
||||
fn write_i16(&mut self, i: i16) {
|
||||
(**self).write_i16(i)
|
||||
}
|
||||
fn write_i32(&mut self, i: i32) {
|
||||
(**self).write_i32(i)
|
||||
}
|
||||
fn write_i64(&mut self, i: i64) {
|
||||
(**self).write_i64(i)
|
||||
}
|
||||
fn write_i128(&mut self, i: i128) {
|
||||
(**self).write_i128(i)
|
||||
}
|
||||
fn write_isize(&mut self, i: isize) {
|
||||
(**self).write_isize(i)
|
||||
}
|
||||
}
|
||||
|
||||
#[stable(feature = "from_for_ptrs", since = "1.6.0")]
|
||||
impl<T> From<T> for Box<T> {
|
||||
fn from(t: T) -> Self {
|
||||
|
@ -121,7 +121,7 @@
|
||||
#![feature(unsize)]
|
||||
#![feature(allocator_internals)]
|
||||
|
||||
#![cfg_attr(not(test), feature(fused, fn_traits, placement_new_protocol, swap_with_slice))]
|
||||
#![cfg_attr(not(test), feature(fused, fn_traits, placement_new_protocol, swap_with_slice, i128))]
|
||||
#![cfg_attr(test, feature(test, box_heap))]
|
||||
|
||||
// Allow testing this library
|
||||
|
@ -50,3 +50,19 @@ fn hash<T: Hash>(t: &T) -> u64 {
|
||||
t.hash(&mut s);
|
||||
s.finish()
|
||||
}
|
||||
|
||||
// FIXME: Instantiated functions with i128 in the signature is not supported in Emscripten.
|
||||
// See https://github.com/kripken/emscripten-fastcomp/issues/169
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
#[test]
|
||||
fn test_boxed_hasher() {
|
||||
let ordinary_hash = hash(&5u32);
|
||||
|
||||
let mut hasher_1 = Box::new(DefaultHasher::new());
|
||||
5u32.hash(&mut hasher_1);
|
||||
assert_eq!(ordinary_hash, hasher_1.finish());
|
||||
|
||||
let mut hasher_2 = Box::new(DefaultHasher::new()) as Box<Hasher>;
|
||||
5u32.hash(&mut hasher_2);
|
||||
assert_eq!(ordinary_hash, hasher_2.finish());
|
||||
}
|
||||
|
@ -359,6 +359,52 @@ pub trait Hasher {
|
||||
}
|
||||
}
|
||||
|
||||
#[stable(feature = "indirect_hasher_impl", since = "1.22.0")]
|
||||
impl<'a, H: Hasher + ?Sized> Hasher for &'a mut H {
|
||||
fn finish(&self) -> u64 {
|
||||
(**self).finish()
|
||||
}
|
||||
fn write(&mut self, bytes: &[u8]) {
|
||||
(**self).write(bytes)
|
||||
}
|
||||
fn write_u8(&mut self, i: u8) {
|
||||
(**self).write_u8(i)
|
||||
}
|
||||
fn write_u16(&mut self, i: u16) {
|
||||
(**self).write_u16(i)
|
||||
}
|
||||
fn write_u32(&mut self, i: u32) {
|
||||
(**self).write_u32(i)
|
||||
}
|
||||
fn write_u64(&mut self, i: u64) {
|
||||
(**self).write_u64(i)
|
||||
}
|
||||
fn write_u128(&mut self, i: u128) {
|
||||
(**self).write_u128(i)
|
||||
}
|
||||
fn write_usize(&mut self, i: usize) {
|
||||
(**self).write_usize(i)
|
||||
}
|
||||
fn write_i8(&mut self, i: i8) {
|
||||
(**self).write_i8(i)
|
||||
}
|
||||
fn write_i16(&mut self, i: i16) {
|
||||
(**self).write_i16(i)
|
||||
}
|
||||
fn write_i32(&mut self, i: i32) {
|
||||
(**self).write_i32(i)
|
||||
}
|
||||
fn write_i64(&mut self, i: i64) {
|
||||
(**self).write_i64(i)
|
||||
}
|
||||
fn write_i128(&mut self, i: i128) {
|
||||
(**self).write_i128(i)
|
||||
}
|
||||
fn write_isize(&mut self, i: isize) {
|
||||
(**self).write_isize(i)
|
||||
}
|
||||
}
|
||||
|
||||
/// A trait for creating instances of [`Hasher`].
|
||||
///
|
||||
/// A `BuildHasher` is typically used (e.g. by [`HashMap`]) to create
|
||||
|
@ -109,3 +109,16 @@ fn test_custom_state() {
|
||||
|
||||
assert_eq!(hash(&Custom { hash: 5 }), 5);
|
||||
}
|
||||
|
||||
// FIXME: Instantiated functions with i128 in the signature is not supported in Emscripten.
|
||||
// See https://github.com/kripken/emscripten-fastcomp/issues/169
|
||||
#[cfg(not(target_os = "emscripten"))]
|
||||
#[test]
|
||||
fn test_indirect_hasher() {
|
||||
let mut hasher = MyHasher { hash: 0 };
|
||||
{
|
||||
let mut indirect_hasher: &mut Hasher = &mut hasher;
|
||||
5u32.hash(&mut indirect_hasher);
|
||||
}
|
||||
assert_eq!(hasher.hash, 5);
|
||||
}
|
||||
|
@ -76,6 +76,7 @@ const EXCEPTION_PATHS: &'static [&'static str] = &[
|
||||
|
||||
// std testing crates, ok for now at least
|
||||
"src/libcore/tests",
|
||||
"src/liballoc/tests/lib.rs",
|
||||
|
||||
// non-std crates
|
||||
"src/test",
|
||||
|
Loading…
Reference in New Issue
Block a user