2014-07-27 06:50:46 -05:00
|
|
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
2012-12-03 18:48:01 -06:00
|
|
|
// 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.
|
|
|
|
|
2014-09-12 05:10:30 -05:00
|
|
|
//! AST walker. Each overridden visit method has full control over what
|
|
|
|
//! happens with its node, it can do its own traversal of the node's children,
|
|
|
|
//! call `visit::walk_*` to apply the default traversal algorithm, or prevent
|
|
|
|
//! deeper traversal by doing nothing.
|
2014-06-09 15:12:30 -05:00
|
|
|
//!
|
|
|
|
//! Note: it is an important invariant that the default visitor walks the body
|
|
|
|
//! of a function in "execution order" (more concretely, reverse post-order
|
|
|
|
//! with respect to the CFG implied by the AST), meaning that if AST node A may
|
|
|
|
//! execute before AST node B, then A is visited first. The borrow checker in
|
|
|
|
//! particular relies on this property.
|
|
|
|
//!
|
2014-07-09 16:48:12 -05:00
|
|
|
//! Note: walking an AST before macro expansion is probably a bad idea. For
|
|
|
|
//! instance, a walker looking for item names in a module will miss all of
|
|
|
|
//! those that are created by the expansion of a macro.
|
|
|
|
|
2014-11-06 02:05:53 -06:00
|
|
|
pub use self::FnKind::*;
|
|
|
|
|
2014-04-02 03:19:41 -05:00
|
|
|
use abi::Abi;
|
2012-09-04 13:37:29 -05:00
|
|
|
use ast::*;
|
2012-12-23 16:41:37 -06:00
|
|
|
use ast;
|
2013-08-31 11:13:04 -05:00
|
|
|
use codemap::Span;
|
2014-09-13 11:06:01 -05:00
|
|
|
use ptr::P;
|
2014-03-19 09:52:37 -05:00
|
|
|
use owned_slice::OwnedSlice;
|
2012-12-23 16:41:37 -06:00
|
|
|
|
2015-01-03 21:54:18 -06:00
|
|
|
#[derive(Copy)]
|
2014-01-09 07:05:33 -06:00
|
|
|
pub enum FnKind<'a> {
|
2014-06-09 15:12:30 -05:00
|
|
|
/// fn foo() or extern "Abi" fn foo()
|
2014-12-09 09:36:46 -06:00
|
|
|
FkItemFn(Ident, &'a Generics, Unsafety, Abi),
|
2013-03-13 21:25:28 -05:00
|
|
|
|
2014-06-09 15:12:30 -05:00
|
|
|
/// fn foo(&self)
|
2015-03-10 05:28:44 -05:00
|
|
|
FkMethod(Ident, &'a Method),
|
2013-03-13 21:25:28 -05:00
|
|
|
|
2014-06-09 15:12:30 -05:00
|
|
|
/// |x, y| ...
|
|
|
|
/// proc(x, y) ...
|
2014-01-09 07:05:33 -06:00
|
|
|
FkFnBlock,
|
2011-12-29 22:07:55 -06:00
|
|
|
}
|
|
|
|
|
2014-05-12 11:58:23 -05:00
|
|
|
/// Each method of the Visitor trait is a hook to be potentially
|
2014-06-08 23:00:52 -05:00
|
|
|
/// overridden. Each method's default implementation recursively visits
|
2014-05-12 11:58:23 -05:00
|
|
|
/// the substructure of the input via the corresponding `walk` method;
|
|
|
|
/// e.g. the `visit_mod` method by default calls `visit::walk_mod`.
|
|
|
|
///
|
|
|
|
/// If you want to ensure that your code handles every variant
|
|
|
|
/// explicitly, you need to override each method. (And you also need
|
|
|
|
/// to monitor future changes to `Visitor` in case a new method with a
|
|
|
|
/// new default implementation gets introduced.)
|
2014-12-18 14:27:41 -06:00
|
|
|
pub trait Visitor<'v> : Sized {
|
2014-11-18 10:39:16 -06:00
|
|
|
fn visit_name(&mut self, _span: Span, _name: Name) {
|
|
|
|
// Nothing to do.
|
|
|
|
}
|
|
|
|
fn visit_ident(&mut self, span: Span, ident: Ident) {
|
|
|
|
self.visit_name(span, ident.name);
|
2013-11-22 06:24:49 -06:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
fn visit_mod(&mut self, m: &'v Mod, _s: Span, _n: NodeId) { walk_mod(self, m) }
|
|
|
|
fn visit_foreign_item(&mut self, i: &'v ForeignItem) { walk_foreign_item(self, i) }
|
|
|
|
fn visit_item(&mut self, i: &'v Item) { walk_item(self, i) }
|
|
|
|
fn visit_local(&mut self, l: &'v Local) { walk_local(self, l) }
|
|
|
|
fn visit_block(&mut self, b: &'v Block) { walk_block(self, b) }
|
|
|
|
fn visit_stmt(&mut self, s: &'v Stmt) { walk_stmt(self, s) }
|
|
|
|
fn visit_arm(&mut self, a: &'v Arm) { walk_arm(self, a) }
|
|
|
|
fn visit_pat(&mut self, p: &'v Pat) { walk_pat(self, p) }
|
|
|
|
fn visit_decl(&mut self, d: &'v Decl) { walk_decl(self, d) }
|
|
|
|
fn visit_expr(&mut self, ex: &'v Expr) { walk_expr(self, ex) }
|
|
|
|
fn visit_expr_post(&mut self, _ex: &'v Expr) { }
|
|
|
|
fn visit_ty(&mut self, t: &'v Ty) { walk_ty(self, t) }
|
|
|
|
fn visit_generics(&mut self, g: &'v Generics) { walk_generics(self, g) }
|
|
|
|
fn visit_fn(&mut self, fk: FnKind<'v>, fd: &'v FnDecl, b: &'v Block, s: Span, _: NodeId) {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_fn(self, fk, fd, b, s)
|
|
|
|
}
|
2015-03-10 05:28:44 -05:00
|
|
|
fn visit_trait_item(&mut self, ti: &'v TraitItem) { walk_trait_item(self, ti) }
|
|
|
|
fn visit_impl_item(&mut self, ii: &'v ImplItem) { walk_impl_item(self, ii) }
|
2014-11-07 05:53:45 -06:00
|
|
|
fn visit_trait_ref(&mut self, t: &'v TraitRef) { walk_trait_ref(self, t) }
|
2014-11-15 15:55:27 -06:00
|
|
|
fn visit_ty_param_bound(&mut self, bounds: &'v TyParamBound) {
|
|
|
|
walk_ty_param_bound(self, bounds)
|
|
|
|
}
|
2014-12-24 00:38:10 -06:00
|
|
|
fn visit_poly_trait_ref(&mut self, t: &'v PolyTraitRef, m: &'v TraitBoundModifier) {
|
|
|
|
walk_poly_trait_ref(self, t, m)
|
2014-11-07 05:53:45 -06:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
fn visit_struct_def(&mut self, s: &'v StructDef, _: Ident, _: &'v Generics, _: NodeId) {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_struct_def(self, s)
|
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
fn visit_struct_field(&mut self, s: &'v StructField) { walk_struct_field(self, s) }
|
|
|
|
fn visit_variant(&mut self, v: &'v Variant, g: &'v Generics) { walk_variant(self, v, g) }
|
2014-11-25 20:17:11 -06:00
|
|
|
|
|
|
|
/// Visits an optional reference to a lifetime. The `span` is the span of some surrounding
|
|
|
|
/// reference should opt_lifetime be None.
|
2013-10-28 16:37:10 -05:00
|
|
|
fn visit_opt_lifetime_ref(&mut self,
|
|
|
|
_span: Span,
|
2014-09-09 17:54:36 -05:00
|
|
|
opt_lifetime: &'v Option<Lifetime>) {
|
2013-10-28 16:37:10 -05:00
|
|
|
match *opt_lifetime {
|
2014-09-12 05:10:30 -05:00
|
|
|
Some(ref l) => self.visit_lifetime_ref(l),
|
2013-10-28 16:37:10 -05:00
|
|
|
None => ()
|
|
|
|
}
|
|
|
|
}
|
2014-12-13 04:34:34 -06:00
|
|
|
fn visit_lifetime_bound(&mut self, lifetime: &'v Lifetime) {
|
|
|
|
walk_lifetime_bound(self, lifetime)
|
|
|
|
}
|
2014-11-18 10:39:16 -06:00
|
|
|
fn visit_lifetime_ref(&mut self, lifetime: &'v Lifetime) {
|
2014-12-13 04:34:34 -06:00
|
|
|
walk_lifetime_ref(self, lifetime)
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2014-11-18 10:39:16 -06:00
|
|
|
fn visit_lifetime_def(&mut self, lifetime: &'v LifetimeDef) {
|
|
|
|
walk_lifetime_def(self, lifetime)
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
fn visit_explicit_self(&mut self, es: &'v ExplicitSelf) {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_explicit_self(self, es)
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2015-01-02 15:39:05 -06:00
|
|
|
fn visit_mac(&mut self, _mac: &'v Mac) {
|
2014-10-09 14:17:22 -05:00
|
|
|
panic!("visit_mac disabled by default");
|
2014-07-09 16:48:12 -05:00
|
|
|
// NB: see note about macros above.
|
|
|
|
// if you really want a visitor that
|
|
|
|
// works on macros, use this
|
|
|
|
// definition in your trait impl:
|
2015-01-02 15:39:05 -06:00
|
|
|
// visit::walk_mac(self, _mac)
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
fn visit_path(&mut self, path: &'v Path, _id: ast::NodeId) {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_path(self, path)
|
2013-12-08 13:25:35 -06:00
|
|
|
}
|
2014-11-15 15:55:27 -06:00
|
|
|
fn visit_path_segment(&mut self, path_span: Span, path_segment: &'v PathSegment) {
|
|
|
|
walk_path_segment(self, path_span, path_segment)
|
|
|
|
}
|
|
|
|
fn visit_path_parameters(&mut self, path_span: Span, path_parameters: &'v PathParameters) {
|
|
|
|
walk_path_parameters(self, path_span, path_parameters)
|
|
|
|
}
|
2014-12-28 09:07:21 -06:00
|
|
|
fn visit_assoc_type_binding(&mut self, type_binding: &'v TypeBinding) {
|
|
|
|
walk_assoc_type_binding(self, type_binding)
|
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
fn visit_attribute(&mut self, _attr: &'v Attribute) {}
|
2013-08-08 07:23:25 -05:00
|
|
|
}
|
|
|
|
|
2014-08-05 21:44:21 -05:00
|
|
|
pub fn walk_inlined_item<'v,V>(visitor: &mut V, item: &'v InlinedItem)
|
|
|
|
where V: Visitor<'v> {
|
2014-01-15 13:39:08 -06:00
|
|
|
match *item {
|
2014-09-09 17:54:36 -05:00
|
|
|
IIItem(ref i) => visitor.visit_item(&**i),
|
|
|
|
IIForeign(ref i) => visitor.visit_foreign_item(&**i),
|
2014-09-13 11:06:01 -05:00
|
|
|
IITraitItem(_, ref ti) => visitor.visit_trait_item(ti),
|
2015-03-10 05:28:44 -05:00
|
|
|
IIImplItem(_, ref ii) => visitor.visit_impl_item(ii),
|
2014-01-15 13:39:08 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_crate<'v, V: Visitor<'v>>(visitor: &mut V, krate: &'v Crate) {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_mod(&krate.module, krate.span, CRATE_NODE_ID);
|
2015-01-31 11:20:46 -06:00
|
|
|
for attr in &krate.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_mod<'v, V: Visitor<'v>>(visitor: &mut V, module: &'v Mod) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for item in &module.items {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_item(&**item)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_local<'v, V: Visitor<'v>>(visitor: &mut V, local: &'v Local) {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&*local.pat);
|
2015-01-02 05:55:31 -06:00
|
|
|
walk_ty_opt(visitor, &local.ty);
|
2014-09-09 17:54:36 -05:00
|
|
|
walk_expr_opt(visitor, &local.init);
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
|
|
|
|
2014-11-18 10:39:16 -06:00
|
|
|
pub fn walk_lifetime_def<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
lifetime_def: &'v LifetimeDef) {
|
2014-11-19 16:06:53 -06:00
|
|
|
visitor.visit_name(lifetime_def.lifetime.span, lifetime_def.lifetime.name);
|
2015-01-31 11:20:46 -06:00
|
|
|
for bound in &lifetime_def.bounds {
|
2014-12-13 04:34:34 -06:00
|
|
|
visitor.visit_lifetime_bound(bound);
|
2014-11-18 10:39:16 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-13 04:34:34 -06:00
|
|
|
pub fn walk_lifetime_bound<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
lifetime_ref: &'v Lifetime) {
|
|
|
|
visitor.visit_lifetime_ref(lifetime_ref)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn walk_lifetime_ref<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
lifetime_ref: &'v Lifetime) {
|
|
|
|
visitor.visit_name(lifetime_ref.span, lifetime_ref.name)
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_explicit_self<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
explicit_self: &'v ExplicitSelf) {
|
2013-10-28 16:37:10 -05:00
|
|
|
match explicit_self.node {
|
2014-07-23 12:21:50 -05:00
|
|
|
SelfStatic | SelfValue(_) => {},
|
2014-07-06 17:10:57 -05:00
|
|
|
SelfRegion(ref lifetime, _, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_opt_lifetime_ref(explicit_self.span, lifetime)
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
SelfExplicit(ref typ, _) => visitor.visit_ty(&**typ),
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-07 05:53:45 -06:00
|
|
|
pub fn walk_poly_trait_ref<'v, V>(visitor: &mut V,
|
2014-12-24 00:38:10 -06:00
|
|
|
trait_ref: &'v PolyTraitRef,
|
|
|
|
_modifier: &'v TraitBoundModifier)
|
2014-11-07 05:53:45 -06:00
|
|
|
where V: Visitor<'v>
|
|
|
|
{
|
2014-11-15 15:55:27 -06:00
|
|
|
walk_lifetime_decls_helper(visitor, &trait_ref.bound_lifetimes);
|
2014-11-07 05:53:45 -06:00
|
|
|
visitor.visit_trait_ref(&trait_ref.trait_ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn walk_trait_ref<'v,V>(visitor: &mut V,
|
|
|
|
trait_ref: &'v TraitRef)
|
|
|
|
where V: Visitor<'v>
|
|
|
|
{
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_path(&trait_ref.path, trait_ref.ref_id)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_item<'v, V: Visitor<'v>>(visitor: &mut V, item: &'v Item) {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ident(item.span, item.ident);
|
2013-07-19 20:42:11 -05:00
|
|
|
match item.node {
|
2015-01-13 09:30:17 -06:00
|
|
|
ItemExternCrate(..) => {}
|
|
|
|
ItemUse(ref vp) => {
|
|
|
|
match vp.node {
|
|
|
|
ViewPathSimple(ident, ref path) => {
|
|
|
|
visitor.visit_ident(vp.span, ident);
|
|
|
|
visitor.visit_path(path, item.id);
|
|
|
|
}
|
|
|
|
ViewPathGlob(ref path) => {
|
|
|
|
visitor.visit_path(path, item.id);
|
|
|
|
}
|
|
|
|
ViewPathList(ref prefix, ref list) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for id in list {
|
2015-01-13 09:30:17 -06:00
|
|
|
match id.node {
|
|
|
|
PathListIdent { name, .. } => {
|
|
|
|
visitor.visit_ident(id.span, name);
|
|
|
|
}
|
|
|
|
PathListMod { .. } => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note that the `prefix` here is not a complete
|
|
|
|
// path, so we don't use `visit_path`.
|
|
|
|
walk_path(visitor, prefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
rustc: Add `const` globals to the language
This change is an implementation of [RFC 69][rfc] which adds a third kind of
global to the language, `const`. This global is most similar to what the old
`static` was, and if you're unsure about what to use then you should use a
`const`.
The semantics of these three kinds of globals are:
* A `const` does not represent a memory location, but only a value. Constants
are translated as rvalues, which means that their values are directly inlined
at usage location (similar to a #define in C/C++). Constant values are, well,
constant, and can not be modified. Any "modification" is actually a
modification to a local value on the stack rather than the actual constant
itself.
Almost all values are allowed inside constants, whether they have interior
mutability or not. There are a few minor restrictions listed in the RFC, but
they should in general not come up too often.
* A `static` now always represents a memory location (unconditionally). Any
references to the same `static` are actually a reference to the same memory
location. Only values whose types ascribe to `Sync` are allowed in a `static`.
This restriction is in place because many threads may access a `static`
concurrently. Lifting this restriction (and allowing unsafe access) is a
future extension not implemented at this time.
* A `static mut` continues to always represent a memory location. All references
to a `static mut` continue to be `unsafe`.
This is a large breaking change, and many programs will need to be updated
accordingly. A summary of the breaking changes is:
* Statics may no longer be used in patterns. Statics now always represent a
memory location, which can sometimes be modified. To fix code, repurpose the
matched-on-`static` to a `const`.
static FOO: uint = 4;
match n {
FOO => { /* ... */ }
_ => { /* ... */ }
}
change this code to:
const FOO: uint = 4;
match n {
FOO => { /* ... */ }
_ => { /* ... */ }
}
* Statics may no longer refer to other statics by value. Due to statics being
able to change at runtime, allowing them to reference one another could
possibly lead to confusing semantics. If you are in this situation, use a
constant initializer instead. Note, however, that statics may reference other
statics by address, however.
* Statics may no longer be used in constant expressions, such as array lengths.
This is due to the same restrictions as listed above. Use a `const` instead.
[breaking-change]
[rfc]: https://github.com/rust-lang/rfcs/pull/246
2014-10-06 10:17:01 -05:00
|
|
|
ItemStatic(ref typ, _, ref expr) |
|
|
|
|
ItemConst(ref typ, ref expr) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&**typ);
|
|
|
|
visitor.visit_expr(&**expr);
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
ItemFn(ref declaration, fn_style, abi, ref generics, ref body) => {
|
|
|
|
visitor.visit_fn(FkItemFn(item.ident, generics, fn_style, abi),
|
|
|
|
&**declaration,
|
|
|
|
&**body,
|
2013-07-19 20:42:11 -05:00
|
|
|
item.span,
|
2014-09-12 05:10:30 -05:00
|
|
|
item.id)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
ItemMod(ref module) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_mod(module, item.span, item.id)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
ItemForeignMod(ref foreign_module) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for foreign_item in &foreign_module.items {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_foreign_item(&**foreign_item)
|
2013-07-02 14:47:32 -05:00
|
|
|
}
|
2011-12-20 09:33:55 -06:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ItemTy(ref typ, ref type_parameters) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&**typ);
|
|
|
|
visitor.visit_generics(type_parameters)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
ItemEnum(ref enum_definition, ref type_parameters) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_generics(type_parameters);
|
|
|
|
walk_enum_def(visitor, enum_definition, type_parameters)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2015-02-07 07:24:34 -06:00
|
|
|
ItemDefaultImpl(_, ref trait_ref) => {
|
2015-01-22 15:14:52 -06:00
|
|
|
visitor.visit_trait_ref(trait_ref)
|
|
|
|
}
|
2014-12-28 16:33:18 -06:00
|
|
|
ItemImpl(_, _,
|
2014-12-10 05:15:06 -06:00
|
|
|
ref type_parameters,
|
2014-01-09 02:42:28 -06:00
|
|
|
ref trait_reference,
|
2014-09-09 17:54:36 -05:00
|
|
|
ref typ,
|
2014-08-04 15:56:56 -05:00
|
|
|
ref impl_items) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_generics(type_parameters);
|
2014-01-09 02:42:28 -06:00
|
|
|
match *trait_reference {
|
2014-11-07 05:53:45 -06:00
|
|
|
Some(ref trait_reference) => visitor.visit_trait_ref(trait_reference),
|
2014-01-09 02:42:28 -06:00
|
|
|
None => ()
|
2013-03-27 05:16:28 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
visitor.visit_ty(&**typ);
|
2015-01-31 11:20:46 -06:00
|
|
|
for impl_item in impl_items {
|
2015-03-10 05:28:44 -05:00
|
|
|
visitor.visit_impl_item(impl_item);
|
2013-02-18 23:25:44 -06:00
|
|
|
}
|
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ItemStruct(ref struct_definition, ref generics) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_generics(generics);
|
2014-05-16 02:16:13 -05:00
|
|
|
visitor.visit_struct_def(&**struct_definition,
|
2013-07-19 20:42:11 -05:00
|
|
|
item.ident,
|
|
|
|
generics,
|
2014-09-12 05:10:30 -05:00
|
|
|
item.id)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-12-24 00:38:10 -06:00
|
|
|
ItemTrait(_, ref generics, ref bounds, ref methods) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_generics(generics);
|
2014-11-15 15:55:27 -06:00
|
|
|
walk_ty_param_bounds_helper(visitor, bounds);
|
2015-01-31 11:20:46 -06:00
|
|
|
for method in methods {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_trait_item(method)
|
2013-02-18 23:25:44 -06:00
|
|
|
}
|
|
|
|
}
|
2015-01-02 15:39:05 -06:00
|
|
|
ItemMac(ref mac) => visitor.visit_mac(mac),
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
for attr in &item.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_enum_def<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
enum_definition: &'v EnumDef,
|
|
|
|
generics: &'v Generics) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for variant in &enum_definition.variants {
|
2014-09-09 17:54:36 -05:00
|
|
|
visitor.visit_variant(&**variant, generics);
|
2013-11-11 22:17:47 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_variant<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
variant: &'v Variant,
|
|
|
|
generics: &'v Generics) {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ident(variant.span, variant.node.name);
|
2013-11-22 06:24:49 -06:00
|
|
|
|
2013-11-11 22:17:47 -06:00
|
|
|
match variant.node.kind {
|
2014-01-09 07:05:33 -06:00
|
|
|
TupleVariantKind(ref variant_arguments) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for variant_argument in variant_arguments {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&*variant_argument.ty)
|
2012-08-08 16:17:52 -05:00
|
|
|
}
|
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
StructVariantKind(ref struct_definition) => {
|
|
|
|
visitor.visit_struct_def(&**struct_definition,
|
2013-11-11 22:17:47 -06:00
|
|
|
variant.node.name,
|
|
|
|
generics,
|
2014-09-12 05:10:30 -05:00
|
|
|
variant.node.id)
|
2013-11-11 22:17:47 -06:00
|
|
|
}
|
2012-08-08 16:17:52 -05:00
|
|
|
}
|
2013-12-16 01:13:54 -06:00
|
|
|
match variant.node.disr_expr {
|
2014-09-12 05:10:30 -05:00
|
|
|
Some(ref expr) => visitor.visit_expr(&**expr),
|
2013-12-16 01:13:54 -06:00
|
|
|
None => ()
|
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
for attr in &variant.node.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2012-08-08 16:17:52 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn skip_ty<'v, V: Visitor<'v>>(_: &mut V, _: &'v Ty) {
|
2013-07-19 20:42:11 -05:00
|
|
|
// Empty!
|
|
|
|
}
|
2011-11-22 03:57:47 -06:00
|
|
|
|
2015-01-02 05:55:31 -06:00
|
|
|
pub fn walk_ty_opt<'v, V: Visitor<'v>>(visitor: &mut V, optional_type: &'v Option<P<Ty>>) {
|
|
|
|
match *optional_type {
|
|
|
|
Some(ref ty) => visitor.visit_ty(&**ty),
|
|
|
|
None => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_ty<'v, V: Visitor<'v>>(visitor: &mut V, typ: &'v Ty) {
|
2013-07-19 20:42:11 -05:00
|
|
|
match typ.node {
|
2014-11-09 09:14:15 -06:00
|
|
|
TyVec(ref ty) | TyParen(ref ty) => {
|
2014-09-09 17:54:36 -05:00
|
|
|
visitor.visit_ty(&**ty)
|
2013-12-16 12:01:40 -06:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
TyPtr(ref mutable_type) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&*mutable_type.ty)
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
TyRptr(ref lifetime, ref mutable_type) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_opt_lifetime_ref(typ.span, lifetime);
|
|
|
|
visitor.visit_ty(&*mutable_type.ty)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
TyTup(ref tuple_element_types) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for tuple_element_type in tuple_element_types {
|
2014-09-09 17:54:36 -05:00
|
|
|
visitor.visit_ty(&**tuple_element_type)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
TyBareFn(ref function_declaration) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for argument in &function_declaration.decl.inputs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&*argument.ty)
|
2013-07-02 14:47:32 -05:00
|
|
|
}
|
2014-11-09 09:14:15 -06:00
|
|
|
walk_fn_ret_ty(visitor, &function_declaration.decl.output);
|
2014-11-15 15:55:27 -06:00
|
|
|
walk_lifetime_decls_helper(visitor, &function_declaration.lifetimes);
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2015-02-17 11:29:13 -06:00
|
|
|
TyPath(ref maybe_qself, ref path) => {
|
|
|
|
if let Some(ref qself) = *maybe_qself {
|
|
|
|
visitor.visit_ty(&qself.ty);
|
|
|
|
}
|
2015-01-29 13:18:17 -06:00
|
|
|
visitor.visit_path(path, typ.id);
|
2014-11-20 14:08:48 -06:00
|
|
|
}
|
|
|
|
TyObjectSum(ref ty, ref bounds) => {
|
|
|
|
visitor.visit_ty(&**ty);
|
|
|
|
walk_ty_param_bounds_helper(visitor, bounds);
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
TyFixedLengthVec(ref ty, ref expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&**ty);
|
|
|
|
visitor.visit_expr(&**expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-11-15 15:55:27 -06:00
|
|
|
TyPolyTraitRef(ref bounds) => {
|
|
|
|
walk_ty_param_bounds_helper(visitor, bounds)
|
2014-11-07 05:53:45 -06:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
TyTypeof(ref expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**expression)
|
2013-08-22 16:00:02 -05:00
|
|
|
}
|
2014-11-09 09:14:15 -06:00
|
|
|
TyInfer => {}
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 15:55:27 -06:00
|
|
|
pub fn walk_lifetime_decls_helper<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
lifetimes: &'v Vec<LifetimeDef>) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for l in lifetimes {
|
2014-11-18 10:39:16 -06:00
|
|
|
visitor.visit_lifetime_def(l);
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_path<'v, V: Visitor<'v>>(visitor: &mut V, path: &'v Path) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for segment in &path.segments {
|
2014-11-15 15:55:27 -06:00
|
|
|
visitor.visit_path_segment(path.span, segment);
|
|
|
|
}
|
|
|
|
}
|
2013-11-22 06:24:49 -06:00
|
|
|
|
2014-11-15 15:55:27 -06:00
|
|
|
pub fn walk_path_segment<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
path_span: Span,
|
|
|
|
segment: &'v PathSegment) {
|
|
|
|
visitor.visit_ident(path_span, segment.identifier);
|
|
|
|
visitor.visit_path_parameters(path_span, &segment.parameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn walk_path_parameters<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
_path_span: Span,
|
|
|
|
path_parameters: &'v PathParameters) {
|
|
|
|
match *path_parameters {
|
|
|
|
ast::AngleBracketedParameters(ref data) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for typ in &*data.types {
|
2014-11-15 15:55:27 -06:00
|
|
|
visitor.visit_ty(&**typ);
|
2014-11-03 20:52:52 -06:00
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
for lifetime in &data.lifetimes {
|
2014-11-15 15:55:27 -06:00
|
|
|
visitor.visit_lifetime_ref(lifetime);
|
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
for binding in &*data.bindings {
|
2014-12-28 09:07:21 -06:00
|
|
|
visitor.visit_assoc_type_binding(&**binding);
|
|
|
|
}
|
2014-11-15 15:55:27 -06:00
|
|
|
}
|
|
|
|
ast::ParenthesizedParameters(ref data) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for typ in &data.inputs {
|
2014-11-15 15:55:27 -06:00
|
|
|
visitor.visit_ty(&**typ);
|
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
if let Some(ref typ) = data.output {
|
2014-11-15 15:55:27 -06:00
|
|
|
visitor.visit_ty(&**typ);
|
2014-11-03 20:52:52 -06:00
|
|
|
}
|
2013-08-07 11:47:28 -05:00
|
|
|
}
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2011-12-13 06:19:56 -06:00
|
|
|
}
|
|
|
|
|
2014-12-28 09:07:21 -06:00
|
|
|
pub fn walk_assoc_type_binding<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
type_binding: &'v TypeBinding) {
|
|
|
|
visitor.visit_ident(type_binding.span, type_binding.ident);
|
|
|
|
visitor.visit_ty(&*type_binding.ty);
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_pat<'v, V: Visitor<'v>>(visitor: &mut V, pattern: &'v Pat) {
|
2013-07-19 20:42:11 -05:00
|
|
|
match pattern.node {
|
2013-09-01 20:45:37 -05:00
|
|
|
PatEnum(ref path, ref children) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_path(path, pattern.id);
|
2015-01-31 11:20:46 -06:00
|
|
|
if let Some(ref children) = *children {
|
|
|
|
for child in children {
|
|
|
|
visitor.visit_pat(&*child)
|
2013-06-15 19:26:59 -05:00
|
|
|
}
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2012-08-06 19:01:14 -05:00
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
PatStruct(ref path, ref fields, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_path(path, pattern.id);
|
2015-01-31 11:20:46 -06:00
|
|
|
for field in fields {
|
2014-10-05 19:36:53 -05:00
|
|
|
visitor.visit_pat(&*field.node.pat)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
PatTup(ref tuple_elements) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for tuple_element in tuple_elements {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**tuple_element)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
PatBox(ref subpattern) |
|
2014-12-05 17:56:25 -06:00
|
|
|
PatRegion(ref subpattern, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**subpattern)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-06-30 20:02:14 -05:00
|
|
|
PatIdent(_, ref pth1, ref optional_subpattern) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ident(pth1.span, pth1.node);
|
2013-07-19 20:42:11 -05:00
|
|
|
match *optional_subpattern {
|
|
|
|
None => {}
|
2014-09-12 05:10:30 -05:00
|
|
|
Some(ref subpattern) => visitor.visit_pat(&**subpattern),
|
2013-06-15 19:26:59 -05:00
|
|
|
}
|
2012-12-08 14:22:43 -06:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
PatLit(ref expression) => visitor.visit_expr(&**expression),
|
2014-05-16 02:16:13 -05:00
|
|
|
PatRange(ref lower_bound, ref upper_bound) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**lower_bound);
|
|
|
|
visitor.visit_expr(&**upper_bound)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2014-08-06 10:04:44 -05:00
|
|
|
PatWild(_) => (),
|
2013-09-01 20:45:37 -05:00
|
|
|
PatVec(ref prepattern, ref slice_pattern, ref postpatterns) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for prepattern in prepattern {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**prepattern)
|
2013-02-26 12:58:46 -06:00
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
if let Some(ref slice_pattern) = *slice_pattern {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**slice_pattern)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
for postpattern in postpatterns {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**postpattern)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2012-12-08 14:22:43 -06:00
|
|
|
}
|
2015-01-02 15:39:05 -06:00
|
|
|
PatMac(ref mac) => visitor.visit_mac(mac),
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_foreign_item<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
foreign_item: &'v ForeignItem) {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ident(foreign_item.span, foreign_item.ident);
|
2013-11-22 06:24:49 -06:00
|
|
|
|
2013-07-19 20:42:11 -05:00
|
|
|
match foreign_item.node {
|
2014-05-16 02:16:13 -05:00
|
|
|
ForeignItemFn(ref function_declaration, ref generics) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_fn_decl(visitor, &**function_declaration);
|
|
|
|
visitor.visit_generics(generics)
|
2013-02-18 23:25:44 -06:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
ForeignItemStatic(ref typ, _) => visitor.visit_ty(&**typ),
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
|
|
|
|
2015-01-31 11:20:46 -06:00
|
|
|
for attr in &foreign_item.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 15:55:27 -06:00
|
|
|
pub fn walk_ty_param_bounds_helper<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
bounds: &'v OwnedSlice<TyParamBound>) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for bound in &**bounds {
|
2014-11-15 15:55:27 -06:00
|
|
|
visitor.visit_ty_param_bound(bound)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn walk_ty_param_bound<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
bound: &'v TyParamBound) {
|
|
|
|
match *bound {
|
2014-12-24 00:38:10 -06:00
|
|
|
TraitTyParamBound(ref typ, ref modifier) => {
|
|
|
|
visitor.visit_poly_trait_ref(typ, modifier);
|
2014-11-15 15:55:27 -06:00
|
|
|
}
|
|
|
|
RegionTyParamBound(ref lifetime) => {
|
2014-12-13 04:34:34 -06:00
|
|
|
visitor.visit_lifetime_bound(lifetime);
|
2013-01-10 13:16:54 -06:00
|
|
|
}
|
2012-08-06 20:54:20 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_generics<'v, V: Visitor<'v>>(visitor: &mut V, generics: &'v Generics) {
|
2015-03-10 05:28:44 -05:00
|
|
|
for param in &*generics.ty_params {
|
|
|
|
visitor.visit_ident(param.span, param.ident);
|
|
|
|
walk_ty_param_bounds_helper(visitor, ¶m.bounds);
|
|
|
|
walk_ty_opt(visitor, ¶m.default);
|
2011-12-28 10:50:12 -06:00
|
|
|
}
|
2014-11-15 15:55:27 -06:00
|
|
|
walk_lifetime_decls_helper(visitor, &generics.lifetimes);
|
2015-01-31 11:20:46 -06:00
|
|
|
for predicate in &generics.where_clause.predicates {
|
2014-11-28 22:08:30 -06:00
|
|
|
match predicate {
|
2014-12-20 04:29:19 -06:00
|
|
|
&ast::WherePredicate::BoundPredicate(ast::WhereBoundPredicate{ref bounded_ty,
|
2014-11-28 22:08:30 -06:00
|
|
|
ref bounds,
|
|
|
|
..}) => {
|
2014-12-20 04:29:19 -06:00
|
|
|
visitor.visit_ty(&**bounded_ty);
|
2014-11-28 22:08:30 -06:00
|
|
|
walk_ty_param_bounds_helper(visitor, bounds);
|
|
|
|
}
|
2014-12-20 04:29:19 -06:00
|
|
|
&ast::WherePredicate::RegionPredicate(ast::WhereRegionPredicate{ref lifetime,
|
2014-12-20 04:48:43 -06:00
|
|
|
ref bounds,
|
2014-12-20 04:29:19 -06:00
|
|
|
..}) => {
|
|
|
|
visitor.visit_lifetime_ref(lifetime);
|
2014-12-20 04:48:43 -06:00
|
|
|
|
2015-01-31 11:20:46 -06:00
|
|
|
for bound in bounds {
|
2014-12-20 04:48:43 -06:00
|
|
|
visitor.visit_lifetime_ref(bound);
|
|
|
|
}
|
2014-12-20 04:29:19 -06:00
|
|
|
}
|
2014-11-28 22:08:30 -06:00
|
|
|
&ast::WherePredicate::EqPredicate(ast::WhereEqPredicate{id,
|
|
|
|
ref path,
|
|
|
|
ref ty,
|
|
|
|
..}) => {
|
|
|
|
visitor.visit_path(path, id);
|
|
|
|
visitor.visit_ty(&**ty);
|
|
|
|
}
|
|
|
|
}
|
2014-08-11 11:32:26 -05:00
|
|
|
}
|
2011-12-28 10:50:12 -06:00
|
|
|
}
|
|
|
|
|
2014-11-09 09:14:15 -06:00
|
|
|
pub fn walk_fn_ret_ty<'v, V: Visitor<'v>>(visitor: &mut V, ret_ty: &'v FunctionRetTy) {
|
|
|
|
if let Return(ref output_ty) = *ret_ty {
|
|
|
|
visitor.visit_ty(&**output_ty)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_fn_decl<'v, V: Visitor<'v>>(visitor: &mut V, function_declaration: &'v FnDecl) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for argument in &function_declaration.inputs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&*argument.pat);
|
|
|
|
visitor.visit_ty(&*argument.ty)
|
2012-11-06 20:41:06 -06:00
|
|
|
}
|
2014-11-09 09:14:15 -06:00
|
|
|
walk_fn_ret_ty(visitor, &function_declaration.output)
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
|
2012-03-01 12:36:22 -06:00
|
|
|
// Note: there is no visit_method() method in the visitor, instead override
|
2014-01-09 07:05:33 -06:00
|
|
|
// visit_fn() and check for FkMethod(). I named this visit_method_helper()
|
2012-03-01 12:36:22 -06:00
|
|
|
// because it is not a default impl of any method, though I doubt that really
|
|
|
|
// clarifies anything. - Niko
|
2015-03-10 05:28:44 -05:00
|
|
|
fn walk_method_helper<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
id: NodeId,
|
|
|
|
ident: Ident,
|
|
|
|
span: Span,
|
|
|
|
method: &'v Method) {
|
|
|
|
match *method {
|
2015-03-11 01:38:27 -05:00
|
|
|
MethDecl(ref sig, ref body) => {
|
2015-03-10 05:28:44 -05:00
|
|
|
visitor.visit_ident(span, ident);
|
|
|
|
visitor.visit_fn(FkMethod(ident, method),
|
2015-03-11 01:38:27 -05:00
|
|
|
&sig.decl,
|
|
|
|
body,
|
2015-03-10 05:28:44 -05:00
|
|
|
span,
|
|
|
|
id);
|
2014-07-11 23:22:11 -05:00
|
|
|
},
|
2014-09-12 05:10:30 -05:00
|
|
|
MethMac(ref mac) => visitor.visit_mac(mac)
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2013-02-18 23:25:44 -06:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_fn<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
function_kind: FnKind<'v>,
|
|
|
|
function_declaration: &'v FnDecl,
|
|
|
|
function_body: &'v Block,
|
|
|
|
_span: Span) {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_fn_decl(visitor, function_declaration);
|
2013-10-28 16:37:10 -05:00
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
match function_kind {
|
2014-01-09 07:05:33 -06:00
|
|
|
FkItemFn(_, generics, _, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_generics(generics);
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2015-03-10 05:28:44 -05:00
|
|
|
FkMethod(_, method) => {
|
|
|
|
match *method {
|
2015-03-11 01:38:27 -05:00
|
|
|
MethDecl(ref sig, _) => {
|
|
|
|
visitor.visit_generics(&sig.generics);
|
|
|
|
visitor.visit_explicit_self(&sig.explicit_self);
|
2015-03-10 05:28:44 -05:00
|
|
|
}
|
|
|
|
MethMac(ref mac) => visitor.visit_mac(mac)
|
2014-07-11 23:22:11 -05:00
|
|
|
}
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
2014-01-09 07:05:33 -06:00
|
|
|
FkFnBlock(..) => {}
|
2013-10-28 16:37:10 -05:00
|
|
|
}
|
|
|
|
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_block(function_body)
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
|
2015-03-10 05:28:44 -05:00
|
|
|
pub fn walk_trait_item<'v, V: Visitor<'v>>(visitor: &mut V, trait_item: &'v TraitItem) {
|
|
|
|
visitor.visit_ident(trait_item.span, trait_item.ident);
|
|
|
|
for attr in &trait_item.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2015-03-10 05:28:44 -05:00
|
|
|
match trait_item.node {
|
2015-03-11 01:38:27 -05:00
|
|
|
RequiredMethod(ref sig) => {
|
|
|
|
visitor.visit_explicit_self(&sig.explicit_self);
|
|
|
|
visitor.visit_generics(&sig.generics);
|
|
|
|
walk_fn_decl(visitor, &sig.decl);
|
2015-03-10 05:28:44 -05:00
|
|
|
}
|
|
|
|
ProvidedMethod(ref method) => {
|
|
|
|
walk_method_helper(visitor,
|
|
|
|
trait_item.id,
|
|
|
|
trait_item.ident,
|
|
|
|
trait_item.span,
|
|
|
|
method);
|
|
|
|
}
|
|
|
|
TypeTraitItem(ref bounds, ref default) => {
|
|
|
|
walk_ty_param_bounds_helper(visitor, bounds);
|
|
|
|
walk_ty_opt(visitor, default);
|
|
|
|
}
|
|
|
|
}
|
2012-07-11 12:28:30 -05:00
|
|
|
}
|
|
|
|
|
2015-03-10 05:28:44 -05:00
|
|
|
pub fn walk_impl_item<'v, V: Visitor<'v>>(visitor: &mut V, impl_item: &'v ImplItem) {
|
|
|
|
visitor.visit_ident(impl_item.span, impl_item.ident);
|
|
|
|
for attr in &impl_item.attrs {
|
|
|
|
visitor.visit_attribute(attr);
|
|
|
|
}
|
|
|
|
match impl_item.node {
|
|
|
|
MethodImplItem(ref method) => {
|
|
|
|
walk_method_helper(visitor,
|
|
|
|
impl_item.id,
|
|
|
|
impl_item.ident,
|
|
|
|
impl_item.span,
|
|
|
|
method);
|
|
|
|
}
|
|
|
|
TypeImplItem(ref ty) => {
|
|
|
|
visitor.visit_ty(ty);
|
2014-08-05 21:44:21 -05:00
|
|
|
}
|
2012-07-10 15:44:20 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_struct_def<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
struct_definition: &'v StructDef) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for field in &struct_definition.fields {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_struct_field(field)
|
2012-08-15 17:53:58 -05:00
|
|
|
}
|
2012-08-07 17:54:59 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_struct_field<'v, V: Visitor<'v>>(visitor: &mut V,
|
|
|
|
struct_field: &'v StructField) {
|
2014-11-29 15:41:21 -06:00
|
|
|
if let NamedField(name, _) = struct_field.node.kind {
|
|
|
|
visitor.visit_ident(struct_field.span, name);
|
2013-11-22 06:24:49 -06:00
|
|
|
}
|
|
|
|
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&*struct_field.node.ty);
|
2014-06-08 01:02:48 -05:00
|
|
|
|
2015-01-31 11:20:46 -06:00
|
|
|
for attr in &struct_field.node.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2012-08-15 17:53:58 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_block<'v, V: Visitor<'v>>(visitor: &mut V, block: &'v Block) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for statement in &block.stmts {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_stmt(&**statement)
|
2012-09-19 18:55:01 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
walk_expr_opt(visitor, &block.expr)
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_stmt<'v, V: Visitor<'v>>(visitor: &mut V, statement: &'v Stmt) {
|
2013-07-19 20:42:11 -05:00
|
|
|
match statement.node {
|
2014-09-12 05:10:30 -05:00
|
|
|
StmtDecl(ref declaration, _) => visitor.visit_decl(&**declaration),
|
2014-05-16 02:16:13 -05:00
|
|
|
StmtExpr(ref expression, _) | StmtSemi(ref expression, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2015-01-02 15:39:05 -06:00
|
|
|
StmtMac(ref mac, _) => visitor.visit_mac(&**mac),
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_decl<'v, V: Visitor<'v>>(visitor: &mut V, declaration: &'v Decl) {
|
2013-07-19 20:42:11 -05:00
|
|
|
match declaration.node {
|
2014-09-12 05:10:30 -05:00
|
|
|
DeclLocal(ref local) => visitor.visit_local(&**local),
|
|
|
|
DeclItem(ref item) => visitor.visit_item(&**item),
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_expr_opt<'v, V: Visitor<'v>>(visitor: &mut V,
|
2014-09-13 11:06:01 -05:00
|
|
|
optional_expression: &'v Option<P<Expr>>) {
|
2014-09-09 17:54:36 -05:00
|
|
|
match *optional_expression {
|
2013-07-19 20:42:11 -05:00
|
|
|
None => {}
|
2014-09-12 05:10:30 -05:00
|
|
|
Some(ref expression) => visitor.visit_expr(&**expression),
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
|
2014-09-13 11:06:01 -05:00
|
|
|
pub fn walk_exprs<'v, V: Visitor<'v>>(visitor: &mut V, expressions: &'v [P<Expr>]) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for expression in expressions {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_mac<'v, V: Visitor<'v>>(_: &mut V, _: &'v Mac) {
|
2013-07-19 20:42:11 -05:00
|
|
|
// Empty!
|
2011-07-08 18:35:09 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_expr<'v, V: Visitor<'v>>(visitor: &mut V, expression: &'v Expr) {
|
2013-07-19 20:42:11 -05:00
|
|
|
match expression.node {
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprBox(ref place, ref subexpression) => {
|
2014-12-16 07:30:30 -06:00
|
|
|
place.as_ref().map(|e|visitor.visit_expr(&**e));
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression)
|
2013-12-17 18:46:18 -06:00
|
|
|
}
|
2014-04-04 05:12:18 -05:00
|
|
|
ExprVec(ref subexpressions) => {
|
2015-02-01 20:53:25 -06:00
|
|
|
walk_exprs(visitor, subexpressions)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprRepeat(ref element, ref count) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**element);
|
|
|
|
visitor.visit_expr(&**count)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
ExprStruct(ref path, ref fields, ref optional_base) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_path(path, expression.id);
|
2015-01-31 11:20:46 -06:00
|
|
|
for field in fields {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&*field.expr)
|
2013-06-15 19:26:59 -05:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_expr_opt(visitor, optional_base)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ExprTup(ref subexpressions) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for subexpression in subexpressions {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprCall(ref callee_expression, ref arguments) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for argument in arguments {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**argument)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**callee_expression)
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2014-02-26 08:06:45 -06:00
|
|
|
ExprMethodCall(_, ref types, ref arguments) => {
|
2015-02-01 20:53:25 -06:00
|
|
|
walk_exprs(visitor, arguments);
|
2015-01-31 11:20:46 -06:00
|
|
|
for typ in types {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_ty(&**typ)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprBinary(_, ref left_expression, ref right_expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**left_expression);
|
|
|
|
visitor.visit_expr(&**right_expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprAddrOf(_, ref subexpression) | ExprUnary(_, ref subexpression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ExprLit(_) => {}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprCast(ref subexpression, ref typ) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression);
|
|
|
|
visitor.visit_ty(&**typ)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
ExprIf(ref head_expression, ref if_block, ref optional_else) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**head_expression);
|
|
|
|
visitor.visit_block(&**if_block);
|
|
|
|
walk_expr_opt(visitor, optional_else)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-07-25 19:12:51 -05:00
|
|
|
ExprWhile(ref subexpression, ref block, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression);
|
|
|
|
visitor.visit_block(&**block)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-08-24 20:04:29 -05:00
|
|
|
ExprIfLet(ref pattern, ref subexpression, ref if_block, ref optional_else) => {
|
|
|
|
visitor.visit_pat(&**pattern);
|
|
|
|
visitor.visit_expr(&**subexpression);
|
|
|
|
visitor.visit_block(&**if_block);
|
|
|
|
walk_expr_opt(visitor, optional_else);
|
|
|
|
}
|
2014-10-02 21:45:46 -05:00
|
|
|
ExprWhileLet(ref pattern, ref subexpression, ref block, _) => {
|
|
|
|
visitor.visit_pat(&**pattern);
|
|
|
|
visitor.visit_expr(&**subexpression);
|
|
|
|
visitor.visit_block(&**block);
|
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprForLoop(ref pattern, ref subexpression, ref block, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**pattern);
|
|
|
|
visitor.visit_expr(&**subexpression);
|
|
|
|
visitor.visit_block(&**block)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
ExprLoop(ref block, _) => visitor.visit_block(&**block),
|
2014-08-24 20:04:29 -05:00
|
|
|
ExprMatch(ref subexpression, ref arms, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression);
|
2015-01-31 11:20:46 -06:00
|
|
|
for arm in arms {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_arm(arm)
|
2013-06-15 19:26:59 -05:00
|
|
|
}
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2015-02-03 10:34:05 -06:00
|
|
|
ExprClosure(_, ref function_declaration, ref body) => {
|
2014-09-09 17:54:36 -05:00
|
|
|
visitor.visit_fn(FkFnBlock,
|
2014-05-29 00:26:56 -05:00
|
|
|
&**function_declaration,
|
|
|
|
&**body,
|
|
|
|
expression.span,
|
2014-09-12 05:10:30 -05:00
|
|
|
expression.id)
|
2014-05-29 00:26:56 -05:00
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
ExprBlock(ref block) => visitor.visit_block(&**block),
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprAssign(ref left_hand_expression, ref right_hand_expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**right_hand_expression);
|
|
|
|
visitor.visit_expr(&**left_hand_expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprAssignOp(_, ref left_expression, ref right_expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**right_expression);
|
|
|
|
visitor.visit_expr(&**left_expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-11-23 05:14:35 -06:00
|
|
|
ExprField(ref subexpression, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression);
|
2013-02-18 00:20:36 -06:00
|
|
|
}
|
2014-11-23 05:14:35 -06:00
|
|
|
ExprTupField(ref subexpression, _) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression);
|
2014-08-09 22:54:33 -05:00
|
|
|
}
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprIndex(ref main_expression, ref index_expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**main_expression);
|
|
|
|
visitor.visit_expr(&**index_expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-12-12 23:41:02 -06:00
|
|
|
ExprRange(ref start, ref end) => {
|
2014-12-17 22:55:04 -06:00
|
|
|
walk_expr_opt(visitor, start);
|
2014-12-12 23:41:02 -06:00
|
|
|
walk_expr_opt(visitor, end)
|
|
|
|
}
|
2015-02-17 11:29:13 -06:00
|
|
|
ExprPath(ref maybe_qself, ref path) => {
|
|
|
|
if let Some(ref qself) = *maybe_qself {
|
|
|
|
visitor.visit_ty(&qself.ty);
|
|
|
|
}
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_path(path, expression.id)
|
2013-12-08 13:25:35 -06:00
|
|
|
}
|
2014-01-27 06:18:36 -06:00
|
|
|
ExprBreak(_) | ExprAgain(_) => {}
|
2014-09-09 17:54:36 -05:00
|
|
|
ExprRet(ref optional_expression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
walk_expr_opt(visitor, optional_expression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2015-01-02 15:39:05 -06:00
|
|
|
ExprMac(ref mac) => visitor.visit_mac(mac),
|
2014-05-16 02:16:13 -05:00
|
|
|
ExprParen(ref subexpression) => {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**subexpression)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-08-19 14:39:26 -05:00
|
|
|
ExprInlineAsm(ref ia) => {
|
2015-01-31 11:20:46 -06:00
|
|
|
for input in &ia.inputs {
|
2014-09-09 17:54:36 -05:00
|
|
|
let (_, ref input) = *input;
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**input)
|
2013-03-12 19:53:25 -05:00
|
|
|
}
|
2015-01-31 11:20:46 -06:00
|
|
|
for output in &ia.outputs {
|
2014-09-09 17:54:36 -05:00
|
|
|
let (_, ref output, _) = *output;
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&**output)
|
2013-03-12 19:53:25 -05:00
|
|
|
}
|
|
|
|
}
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
2013-07-19 20:42:11 -05:00
|
|
|
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr_post(expression)
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|
|
|
|
|
2014-09-09 17:54:36 -05:00
|
|
|
pub fn walk_arm<'v, V: Visitor<'v>>(visitor: &mut V, arm: &'v Arm) {
|
2015-01-31 11:20:46 -06:00
|
|
|
for pattern in &arm.pats {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_pat(&**pattern)
|
2013-07-19 20:42:11 -05:00
|
|
|
}
|
2014-09-09 17:54:36 -05:00
|
|
|
walk_expr_opt(visitor, &arm.guard);
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_expr(&*arm.body);
|
2015-01-31 11:20:46 -06:00
|
|
|
for attr in &arm.attrs {
|
2014-09-12 05:10:30 -05:00
|
|
|
visitor.visit_attribute(attr);
|
2014-06-08 01:02:48 -05:00
|
|
|
}
|
2011-06-08 15:48:19 -05:00
|
|
|
}
|