A number of trait methods like PartialEq::eq or Hash::hash don't actually need a distinct arm for each variant, because the code within the arm only depends on the number and types of the fields in the variants. We can easily exploit this fact to create less and better code for enums with multiple variants that have no fields at all, the extreme case being C-like enums. For nickel.rs and its by now infamous 800 variant enum, this reduces optimized compile times by 25% and non-optimized compile times by 40%. Also peak memory usage is down by almost 40% (310MB down to 190MB). To be fair, most other crates don't benefit nearly as much, because they don't have as huge enums. The crates in the Rust distribution that I measured saw basically no change in compile times (I only tried optimized builds) and only 1-2% reduction in peak memory usage.
104 lines
3.7 KiB
Rust
104 lines
3.7 KiB
Rust
// 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 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
use deriving;
|
|
use deriving::generic::*;
|
|
use deriving::generic::ty::*;
|
|
|
|
use syntax::ast::{MetaItem, Expr, Mutability};
|
|
use syntax::codemap::Span;
|
|
use syntax::ext::base::{ExtCtxt, Annotatable};
|
|
use syntax::ext::build::AstBuilder;
|
|
use syntax::ptr::P;
|
|
|
|
pub fn expand_deriving_hash(cx: &mut ExtCtxt,
|
|
span: Span,
|
|
mitem: &MetaItem,
|
|
item: &Annotatable,
|
|
push: &mut FnMut(Annotatable))
|
|
{
|
|
|
|
let path = Path::new_(pathvec_std!(cx, core::hash::Hash), None,
|
|
vec!(), true);
|
|
|
|
let typaram = &*deriving::hygienic_type_parameter(item, "__H");
|
|
|
|
let arg = Path::new_local(typaram);
|
|
let hash_trait_def = TraitDef {
|
|
span: span,
|
|
attributes: Vec::new(),
|
|
path: path,
|
|
additional_bounds: Vec::new(),
|
|
generics: LifetimeBounds::empty(),
|
|
is_unsafe: false,
|
|
methods: vec!(
|
|
MethodDef {
|
|
name: "hash",
|
|
generics: LifetimeBounds {
|
|
lifetimes: Vec::new(),
|
|
bounds: vec![(typaram,
|
|
vec![path_std!(cx, core::hash::Hasher)])],
|
|
},
|
|
explicit_self: borrowed_explicit_self(),
|
|
args: vec!(Ptr(Box::new(Literal(arg)), Borrowed(None, Mutability::Mutable))),
|
|
ret_ty: nil_ty(),
|
|
attributes: vec![],
|
|
is_unsafe: false,
|
|
unify_fieldless_variants: true,
|
|
combine_substructure: combine_substructure(Box::new(|a, b, c| {
|
|
hash_substructure(a, b, c)
|
|
}))
|
|
}
|
|
),
|
|
associated_types: Vec::new(),
|
|
};
|
|
|
|
hash_trait_def.expand(cx, mitem, item, push);
|
|
}
|
|
|
|
fn hash_substructure(cx: &mut ExtCtxt, trait_span: Span, substr: &Substructure) -> P<Expr> {
|
|
let state_expr = match (substr.nonself_args.len(), substr.nonself_args.get(0)) {
|
|
(1, Some(o_f)) => o_f,
|
|
_ => cx.span_bug(trait_span, "incorrect number of arguments in `derive(Hash)`")
|
|
};
|
|
let call_hash = |span, thing_expr| {
|
|
let hash_path = {
|
|
let strs = cx.std_path(&["hash", "Hash", "hash"]);
|
|
|
|
cx.expr_path(cx.path_global(span, strs))
|
|
};
|
|
let ref_thing = cx.expr_addr_of(span, thing_expr);
|
|
let expr = cx.expr_call(span, hash_path, vec!(ref_thing, state_expr.clone()));
|
|
cx.stmt_expr(expr)
|
|
};
|
|
let mut stmts = Vec::new();
|
|
|
|
let fields = match *substr.fields {
|
|
Struct(_, ref fs) => fs,
|
|
EnumMatching(_, _, ref fs) => {
|
|
let variant_value = deriving::call_intrinsic(cx,
|
|
trait_span,
|
|
"discriminant_value",
|
|
vec![cx.expr_self(trait_span)]);
|
|
|
|
stmts.push(call_hash(trait_span, variant_value));
|
|
|
|
fs
|
|
}
|
|
_ => cx.span_bug(trait_span, "impossible substructure in `derive(Hash)`")
|
|
};
|
|
|
|
for &FieldInfo { ref self_, span, .. } in fields {
|
|
stmts.push(call_hash(span, self_.clone()));
|
|
}
|
|
|
|
cx.expr_block(cx.block(trait_span, stmts, None))
|
|
}
|