// Copyright 2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! An efficient hash map for node IDs use collections::{HashMap, HashSet}; use std::hash::{Hasher, Hash}; use std::io; use syntax::ast; pub type FnvHashMap = HashMap; pub type FnvHashSet = HashSet; pub type NodeMap = FnvHashMap; pub type DefIdMap = FnvHashMap; pub type NodeSet = FnvHashSet; pub type DefIdSet = FnvHashSet; // Hacks to get good names pub mod FnvHashMap { use std::hash::Hash; use collections::HashMap; pub fn new + Eq, V>() -> super::FnvHashMap { HashMap::with_hasher(super::FnvHasher) } } pub mod FnvHashSet { use std::hash::Hash; use collections::HashSet; pub fn new + Eq>() -> super::FnvHashSet { HashSet::with_hasher(super::FnvHasher) } } pub mod NodeMap { pub fn new() -> super::NodeMap { super::FnvHashMap::new() } } pub mod DefIdMap { pub fn new() -> super::DefIdMap { super::FnvHashMap::new() } } pub mod NodeSet { pub fn new() -> super::NodeSet { super::FnvHashSet::new() } } pub mod DefIdSet { pub fn new() -> super::DefIdSet { super::FnvHashSet::new() } } /// A speedy hash algorithm for node ids and def ids. The hashmap in /// libcollections by default uses SipHash which isn't quite as speedy as we /// want. In the compiler we're not really worried about DOS attempts, so we /// just default to a non-cryptographic hash. /// /// This uses FNV hashing, as described here: /// http://en.wikipedia.org/wiki/Fowler%E2%80%93Noll%E2%80%93Vo_hash_function #[deriving(Clone)] pub struct FnvHasher; pub struct FnvState(u64); impl Hasher for FnvHasher { fn hash>(&self, t: &T) -> u64 { let mut state = FnvState(0xcbf29ce484222325); t.hash(&mut state); let FnvState(ret) = state; return ret; } } impl Writer for FnvState { fn write(&mut self, bytes: &[u8]) -> io::IoResult<()> { let FnvState(mut hash) = *self; for byte in bytes.iter() { hash = hash ^ (*byte as u64); hash = hash * 0x100000001b3; } *self = FnvState(hash); Ok(()) } }