lib.rs 149.1 KB
Newer Older
1
// Copyright 2012-2015 The Rust Project Developers. See the COPYRIGHT
2 3 4 5 6 7 8 9 10
// 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.

11
#![crate_name = "rustc_resolve"]
12
#![unstable(feature = "rustc_private", issue = "27812")]
13 14
#![crate_type = "dylib"]
#![crate_type = "rlib"]
15
#![doc(html_logo_url = "https://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
A
Alex Crichton 已提交
16
      html_favicon_url = "https://doc.rust-lang.org/favicon.ico",
17
      html_root_url = "https://doc.rust-lang.org/nightly/")]
18
#![cfg_attr(not(stage0), deny(warnings))]
19

T
Fallout  
Tamir Duberstein 已提交
20
#![feature(associated_consts)]
21
#![feature(borrow_state)]
A
Alex Crichton 已提交
22
#![feature(rustc_diagnostic_macros)]
23
#![feature(rustc_private)]
A
Alex Crichton 已提交
24
#![feature(staged_api)]
25

C
corentih 已提交
26 27 28 29
#[macro_use]
extern crate log;
#[macro_use]
extern crate syntax;
30
extern crate arena;
C
corentih 已提交
31 32 33
#[macro_use]
#[no_link]
extern crate rustc_bitflags;
34
extern crate rustc_front;
35 36
extern crate rustc;

S
Steven Fackler 已提交
37 38 39 40 41 42 43 44
use self::PatternBindingMode::*;
use self::Namespace::*;
use self::ResolveResult::*;
use self::FallbackSuggestion::*;
use self::TypeParameters::*;
use self::RibKind::*;
use self::UseLexicalScopeFlag::*;
use self::ModulePrefixResult::*;
45
use self::AssocItemResolveResult::*;
S
Steven Fackler 已提交
46 47 48 49
use self::BareIdentifierPatternResolution::*;
use self::ParentLink::*;
use self::FallbackChecks::*;

50
use rustc::dep_graph::DepNode;
51
use rustc::front::map as hir_map;
52 53
use rustc::session::Session;
use rustc::lint;
54
use rustc::middle::cstore::{CrateStore, DefLike, DlDef};
55
use rustc::middle::def::*;
56
use rustc::middle::def_id::DefId;
57
use rustc::middle::pat_util::pat_bindings;
58 59
use rustc::middle::privacy::*;
use rustc::middle::subst::{ParamSpace, FnSpace, TypeSpace};
60
use rustc::middle::ty::{Freevar, FreevarMap, TraitMap, GlobMap};
61
use rustc::util::nodemap::{NodeMap, DefIdSet, FnvHashMap};
62

P
Patrick Walton 已提交
63
use syntax::ast;
J
Jeffrey Seyfried 已提交
64
use syntax::ast::{CRATE_NODE_ID, Name, NodeId, CrateNum, TyIs, TyI8, TyI16, TyI32, TyI64};
65
use syntax::ast::{TyUs, TyU8, TyU16, TyU32, TyU64, TyF64, TyF32};
66
use syntax::attr::AttrMetaMethods;
67
use syntax::codemap::{self, Span, Pos};
N
Nick Cameron 已提交
68 69
use syntax::errors::DiagnosticBuilder;
use syntax::parse::token::{self, special_names, special_idents};
70
use syntax::util::lev_distance::find_best_match_for_name;
71

72
use rustc_front::intravisit::{self, FnKind, Visitor};
73 74
use rustc_front::hir;
use rustc_front::hir::{Arm, BindByRef, BindByValue, BindingMode, Block};
75
use rustc_front::hir::Crate;
76
use rustc_front::hir::{Expr, ExprAgain, ExprBreak, ExprCall, ExprField};
77 78 79 80 81 82
use rustc_front::hir::{ExprLoop, ExprWhile, ExprMethodCall};
use rustc_front::hir::{ExprPath, ExprStruct, FnDecl};
use rustc_front::hir::{ForeignItemFn, ForeignItemStatic, Generics};
use rustc_front::hir::{ImplItem, Item, ItemConst, ItemEnum, ItemExternCrate};
use rustc_front::hir::{ItemFn, ItemForeignMod, ItemImpl, ItemMod, ItemStatic, ItemDefaultImpl};
use rustc_front::hir::{ItemStruct, ItemTrait, ItemTy, ItemUse};
83
use rustc_front::hir::Local;
84 85
use rustc_front::hir::{Pat, PatEnum, PatIdent, PatLit, PatQPath};
use rustc_front::hir::{PatRange, PatStruct, Path, PrimTy};
86 87
use rustc_front::hir::{TraitRef, Ty, TyBool, TyChar, TyFloat, TyInt};
use rustc_front::hir::{TyRptr, TyStr, TyUint, TyPath, TyPtr};
88
use rustc_front::util::walk_pat;
89

90
use std::collections::{HashMap, HashSet};
91
use std::cell::{Cell, RefCell};
92
use std::fmt;
93
use std::mem::replace;
94

95
use resolve_imports::{ImportDirective, NameResolution};
96

97 98 99 100
// NB: This module needs to be declared first so diagnostics are
// registered before they are used.
pub mod diagnostics;

A
Alex Crichton 已提交
101
mod check_unused;
102
mod build_reduced_graph;
103
mod resolve_imports;
104

105 106 107 108 109 110 111 112 113 114 115
// Perform the callback, not walking deeper if the return is true
macro_rules! execute_callback {
    ($node: expr, $walker: expr) => (
        if let Some(ref callback) = $walker.callback {
            if callback($node, &mut $walker.resolved) {
                return;
            }
        }
    )
}

116 117
enum SuggestionType {
    Macro(String),
118
    Function(token::InternedString),
119 120 121
    NotFound,
}

122
pub enum ResolutionError<'a> {
123
    /// error E0401: can't use type parameters from outer function
124
    TypeParametersFromOuterFunction,
125
    /// error E0402: cannot use an outer type parameter in this context
126
    OuterTypeParameterContext,
127
    /// error E0403: the name is already used for a type parameter in this type parameter list
128
    NameAlreadyUsedInTypeParameterList(Name),
129
    /// error E0404: is not a trait
130
    IsNotATrait(&'a str),
131
    /// error E0405: use of undeclared trait name
132
    UndeclaredTraitName(&'a str),
133
    /// error E0406: undeclared associated type
134
    UndeclaredAssociatedType,
135
    /// error E0407: method is not a member of trait
136
    MethodNotMemberOfTrait(Name, &'a str),
137 138 139 140
    /// error E0437: type is not a member of trait
    TypeNotMemberOfTrait(Name, &'a str),
    /// error E0438: const is not a member of trait
    ConstNotMemberOfTrait(Name, &'a str),
141
    /// error E0408: variable `{}` from pattern #1 is not bound in pattern
142
    VariableNotBoundInPattern(Name, usize),
143
    /// error E0409: variable is bound with different mode in pattern #{} than in pattern #1
144
    VariableBoundWithDifferentMode(Name, usize),
145
    /// error E0410: variable from pattern is not bound in pattern #1
146
    VariableNotBoundInParentPattern(Name, usize),
147
    /// error E0411: use of `Self` outside of an impl or trait
148
    SelfUsedOutsideImplOrTrait,
149
    /// error E0412: use of undeclared
150
    UseOfUndeclared(&'a str, &'a str),
151
    /// error E0413: declaration shadows an enum variant or unit-like struct in scope
152
    DeclarationShadowsEnumVariantOrUnitLikeStruct(Name),
153
    /// error E0414: only irrefutable patterns allowed here
154
    OnlyIrrefutablePatternsAllowedHere(DefId, Name),
155
    /// error E0415: identifier is bound more than once in this parameter list
156
    IdentifierBoundMoreThanOnceInParameterList(&'a str),
157
    /// error E0416: identifier is bound more than once in the same pattern
158
    IdentifierBoundMoreThanOnceInSamePattern(&'a str),
159
    /// error E0417: static variables cannot be referenced in a pattern
160
    StaticVariableReference,
161
    /// error E0418: is not an enum variant, struct or const
162
    NotAnEnumVariantStructOrConst(&'a str),
163
    /// error E0419: unresolved enum variant, struct or const
164
    UnresolvedEnumVariantStructOrConst(&'a str),
165
    /// error E0420: is not an associated const
166
    NotAnAssociatedConst(&'a str),
167
    /// error E0421: unresolved associated const
168
    UnresolvedAssociatedConst(&'a str),
169
    /// error E0422: does not name a struct
170
    DoesNotNameAStruct(&'a str),
171
    /// error E0423: is a struct variant name, but this expression uses it like a function name
172
    StructVariantUsedAsFunction(&'a str),
173
    /// error E0424: `self` is not available in a static method
174
    SelfNotAvailableInStaticMethod,
175
    /// error E0425: unresolved name
176
    UnresolvedName(&'a str, &'a str, UnresolvedNameContext),
177
    /// error E0426: use of undeclared label
178
    UndeclaredLabel(&'a str),
179
    /// error E0427: cannot use `ref` binding mode with ...
180
    CannotUseRefBindingModeWith(&'a str),
181
    /// error E0428: duplicate definition
182
    DuplicateDefinition(&'a str, Name),
183
    /// error E0429: `self` imports are only allowed within a { } list
184
    SelfImportsOnlyAllowedWithin,
185
    /// error E0430: `self` import can only appear once in the list
186
    SelfImportCanOnlyAppearOnceInTheList,
187
    /// error E0431: `self` import can only appear in an import list with a non-empty prefix
188
    SelfImportOnlyInImportListWithNonEmptyPrefix,
189
    /// error E0432: unresolved import
190
    UnresolvedImport(Option<(&'a str, &'a str)>),
191
    /// error E0433: failed to resolve
192
    FailedToResolve(&'a str),
193
    /// error E0434: can't capture dynamic environment in a fn item
194
    CannotCaptureDynamicEnvironmentInFnItem,
195
    /// error E0435: attempt to use a non-constant value in a constant
196
    AttemptToUseNonConstantValueInConstant,
197 198
}

199
/// Context of where `ResolutionError::UnresolvedName` arose.
200 201
#[derive(Clone, PartialEq, Eq, Debug)]
pub enum UnresolvedNameContext {
202 203 204 205
    /// `PathIsMod(id)` indicates that a given path, used in
    /// expression context, actually resolved to a module rather than
    /// a value. The `id` attached to the variant is the node id of
    /// the erroneous path expression.
206
    PathIsMod(ast::NodeId),
207 208 209 210

    /// `Other` means we have no extra information about the context
    /// of the unresolved name error. (Maybe we could eliminate all
    /// such cases; but for now, this is an information-free default.)
211 212 213
    Other,
}

C
corentih 已提交
214 215 216
fn resolve_error<'b, 'a: 'b, 'tcx: 'a>(resolver: &'b Resolver<'a, 'tcx>,
                                       span: syntax::codemap::Span,
                                       resolution_error: ResolutionError<'b>) {
N
Nick Cameron 已提交
217
    resolve_struct_error(resolver, span, resolution_error).emit();
N
Nick Cameron 已提交
218 219 220 221 222
}

fn resolve_struct_error<'b, 'a: 'b, 'tcx: 'a>(resolver: &'b Resolver<'a, 'tcx>,
                                              span: syntax::codemap::Span,
                                              resolution_error: ResolutionError<'b>)
N
Nick Cameron 已提交
223
                                              -> DiagnosticBuilder<'a> {
224
    if !resolver.emit_errors {
N
Nick Cameron 已提交
225
        return resolver.session.diagnostic().struct_dummy();
226
    }
N
Nick Cameron 已提交
227

N
Nick Cameron 已提交
228
    match resolution_error {
229
        ResolutionError::TypeParametersFromOuterFunction => {
N
Nick Cameron 已提交
230 231 232 233 234
            struct_span_err!(resolver.session,
                             span,
                             E0401,
                             "can't use type parameters from outer function; try using a local \
                              type parameter instead")
C
corentih 已提交
235
        }
236
        ResolutionError::OuterTypeParameterContext => {
N
Nick Cameron 已提交
237 238 239 240
            struct_span_err!(resolver.session,
                             span,
                             E0402,
                             "cannot use an outer type parameter in this context")
C
corentih 已提交
241
        }
242
        ResolutionError::NameAlreadyUsedInTypeParameterList(name) => {
N
Nick Cameron 已提交
243 244 245 246 247 248
            struct_span_err!(resolver.session,
                             span,
                             E0403,
                             "the name `{}` is already used for a type parameter in this type \
                              parameter list",
                             name)
C
corentih 已提交
249
        }
250
        ResolutionError::IsNotATrait(name) => {
N
Nick Cameron 已提交
251
            struct_span_err!(resolver.session, span, E0404, "`{}` is not a trait", name)
C
corentih 已提交
252
        }
253
        ResolutionError::UndeclaredTraitName(name) => {
N
Nick Cameron 已提交
254 255 256 257 258
            struct_span_err!(resolver.session,
                             span,
                             E0405,
                             "use of undeclared trait name `{}`",
                             name)
C
corentih 已提交
259
        }
260
        ResolutionError::UndeclaredAssociatedType => {
N
Nick Cameron 已提交
261
            struct_span_err!(resolver.session, span, E0406, "undeclared associated type")
C
corentih 已提交
262
        }
263
        ResolutionError::MethodNotMemberOfTrait(method, trait_) => {
N
Nick Cameron 已提交
264 265 266 267 268 269
            struct_span_err!(resolver.session,
                             span,
                             E0407,
                             "method `{}` is not a member of trait `{}`",
                             method,
                             trait_)
C
corentih 已提交
270
        }
271
        ResolutionError::TypeNotMemberOfTrait(type_, trait_) => {
N
Nick Cameron 已提交
272 273 274 275 276 277
            struct_span_err!(resolver.session,
                             span,
                             E0437,
                             "type `{}` is not a member of trait `{}`",
                             type_,
                             trait_)
C
corentih 已提交
278
        }
279
        ResolutionError::ConstNotMemberOfTrait(const_, trait_) => {
N
Nick Cameron 已提交
280 281 282 283 284 285
            struct_span_err!(resolver.session,
                             span,
                             E0438,
                             "const `{}` is not a member of trait `{}`",
                             const_,
                             trait_)
C
corentih 已提交
286
        }
287
        ResolutionError::VariableNotBoundInPattern(variable_name, pattern_number) => {
N
Nick Cameron 已提交
288 289 290 291 292 293
            struct_span_err!(resolver.session,
                             span,
                             E0408,
                             "variable `{}` from pattern #1 is not bound in pattern #{}",
                             variable_name,
                             pattern_number)
C
corentih 已提交
294
        }
295
        ResolutionError::VariableBoundWithDifferentMode(variable_name, pattern_number) => {
N
Nick Cameron 已提交
296 297 298 299 300 301 302
            struct_span_err!(resolver.session,
                             span,
                             E0409,
                             "variable `{}` is bound with different mode in pattern #{} than in \
                              pattern #1",
                             variable_name,
                             pattern_number)
C
corentih 已提交
303
        }
304
        ResolutionError::VariableNotBoundInParentPattern(variable_name, pattern_number) => {
N
Nick Cameron 已提交
305 306 307 308 309 310
            struct_span_err!(resolver.session,
                             span,
                             E0410,
                             "variable `{}` from pattern #{} is not bound in pattern #1",
                             variable_name,
                             pattern_number)
C
corentih 已提交
311
        }
312
        ResolutionError::SelfUsedOutsideImplOrTrait => {
N
Nick Cameron 已提交
313 314 315 316
            struct_span_err!(resolver.session,
                             span,
                             E0411,
                             "use of `Self` outside of an impl or trait")
C
corentih 已提交
317
        }
318
        ResolutionError::UseOfUndeclared(kind, name) => {
N
Nick Cameron 已提交
319 320 321 322 323 324
            struct_span_err!(resolver.session,
                             span,
                             E0412,
                             "use of undeclared {} `{}`",
                             kind,
                             name)
C
corentih 已提交
325
        }
326
        ResolutionError::DeclarationShadowsEnumVariantOrUnitLikeStruct(name) => {
N
Nick Cameron 已提交
327 328 329 330 331 332
            struct_span_err!(resolver.session,
                             span,
                             E0413,
                             "declaration of `{}` shadows an enum variant \
                              or unit-like struct in scope",
                             name)
C
corentih 已提交
333
        }
334
        ResolutionError::OnlyIrrefutablePatternsAllowedHere(did, name) => {
N
Nick Cameron 已提交
335 336 337 338 339 340 341
            let mut err = struct_span_err!(resolver.session,
                                           span,
                                           E0414,
                                           "only irrefutable patterns allowed here");
            err.span_note(span,
                          "there already is a constant in scope sharing the same \
                           name as this pattern");
342
            if let Some(sp) = resolver.ast_map.span_if_local(did) {
N
Nick Cameron 已提交
343
                err.span_note(sp, "constant defined here");
344
            }
345 346
            if let Success(binding) = resolver.current_module.resolve_name(name, ValueNS, true) {
                if binding.is_import() {
347 348
                    err.span_note(binding.span.unwrap(), "constant imported here");
                }
349
            }
N
Nick Cameron 已提交
350
            err
C
corentih 已提交
351
        }
352
        ResolutionError::IdentifierBoundMoreThanOnceInParameterList(identifier) => {
N
Nick Cameron 已提交
353 354 355 356 357
            struct_span_err!(resolver.session,
                             span,
                             E0415,
                             "identifier `{}` is bound more than once in this parameter list",
                             identifier)
C
corentih 已提交
358
        }
359
        ResolutionError::IdentifierBoundMoreThanOnceInSamePattern(identifier) => {
N
Nick Cameron 已提交
360 361 362 363 364
            struct_span_err!(resolver.session,
                             span,
                             E0416,
                             "identifier `{}` is bound more than once in the same pattern",
                             identifier)
C
corentih 已提交
365
        }
366
        ResolutionError::StaticVariableReference => {
N
Nick Cameron 已提交
367 368 369 370 371
            struct_span_err!(resolver.session,
                             span,
                             E0417,
                             "static variables cannot be referenced in a pattern, use a \
                              `const` instead")
C
corentih 已提交
372
        }
373
        ResolutionError::NotAnEnumVariantStructOrConst(name) => {
N
Nick Cameron 已提交
374 375 376 377 378
            struct_span_err!(resolver.session,
                             span,
                             E0418,
                             "`{}` is not an enum variant, struct or const",
                             name)
C
corentih 已提交
379
        }
380
        ResolutionError::UnresolvedEnumVariantStructOrConst(name) => {
N
Nick Cameron 已提交
381 382 383 384 385
            struct_span_err!(resolver.session,
                             span,
                             E0419,
                             "unresolved enum variant, struct or const `{}`",
                             name)
C
corentih 已提交
386
        }
387
        ResolutionError::NotAnAssociatedConst(name) => {
N
Nick Cameron 已提交
388 389 390 391 392
            struct_span_err!(resolver.session,
                             span,
                             E0420,
                             "`{}` is not an associated const",
                             name)
C
corentih 已提交
393
        }
394
        ResolutionError::UnresolvedAssociatedConst(name) => {
N
Nick Cameron 已提交
395 396 397 398 399
            struct_span_err!(resolver.session,
                             span,
                             E0421,
                             "unresolved associated const `{}`",
                             name)
C
corentih 已提交
400
        }
401
        ResolutionError::DoesNotNameAStruct(name) => {
N
Nick Cameron 已提交
402 403 404 405 406
            struct_span_err!(resolver.session,
                             span,
                             E0422,
                             "`{}` does not name a structure",
                             name)
C
corentih 已提交
407
        }
408
        ResolutionError::StructVariantUsedAsFunction(path_name) => {
N
Nick Cameron 已提交
409 410 411 412 413 414
            struct_span_err!(resolver.session,
                             span,
                             E0423,
                             "`{}` is the name of a struct or struct variant, but this expression \
                             uses it like a function name",
                             path_name)
C
corentih 已提交
415
        }
416
        ResolutionError::SelfNotAvailableInStaticMethod => {
N
Nick Cameron 已提交
417 418 419 420 421
            struct_span_err!(resolver.session,
                             span,
                             E0424,
                             "`self` is not available in a static method. Maybe a `self` \
                             argument is missing?")
C
corentih 已提交
422
        }
423
        ResolutionError::UnresolvedName(path, msg, context) => {
N
Nick Cameron 已提交
424 425 426 427 428 429
            let mut err = struct_span_err!(resolver.session,
                                           span,
                                           E0425,
                                           "unresolved name `{}`{}",
                                           path,
                                           msg);
430 431

            match context {
432
                UnresolvedNameContext::Other => { } // no help available
433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451
                UnresolvedNameContext::PathIsMod(id) => {
                    let mut help_msg = String::new();
                    let parent_id = resolver.ast_map.get_parent_node(id);
                    if let Some(hir_map::Node::NodeExpr(e)) = resolver.ast_map.find(parent_id) {
                        match e.node {
                            ExprField(_, ident) => {
                                help_msg = format!("To reference an item from the \
                                                    `{module}` module, use \
                                                    `{module}::{ident}`",
                                                   module = &*path,
                                                   ident = ident.node);
                            }
                            ExprMethodCall(ident, _, _) => {
                                help_msg = format!("To call a function from the \
                                                    `{module}` module, use \
                                                    `{module}::{ident}(..)`",
                                                   module = &*path,
                                                   ident = ident.node);
                            }
452 453 454 455 456
                            ExprCall(_, _) => {
                                help_msg = format!("No function corresponds to `{module}(..)`",
                                                   module = &*path);
                            }
                            _ => { } // no help available
457
                        }
458 459 460
                    } else {
                        help_msg = format!("Module `{module}` cannot be the value of an expression",
                                           module = &*path);
461 462 463
                    }

                    if !help_msg.is_empty() {
N
Nick Cameron 已提交
464
                        err.fileline_help(span, &help_msg);
465 466 467
                    }
                }
            }
N
Nick Cameron 已提交
468
            err
C
corentih 已提交
469
        }
470
        ResolutionError::UndeclaredLabel(name) => {
N
Nick Cameron 已提交
471 472 473 474 475
            struct_span_err!(resolver.session,
                             span,
                             E0426,
                             "use of undeclared label `{}`",
                             name)
C
corentih 已提交
476
        }
477
        ResolutionError::CannotUseRefBindingModeWith(descr) => {
N
Nick Cameron 已提交
478 479 480 481 482
            struct_span_err!(resolver.session,
                             span,
                             E0427,
                             "cannot use `ref` binding mode with {}",
                             descr)
C
corentih 已提交
483
        }
484
        ResolutionError::DuplicateDefinition(namespace, name) => {
N
Nick Cameron 已提交
485 486 487 488 489 490
            struct_span_err!(resolver.session,
                             span,
                             E0428,
                             "duplicate definition of {} `{}`",
                             namespace,
                             name)
C
corentih 已提交
491
        }
492
        ResolutionError::SelfImportsOnlyAllowedWithin => {
N
Nick Cameron 已提交
493 494 495 496 497
            struct_span_err!(resolver.session,
                             span,
                             E0429,
                             "{}",
                             "`self` imports are only allowed within a { } list")
C
corentih 已提交
498
        }
499
        ResolutionError::SelfImportCanOnlyAppearOnceInTheList => {
N
Nick Cameron 已提交
500 501 502 503
            struct_span_err!(resolver.session,
                             span,
                             E0430,
                             "`self` import can only appear once in the list")
C
corentih 已提交
504
        }
505
        ResolutionError::SelfImportOnlyInImportListWithNonEmptyPrefix => {
N
Nick Cameron 已提交
506 507 508 509 510
            struct_span_err!(resolver.session,
                             span,
                             E0431,
                             "`self` import can only appear in an import list with a \
                              non-empty prefix")
511
        }
512
        ResolutionError::UnresolvedImport(name) => {
513
            let msg = match name {
514
                Some((n, p)) => format!("unresolved import `{}`{}", n, p),
C
corentih 已提交
515
                None => "unresolved import".to_owned(),
516
            };
N
Nick Cameron 已提交
517
            struct_span_err!(resolver.session, span, E0432, "{}", msg)
C
corentih 已提交
518
        }
519
        ResolutionError::FailedToResolve(msg) => {
N
Nick Cameron 已提交
520
            struct_span_err!(resolver.session, span, E0433, "failed to resolve. {}", msg)
C
corentih 已提交
521
        }
522
        ResolutionError::CannotCaptureDynamicEnvironmentInFnItem => {
N
Nick Cameron 已提交
523 524 525 526 527 528
            struct_span_err!(resolver.session,
                             span,
                             E0434,
                             "{}",
                             "can't capture dynamic environment in a fn item; use the || { ... } \
                              closure form instead")
C
corentih 已提交
529 530
        }
        ResolutionError::AttemptToUseNonConstantValueInConstant => {
N
Nick Cameron 已提交
531 532 533 534
            struct_span_err!(resolver.session,
                             span,
                             E0435,
                             "attempt to use a non-constant value in a constant")
C
corentih 已提交
535
        }
N
Nick Cameron 已提交
536
    }
537 538
}

N
Niko Matsakis 已提交
539
#[derive(Copy, Clone)]
540
struct BindingInfo {
541
    span: Span,
542
    binding_mode: BindingMode,
543 544 545
}

// Map from the name in a pattern to its binding mode.
546
type BindingMap = HashMap<Name, BindingInfo>;
547

N
Niko Matsakis 已提交
548
#[derive(Copy, Clone, PartialEq)]
F
Felix S. Klock II 已提交
549
enum PatternBindingMode {
550
    RefutableMode,
551
    LocalIrrefutableMode,
552
    ArgumentIrrefutableMode,
553 554
}

N
Niko Matsakis 已提交
555
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
G
Garming Sam 已提交
556
pub enum Namespace {
557
    TypeNS,
C
corentih 已提交
558
    ValueNS,
559 560
}

561
impl<'a, 'v, 'tcx> Visitor<'v> for Resolver<'a, 'tcx> {
562 563 564
    fn visit_nested_item(&mut self, item: hir::ItemId) {
        self.visit_item(self.ast_map.expect_item(item.id))
    }
565
    fn visit_item(&mut self, item: &Item) {
566
        execute_callback!(hir_map::Node::NodeItem(item), self);
A
Alex Crichton 已提交
567
        self.resolve_item(item);
568
    }
569
    fn visit_arm(&mut self, arm: &Arm) {
A
Alex Crichton 已提交
570
        self.resolve_arm(arm);
571
    }
572
    fn visit_block(&mut self, block: &Block) {
573
        execute_callback!(hir_map::Node::NodeBlock(block), self);
A
Alex Crichton 已提交
574
        self.resolve_block(block);
575
    }
576
    fn visit_expr(&mut self, expr: &Expr) {
577
        execute_callback!(hir_map::Node::NodeExpr(expr), self);
A
Alex Crichton 已提交
578
        self.resolve_expr(expr);
579
    }
580
    fn visit_local(&mut self, local: &Local) {
581
        execute_callback!(hir_map::Node::NodeLocal(&*local.pat), self);
A
Alex Crichton 已提交
582
        self.resolve_local(local);
583
    }
584
    fn visit_ty(&mut self, ty: &Ty) {
A
Alex Crichton 已提交
585
        self.resolve_type(ty);
586
    }
587 588 589
    fn visit_generics(&mut self, generics: &Generics) {
        self.resolve_generics(generics);
    }
C
corentih 已提交
590
    fn visit_poly_trait_ref(&mut self, tref: &hir::PolyTraitRef, m: &hir::TraitBoundModifier) {
591 592
        match self.resolve_trait_reference(tref.trait_ref.ref_id, &tref.trait_ref.path, 0) {
            Ok(def) => self.record_def(tref.trait_ref.ref_id, def),
C
corentih 已提交
593 594
            Err(_) => {
                // error already reported
595
                self.record_def(tref.trait_ref.ref_id, err_path_resolution())
C
corentih 已提交
596
            }
597
        }
598
        intravisit::walk_poly_trait_ref(self, tref, m);
599
    }
C
corentih 已提交
600 601 602 603
    fn visit_variant(&mut self,
                     variant: &hir::Variant,
                     generics: &Generics,
                     item_id: ast::NodeId) {
604
        execute_callback!(hir_map::Node::NodeVariant(variant), self);
605 606 607
        if let Some(ref dis_expr) = variant.node.disr_expr {
            // resolve the discriminator expr as a constant
            self.with_constant_rib(|this| {
608
                this.visit_expr(dis_expr);
609 610 611
            });
        }

612
        // `intravisit::walk_variant` without the discriminant expression.
C
corentih 已提交
613 614 615 616 617
        self.visit_variant_data(&variant.node.data,
                                variant.node.name,
                                generics,
                                item_id,
                                variant.span);
618
    }
619 620
    fn visit_foreign_item(&mut self, foreign_item: &hir::ForeignItem) {
        execute_callback!(hir_map::Node::NodeForeignItem(foreign_item), self);
621 622 623 624
        let type_parameters = match foreign_item.node {
            ForeignItemFn(_, ref generics) => {
                HasTypeParameters(generics, FnSpace, ItemRibKind)
            }
C
corentih 已提交
625
            ForeignItemStatic(..) => NoTypeParameters,
626 627
        };
        self.with_type_parameter_rib(type_parameters, |this| {
628
            intravisit::walk_foreign_item(this, foreign_item);
629 630 631
        });
    }
    fn visit_fn(&mut self,
632
                function_kind: FnKind<'v>,
633 634 635 636 637
                declaration: &'v FnDecl,
                block: &'v Block,
                _: Span,
                node_id: NodeId) {
        let rib_kind = match function_kind {
638
            FnKind::ItemFn(_, generics, _, _, _, _) => {
639 640 641
                self.visit_generics(generics);
                ItemRibKind
            }
642
            FnKind::Method(_, sig, _) => {
643 644
                self.visit_generics(&sig.generics);
                self.visit_explicit_self(&sig.explicit_self);
645 646
                MethodRibKind
            }
V
Vadim Petrochenkov 已提交
647
            FnKind::Closure => ClosureRibKind(node_id),
648 649 650
        };
        self.resolve_function(rib_kind, declaration, block);
    }
651
}
652

653
pub type ErrorMessage = Option<(Span, String)>;
654

655
#[derive(Clone, PartialEq, Eq)]
656
pub enum ResolveResult<T> {
C
corentih 已提交
657 658 659
    Failed(ErrorMessage), // Failed to resolve the name, optional helpful error message.
    Indeterminate, // Couldn't determine due to unresolved globs.
    Success(T), // Successfully resolved the import.
660 661
}

662
impl<T> ResolveResult<T> {
663 664 665 666 667
    fn and_then<U, F: FnOnce(T) -> ResolveResult<U>>(self, f: F) -> ResolveResult<U> {
        match self {
            Failed(msg) => Failed(msg),
            Indeterminate => Indeterminate,
            Success(t) => f(t),
C
corentih 已提交
668
        }
669
    }
670 671 672 673 674 675 676

    fn success(self) -> Option<T> {
        match self {
            Success(t) => Some(t),
            _ => None,
        }
    }
677 678
}

679 680 681 682
enum FallbackSuggestion {
    NoSuggestion,
    Field,
    Method,
683
    TraitItem,
684
    StaticMethod(String),
685
    TraitMethod(String),
686 687
}

N
Niko Matsakis 已提交
688
#[derive(Copy, Clone)]
689
enum TypeParameters<'tcx, 'a> {
690
    NoTypeParameters,
C
corentih 已提交
691 692
    HasTypeParameters(// Type parameters.
                      &'a Generics,
693

C
corentih 已提交
694 695 696
                      // Identifies the things that these parameters
                      // were declared on (type, fn, etc)
                      ParamSpace,
697

C
corentih 已提交
698
                      // The kind of the rib used for type parameters.
699
                      RibKind<'tcx>),
700 701
}

702
// The rib kind controls the translation of local
703
// definitions (`Def::Local`) to upvars (`Def::Upvar`).
N
Niko Matsakis 已提交
704
#[derive(Copy, Clone, Debug)]
705
enum RibKind<'a> {
706 707
    // No translation needs to be applied.
    NormalRibKind,
708

709 710
    // We passed through a closure scope at the given node ID.
    // Translate upvars as appropriate.
711
    ClosureRibKind(NodeId /* func id */),
712

713
    // We passed through an impl or trait and are now in one of its
714
    // methods. Allow references to ty params that impl or trait
715 716
    // binds. Disallow any other upvars (including other ty params that are
    // upvars).
717
    MethodRibKind,
718

719 720
    // We passed through an item scope. Disallow upvars.
    ItemRibKind,
721 722

    // We're in a constant item. Can't refer to dynamic stuff.
C
corentih 已提交
723
    ConstantItemRibKind,
724 725 726

    // We passed through an anonymous module.
    AnonymousModuleRibKind(Module<'a>),
727 728
}

N
Niko Matsakis 已提交
729
#[derive(Copy, Clone)]
F
Felix S. Klock II 已提交
730
enum UseLexicalScopeFlag {
731
    DontUseLexicalScope,
C
corentih 已提交
732
    UseLexicalScope,
733 734
}

735
enum ModulePrefixResult<'a> {
736
    NoPrefixFound,
737
    PrefixFound(Module<'a>, usize),
738 739
}

740 741 742 743 744 745 746 747 748
#[derive(Copy, Clone)]
enum AssocItemResolveResult {
    /// Syntax such as `<T>::item`, which can't be resolved until type
    /// checking.
    TypecheckRequired,
    /// We should have been able to resolve the associated item.
    ResolveAttempt(Option<PathResolution>),
}

N
Niko Matsakis 已提交
749
#[derive(Copy, Clone)]
F
Felix S. Klock II 已提交
750
enum BareIdentifierPatternResolution {
751
    FoundStructOrEnumVariant(Def, LastPrivate),
752
    FoundConst(Def, LastPrivate, Name),
C
corentih 已提交
753
    BareIdentifierPatternUnresolved,
754 755
}

756
/// One local scope.
J
Jorge Aparicio 已提交
757
#[derive(Debug)]
758
struct Rib<'a> {
759
    bindings: HashMap<Name, DefLike>,
760
    kind: RibKind<'a>,
B
Brian Anderson 已提交
761
}
762

763 764
impl<'a> Rib<'a> {
    fn new(kind: RibKind<'a>) -> Rib<'a> {
765
        Rib {
766
            bindings: HashMap::new(),
C
corentih 已提交
767
            kind: kind,
768
        }
769 770 771
    }
}

772 773 774
/// A definition along with the index of the rib it was found on
struct LocalDef {
    ribs: Option<(Namespace, usize)>,
C
corentih 已提交
775
    def: Def,
776 777 778 779 780 781
}

impl LocalDef {
    fn from_def(def: Def) -> Self {
        LocalDef {
            ribs: None,
C
corentih 已提交
782
            def: def,
783 784 785 786
        }
    }
}

787
/// The link from a module up to its nearest parent node.
J
Jorge Aparicio 已提交
788
#[derive(Clone,Debug)]
789
enum ParentLink<'a> {
790
    NoParentLink,
791 792
    ModuleParentLink(Module<'a>, Name),
    BlockParentLink(Module<'a>, NodeId),
793 794
}

795
/// One node in the tree of modules.
796 797
pub struct ModuleS<'a> {
    parent_link: ParentLink<'a>,
J
Jeffrey Seyfried 已提交
798
    def: Option<Def>,
799
    is_public: bool,
800
    is_extern_crate: bool,
801

802
    children: RefCell<HashMap<(Name, Namespace), NameResolution<'a>>>,
E
Eduard Burtescu 已提交
803
    imports: RefCell<Vec<ImportDirective>>,
804 805 806 807 808 809 810 811 812 813 814 815 816 817 818

    // The anonymous children of this node. Anonymous children are pseudo-
    // modules that are implicitly created around items contained within
    // blocks.
    //
    // For example, if we have this:
    //
    //  fn f() {
    //      fn g() {
    //          ...
    //      }
    //  }
    //
    // There will be an anonymous module created around `g` with the ID of the
    // entry block for `f`.
819
    anonymous_children: RefCell<NodeMap<Module<'a>>>,
820 821

    // The number of unresolved globs that this module exports.
822
    glob_count: Cell<usize>,
823

824 825 826 827 828 829
    // The number of unresolved pub imports (both regular and globs) in this module
    pub_count: Cell<usize>,

    // The number of unresolved pub glob imports in this module
    pub_glob_count: Cell<usize>,

830
    // The index of the import we're resolving.
831
    resolved_import_count: Cell<usize>,
832 833 834 835

    // Whether this module is populated. If not populated, any attempt to
    // access the children must be preceded with a
    // `populate_module_if_necessary` call.
836
    populated: Cell<bool>,
837 838
}

839 840 841 842 843
pub type Module<'a> = &'a ModuleS<'a>;

impl<'a> ModuleS<'a> {
    fn new(parent_link: ParentLink<'a>, def: Option<Def>, external: bool, is_public: bool) -> Self {
        ModuleS {
844
            parent_link: parent_link,
J
Jeffrey Seyfried 已提交
845
            def: def,
846
            is_public: is_public,
847
            is_extern_crate: false,
848
            children: RefCell::new(HashMap::new()),
849
            imports: RefCell::new(Vec::new()),
850
            anonymous_children: RefCell::new(NodeMap()),
851
            glob_count: Cell::new(0),
852 853
            pub_count: Cell::new(0),
            pub_glob_count: Cell::new(0),
854
            resolved_import_count: Cell::new(0),
855
            populated: Cell::new(!external),
856
        }
B
Brian Anderson 已提交
857 858
    }

859 860 861 862 863 864 865 866 867 868
    fn resolve_name(&self, name: Name, ns: Namespace, allow_private_imports: bool)
                    -> ResolveResult<&'a NameBinding<'a>> {
        let glob_count =
            if allow_private_imports { self.glob_count.get() } else { self.pub_glob_count.get() };

        self.children.borrow().get(&(name, ns)).cloned().unwrap_or_default().result(glob_count)
            .and_then(|binding| {
                let allowed = allow_private_imports || !binding.is_import() || binding.is_public();
                if allowed { Success(binding) } else { Failed(None) }
            })
869 870
    }

871
    // Define the name or return the existing binding if there is a collision.
872
    fn try_define_child(&self, name: Name, ns: Namespace, binding: &'a NameBinding<'a>)
873
                        -> Result<(), &'a NameBinding<'a>> {
874 875
        self.children.borrow_mut().entry((name, ns)).or_insert_with(Default::default)
                                                    .try_define(binding)
876 877
    }

878
    fn increment_outstanding_references_for(&self, name: Name, ns: Namespace) {
879 880
        let mut children = self.children.borrow_mut();
        children.entry((name, ns)).or_insert_with(Default::default).outstanding_references += 1;
881 882 883
    }

    fn decrement_outstanding_references_for(&self, name: Name, ns: Namespace) {
884
        match self.children.borrow_mut().get_mut(&(name, ns)).unwrap().outstanding_references {
885 886 887 888 889
            0 => panic!("No more outstanding references!"),
            ref mut outstanding_references => { *outstanding_references -= 1; }
        }
    }

890 891 892 893 894 895
    fn for_each_child<F: FnMut(Name, Namespace, &'a NameBinding<'a>)>(&self, mut f: F) {
        for (&(name, ns), name_resolution) in self.children.borrow().iter() {
            name_resolution.binding.map(|binding| f(name, ns, binding));
        }
    }

896
    fn for_each_local_child<F: FnMut(Name, Namespace, &'a NameBinding<'a>)>(&self, mut f: F) {
897 898
        self.for_each_child(|name, ns, name_binding| {
            if !name_binding.is_import() && !name_binding.is_extern_crate() {
899 900
                f(name, ns, name_binding)
            }
901
        })
902 903
    }

904
    fn def_id(&self) -> Option<DefId> {
J
Jeffrey Seyfried 已提交
905
        self.def.as_ref().map(Def::def_id)
906 907 908
    }

    fn is_normal(&self) -> bool {
J
Jeffrey Seyfried 已提交
909
        match self.def {
910
            Some(Def::Mod(_)) | Some(Def::ForeignMod(_)) => true,
911 912 913 914 915
            _ => false,
        }
    }

    fn is_trait(&self) -> bool {
J
Jeffrey Seyfried 已提交
916
        match self.def {
917
            Some(Def::Trait(_)) => true,
918
            _ => false,
919
        }
B
Brian Anderson 已提交
920 921
    }

F
Felix S. Klock II 已提交
922
    fn all_imports_resolved(&self) -> bool {
923 924 925 926 927 928
        if self.imports.borrow_state() == ::std::cell::BorrowState::Writing {
            // it is currently being resolved ! so nope
            false
        } else {
            self.imports.borrow().len() == self.resolved_import_count.get()
        }
929 930
    }

V
Victor Berger 已提交
931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953
    pub fn inc_glob_count(&self) {
        self.glob_count.set(self.glob_count.get() + 1);
    }
    pub fn dec_glob_count(&self) {
        assert!(self.glob_count.get() > 0);
        self.glob_count.set(self.glob_count.get() - 1);
    }
    pub fn inc_pub_count(&self) {
        self.pub_count.set(self.pub_count.get() + 1);
    }
    pub fn dec_pub_count(&self) {
        assert!(self.pub_count.get() > 0);
        self.pub_count.set(self.pub_count.get() - 1);
    }
    pub fn inc_pub_glob_count(&self) {
        self.pub_glob_count.set(self.pub_glob_count.get() + 1);
    }
    pub fn dec_pub_glob_count(&self) {
        assert!(self.pub_glob_count.get() > 0);
        self.pub_glob_count.set(self.pub_glob_count.get() - 1);
    }
}

954
impl<'a> fmt::Debug for ModuleS<'a> {
955
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
C
corentih 已提交
956
        write!(f,
957 958
               "{:?}, {}",
               self.def,
C
corentih 已提交
959 960 961 962 963
               if self.is_public {
                   "public"
               } else {
                   "private"
               })
964 965 966
    }
}

967
bitflags! {
J
Jorge Aparicio 已提交
968
    #[derive(Debug)]
969
    flags DefModifiers: u8 {
V
Vadim Petrochenkov 已提交
970 971
        // Enum variants are always considered `PUBLIC`, this is needed for `use Enum::Variant`
        // or `use Enum::*` to work on private enums.
T
Fallout  
Tamir Duberstein 已提交
972 973
        const PUBLIC     = 1 << 0,
        const IMPORTABLE = 1 << 1,
V
Vadim Petrochenkov 已提交
974
        // Variants are considered `PUBLIC`, but some of them live in private enums.
975 976
        // We need to track them to prohibit reexports like `pub use PrivEnum::Variant`.
        const PRIVATE_VARIANT = 1 << 2,
J
Jeffrey Seyfried 已提交
977
        const PRELUDE = 1 << 3,
978
        const GLOB_IMPORTED = 1 << 4,
979 980 981
    }
}

982
// Records a possibly-private value, type, or module definition.
983
#[derive(Debug)]
984
pub struct NameBinding<'a> {
985 986
    modifiers: DefModifiers,
    kind: NameBindingKind<'a>,
987
    span: Option<Span>,
988 989
}

990 991
#[derive(Debug)]
enum NameBindingKind<'a> {
992
    Def(Def),
993
    Module(Module<'a>),
994 995 996 997
    Import {
        binding: &'a NameBinding<'a>,
        id: NodeId,
    },
998 999
}

1000
impl<'a> NameBinding<'a> {
1001
    fn create_from_module(module: Module<'a>, span: Option<Span>) -> Self {
1002
        let modifiers = if module.is_public {
T
Fallout  
Tamir Duberstein 已提交
1003 1004 1005 1006
            DefModifiers::PUBLIC
        } else {
            DefModifiers::empty()
        } | DefModifiers::IMPORTABLE;
1007

1008
        NameBinding { modifiers: modifiers, kind: NameBindingKind::Module(module), span: span }
1009 1010
    }

1011
    fn module(&self) -> Option<Module<'a>> {
1012 1013 1014 1015
        match self.kind {
            NameBindingKind::Module(module) => Some(module),
            NameBindingKind::Def(_) => None,
            NameBindingKind::Import { binding, .. } => binding.module(),
1016 1017 1018
        }
    }

1019
    fn def(&self) -> Option<Def> {
1020 1021 1022 1023
        match self.kind {
            NameBindingKind::Def(def) => Some(def),
            NameBindingKind::Module(module) => module.def,
            NameBindingKind::Import { binding, .. } => binding.def(),
1024
        }
1025
    }
1026

1027
    fn defined_with(&self, modifiers: DefModifiers) -> bool {
1028
        self.modifiers.contains(modifiers)
1029 1030 1031 1032 1033 1034 1035 1036 1037 1038
    }

    fn is_public(&self) -> bool {
        self.defined_with(DefModifiers::PUBLIC)
    }

    fn def_and_lp(&self) -> (Def, LastPrivate) {
        let def = self.def().unwrap();
        (def, LastMod(if self.is_public() { AllPublic } else { DependsOn(def.def_id()) }))
    }
1039 1040 1041 1042

    fn is_extern_crate(&self) -> bool {
        self.module().map(|module| module.is_extern_crate).unwrap_or(false)
    }
1043 1044 1045 1046 1047 1048 1049

    fn is_import(&self) -> bool {
        match self.kind {
            NameBindingKind::Import { .. } => true,
            _ => false,
        }
    }
1050 1051
}

1052
/// Interns the names of the primitive types.
F
Felix S. Klock II 已提交
1053
struct PrimitiveTypeTable {
1054
    primitive_types: HashMap<Name, PrimTy>,
1055
}
1056

1057
impl PrimitiveTypeTable {
K
Kevin Butler 已提交
1058
    fn new() -> PrimitiveTypeTable {
C
corentih 已提交
1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075
        let mut table = PrimitiveTypeTable { primitive_types: HashMap::new() };

        table.intern("bool", TyBool);
        table.intern("char", TyChar);
        table.intern("f32", TyFloat(TyF32));
        table.intern("f64", TyFloat(TyF64));
        table.intern("isize", TyInt(TyIs));
        table.intern("i8", TyInt(TyI8));
        table.intern("i16", TyInt(TyI16));
        table.intern("i32", TyInt(TyI32));
        table.intern("i64", TyInt(TyI64));
        table.intern("str", TyStr);
        table.intern("usize", TyUint(TyUs));
        table.intern("u8", TyUint(TyU8));
        table.intern("u16", TyUint(TyU16));
        table.intern("u32", TyUint(TyU32));
        table.intern("u64", TyUint(TyU64));
K
Kevin Butler 已提交
1076 1077 1078 1079

        table
    }

1080
    fn intern(&mut self, string: &str, primitive_type: PrimTy) {
1081
        self.primitive_types.insert(token::intern(string), primitive_type);
1082 1083 1084
    }
}

1085
/// The main resolver class.
C
corentih 已提交
1086
pub struct Resolver<'a, 'tcx: 'a> {
E
Eduard Burtescu 已提交
1087
    session: &'a Session,
1088

1089
    ast_map: &'a hir_map::Map<'tcx>,
1090

1091
    graph_root: Module<'a>,
1092

1093
    trait_item_map: FnvHashMap<(Name, DefId), DefId>,
1094

1095
    structs: FnvHashMap<DefId, Vec<Name>>,
1096

1097
    // The number of imports that are currently unresolved.
1098
    unresolved_imports: usize,
1099 1100

    // The module that represents the current item scope.
1101
    current_module: Module<'a>,
1102 1103

    // The current set of local scopes, for values.
1104
    // FIXME #4948: Reuse ribs to avoid allocation.
1105
    value_ribs: Vec<Rib<'a>>,
1106 1107

    // The current set of local scopes, for types.
1108
    type_ribs: Vec<Rib<'a>>,
1109

1110
    // The current set of local scopes, for labels.
1111
    label_ribs: Vec<Rib<'a>>,
1112

1113
    // The trait that the current context can refer to.
1114 1115 1116 1117
    current_trait_ref: Option<(DefId, TraitRef)>,

    // The current self type if inside an impl (used for better errors).
    current_self_type: Option<Ty>,
1118

1119
    // The idents for the primitive types.
E
Eduard Burtescu 已提交
1120
    primitive_type_table: PrimitiveTypeTable,
1121

J
Jonathan S 已提交
1122
    def_map: RefCell<DefMap>,
1123 1124
    freevars: FreevarMap,
    freevars_seen: NodeMap<NodeMap<usize>>,
1125
    export_map: ExportMap,
1126
    trait_map: TraitMap,
1127
    external_exports: ExternalExports,
1128

1129 1130 1131 1132 1133
    // Whether or not to print error messages. Can be set to true
    // when getting additional info for error message suggestions,
    // so as to avoid printing duplicate errors
    emit_errors: bool,

1134 1135 1136 1137 1138
    make_glob_map: bool,
    // Maps imports to the names of items actually imported (this actually maps
    // all imports, but only glob imports are actually interesting).
    glob_map: GlobMap,

1139
    used_imports: HashSet<(NodeId, Namespace)>,
1140
    used_crates: HashSet<CrateNum>,
G
Garming Sam 已提交
1141 1142

    // Callback function for intercepting walks
1143
    callback: Option<Box<Fn(hir_map::Node, &mut bool) -> bool>>,
G
Garming Sam 已提交
1144 1145 1146
    // The intention is that the callback modifies this flag.
    // Once set, the resolver falls out of the walk, preserving the ribs.
    resolved: bool,
1147 1148 1149 1150 1151 1152

    arenas: &'a ResolverArenas<'a>,
}

pub struct ResolverArenas<'a> {
    modules: arena::TypedArena<ModuleS<'a>>,
1153
    name_bindings: arena::TypedArena<NameBinding<'a>>,
1154 1155
}

1156
#[derive(PartialEq)]
S
Steven Fackler 已提交
1157 1158
enum FallbackChecks {
    Everything,
C
corentih 已提交
1159
    OnlyTraitAndStatics,
S
Steven Fackler 已提交
1160 1161
}

1162 1163
impl<'a, 'tcx> Resolver<'a, 'tcx> {
    fn new(session: &'a Session,
1164
           ast_map: &'a hir_map::Map<'tcx>,
1165 1166
           make_glob_map: MakeGlobMap,
           arenas: &'a ResolverArenas<'a>)
C
corentih 已提交
1167
           -> Resolver<'a, 'tcx> {
1168
        let root_def_id = ast_map.local_def_id(CRATE_NODE_ID);
1169
        let graph_root = ModuleS::new(NoParentLink, Some(Def::Mod(root_def_id)), false, true);
1170
        let graph_root = arenas.modules.alloc(graph_root);
K
Kevin Butler 已提交
1171 1172 1173 1174

        Resolver {
            session: session,

1175 1176
            ast_map: ast_map,

K
Kevin Butler 已提交
1177 1178
            // The outermost module has def ID 0; this is not reflected in the
            // AST.
1179
            graph_root: graph_root,
K
Kevin Butler 已提交
1180

1181 1182
            trait_item_map: FnvHashMap(),
            structs: FnvHashMap(),
K
Kevin Butler 已提交
1183 1184 1185

            unresolved_imports: 0,

1186
            current_module: graph_root,
1187 1188 1189
            value_ribs: Vec::new(),
            type_ribs: Vec::new(),
            label_ribs: Vec::new(),
K
Kevin Butler 已提交
1190 1191 1192 1193 1194 1195

            current_trait_ref: None,
            current_self_type: None,

            primitive_type_table: PrimitiveTypeTable::new(),

1196
            def_map: RefCell::new(NodeMap()),
1197 1198
            freevars: NodeMap(),
            freevars_seen: NodeMap(),
1199 1200
            export_map: NodeMap(),
            trait_map: NodeMap(),
K
Kevin Butler 已提交
1201
            used_imports: HashSet::new(),
1202
            used_crates: HashSet::new(),
1203
            external_exports: DefIdSet(),
K
Kevin Butler 已提交
1204 1205

            emit_errors: true,
1206 1207
            make_glob_map: make_glob_map == MakeGlobMap::Yes,
            glob_map: HashMap::new(),
G
Garming Sam 已提交
1208 1209 1210

            callback: None,
            resolved: false,
1211 1212 1213 1214 1215 1216 1217 1218

            arenas: arenas,
        }
    }

    fn arenas() -> ResolverArenas<'a> {
        ResolverArenas {
            modules: arena::TypedArena::new(),
1219
            name_bindings: arena::TypedArena::new(),
K
Kevin Butler 已提交
1220 1221
        }
    }
1222

1223 1224 1225 1226 1227 1228 1229 1230
    fn new_module(&self,
                  parent_link: ParentLink<'a>,
                  def: Option<Def>,
                  external: bool,
                  is_public: bool) -> Module<'a> {
        self.arenas.modules.alloc(ModuleS::new(parent_link, def, external, is_public))
    }

1231 1232 1233 1234
    fn new_name_binding(&self, name_binding: NameBinding<'a>) -> &'a NameBinding<'a> {
        self.arenas.name_bindings.alloc(name_binding)
    }

1235 1236 1237 1238 1239 1240
    fn new_extern_crate_module(&self, parent_link: ParentLink<'a>, def: Def) -> Module<'a> {
        let mut module = ModuleS::new(parent_link, Some(def), false, true);
        module.is_extern_crate = true;
        self.arenas.modules.alloc(module)
    }

1241 1242 1243 1244
    fn get_ribs<'b>(&'b mut self, ns: Namespace) -> &'b mut Vec<Rib<'a>> {
        match ns { ValueNS => &mut self.value_ribs, TypeNS => &mut self.type_ribs }
    }

1245
    #[inline]
1246 1247 1248 1249 1250 1251
    fn record_use(&mut self, name: Name, ns: Namespace, binding: &'a NameBinding<'a>) {
        // track extern crates for unused_extern_crate lint
        if let Some(DefId { krate, .. }) = binding.module().and_then(ModuleS::def_id) {
            self.used_crates.insert(krate);
        }

1252 1253 1254 1255 1256
        let import_id = match binding.kind {
            NameBindingKind::Import { id, .. } => id,
            _ => return,
        };

1257 1258
        self.used_imports.insert((import_id, ns));

1259 1260 1261 1262
        if !self.make_glob_map {
            return;
        }
        if self.glob_map.contains_key(&import_id) {
1263
            self.glob_map.get_mut(&import_id).unwrap().insert(name);
1264 1265 1266 1267 1268 1269 1270 1271 1272
            return;
        }

        let mut new_set = HashSet::new();
        new_set.insert(name);
        self.glob_map.insert(import_id, new_set);
    }

    fn get_trait_name(&self, did: DefId) -> Name {
1273 1274
        if let Some(node_id) = self.ast_map.as_local_node_id(did) {
            self.ast_map.expect_item(node_id).name
1275
        } else {
1276
            self.session.cstore.item_name(did)
1277 1278 1279
        }
    }

1280
    /// Resolves the given module path from the given root `module_`.
F
Felix S. Klock II 已提交
1281
    fn resolve_module_path_from_root(&mut self,
1282
                                     module_: Module<'a>,
1283
                                     module_path: &[Name],
1284
                                     index: usize,
1285 1286
                                     span: Span,
                                     lp: LastPrivate)
1287
                                     -> ResolveResult<(Module<'a>, LastPrivate)> {
1288
        fn search_parent_externals<'a>(needle: Name, module: Module<'a>) -> Option<Module<'a>> {
1289 1290
            match module.resolve_name(needle, TypeNS, false) {
                Success(binding) if binding.is_extern_crate() => Some(module),
1291
                _ => match module.parent_link {
1292
                    ModuleParentLink(ref parent, _) => {
1293
                        search_parent_externals(needle, parent)
1294
                    }
C
corentih 已提交
1295 1296
                    _ => None,
                },
1297
            }
1298 1299
        }

1300
        let mut search_module = module_;
1301
        let mut index = index;
A
Alex Crichton 已提交
1302
        let module_path_len = module_path.len();
1303
        let mut closest_private = lp;
1304 1305 1306 1307 1308

        // Resolve the module part of the path. This does not involve looking
        // upward though scope chains; we simply resolve names directly in
        // modules as we go.
        while index < module_path_len {
A
Alex Crichton 已提交
1309
            let name = module_path[index];
1310
            match self.resolve_name_in_module(search_module, name, TypeNS, false, true) {
1311
                Failed(None) => {
1312
                    let segment_name = name.as_str();
1313
                    let module_name = module_to_string(search_module);
1314
                    let mut span = span;
1315
                    let msg = if "???" == &module_name {
1316
                        span.hi = span.lo + Pos::from_usize(segment_name.len());
1317

C
corentih 已提交
1318
                        match search_parent_externals(name, &self.current_module) {
1319
                            Some(module) => {
1320 1321
                                let path_str = names_to_string(module_path);
                                let target_mod_str = module_to_string(&*module);
C
corentih 已提交
1322
                                let current_mod_str = module_to_string(&*self.current_module);
1323 1324 1325 1326 1327 1328 1329

                                let prefix = if target_mod_str == current_mod_str {
                                    "self::".to_string()
                                } else {
                                    format!("{}::", target_mod_str)
                                };

1330
                                format!("Did you mean `{}{}`?", prefix, path_str)
C
corentih 已提交
1331 1332
                            }
                            None => format!("Maybe a missing `extern crate {}`?", segment_name),
1333
                        }
1334
                    } else {
C
corentih 已提交
1335
                        format!("Could not find `{}` in `{}`", segment_name, module_name)
1336
                    };
1337

1338
                    return Failed(Some((span, msg)));
1339
                }
1340
                Failed(err) => return Failed(err),
B
Brian Anderson 已提交
1341
                Indeterminate => {
C
corentih 已提交
1342 1343 1344
                    debug!("(resolving module path for import) module resolution is \
                            indeterminate: {}",
                           name);
B
Brian Anderson 已提交
1345
                    return Indeterminate;
1346
                }
1347
                Success(binding) => {
1348 1349
                    // Check to see whether there are type bindings, and, if
                    // so, whether there is a module within.
J
Jeffrey Seyfried 已提交
1350
                    if let Some(module_def) = binding.module() {
1351
                        search_module = module_def;
1352

1353 1354
                        // Keep track of the closest private module used
                        // when resolving this import chain.
1355
                        if !binding.is_public() {
1356
                            if let Some(did) = search_module.def_id() {
1357
                                closest_private = LastMod(DependsOn(did));
1358 1359
                            }
                        }
1360 1361 1362
                    } else {
                        let msg = format!("Not a module `{}`", name);
                        return Failed(Some((span, msg)));
1363 1364 1365 1366
                    }
                }
            }

T
Tim Chevalier 已提交
1367
            index += 1;
1368 1369
        }

1370
        return Success((search_module, closest_private));
1371 1372
    }

1373 1374
    /// Attempts to resolve the module part of an import directive or path
    /// rooted at the given module.
1375 1376 1377
    ///
    /// On success, returns the resolved module, and the closest *private*
    /// module found to the destination when resolving this path.
F
Felix S. Klock II 已提交
1378
    fn resolve_module_path(&mut self,
1379
                           module_: Module<'a>,
1380
                           module_path: &[Name],
1381
                           use_lexical_scope: UseLexicalScopeFlag,
J
Jeffrey Seyfried 已提交
1382
                           span: Span)
1383
                           -> ResolveResult<(Module<'a>, LastPrivate)> {
1384 1385 1386
        if module_path.len() == 0 {
            return Success((self.graph_root, LastMod(AllPublic))) // Use the crate root
        }
1387

1388
        debug!("(resolving module path for import) processing `{}` rooted at `{}`",
1389 1390
               names_to_string(module_path),
               module_to_string(&*module_));
1391

1392
        // Resolve the module prefix, if any.
1393
        let module_prefix_result = self.resolve_module_prefix(module_, module_path);
1394

1395 1396
        let search_module;
        let start_index;
1397
        let last_private;
1398
        match module_prefix_result {
1399
            Failed(None) => {
1400
                let mpath = names_to_string(module_path);
1401
                let mpath = &mpath[..];
1402
                match mpath.rfind(':') {
C
Fix ICE  
Corey Richardson 已提交
1403
                    Some(idx) => {
1404
                        let msg = format!("Could not find `{}` in `{}`",
C
corentih 已提交
1405 1406 1407 1408
                                          // idx +- 1 to account for the
                                          // colons on either side
                                          &mpath[idx + 1..],
                                          &mpath[..idx - 1]);
1409
                        return Failed(Some((span, msg)));
C
corentih 已提交
1410
                    }
1411
                    None => {
C
corentih 已提交
1412
                        return Failed(None);
1413
                    }
1414
                }
1415
            }
1416
            Failed(err) => return Failed(err),
B
Brian Anderson 已提交
1417
            Indeterminate => {
C
corentih 已提交
1418
                debug!("(resolving module path for import) indeterminate; bailing");
B
Brian Anderson 已提交
1419
                return Indeterminate;
1420
            }
1421 1422 1423 1424 1425 1426 1427 1428
            Success(NoPrefixFound) => {
                // There was no prefix, so we're considering the first element
                // of the path. How we handle this depends on whether we were
                // instructed to use lexical scope or not.
                match use_lexical_scope {
                    DontUseLexicalScope => {
                        // This is a crate-relative path. We will start the
                        // resolution process at index zero.
1429
                        search_module = self.graph_root;
1430
                        start_index = 0;
1431
                        last_private = LastMod(AllPublic);
1432 1433 1434 1435 1436
                    }
                    UseLexicalScope => {
                        // This is not a crate-relative path. We resolve the
                        // first component of the path in the current lexical
                        // scope and then proceed to resolve below that.
1437 1438 1439 1440
                        match self.resolve_item_in_lexical_scope(module_,
                                                                 module_path[0],
                                                                 TypeNS,
                                                                 true) {
1441
                            Failed(err) => return Failed(err),
1442
                            Indeterminate => {
C
corentih 已提交
1443
                                debug!("(resolving module path for import) indeterminate; bailing");
1444 1445
                                return Indeterminate;
                            }
1446
                            Success(binding) => match binding.module() {
1447 1448 1449 1450 1451 1452
                                Some(containing_module) => {
                                    search_module = containing_module;
                                    start_index = 1;
                                    last_private = LastMod(AllPublic);
                                }
                                None => return Failed(None),
1453 1454 1455 1456 1457
                            }
                        }
                    }
                }
            }
E
Eduard Burtescu 已提交
1458
            Success(PrefixFound(ref containing_module, index)) => {
1459
                search_module = containing_module;
1460
                start_index = index;
1461
                last_private = LastMod(DependsOn(containing_module.def_id()
1462
                                                                  .unwrap()));
1463 1464 1465
            }
        }

1466 1467 1468 1469
        self.resolve_module_path_from_root(search_module,
                                           module_path,
                                           start_index,
                                           span,
1470
                                           last_private)
1471 1472
    }

1473 1474
    /// Invariant: This must only be called during main resolution, not during
    /// import resolution.
F
Felix S. Klock II 已提交
1475
    fn resolve_item_in_lexical_scope(&mut self,
1476
                                     module_: Module<'a>,
1477
                                     name: Name,
1478 1479
                                     namespace: Namespace,
                                     record_used: bool)
1480
                                     -> ResolveResult<&'a NameBinding<'a>> {
C
corentih 已提交
1481
        debug!("(resolving item in lexical scope) resolving `{}` in namespace {:?} in `{}`",
1482
               name,
1483
               namespace,
1484
               module_to_string(&*module_));
1485

1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497
        // Proceed up the scope chain looking for parent modules.
        let mut search_module = module_;
        loop {
            // Resolve the name in the parent module.
            match self.resolve_name_in_module(search_module, name, namespace, true, record_used) {
                Failed(Some((span, msg))) => {
                    resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
                }
                Failed(None) => (), // Continue up the search chain.
                Indeterminate => {
                    // We couldn't see through the higher scope because of an
                    // unresolved import higher up. Bail.
1498

1499 1500
                    debug!("(resolving item in lexical scope) indeterminate higher scope; bailing");
                    return Indeterminate;
1501
                }
1502
                Success(binding) => {
1503 1504
                    // We found the module.
                    debug!("(resolving item in lexical scope) found name in module, done");
1505
                    return Success(binding);
1506 1507 1508 1509
                }
            }

            // Go to the next parent.
1510
            match search_module.parent_link {
B
Brian Anderson 已提交
1511
                NoParentLink => {
1512
                    // No more parents. This module was unresolved.
1513
                    debug!("(resolving item in lexical scope) unresolved module: no parent module");
1514
                    return Failed(None);
1515
                }
1516
                ModuleParentLink(parent_module_node, _) => {
1517 1518 1519 1520
                    if search_module.is_normal() {
                        // We stop the search here.
                        debug!("(resolving item in lexical scope) unresolved module: not \
                                searching through module parents");
1521
                            return Failed(None);
1522
                    } else {
1523
                        search_module = parent_module_node;
1524 1525
                    }
                }
1526 1527
                BlockParentLink(parent_module_node, _) => {
                    search_module = parent_module_node;
1528 1529 1530 1531 1532
                }
            }
        }
    }

1533
    /// Returns the nearest normal module parent of the given module.
1534
    fn get_nearest_normal_module_parent(&mut self, module_: Module<'a>) -> Option<Module<'a>> {
1535 1536
        let mut module_ = module_;
        loop {
1537
            match module_.parent_link {
1538 1539 1540
                NoParentLink => return None,
                ModuleParentLink(new_module, _) |
                BlockParentLink(new_module, _) => {
1541
                    let new_module = new_module;
1542 1543
                    if new_module.is_normal() {
                        return Some(new_module);
1544
                    }
1545
                    module_ = new_module;
1546 1547 1548 1549 1550
                }
            }
        }
    }

1551 1552
    /// Returns the nearest normal module parent of the given module, or the
    /// module itself if it is a normal module.
1553
    fn get_nearest_normal_module_parent_or_self(&mut self, module_: Module<'a>) -> Module<'a> {
1554 1555 1556
        if module_.is_normal() {
            return module_;
        }
1557
        match self.get_nearest_normal_module_parent(module_) {
1558 1559
            None => module_,
            Some(new_module) => new_module,
1560 1561 1562
        }
    }

1563
    /// Resolves a "module prefix". A module prefix is one or both of (a) `self::`;
1564
    /// (b) some chain of `super::`.
1565
    /// grammar: (SELF MOD_SEP ) ? (SUPER MOD_SEP) *
F
Felix S. Klock II 已提交
1566
    fn resolve_module_prefix(&mut self,
1567
                             module_: Module<'a>,
1568
                             module_path: &[Name])
1569
                             -> ResolveResult<ModulePrefixResult<'a>> {
1570 1571
        // Start at the current module if we see `self` or `super`, or at the
        // top of the crate otherwise.
1572 1573 1574 1575 1576 1577
        let mut i = match &*module_path[0].as_str() {
            "self" => 1,
            "super" => 0,
            _ => return Success(NoPrefixFound),
        };
        let mut containing_module = self.get_nearest_normal_module_parent_or_self(module_);
1578 1579

        // Now loop through all the `super`s we find.
1580
        while i < module_path.len() && "super" == module_path[i].as_str() {
1581
            debug!("(resolving module prefix) resolving `super` at {}",
1582
                   module_to_string(&*containing_module));
1583
            match self.get_nearest_normal_module_parent(containing_module) {
1584
                None => return Failed(None),
1585 1586 1587
                Some(new_module) => {
                    containing_module = new_module;
                    i += 1;
1588 1589 1590 1591
                }
            }
        }

1592
        debug!("(resolving module prefix) finished resolving prefix at {}",
1593
               module_to_string(&*containing_module));
1594 1595

        return Success(PrefixFound(containing_module, i));
1596 1597
    }

1598
    /// Attempts to resolve the supplied name in the given module for the
J
Jeffrey Seyfried 已提交
1599
    /// given namespace. If successful, returns the binding corresponding to
1600
    /// the name.
F
Felix S. Klock II 已提交
1601
    fn resolve_name_in_module(&mut self,
1602
                              module: Module<'a>,
1603
                              name: Name,
1604
                              namespace: Namespace,
1605 1606
                              allow_private_imports: bool,
                              record_used: bool)
1607
                              -> ResolveResult<&'a NameBinding<'a>> {
1608
        debug!("(resolving name in module) resolving `{}` in `{}`", name, module_to_string(module));
1609

1610 1611 1612 1613
        build_reduced_graph::populate_module_if_necessary(self, module);
        module.resolve_name(name, namespace, allow_private_imports).and_then(|binding| {
            if record_used {
                self.record_use(name, namespace, binding);
1614
            }
1615 1616
            Success(binding)
        })
1617 1618
    }

1619
    fn report_unresolved_imports(&mut self, module_: Module<'a>) {
1620
        let index = module_.resolved_import_count.get();
1621 1622
        let imports = module_.imports.borrow();
        let import_count = imports.len();
1623
        if index != import_count {
1624 1625 1626
            resolve_error(self,
                          (*imports)[index].span,
                          ResolutionError::UnresolvedImport(None));
1627 1628 1629
        }

        // Descend into children and anonymous children.
1630
        build_reduced_graph::populate_module_if_necessary(self, module_);
1631

1632
        module_.for_each_local_child(|_, _, child_node| {
1633
            match child_node.module() {
1634 1635 1636 1637 1638
                None => {
                    // Continue.
                }
                Some(child_module) => {
                    self.report_unresolved_imports(child_module);
1639 1640
                }
            }
1641
        });
1642

1643
        for (_, module_) in module_.anonymous_children.borrow().iter() {
1644
            self.report_unresolved_imports(module_);
1645 1646 1647 1648 1649
        }
    }

    // AST resolution
    //
1650
    // We maintain a list of value ribs and type ribs.
1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665
    //
    // Simultaneously, we keep track of the current position in the module
    // graph in the `current_module` pointer. When we go to resolve a name in
    // the value or type namespaces, we first look through all the ribs and
    // then query the module graph. When we resolve a name in the module
    // namespace, we can skip all the ribs (since nested modules are not
    // allowed within blocks in Rust) and jump straight to the current module
    // graph node.
    //
    // Named implementations are handled separately. When we find a method
    // call, we consult the module node to find all of the implementations in
    // scope. This information is lazily cached in the module node. We then
    // generate a fake "implementation scope" containing all the
    // implementations thus found, for compatibility with old resolve pass.

C
corentih 已提交
1666 1667
    fn with_scope<F>(&mut self, name: Option<Name>, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1668
    {
1669
        let orig_module = self.current_module;
1670 1671

        // Move down in the graph.
1672
        match name {
B
Brian Anderson 已提交
1673
            None => {
1674 1675
                // Nothing to do.
            }
B
Brian Anderson 已提交
1676
            Some(name) => {
1677
                build_reduced_graph::populate_module_if_necessary(self, &orig_module);
1678

1679 1680 1681 1682 1683 1684 1685 1686 1687
                if let Success(name_binding) = orig_module.resolve_name(name, TypeNS, false) {
                    match name_binding.module() {
                        None => {
                            debug!("!!! (with scope) didn't find module for `{}` in `{}`",
                                   name,
                                   module_to_string(orig_module));
                        }
                        Some(module) => {
                            self.current_module = module;
1688 1689 1690 1691 1692 1693
                        }
                    }
                }
            }
        }

A
Alex Crichton 已提交
1694
        f(self);
1695 1696 1697 1698

        self.current_module = orig_module;
    }

S
Seo Sanghyeon 已提交
1699 1700
    /// Searches the current set of local scopes for labels.
    /// Stops after meeting a closure.
1701 1702 1703 1704 1705 1706 1707 1708
    fn search_label(&self, name: Name) -> Option<DefLike> {
        for rib in self.label_ribs.iter().rev() {
            match rib.kind {
                NormalRibKind => {
                    // Continue
                }
                _ => {
                    // Do not resolve labels across function boundary
C
corentih 已提交
1709
                    return None;
1710 1711 1712
                }
            }
            let result = rib.bindings.get(&name).cloned();
S
Seo Sanghyeon 已提交
1713
            if result.is_some() {
C
corentih 已提交
1714
                return result;
1715 1716 1717 1718 1719
            }
        }
        None
    }

1720
    fn resolve_crate(&mut self, krate: &hir::Crate) {
1721
        debug!("(resolving crate) starting");
1722

1723
        intravisit::walk_crate(self, krate);
1724 1725
    }

W
we 已提交
1726 1727
    fn check_if_primitive_type_name(&self, name: Name, span: Span) {
        if let Some(_) = self.primitive_type_table.primitive_types.get(&name) {
C
corentih 已提交
1728 1729 1730 1731
            span_err!(self.session,
                      span,
                      E0317,
                      "user-defined types or type parameters cannot shadow the primitive types");
W
we 已提交
1732 1733 1734
        }
    }

1735
    fn resolve_item(&mut self, item: &Item) {
V
Vadim Petrochenkov 已提交
1736
        let name = item.name;
1737

C
corentih 已提交
1738
        debug!("(resolving item) resolving {}", name);
1739

1740
        match item.node {
1741 1742 1743
            ItemEnum(_, ref generics) |
            ItemTy(_, ref generics) |
            ItemStruct(_, ref generics) => {
W
we 已提交
1744 1745
                self.check_if_primitive_type_name(name, item.span);

C
corentih 已提交
1746
                self.with_type_parameter_rib(HasTypeParameters(generics, TypeSpace, ItemRibKind),
1747
                                             |this| intravisit::walk_item(this, item));
1748
            }
1749
            ItemFn(_, _, _, _, ref generics, _) => {
C
corentih 已提交
1750
                self.with_type_parameter_rib(HasTypeParameters(generics, FnSpace, ItemRibKind),
1751
                                             |this| intravisit::walk_item(this, item));
1752 1753
            }

F
Flavio Percoco 已提交
1754
            ItemDefaultImpl(_, ref trait_ref) => {
1755
                self.with_optional_trait_ref(Some(trait_ref), |_, _| {});
1756
            }
C
corentih 已提交
1757
            ItemImpl(_, _, ref generics, ref opt_trait_ref, ref self_type, ref impl_items) => {
1758
                self.resolve_implementation(generics,
1759
                                            opt_trait_ref,
1760
                                            &**self_type,
1761
                                            item.id,
1762
                                            impl_items);
1763 1764
            }

N
Nick Cameron 已提交
1765
            ItemTrait(_, ref generics, ref bounds, ref trait_items) => {
W
we 已提交
1766 1767
                self.check_if_primitive_type_name(name, item.span);

1768 1769 1770 1771 1772
                // Create a new rib for the trait-wide type parameters.
                self.with_type_parameter_rib(HasTypeParameters(generics,
                                                               TypeSpace,
                                                               ItemRibKind),
                                             |this| {
1773
                    let local_def_id = this.ast_map.local_def_id(item.id);
1774
                    this.with_self_rib(Def::SelfTy(Some(local_def_id), None), |this| {
1775
                        this.visit_generics(generics);
1776
                        walk_list!(this, visit_ty_param_bound, bounds);
1777 1778

                        for trait_item in trait_items {
1779
                            match trait_item.node {
1780
                                hir::ConstTraitItem(_, ref default) => {
1781 1782 1783 1784 1785
                                    // Only impose the restrictions of
                                    // ConstRibKind if there's an actual constant
                                    // expression in a provided default.
                                    if default.is_some() {
                                        this.with_constant_rib(|this| {
1786
                                            intravisit::walk_trait_item(this, trait_item)
1787 1788
                                        });
                                    } else {
1789
                                        intravisit::walk_trait_item(this, trait_item)
1790 1791
                                    }
                                }
1792
                                hir::MethodTraitItem(ref sig, _) => {
1793 1794 1795 1796 1797
                                    let type_parameters =
                                        HasTypeParameters(&sig.generics,
                                                          FnSpace,
                                                          MethodRibKind);
                                    this.with_type_parameter_rib(type_parameters, |this| {
1798
                                        intravisit::walk_trait_item(this, trait_item)
1799
                                    });
1800
                                }
1801
                                hir::TypeTraitItem(..) => {
V
Vadim Petrochenkov 已提交
1802
                                    this.check_if_primitive_type_name(trait_item.name,
1803
                                                                      trait_item.span);
1804
                                    this.with_type_parameter_rib(NoTypeParameters, |this| {
1805
                                        intravisit::walk_trait_item(this, trait_item)
1806
                                    });
1807 1808 1809 1810
                                }
                            };
                        }
                    });
1811
                });
1812 1813
            }

1814
            ItemMod(_) | ItemForeignMod(_) => {
1815
                self.with_scope(Some(name), |this| {
1816
                    intravisit::walk_item(this, item);
1817
                });
1818 1819
            }

1820
            ItemConst(..) | ItemStatic(..) => {
A
Alex Crichton 已提交
1821
                self.with_constant_rib(|this| {
1822
                    intravisit::walk_item(this, item);
1823
                });
1824
            }
1825

W
we 已提交
1826 1827
            ItemUse(ref view_path) => {
                // check for imports shadowing primitive types
1828
                let check_rename = |this: &Self, id, name| {
1829
                    match this.def_map.borrow().get(&id).map(|d| d.full_def()) {
1830 1831
                        Some(Def::Enum(..)) | Some(Def::TyAlias(..)) | Some(Def::Struct(..)) |
                        Some(Def::Trait(..)) | None => {
1832
                            this.check_if_primitive_type_name(name, item.span);
W
we 已提交
1833 1834 1835
                        }
                        _ => {}
                    }
1836 1837 1838
                };

                match view_path.node {
1839 1840
                    hir::ViewPathSimple(name, _) => {
                        check_rename(self, item.id, name);
1841
                    }
1842
                    hir::ViewPathList(ref prefix, ref items) => {
1843
                        for item in items {
1844 1845
                            if let Some(name) = item.node.rename() {
                                check_rename(self, item.node.id(), name);
1846 1847 1848 1849 1850 1851 1852 1853
                            }
                        }

                        // Resolve prefix of an import with empty braces (issue #28388)
                        if items.is_empty() && !prefix.segments.is_empty() {
                            match self.resolve_crate_relative_path(prefix.span,
                                                                   &prefix.segments,
                                                                   TypeNS) {
C
corentih 已提交
1854 1855
                                Some((def, lp)) =>
                                    self.record_def(item.id, PathResolution::new(def, lp, 0)),
1856 1857 1858 1859 1860
                                None => {
                                    resolve_error(self,
                                                  prefix.span,
                                                  ResolutionError::FailedToResolve(
                                                      &path_names_to_string(prefix, 0)));
1861
                                    self.record_def(item.id, err_path_resolution());
1862
                                }
1863 1864 1865 1866
                            }
                        }
                    }
                    _ => {}
W
we 已提交
1867 1868 1869
                }
            }

1870
            ItemExternCrate(_) => {
1871
                // do nothing, these are just around to be encoded
1872
            }
1873 1874 1875
        }
    }

1876
    fn with_type_parameter_rib<'b, F>(&'b mut self, type_parameters: TypeParameters<'a, 'b>, f: F)
C
corentih 已提交
1877
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1878
    {
1879
        match type_parameters {
1880
            HasTypeParameters(generics, space, rib_kind) => {
1881
                let mut function_type_rib = Rib::new(rib_kind);
1882
                let mut seen_bindings = HashSet::new();
D
Daniel Micay 已提交
1883
                for (index, type_parameter) in generics.ty_params.iter().enumerate() {
1884
                    let name = type_parameter.name;
1885
                    debug!("with_type_parameter_rib: {}", type_parameter.id);
1886

1887
                    if seen_bindings.contains(&name) {
1888 1889
                        resolve_error(self,
                                      type_parameter.span,
C
corentih 已提交
1890
                                      ResolutionError::NameAlreadyUsedInTypeParameterList(name));
1891
                    }
1892
                    seen_bindings.insert(name);
1893

1894
                    // plain insert (no renaming)
C
corentih 已提交
1895 1896
                    function_type_rib.bindings
                                     .insert(name,
1897
                                             DlDef(Def::TyParam(space,
C
corentih 已提交
1898 1899 1900 1901
                                                              index as u32,
                                                              self.ast_map
                                                                  .local_def_id(type_parameter.id),
                                                              name)));
1902
                }
1903
                self.type_ribs.push(function_type_rib);
1904 1905
            }

B
Brian Anderson 已提交
1906
            NoTypeParameters => {
1907 1908 1909 1910
                // Nothing to do.
            }
        }

A
Alex Crichton 已提交
1911
        f(self);
1912

1913
        match type_parameters {
C
corentih 已提交
1914 1915 1916 1917 1918 1919
            HasTypeParameters(..) => {
                if !self.resolved {
                    self.type_ribs.pop();
                }
            }
            NoTypeParameters => {}
1920 1921 1922
        }
    }

C
corentih 已提交
1923 1924
    fn with_label_rib<F>(&mut self, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1925
    {
1926
        self.label_ribs.push(Rib::new(NormalRibKind));
A
Alex Crichton 已提交
1927
        f(self);
G
Garming Sam 已提交
1928 1929 1930
        if !self.resolved {
            self.label_ribs.pop();
        }
1931
    }
1932

C
corentih 已提交
1933 1934
    fn with_constant_rib<F>(&mut self, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1935
    {
1936 1937
        self.value_ribs.push(Rib::new(ConstantItemRibKind));
        self.type_ribs.push(Rib::new(ConstantItemRibKind));
A
Alex Crichton 已提交
1938
        f(self);
G
Garming Sam 已提交
1939 1940 1941 1942
        if !self.resolved {
            self.type_ribs.pop();
            self.value_ribs.pop();
        }
1943 1944
    }

1945
    fn resolve_function(&mut self, rib_kind: RibKind<'a>, declaration: &FnDecl, block: &Block) {
1946
        // Create a value rib for the function.
1947
        self.value_ribs.push(Rib::new(rib_kind));
1948

1949
        // Create a label rib for the function.
1950
        self.label_ribs.push(Rib::new(rib_kind));
1951

1952 1953 1954
        // Add each argument to the rib.
        let mut bindings_list = HashMap::new();
        for argument in &declaration.inputs {
C
corentih 已提交
1955
            self.resolve_pattern(&*argument.pat, ArgumentIrrefutableMode, &mut bindings_list);
1956

1957
            self.visit_ty(&*argument.ty);
1958

1959 1960
            debug!("(resolving function) recorded argument");
        }
1961
        intravisit::walk_fn_ret_ty(self, &declaration.output);
1962

1963
        // Resolve the function body.
1964
        self.visit_block(block);
1965

1966
        debug!("(resolving function) leaving function");
1967

G
Garming Sam 已提交
1968 1969 1970 1971
        if !self.resolved {
            self.label_ribs.pop();
            self.value_ribs.pop();
        }
1972 1973
    }

F
Felix S. Klock II 已提交
1974
    fn resolve_trait_reference(&mut self,
N
Nick Cameron 已提交
1975
                               id: NodeId,
1976
                               trait_path: &Path,
1977
                               path_depth: usize)
1978 1979
                               -> Result<PathResolution, ()> {
        if let Some(path_res) = self.resolve_path(id, trait_path, path_depth, TypeNS, true) {
1980
            if let Def::Trait(_) = path_res.base_def {
1981 1982 1983
                debug!("(resolving trait) found trait def: {:?}", path_res);
                Ok(path_res)
            } else {
N
Nick Cameron 已提交
1984 1985 1986 1987 1988
                let mut err =
                    resolve_struct_error(self,
                                  trait_path.span,
                                  ResolutionError::IsNotATrait(&*path_names_to_string(trait_path,
                                                                                      path_depth)));
1989 1990

                // If it's a typedef, give a note
1991
                if let Def::TyAlias(..) = path_res.base_def {
N
Nick Cameron 已提交
1992 1993
                    err.span_note(trait_path.span,
                                  "`type` aliases cannot be used for traits");
1994
                }
N
Nick Cameron 已提交
1995
                err.emit();
1996 1997
                Err(())
            }
1998
        } else {
1999 2000
            resolve_error(self,
                          trait_path.span,
C
corentih 已提交
2001 2002
                          ResolutionError::UndeclaredTraitName(&*path_names_to_string(trait_path,
                                                                                      path_depth)));
2003
            Err(())
2004 2005 2006
        }
    }

2007
    fn resolve_generics(&mut self, generics: &Generics) {
2008
        for type_parameter in generics.ty_params.iter() {
2009
            self.check_if_primitive_type_name(type_parameter.name, type_parameter.span);
2010 2011
        }
        for predicate in &generics.where_clause.predicates {
2012
            match predicate {
2013 2014 2015
                &hir::WherePredicate::BoundPredicate(_) |
                &hir::WherePredicate::RegionPredicate(_) => {}
                &hir::WherePredicate::EqPredicate(ref eq_pred) => {
2016
                    let path_res = self.resolve_path(eq_pred.id, &eq_pred.path, 0, TypeNS, true);
2017
                    if let Some(PathResolution { base_def: Def::TyParam(..), .. }) = path_res {
2018 2019
                        self.record_def(eq_pred.id, path_res.unwrap());
                    } else {
2020 2021
                        resolve_error(self,
                                      eq_pred.span,
2022
                                      ResolutionError::UndeclaredAssociatedType);
2023
                        self.record_def(eq_pred.id, err_path_resolution());
2024 2025
                    }
                }
2026 2027
            }
        }
2028
        intravisit::walk_generics(self, generics);
2029 2030
    }

2031 2032
    fn with_current_self_type<T, F>(&mut self, self_type: &Ty, f: F) -> T
        where F: FnOnce(&mut Resolver) -> T
J
Jorge Aparicio 已提交
2033
    {
2034 2035 2036 2037 2038 2039 2040
        // Handle nested impls (inside fn bodies)
        let previous_value = replace(&mut self.current_self_type, Some(self_type.clone()));
        let result = f(self);
        self.current_self_type = previous_value;
        result
    }

C
corentih 已提交
2041
    fn with_optional_trait_ref<T, F>(&mut self, opt_trait_ref: Option<&TraitRef>, f: F) -> T
2042
        where F: FnOnce(&mut Resolver, Option<DefId>) -> T
J
Jorge Aparicio 已提交
2043
    {
2044
        let mut new_val = None;
2045
        let mut new_id = None;
E
Eduard Burtescu 已提交
2046
        if let Some(trait_ref) = opt_trait_ref {
2047
            if let Ok(path_res) = self.resolve_trait_reference(trait_ref.ref_id,
C
corentih 已提交
2048 2049
                                                               &trait_ref.path,
                                                               0) {
2050 2051 2052 2053
                assert!(path_res.depth == 0);
                self.record_def(trait_ref.ref_id, path_res);
                new_val = Some((path_res.base_def.def_id(), trait_ref.clone()));
                new_id = Some(path_res.base_def.def_id());
2054 2055
            } else {
                self.record_def(trait_ref.ref_id, err_path_resolution());
2056
            }
2057
            intravisit::walk_trait_ref(self, trait_ref);
2058
        }
2059
        let original_trait_ref = replace(&mut self.current_trait_ref, new_val);
2060
        let result = f(self, new_id);
2061 2062 2063 2064
        self.current_trait_ref = original_trait_ref;
        result
    }

2065 2066 2067 2068 2069 2070 2071 2072 2073 2074
    fn with_self_rib<F>(&mut self, self_def: Def, f: F)
        where F: FnOnce(&mut Resolver)
    {
        let mut self_type_rib = Rib::new(NormalRibKind);

        // plain insert (no renaming, types are not currently hygienic....)
        let name = special_names::type_self;
        self_type_rib.bindings.insert(name, DlDef(self_def));
        self.type_ribs.push(self_type_rib);
        f(self);
G
Garming Sam 已提交
2075 2076 2077
        if !self.resolved {
            self.type_ribs.pop();
        }
2078 2079
    }

F
Felix S. Klock II 已提交
2080
    fn resolve_implementation(&mut self,
2081 2082 2083
                              generics: &Generics,
                              opt_trait_reference: &Option<TraitRef>,
                              self_type: &Ty,
2084
                              item_id: NodeId,
2085
                              impl_items: &[ImplItem]) {
2086
        // If applicable, create a rib for the type parameters.
2087
        self.with_type_parameter_rib(HasTypeParameters(generics,
2088
                                                       TypeSpace,
2089
                                                       ItemRibKind),
2090
                                     |this| {
2091
            // Resolve the type parameters.
2092
            this.visit_generics(generics);
2093

2094
            // Resolve the trait reference, if necessary.
2095
            this.with_optional_trait_ref(opt_trait_reference.as_ref(), |this, trait_id| {
2096
                // Resolve the self type.
2097
                this.visit_ty(self_type);
2098

2099
                this.with_self_rib(Def::SelfTy(trait_id, Some((item_id, self_type.id))), |this| {
2100 2101 2102
                    this.with_current_self_type(self_type, |this| {
                        for impl_item in impl_items {
                            match impl_item.node {
2103
                                hir::ImplItemKind::Const(..) => {
2104
                                    // If this is a trait impl, ensure the const
2105
                                    // exists in trait
V
Vadim Petrochenkov 已提交
2106
                                    this.check_trait_item(impl_item.name,
2107 2108
                                                          impl_item.span,
                                        |n, s| ResolutionError::ConstNotMemberOfTrait(n, s));
2109
                                    this.with_constant_rib(|this| {
2110
                                        intravisit::walk_impl_item(this, impl_item);
2111 2112
                                    });
                                }
2113
                                hir::ImplItemKind::Method(ref sig, _) => {
2114 2115
                                    // If this is a trait impl, ensure the method
                                    // exists in trait
V
Vadim Petrochenkov 已提交
2116
                                    this.check_trait_item(impl_item.name,
2117 2118
                                                          impl_item.span,
                                        |n, s| ResolutionError::MethodNotMemberOfTrait(n, s));
2119 2120 2121 2122 2123 2124 2125 2126

                                    // We also need a new scope for the method-
                                    // specific type parameters.
                                    let type_parameters =
                                        HasTypeParameters(&sig.generics,
                                                          FnSpace,
                                                          MethodRibKind);
                                    this.with_type_parameter_rib(type_parameters, |this| {
2127
                                        intravisit::walk_impl_item(this, impl_item);
2128 2129
                                    });
                                }
2130
                                hir::ImplItemKind::Type(ref ty) => {
2131
                                    // If this is a trait impl, ensure the type
2132
                                    // exists in trait
V
Vadim Petrochenkov 已提交
2133
                                    this.check_trait_item(impl_item.name,
2134 2135
                                                          impl_item.span,
                                        |n, s| ResolutionError::TypeNotMemberOfTrait(n, s));
2136

2137 2138
                                    this.visit_ty(ty);
                                }
2139
                            }
2140
                        }
2141
                    });
2142 2143
                });
            });
2144
        });
2145 2146
    }

2147
    fn check_trait_item<F>(&self, name: Name, span: Span, err: F)
C
corentih 已提交
2148 2149 2150 2151
        where F: FnOnce(Name, &str) -> ResolutionError
    {
        // If there is a TraitRef in scope for an impl, then the method must be in the
        // trait.
2152
        if let Some((did, ref trait_ref)) = self.current_trait_ref {
2153
            if !self.trait_item_map.contains_key(&(name, did)) {
2154
                let path_str = path_names_to_string(&trait_ref.path, 0);
C
corentih 已提交
2155
                resolve_error(self, span, err(name, &*path_str));
2156 2157 2158 2159
            }
        }
    }

E
Eduard Burtescu 已提交
2160
    fn resolve_local(&mut self, local: &Local) {
2161
        // Resolve the type.
2162
        walk_list!(self, visit_ty, &local.ty);
2163

2164
        // Resolve the initializer.
2165
        walk_list!(self, visit_expr, &local.init);
2166 2167

        // Resolve the pattern.
C
corentih 已提交
2168
        self.resolve_pattern(&*local.pat, LocalIrrefutableMode, &mut HashMap::new());
2169 2170
    }

J
John Clements 已提交
2171 2172 2173 2174
    // build a map from pattern identifiers to binding-info's.
    // this is done hygienically. This could arise for a macro
    // that expands into an or-pattern where one 'x' was from the
    // user and one 'x' came from the macro.
E
Eduard Burtescu 已提交
2175
    fn binding_mode_map(&mut self, pat: &Pat) -> BindingMap {
2176
        let mut result = HashMap::new();
2177 2178
        pat_bindings(&self.def_map, pat, |binding_mode, _id, sp, path1| {
            let name = path1.node;
C
corentih 已提交
2179 2180 2181 2182 2183
            result.insert(name,
                          BindingInfo {
                              span: sp,
                              binding_mode: binding_mode,
                          });
2184
        });
2185
        return result;
2186 2187
    }

J
John Clements 已提交
2188 2189
    // check that all of the arms in an or-pattern have exactly the
    // same set of bindings, with the same binding modes for each.
F
Felix S. Klock II 已提交
2190
    fn check_consistent_bindings(&mut self, arm: &Arm) {
2191
        if arm.pats.is_empty() {
C
corentih 已提交
2192
            return;
2193
        }
2194
        let map_0 = self.binding_mode_map(&*arm.pats[0]);
D
Daniel Micay 已提交
2195
        for (i, p) in arm.pats.iter().enumerate() {
2196
            let map_i = self.binding_mode_map(&**p);
2197

2198
            for (&key, &binding_0) in &map_0 {
2199
                match map_i.get(&key) {
C
corentih 已提交
2200
                    None => {
2201
                        resolve_error(self,
C
corentih 已提交
2202 2203 2204 2205 2206 2207 2208 2209 2210 2211
                                      p.span,
                                      ResolutionError::VariableNotBoundInPattern(key, i + 1));
                    }
                    Some(binding_i) => {
                        if binding_0.binding_mode != binding_i.binding_mode {
                            resolve_error(self,
                                          binding_i.span,
                                          ResolutionError::VariableBoundWithDifferentMode(key,
                                                                                          i + 1));
                        }
2212
                    }
2213 2214 2215
                }
            }

2216
            for (&key, &binding) in &map_i {
2217
                if !map_0.contains_key(&key) {
2218 2219
                    resolve_error(self,
                                  binding.span,
C
corentih 已提交
2220
                                  ResolutionError::VariableNotBoundInParentPattern(key, i + 1));
2221 2222 2223
                }
            }
        }
2224 2225
    }

F
Felix S. Klock II 已提交
2226
    fn resolve_arm(&mut self, arm: &Arm) {
2227
        self.value_ribs.push(Rib::new(NormalRibKind));
2228

2229
        let mut bindings_list = HashMap::new();
2230
        for pattern in &arm.pats {
2231
            self.resolve_pattern(&**pattern, RefutableMode, &mut bindings_list);
2232 2233
        }

2234 2235 2236 2237
        // This has to happen *after* we determine which
        // pat_idents are variants
        self.check_consistent_bindings(arm);

2238
        walk_list!(self, visit_expr, &arm.guard);
2239
        self.visit_expr(&*arm.body);
2240

G
Garming Sam 已提交
2241 2242 2243
        if !self.resolved {
            self.value_ribs.pop();
        }
2244 2245
    }

E
Eduard Burtescu 已提交
2246
    fn resolve_block(&mut self, block: &Block) {
2247
        debug!("(resolving block) entering block");
2248
        // Move down in the graph, if there's an anonymous module rooted here.
2249
        let orig_module = self.current_module;
2250 2251 2252 2253 2254 2255 2256 2257 2258 2259
        let anonymous_module =
            orig_module.anonymous_children.borrow().get(&block.id).map(|module| *module);

        if let Some(anonymous_module) = anonymous_module {
            debug!("(resolving block) found anonymous module, moving down");
            self.value_ribs.push(Rib::new(AnonymousModuleRibKind(anonymous_module)));
            self.type_ribs.push(Rib::new(AnonymousModuleRibKind(anonymous_module)));
            self.current_module = anonymous_module;
        } else {
            self.value_ribs.push(Rib::new(NormalRibKind));
2260 2261 2262
        }

        // Descend into the block.
2263
        intravisit::walk_block(self, block);
2264 2265

        // Move back up.
G
Garming Sam 已提交
2266
        if !self.resolved {
2267
            self.current_module = orig_module;
G
Garming Sam 已提交
2268
            self.value_ribs.pop();
2269 2270 2271
            if let Some(_) = anonymous_module {
                self.type_ribs.pop();
            }
G
Garming Sam 已提交
2272
        }
2273
        debug!("(resolving block) leaving block");
2274 2275
    }

F
Felix S. Klock II 已提交
2276
    fn resolve_type(&mut self, ty: &Ty) {
2277
        match ty.node {
2278
            TyPath(ref maybe_qself, ref path) => {
C
corentih 已提交
2279 2280 2281 2282 2283 2284 2285 2286
                let resolution = match self.resolve_possibly_assoc_item(ty.id,
                                                                        maybe_qself.as_ref(),
                                                                        path,
                                                                        TypeNS,
                                                                        true) {
                    // `<T>::a::b::c` is resolved by typeck alone.
                    TypecheckRequired => {
                        // Resolve embedded types.
2287
                        intravisit::walk_ty(self, ty);
C
corentih 已提交
2288 2289 2290 2291
                        return;
                    }
                    ResolveAttempt(resolution) => resolution,
                };
2292 2293

                // This is a path in the type namespace. Walk through scopes
2294
                // looking for it.
2295
                match resolution {
B
Brian Anderson 已提交
2296
                    Some(def) => {
2297
                        // Write the result into the def map.
C
corentih 已提交
2298
                        debug!("(resolving type) writing resolution for `{}` (id {}) = {:?}",
2299
                               path_names_to_string(path, 0),
C
corentih 已提交
2300 2301
                               ty.id,
                               def);
2302
                        self.record_def(ty.id, def);
2303
                    }
B
Brian Anderson 已提交
2304
                    None => {
2305 2306
                        self.record_def(ty.id, err_path_resolution());

2307 2308 2309
                        // Keep reporting some errors even if they're ignored above.
                        self.resolve_path(ty.id, path, 0, TypeNS, true);

2310 2311 2312 2313
                        let kind = if maybe_qself.is_some() {
                            "associated type"
                        } else {
                            "type name"
2314
                        };
2315

2316
                        let self_type_name = special_idents::type_self.name;
C
corentih 已提交
2317 2318 2319 2320
                        let is_invalid_self_type_name = path.segments.len() > 0 &&
                                                        maybe_qself.is_none() &&
                                                        path.segments[0].identifier.name ==
                                                        self_type_name;
G
Guillaume Gomez 已提交
2321
                        if is_invalid_self_type_name {
2322 2323
                            resolve_error(self,
                                          ty.span,
2324
                                          ResolutionError::SelfUsedOutsideImplOrTrait);
2325
                        } else {
2326 2327
                            resolve_error(self,
                                          ty.span,
2328
                                          ResolutionError::UseOfUndeclared(
2329 2330 2331 2332
                                                                    kind,
                                                                    &*path_names_to_string(path,
                                                                                           0))
                                         );
G
Guillaume Gomez 已提交
2333
                        }
2334 2335
                    }
                }
2336
            }
2337
            _ => {}
2338
        }
2339
        // Resolve embedded types.
2340
        intravisit::walk_ty(self, ty);
2341 2342
    }

F
Felix S. Klock II 已提交
2343
    fn resolve_pattern(&mut self,
E
Eduard Burtescu 已提交
2344
                       pattern: &Pat,
2345 2346 2347
                       mode: PatternBindingMode,
                       // Maps idents to the node ID for the (outermost)
                       // pattern that binds them
2348
                       bindings_list: &mut HashMap<Name, NodeId>) {
2349
        let pat_id = pattern.id;
2350
        walk_pat(pattern, |pattern| {
2351
            match pattern.node {
2352 2353
                PatIdent(binding_mode, ref path1, ref at_rhs) => {
                    // The meaning of PatIdent with no type parameters
2354 2355 2356 2357
                    // depends on whether an enum variant or unit-like struct
                    // with that name is in scope. The probing lookup has to
                    // be careful not to emit spurious errors. Only matching
                    // patterns (match) can match nullary variants or
2358 2359 2360 2361
                    // unit-like structs. For binding patterns (let
                    // and the LHS of @-patterns), matching such a value is
                    // simply disallowed (since it's rarely what you want).
                    let const_ok = mode == RefutableMode && at_rhs.is_none();
2362

2363
                    let ident = path1.node;
2364
                    let renamed = ident.name;
2365

2366 2367
                    match self.resolve_bare_identifier_pattern(ident.unhygienic_name,
                                                               pattern.span) {
2368
                        FoundStructOrEnumVariant(def, lp) if const_ok => {
C
corentih 已提交
2369
                            debug!("(resolving pattern) resolving `{}` to struct or enum variant",
2370
                                   renamed);
2371

C
corentih 已提交
2372 2373 2374 2375 2376 2377 2378 2379 2380
                            self.enforce_default_binding_mode(pattern,
                                                              binding_mode,
                                                              "an enum variant");
                            self.record_def(pattern.id,
                                            PathResolution {
                                                base_def: def,
                                                last_private: lp,
                                                depth: 0,
                                            });
2381
                        }
A
Alex Crichton 已提交
2382
                        FoundStructOrEnumVariant(..) => {
2383
                            resolve_error(
2384
                                self,
2385
                                pattern.span,
2386
                                ResolutionError::DeclarationShadowsEnumVariantOrUnitLikeStruct(
2387 2388
                                    renamed)
                            );
2389
                            self.record_def(pattern.id, err_path_resolution());
2390
                        }
2391
                        FoundConst(def, lp, _) if const_ok => {
C
corentih 已提交
2392 2393 2394 2395 2396 2397 2398 2399 2400
                            debug!("(resolving pattern) resolving `{}` to constant", renamed);

                            self.enforce_default_binding_mode(pattern, binding_mode, "a constant");
                            self.record_def(pattern.id,
                                            PathResolution {
                                                base_def: def,
                                                last_private: lp,
                                                depth: 0,
                                            });
2401
                        }
2402
                        FoundConst(def, _, name) => {
2403
                            resolve_error(
2404 2405
                                self,
                                pattern.span,
M
Manish Goregaokar 已提交
2406 2407
                                ResolutionError::OnlyIrrefutablePatternsAllowedHere(def.def_id(),
                                                                                    name)
2408
                            );
2409
                            self.record_def(pattern.id, err_path_resolution());
2410
                        }
2411
                        BareIdentifierPatternUnresolved => {
C
corentih 已提交
2412
                            debug!("(resolving pattern) binding `{}`", renamed);
2413

2414
                            let def_id = self.ast_map.local_def_id(pattern.id);
2415
                            let def = Def::Local(def_id, pattern.id);
2416 2417 2418 2419 2420

                            // Record the definition so that later passes
                            // will be able to distinguish variants from
                            // locals in patterns.

C
corentih 已提交
2421 2422 2423 2424 2425 2426
                            self.record_def(pattern.id,
                                            PathResolution {
                                                base_def: def,
                                                last_private: LastMod(AllPublic),
                                                depth: 0,
                                            });
2427 2428 2429 2430 2431 2432

                            // Add the binding to the local ribs, if it
                            // doesn't already exist in the bindings list. (We
                            // must not add it if it's in the bindings list
                            // because that breaks the assumptions later
                            // passes make about or-patterns.)
2433 2434
                            if !bindings_list.contains_key(&renamed) {
                                let this = &mut *self;
2435 2436
                                let last_rib = this.value_ribs.last_mut().unwrap();
                                last_rib.bindings.insert(renamed, DlDef(def));
2437
                                bindings_list.insert(renamed, pat_id);
2438
                            } else if mode == ArgumentIrrefutableMode &&
C
corentih 已提交
2439
                               bindings_list.contains_key(&renamed) {
2440 2441
                                // Forbid duplicate bindings in the same
                                // parameter list.
2442
                                resolve_error(
2443 2444
                                    self,
                                    pattern.span,
2445
                                    ResolutionError::IdentifierBoundMoreThanOnceInParameterList(
2446
                                        &ident.name.as_str())
2447
                                );
C
corentih 已提交
2448
                            } else if bindings_list.get(&renamed) == Some(&pat_id) {
2449 2450
                                // Then this is a duplicate variable in the
                                // same disjunction, which is an error.
2451
                                resolve_error(
2452 2453
                                    self,
                                    pattern.span,
2454
                                    ResolutionError::IdentifierBoundMoreThanOnceInSamePattern(
2455
                                        &ident.name.as_str())
2456
                                );
2457
                            }
2458 2459
                            // Else, not bound in the same pattern: do
                            // nothing.
2460 2461 2462 2463
                        }
                    }
                }

2464
                PatEnum(ref path, _) => {
2465
                    // This must be an enum variant, struct or const.
C
corentih 已提交
2466 2467 2468 2469 2470 2471 2472 2473 2474 2475 2476 2477 2478 2479 2480 2481 2482 2483
                    let resolution = match self.resolve_possibly_assoc_item(pat_id,
                                                                            None,
                                                                            path,
                                                                            ValueNS,
                                                                            false) {
                        // The below shouldn't happen because all
                        // qualified paths should be in PatQPath.
                        TypecheckRequired =>
                            self.session.span_bug(path.span,
                                                  "resolve_possibly_assoc_item claimed
                                     \
                                                   that a path in PatEnum requires typecheck
                                     \
                                                   to resolve, but qualified paths should be
                                     \
                                                   PatQPath"),
                        ResolveAttempt(resolution) => resolution,
                    };
2484
                    if let Some(path_res) = resolution {
2485
                        match path_res.base_def {
2486
                            Def::Struct(..) if path_res.depth == 0 => {
2487 2488
                                self.record_def(pattern.id, path_res);
                            }
2489
                            Def::Variant(..) | Def::Const(..) => {
2490 2491
                                self.record_def(pattern.id, path_res);
                            }
2492
                            Def::Static(..) => {
2493 2494
                                resolve_error(&self,
                                              path.span,
2495
                                              ResolutionError::StaticVariableReference);
2496
                                self.record_def(pattern.id, err_path_resolution());
2497
                            }
2498 2499 2500 2501
                            _ => {
                                // If anything ends up here entirely resolved,
                                // it's an error. If anything ends up here
                                // partially resolved, that's OK, because it may
2502
                                // be a `T::CONST` that typeck will resolve.
2503
                                if path_res.depth == 0 {
2504
                                    resolve_error(
2505
                                        self,
2506
                                        path.span,
2507
                                        ResolutionError::NotAnEnumVariantStructOrConst(
2508 2509 2510 2511 2512 2513
                                            &path.segments
                                                 .last()
                                                 .unwrap()
                                                 .identifier
                                                 .name
                                                 .as_str())
2514
                                    );
2515
                                    self.record_def(pattern.id, err_path_resolution());
2516
                                } else {
C
corentih 已提交
2517 2518 2519 2520 2521
                                    let const_name = path.segments
                                                         .last()
                                                         .unwrap()
                                                         .identifier
                                                         .name;
2522 2523
                                    let traits = self.get_traits_containing_item(const_name);
                                    self.trait_map.insert(pattern.id, traits);
2524 2525 2526 2527 2528
                                    self.record_def(pattern.id, path_res);
                                }
                            }
                        }
                    } else {
2529
                        resolve_error(
2530 2531
                            self,
                            path.span,
2532
                            ResolutionError::UnresolvedEnumVariantStructOrConst(
2533
                                &path.segments.last().unwrap().identifier.name.as_str())
2534
                        );
2535
                        self.record_def(pattern.id, err_path_resolution());
2536
                    }
2537
                    intravisit::walk_path(self, path);
2538 2539 2540 2541
                }

                PatQPath(ref qself, ref path) => {
                    // Associated constants only.
C
corentih 已提交
2542 2543 2544 2545 2546 2547 2548 2549 2550 2551 2552 2553 2554 2555 2556 2557
                    let resolution = match self.resolve_possibly_assoc_item(pat_id,
                                                                            Some(qself),
                                                                            path,
                                                                            ValueNS,
                                                                            false) {
                        TypecheckRequired => {
                            // All `<T>::CONST` should end up here, and will
                            // require use of the trait map to resolve
                            // during typechecking.
                            let const_name = path.segments
                                                 .last()
                                                 .unwrap()
                                                 .identifier
                                                 .name;
                            let traits = self.get_traits_containing_item(const_name);
                            self.trait_map.insert(pattern.id, traits);
2558
                            intravisit::walk_pat(self, pattern);
C
corentih 已提交
2559 2560 2561 2562
                            return true;
                        }
                        ResolveAttempt(resolution) => resolution,
                    };
2563 2564 2565 2566
                    if let Some(path_res) = resolution {
                        match path_res.base_def {
                            // All `<T as Trait>::CONST` should end up here, and
                            // have the trait already selected.
2567
                            Def::AssociatedConst(..) => {
2568 2569
                                self.record_def(pattern.id, path_res);
                            }
2570
                            _ => {
2571
                                resolve_error(
2572 2573
                                    self,
                                    path.span,
2574
                                    ResolutionError::NotAnAssociatedConst(
2575
                                        &path.segments.last().unwrap().identifier.name.as_str()
2576 2577
                                    )
                                );
2578
                                self.record_def(pattern.id, err_path_resolution());
2579
                            }
2580
                        }
2581
                    } else {
C
corentih 已提交
2582 2583 2584 2585 2586 2587 2588 2589
                        resolve_error(self,
                                      path.span,
                                      ResolutionError::UnresolvedAssociatedConst(&path.segments
                                                                                      .last()
                                                                                      .unwrap()
                                                                                      .identifier
                                                                                      .name
                                                                                      .as_str()));
2590
                        self.record_def(pattern.id, err_path_resolution());
2591
                    }
2592
                    intravisit::walk_pat(self, pattern);
2593 2594
                }

2595
                PatStruct(ref path, _, _) => {
2596
                    match self.resolve_path(pat_id, path, 0, TypeNS, false) {
2597
                        Some(definition) => {
2598 2599
                            self.record_def(pattern.id, definition);
                        }
2600
                        result => {
C
corentih 已提交
2601
                            debug!("(resolving pattern) didn't find struct def: {:?}", result);
2602 2603 2604
                            resolve_error(
                                self,
                                path.span,
2605
                                ResolutionError::DoesNotNameAStruct(
2606 2607
                                    &*path_names_to_string(path, 0))
                            );
2608
                            self.record_def(pattern.id, err_path_resolution());
2609 2610
                        }
                    }
2611
                    intravisit::walk_path(self, path);
2612 2613 2614
                }

                PatLit(_) | PatRange(..) => {
2615
                    intravisit::walk_pat(self, pattern);
2616 2617
                }

2618
                _ => {
2619 2620 2621
                    // Nothing to do.
                }
            }
2622
            true
2623
        });
2624 2625
    }

C
corentih 已提交
2626 2627 2628
    fn resolve_bare_identifier_pattern(&mut self,
                                       name: Name,
                                       span: Span)
E
Eduard Burtescu 已提交
2629
                                       -> BareIdentifierPatternResolution {
2630
        let module = self.current_module;
2631
        match self.resolve_item_in_lexical_scope(module, name, ValueNS, true) {
2632
            Success(binding) => {
C
corentih 已提交
2633 2634
                debug!("(resolve bare identifier pattern) succeeded in finding {} at {:?}",
                       name,
2635
                       binding);
J
Jeffrey Seyfried 已提交
2636
                match binding.def() {
B
Brian Anderson 已提交
2637
                    None => {
C
corentih 已提交
2638 2639
                        panic!("resolved name in the value namespace to a set of name bindings \
                                with no def?!");
2640
                    }
2641 2642 2643
                    // For the two success cases, this lookup can be
                    // considered as not having a private component because
                    // the lookup happened only within the current module.
2644
                    Some(def @ Def::Variant(..)) | Some(def @ Def::Struct(..)) => {
2645 2646
                        return FoundStructOrEnumVariant(def, LastMod(AllPublic));
                    }
2647
                    Some(def @ Def::Const(..)) | Some(def @ Def::AssociatedConst(..)) => {
2648
                        return FoundConst(def, LastMod(AllPublic), name);
2649
                    }
2650
                    Some(Def::Static(..)) => {
2651 2652
                        resolve_error(self, span, ResolutionError::StaticVariableReference);
                        return BareIdentifierPatternUnresolved;
2653
                    }
2654
                    _ => return BareIdentifierPatternUnresolved
2655 2656 2657
                }
            }

2658
            Indeterminate => return BareIdentifierPatternUnresolved,
2659 2660 2661
            Failed(err) => {
                match err {
                    Some((span, msg)) => {
2662
                        resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
2663
                    }
C
corentih 已提交
2664
                    None => (),
2665
                }
2666

C
corentih 已提交
2667
                debug!("(resolve bare identifier pattern) failed to find {}", name);
2668
                return BareIdentifierPatternUnresolved;
2669 2670 2671 2672
            }
        }
    }

2673 2674 2675
    /// Handles paths that may refer to associated items
    fn resolve_possibly_assoc_item(&mut self,
                                   id: NodeId,
2676
                                   maybe_qself: Option<&hir::QSelf>,
2677 2678 2679
                                   path: &Path,
                                   namespace: Namespace,
                                   check_ribs: bool)
C
corentih 已提交
2680
                                   -> AssocItemResolveResult {
2681 2682
        let max_assoc_types;

2683
        match maybe_qself {
2684 2685 2686 2687 2688 2689 2690 2691 2692 2693 2694
            Some(qself) => {
                if qself.position == 0 {
                    return TypecheckRequired;
                }
                max_assoc_types = path.segments.len() - qself.position;
                // Make sure the trait is valid.
                let _ = self.resolve_trait_reference(id, path, max_assoc_types);
            }
            None => {
                max_assoc_types = path.segments.len();
            }
2695 2696 2697 2698 2699 2700 2701 2702 2703 2704
        }

        let mut resolution = self.with_no_errors(|this| {
            this.resolve_path(id, path, 0, namespace, check_ribs)
        });
        for depth in 1..max_assoc_types {
            if resolution.is_some() {
                break;
            }
            self.with_no_errors(|this| {
C
corentih 已提交
2705
                resolution = this.resolve_path(id, path, depth, TypeNS, true);
2706 2707
            });
        }
2708
        if let Some(Def::Mod(_)) = resolution.map(|r| r.base_def) {
2709 2710 2711 2712 2713 2714
            // A module is not a valid type or value.
            resolution = None;
        }
        ResolveAttempt(resolution)
    }

2715 2716
    /// If `check_ribs` is true, checks the local definitions first; i.e.
    /// doesn't skip straight to the containing module.
2717 2718
    /// Skips `path_depth` trailing segments, which is also reflected in the
    /// returned value. See `middle::def::PathResolution` for more info.
G
Garming Sam 已提交
2719 2720 2721 2722 2723
    pub fn resolve_path(&mut self,
                        id: NodeId,
                        path: &Path,
                        path_depth: usize,
                        namespace: Namespace,
C
corentih 已提交
2724 2725
                        check_ribs: bool)
                        -> Option<PathResolution> {
2726
        let span = path.span;
C
corentih 已提交
2727
        let segments = &path.segments[..path.segments.len() - path_depth];
2728

N
Nick Cameron 已提交
2729
        let mk_res = |(def, lp)| PathResolution::new(def, lp, path_depth);
2730

2731
        if path.global {
2732
            let def = self.resolve_crate_relative_path(span, segments, namespace);
2733
            return def.map(mk_res);
2734 2735
        }

2736
        // Try to find a path to an item in a module.
2737
        let last_ident = segments.last().unwrap().identifier;
N
Nick Cameron 已提交
2738
        if segments.len() <= 1 {
2739
            let unqualified_def = self.resolve_identifier(last_ident, namespace, check_ribs, true);
C
corentih 已提交
2740 2741 2742 2743
            return unqualified_def.and_then(|def| self.adjust_local_def(def, span))
                                  .map(|def| {
                                      PathResolution::new(def, LastMod(AllPublic), path_depth)
                                  });
N
Nick Cameron 已提交
2744
        }
2745

2746
        let unqualified_def = self.resolve_identifier(last_ident, namespace, check_ribs, false);
N
Nick Cameron 已提交
2747 2748
        let def = self.resolve_module_relative_path(span, segments, namespace);
        match (def, unqualified_def) {
2749
            (Some((ref d, _)), Some(ref ud)) if *d == ud.def => {
N
Nick Cameron 已提交
2750 2751
                self.session
                    .add_lint(lint::builtin::UNUSED_QUALIFICATIONS,
C
corentih 已提交
2752 2753
                              id,
                              span,
N
Nick Cameron 已提交
2754 2755 2756
                              "unnecessary qualification".to_string());
            }
            _ => {}
2757
        }
N
Nick Cameron 已提交
2758 2759

        def.map(mk_res)
2760 2761
    }

2762
    // Resolve a single identifier
F
Felix S. Klock II 已提交
2763
    fn resolve_identifier(&mut self,
2764
                          identifier: hir::Ident,
2765
                          namespace: Namespace,
2766 2767
                          check_ribs: bool,
                          record_used: bool)
2768
                          -> Option<LocalDef> {
2769
        if identifier.name == special_idents::invalid.name {
2770
            return Some(LocalDef::from_def(Def::Err));
2771 2772
        }

2773 2774 2775 2776
        // First, check to see whether the name is a primitive type.
        if namespace == TypeNS {
            if let Some(&prim_ty) = self.primitive_type_table
                                        .primitive_types
2777
                                        .get(&identifier.unhygienic_name) {
2778
                return Some(LocalDef::from_def(Def::PrimTy(prim_ty)));
2779 2780 2781
            }
        }

2782
        if check_ribs {
C
corentih 已提交
2783
            if let Some(def) = self.resolve_identifier_in_local_ribs(identifier, namespace) {
2784
                return Some(def);
2785 2786 2787
            }
        }

2788 2789
        // Check the items.
        let module = self.current_module;
2790
        let name = identifier.unhygienic_name;
2791
        match self.resolve_item_in_lexical_scope(module, name, namespace, record_used) {
2792
            Success(binding) => binding.def().map(LocalDef::from_def),
2793 2794 2795 2796 2797 2798
            Failed(Some((span, msg))) => {
                resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
                None
            }
            _ => None,
        }
2799 2800 2801
    }

    // Resolve a local definition, potentially adjusting for closures.
2802
    fn adjust_local_def(&mut self, local_def: LocalDef, span: Span) -> Option<Def> {
2803
        let ribs = match local_def.ribs {
C
corentih 已提交
2804 2805 2806
            Some((TypeNS, i)) => &self.type_ribs[i + 1..],
            Some((ValueNS, i)) => &self.value_ribs[i + 1..],
            _ => &[] as &[_],
2807 2808 2809
        };
        let mut def = local_def.def;
        match def {
2810
            Def::Upvar(..) => {
C
corentih 已提交
2811
                self.session.span_bug(span, &format!("unexpected {:?} in bindings", def))
2812
            }
2813
            Def::Local(_, node_id) => {
2814 2815
                for rib in ribs {
                    match rib.kind {
2816
                        NormalRibKind | AnonymousModuleRibKind(..) => {
2817 2818 2819 2820 2821 2822
                            // Nothing to do. Continue.
                        }
                        ClosureRibKind(function_id) => {
                            let prev_def = def;
                            let node_def_id = self.ast_map.local_def_id(node_id);

C
corentih 已提交
2823 2824 2825
                            let seen = self.freevars_seen
                                           .entry(function_id)
                                           .or_insert_with(|| NodeMap());
2826
                            if let Some(&index) = seen.get(&node_id) {
2827
                                def = Def::Upvar(node_def_id, node_id, index, function_id);
2828 2829
                                continue;
                            }
C
corentih 已提交
2830 2831 2832
                            let vec = self.freevars
                                          .entry(function_id)
                                          .or_insert_with(|| vec![]);
2833
                            let depth = vec.len();
C
corentih 已提交
2834 2835 2836 2837
                            vec.push(Freevar {
                                def: prev_def,
                                span: span,
                            });
2838

2839
                            def = Def::Upvar(node_def_id, node_id, depth, function_id);
2840 2841 2842 2843 2844 2845
                            seen.insert(node_id, depth);
                        }
                        ItemRibKind | MethodRibKind => {
                            // This was an attempt to access an upvar inside a
                            // named function item. This is not allowed, so we
                            // report an error.
C
corentih 已提交
2846 2847 2848
                            resolve_error(self,
                                          span,
                                          ResolutionError::CannotCaptureDynamicEnvironmentInFnItem);
2849 2850 2851 2852
                            return None;
                        }
                        ConstantItemRibKind => {
                            // Still doesn't deal with upvars
C
corentih 已提交
2853 2854 2855
                            resolve_error(self,
                                          span,
                                          ResolutionError::AttemptToUseNonConstantValueInConstant);
2856 2857 2858 2859 2860
                            return None;
                        }
                    }
                }
            }
2861
            Def::TyParam(..) | Def::SelfTy(..) => {
2862 2863
                for rib in ribs {
                    match rib.kind {
2864 2865
                        NormalRibKind | MethodRibKind | ClosureRibKind(..) |
                        AnonymousModuleRibKind(..) => {
2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885 2886 2887
                            // Nothing to do. Continue.
                        }
                        ItemRibKind => {
                            // This was an attempt to use a type parameter outside
                            // its scope.

                            resolve_error(self,
                                          span,
                                          ResolutionError::TypeParametersFromOuterFunction);
                            return None;
                        }
                        ConstantItemRibKind => {
                            // see #9186
                            resolve_error(self, span, ResolutionError::OuterTypeParameterContext);
                            return None;
                        }
                    }
                }
            }
            _ => {}
        }
        return Some(def);
2888 2889
    }

2890
    // resolve a "module-relative" path, e.g. a::b::c
F
Felix S. Klock II 已提交
2891
    fn resolve_module_relative_path(&mut self,
2892
                                    span: Span,
2893
                                    segments: &[hir::PathSegment],
2894 2895
                                    namespace: Namespace)
                                    -> Option<(Def, LastPrivate)> {
C
corentih 已提交
2896 2897 2898 2899 2900 2901
        let module_path = segments.split_last()
                                  .unwrap()
                                  .1
                                  .iter()
                                  .map(|ps| ps.identifier.name)
                                  .collect::<Vec<_>>();
2902

2903
        let containing_module;
2904
        let last_private;
2905
        let current_module = self.current_module;
J
Jeffrey Seyfried 已提交
2906
        match self.resolve_module_path(current_module, &module_path, UseLexicalScope, span) {
2907 2908 2909 2910
            Failed(err) => {
                let (span, msg) = match err {
                    Some((span, msg)) => (span, msg),
                    None => {
2911
                        let msg = format!("Use of undeclared type or module `{}`",
2912
                                          names_to_string(&module_path));
2913
                        (span, msg)
2914 2915
                    }
                };
2916

2917
                resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
2918
                return None;
2919
            }
2920
            Indeterminate => return None,
2921
            Success((resulting_module, resulting_last_private)) => {
2922
                containing_module = resulting_module;
2923
                last_private = resulting_last_private;
2924 2925 2926
            }
        }

2927
        let name = segments.last().unwrap().identifier.name;
2928 2929
        let result = self.resolve_name_in_module(containing_module, name, namespace, false, true);
        let def = match result {
2930
            Success(binding) => {
2931
                let (def, lp) = binding.def_and_lp();
2932
                (def, last_private.or(lp))
2933
            }
2934
            _ => return None,
2935 2936
        };
        return Some(def);
2937 2938
    }

2939 2940
    /// Invariant: This must be called only during main resolution, not during
    /// import resolution.
F
Felix S. Klock II 已提交
2941
    fn resolve_crate_relative_path(&mut self,
2942
                                   span: Span,
2943
                                   segments: &[hir::PathSegment],
2944
                                   namespace: Namespace)
C
corentih 已提交
2945 2946 2947 2948 2949 2950 2951
                                   -> Option<(Def, LastPrivate)> {
        let module_path = segments.split_last()
                                  .unwrap()
                                  .1
                                  .iter()
                                  .map(|ps| ps.identifier.name)
                                  .collect::<Vec<_>>();
2952

2953
        let root_module = self.graph_root;
2954

2955
        let containing_module;
2956
        let last_private;
2957
        match self.resolve_module_path_from_root(root_module,
2958
                                                 &module_path,
2959
                                                 0,
2960
                                                 span,
2961
                                                 LastMod(AllPublic)) {
2962 2963 2964 2965 2966
            Failed(err) => {
                let (span, msg) = match err {
                    Some((span, msg)) => (span, msg),
                    None => {
                        let msg = format!("Use of undeclared module `::{}`",
2967
                                          names_to_string(&module_path));
2968
                        (span, msg)
2969 2970 2971
                    }
                };

2972
                resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
B
Brian Anderson 已提交
2973
                return None;
2974 2975
            }

2976
            Indeterminate => return None,
2977

2978
            Success((resulting_module, resulting_last_private)) => {
2979
                containing_module = resulting_module;
2980
                last_private = resulting_last_private;
2981 2982 2983
            }
        }

2984
        let name = segments.last().unwrap().identifier.name;
2985
        match self.resolve_name_in_module(containing_module, name, namespace, false, true) {
2986
            Success(binding) => {
2987 2988
                let (def, lp) = binding.def_and_lp();
                Some((def, last_private.or(lp)))
2989
            }
2990
            _ => None,
2991 2992 2993
        }
    }

F
Felix S. Klock II 已提交
2994
    fn resolve_identifier_in_local_ribs(&mut self,
2995
                                        ident: hir::Ident,
2996 2997
                                        namespace: Namespace)
                                        -> Option<LocalDef> {
2998
        // Check the local set of ribs.
2999
        let name = match namespace { ValueNS => ident.name, TypeNS => ident.unhygienic_name };
3000

3001 3002
        for i in (0 .. self.get_ribs(namespace).len()).rev() {
            if let Some(def_like) = self.get_ribs(namespace)[i].bindings.get(&name).cloned() {
3003 3004 3005
                match def_like {
                    DlDef(def) => {
                        debug!("(resolving path in local ribs) resolved `{}` to {:?} at {}",
C
corentih 已提交
3006 3007 3008
                               name,
                               def,
                               i);
3009 3010
                        return Some(LocalDef {
                            ribs: Some((namespace, i)),
C
corentih 已提交
3011
                            def: def,
3012 3013 3014 3015
                        });
                    }
                    def_like => {
                        debug!("(resolving path in local ribs) resolved `{}` to pseudo-def {:?}",
C
corentih 已提交
3016 3017
                               name,
                               def_like);
3018 3019 3020
                        return None;
                    }
                }
3021
            }
3022 3023

            if let AnonymousModuleRibKind(module) = self.get_ribs(namespace)[i].kind {
3024 3025 3026 3027 3028
                if let Success(binding) = self.resolve_name_in_module(module,
                                                                      ident.unhygienic_name,
                                                                      namespace,
                                                                      true,
                                                                      true) {
J
Jeffrey Seyfried 已提交
3029
                    if let Some(def) = binding.def() {
3030 3031 3032 3033
                        return Some(LocalDef::from_def(def));
                    }
                }
            }
3034
        }
3035 3036

        None
3037 3038
    }

C
corentih 已提交
3039 3040
    fn with_no_errors<T, F>(&mut self, f: F) -> T
        where F: FnOnce(&mut Resolver) -> T
J
Jorge Aparicio 已提交
3041
    {
3042
        self.emit_errors = false;
A
Alex Crichton 已提交
3043
        let rs = f(self);
3044 3045 3046 3047
        self.emit_errors = true;
        rs
    }

3048
    fn find_fallback_in_self_type(&mut self, name: Name) -> FallbackSuggestion {
C
corentih 已提交
3049 3050 3051
        fn extract_path_and_node_id(t: &Ty,
                                    allow: FallbackChecks)
                                    -> Option<(Path, NodeId, FallbackChecks)> {
3052
            match t.node {
3053
                TyPath(None, ref path) => Some((path.clone(), t.id, allow)),
3054 3055
                TyPtr(ref mut_ty) => extract_path_and_node_id(&*mut_ty.ty, OnlyTraitAndStatics),
                TyRptr(_, ref mut_ty) => extract_path_and_node_id(&*mut_ty.ty, allow),
3056 3057 3058 3059 3060 3061 3062
                // This doesn't handle the remaining `Ty` variants as they are not
                // that commonly the self_type, it might be interesting to provide
                // support for those in future.
                _ => None,
            }
        }

3063 3064 3065 3066 3067
        fn get_module<'a, 'tcx>(this: &mut Resolver<'a, 'tcx>,
                                span: Span,
                                name_path: &[ast::Name])
                                -> Option<Module<'a>> {
            let root = this.current_module;
3068
            let last_name = name_path.last().unwrap();
3069

3070
            if name_path.len() == 1 {
3071
                match this.primitive_type_table.primitive_types.get(last_name) {
3072
                    Some(_) => None,
3073
                    None => this.current_module.resolve_name(*last_name, TypeNS, true).success()
3074
                                               .and_then(NameBinding::module)
3075 3076
                }
            } else {
J
Jeffrey Seyfried 已提交
3077
                match this.resolve_module_path(root, &name_path, UseLexicalScope, span) {
3078
                    Success((module, _)) => Some(module),
C
corentih 已提交
3079
                    _ => None,
3080 3081 3082 3083
                }
            }
        }

3084
        fn is_static_method(this: &Resolver, did: DefId) -> bool {
3085 3086
            if let Some(node_id) = this.ast_map.as_local_node_id(did) {
                let sig = match this.ast_map.get(node_id) {
3087 3088
                    hir_map::NodeTraitItem(trait_item) => match trait_item.node {
                        hir::MethodTraitItem(ref sig, _) => sig,
C
corentih 已提交
3089
                        _ => return false,
3090
                    },
3091
                    hir_map::NodeImplItem(impl_item) => match impl_item.node {
3092
                        hir::ImplItemKind::Method(ref sig, _) => sig,
C
corentih 已提交
3093
                        _ => return false,
3094
                    },
C
corentih 已提交
3095
                    _ => return false,
3096
                };
3097
                sig.explicit_self.node == hir::SelfStatic
3098
            } else {
3099
                this.session.cstore.is_static_method(did)
3100 3101 3102
            }
        }

3103 3104 3105 3106
        let (path, node_id, allowed) = match self.current_self_type {
            Some(ref ty) => match extract_path_and_node_id(ty, Everything) {
                Some(x) => x,
                None => return NoSuggestion,
3107 3108 3109 3110
            },
            None => return NoSuggestion,
        };

3111 3112
        if allowed == Everything {
            // Look for a field with the same name in the current self_type.
3113
            match self.def_map.borrow().get(&node_id).map(|d| d.full_def()) {
3114 3115 3116 3117
                Some(Def::Enum(did)) |
                Some(Def::TyAlias(did)) |
                Some(Def::Struct(did)) |
                Some(Def::Variant(_, did)) => match self.structs.get(&did) {
3118 3119 3120 3121 3122
                    None => {}
                    Some(fields) => {
                        if fields.iter().any(|&field_name| name == field_name) {
                            return Field;
                        }
3123
                    }
3124 3125 3126
                },
                _ => {} // Self type didn't resolve properly
            }
3127 3128
        }

3129
        let name_path = path.segments.iter().map(|seg| seg.identifier.name).collect::<Vec<_>>();
3130 3131

        // Look for a method in the current self type's impl module.
3132
        if let Some(module) = get_module(self, path.span, &name_path) {
3133
            if let Success(binding) = module.resolve_name(name, ValueNS, true) {
3134
                if let Some(Def::Method(did)) = binding.def() {
3135
                    if is_static_method(self, did) {
C
corentih 已提交
3136
                        return StaticMethod(path_names_to_string(&path, 0));
3137 3138 3139 3140 3141
                    }
                    if self.current_trait_ref.is_some() {
                        return TraitItem;
                    } else if allowed == Everything {
                        return Method;
3142 3143
                    }
                }
3144
            }
3145 3146 3147
        }

        // Look for a method in the current trait.
3148 3149 3150
        if let Some((trait_did, ref trait_ref)) = self.current_trait_ref {
            if let Some(&did) = self.trait_item_map.get(&(name, trait_did)) {
                if is_static_method(self, did) {
3151
                    return TraitMethod(path_names_to_string(&trait_ref.path, 0));
3152 3153
                } else {
                    return TraitItem;
3154 3155 3156 3157 3158 3159 3160
                }
            }
        }

        NoSuggestion
    }

3161
    fn find_best_match(&mut self, name: &str) -> SuggestionType {
3162
        if let Some(macro_name) = self.session.available_macros
3163
                                  .borrow().iter().find(|n| n.as_str() == name) {
3164 3165 3166
            return SuggestionType::Macro(format!("{}!", macro_name));
        }

3167 3168 3169 3170
        let names = self.value_ribs
                    .iter()
                    .rev()
                    .flat_map(|rib| rib.bindings.keys());
3171

3172 3173 3174
        if let Some(found) = find_best_match_for_name(names, name, None) {
            if name != &*found {
                return SuggestionType::Function(found);
3175
            }
3176
        } SuggestionType::NotFound
3177 3178
    }

E
Eduard Burtescu 已提交
3179
    fn resolve_expr(&mut self, expr: &Expr) {
P
Patrick Walton 已提交
3180 3181
        // First, record candidate traits for this expression if it could
        // result in the invocation of a method call.
3182 3183 3184

        self.record_candidate_traits_for_expr_if_necessary(expr);

3185
        // Next, resolve the node.
3186
        match expr.node {
3187
            ExprPath(ref maybe_qself, ref path) => {
C
corentih 已提交
3188 3189 3190 3191 3192 3193 3194 3195 3196 3197
                let resolution = match self.resolve_possibly_assoc_item(expr.id,
                                                                        maybe_qself.as_ref(),
                                                                        path,
                                                                        ValueNS,
                                                                        true) {
                    // `<T>::a::b::c` is resolved by typeck alone.
                    TypecheckRequired => {
                        let method_name = path.segments.last().unwrap().identifier.name;
                        let traits = self.get_traits_containing_item(method_name);
                        self.trait_map.insert(expr.id, traits);
3198
                        intravisit::walk_expr(self, expr);
C
corentih 已提交
3199 3200 3201 3202
                        return;
                    }
                    ResolveAttempt(resolution) => resolution,
                };
3203

3204 3205
                // This is a local path in the value namespace. Walk through
                // scopes looking for it.
3206
                if let Some(path_res) = resolution {
3207
                    // Check if struct variant
3208
                    let is_struct_variant = if let Def::Variant(_, variant_id) = path_res.base_def {
3209 3210 3211 3212 3213 3214
                        self.structs.contains_key(&variant_id)
                    } else {
                        false
                    };
                    if is_struct_variant {
                        let _ = self.structs.contains_key(&path_res.base_def.def_id());
3215
                        let path_name = path_names_to_string(path, 0);
3216

N
Nick Cameron 已提交
3217 3218 3219
                        let mut err = resolve_struct_error(self,
                                        expr.span,
                                        ResolutionError::StructVariantUsedAsFunction(&*path_name));
3220

C
corentih 已提交
3221
                        let msg = format!("did you mean to write: `{} {{ /* fields */ }}`?",
3222 3223
                                          path_name);
                        if self.emit_errors {
N
Nick Cameron 已提交
3224
                            err.fileline_help(expr.span, &msg);
3225
                        } else {
N
Nick Cameron 已提交
3226
                            err.span_help(expr.span, &msg);
3227
                        }
N
Nick Cameron 已提交
3228
                        err.emit();
3229
                        self.record_def(expr.id, err_path_resolution());
3230
                    } else {
3231
                        // Write the result into the def map.
3232
                        debug!("(resolving expr) resolved `{}`",
3233
                               path_names_to_string(path, 0));
3234

3235 3236
                        // Partial resolutions will need the set of traits in scope,
                        // so they can be completed during typeck.
3237
                        if path_res.depth != 0 {
3238
                            let method_name = path.segments.last().unwrap().identifier.name;
3239
                            let traits = self.get_traits_containing_item(method_name);
3240 3241 3242
                            self.trait_map.insert(expr.id, traits);
                        }

3243
                        self.record_def(expr.id, path_res);
3244
                    }
3245 3246 3247 3248 3249
                } else {
                    // Be helpful if the name refers to a struct
                    // (The pattern matching def_tys where the id is in self.structs
                    // matches on regular structs while excluding tuple- and enum-like
                    // structs, which wouldn't result in this error.)
3250
                    let path_name = path_names_to_string(path, 0);
3251 3252 3253
                    let type_res = self.with_no_errors(|this| {
                        this.resolve_path(expr.id, path, 0, TypeNS, false)
                    });
3254 3255

                    self.record_def(expr.id, err_path_resolution());
3256
                    match type_res.map(|r| r.base_def) {
3257
                        Some(Def::Struct(..)) => {
N
Nick Cameron 已提交
3258 3259 3260
                            let mut err = resolve_struct_error(self,
                                expr.span,
                                ResolutionError::StructVariantUsedAsFunction(&*path_name));
3261

C
corentih 已提交
3262 3263 3264
                            let msg = format!("did you mean to write: `{} {{ /* fields */ }}`?",
                                              path_name);
                            if self.emit_errors {
N
Nick Cameron 已提交
3265
                                err.fileline_help(expr.span, &msg);
C
corentih 已提交
3266
                            } else {
N
Nick Cameron 已提交
3267
                                err.span_help(expr.span, &msg);
3268
                            }
N
Nick Cameron 已提交
3269
                            err.emit();
C
corentih 已提交
3270
                        }
3271 3272 3273 3274 3275 3276 3277 3278 3279 3280 3281 3282 3283
                        _ => {
                            // Keep reporting some errors even if they're ignored above.
                            self.resolve_path(expr.id, path, 0, ValueNS, true);

                            let mut method_scope = false;
                            self.value_ribs.iter().rev().all(|rib| {
                                method_scope = match rib.kind {
                                    MethodRibKind => true,
                                    ItemRibKind | ConstantItemRibKind => false,
                                    _ => return true, // Keep advancing
                                };
                                false // Stop advancing
                            });
3284

3285
                            if method_scope && special_names::self_.as_str() == &path_name[..] {
C
corentih 已提交
3286 3287 3288
                                resolve_error(self,
                                              expr.span,
                                              ResolutionError::SelfNotAvailableInStaticMethod);
3289 3290 3291 3292 3293 3294
                            } else {
                                let last_name = path.segments.last().unwrap().identifier.name;
                                let mut msg = match self.find_fallback_in_self_type(last_name) {
                                    NoSuggestion => {
                                        // limit search to 5 to reduce the number
                                        // of stupid suggestions
3295
                                        match self.find_best_match(&path_name) {
3296 3297 3298 3299 3300 3301
                                            SuggestionType::Macro(s) => {
                                                format!("the macro `{}`", s)
                                            }
                                            SuggestionType::Function(s) => format!("`{}`", s),
                                            SuggestionType::NotFound => "".to_string(),
                                        }
3302 3303 3304
                                    }
                                    Field => format!("`self.{}`", path_name),
                                    Method |
C
corentih 已提交
3305
                                    TraitItem => format!("to call `self.{}`", path_name),
3306 3307
                                    TraitMethod(path_str) |
                                    StaticMethod(path_str) =>
C
corentih 已提交
3308
                                        format!("to call `{}::{}`", path_str, path_name),
3309 3310
                                };

3311
                                let mut context =  UnresolvedNameContext::Other;
3312
                                if !msg.is_empty() {
3313 3314 3315 3316 3317 3318 3319
                                    msg = format!(". Did you mean {}?", msg);
                                } else {
                                    // we check if this a module and if so, we display a help
                                    // message
                                    let name_path = path.segments.iter()
                                                        .map(|seg| seg.identifier.name)
                                                        .collect::<Vec<_>>();
3320
                                    let current_module = self.current_module;
3321 3322

                                    match self.resolve_module_path(current_module,
J
Jeffrey Seyfried 已提交
3323 3324 3325
                                                                   &name_path[..],
                                                                   UseLexicalScope,
                                                                   expr.span) {
3326 3327 3328 3329 3330
                                        Success(_) => {
                                            context = UnresolvedNameContext::PathIsMod(expr.id);
                                        },
                                        _ => {},
                                    };
3331
                                }
3332

3333 3334
                                resolve_error(self,
                                              expr.span,
3335 3336
                                              ResolutionError::UnresolvedName(
                                                  &*path_name, &*msg, context));
3337
                            }
V
Vincent Belliard 已提交
3338
                        }
3339 3340 3341
                    }
                }

3342
                intravisit::walk_expr(self, expr);
3343 3344
            }

3345
            ExprStruct(ref path, _, _) => {
3346 3347 3348
                // Resolve the path to the structure it goes to. We don't
                // check to ensure that the path is actually a structure; that
                // is checked later during typeck.
3349
                match self.resolve_path(expr.id, path, 0, TypeNS, false) {
3350
                    Some(definition) => self.record_def(expr.id, definition),
3351 3352
                    None => {
                        debug!("(resolving expression) didn't find struct def",);
3353

3354 3355
                        resolve_error(self,
                                      path.span,
3356
                                      ResolutionError::DoesNotNameAStruct(
3357 3358
                                                                &*path_names_to_string(path, 0))
                                     );
3359
                        self.record_def(expr.id, err_path_resolution());
3360 3361 3362
                    }
                }

3363
                intravisit::walk_expr(self, expr);
3364 3365
            }

P
Pythoner6 已提交
3366
            ExprLoop(_, Some(label)) | ExprWhile(_, _, Some(label)) => {
3367
                self.with_label_rib(|this| {
3368
                    let def_like = DlDef(Def::Label(expr.id));
3369

3370
                    {
3371
                        let rib = this.label_ribs.last_mut().unwrap();
3372
                        rib.bindings.insert(label.name, def_like);
3373
                    }
3374

3375
                    intravisit::walk_expr(this, expr);
3376
                })
3377 3378
            }

3379
            ExprBreak(Some(label)) | ExprAgain(Some(label)) => {
3380
                match self.search_label(label.node.name) {
3381
                    None => {
3382
                        self.record_def(expr.id, err_path_resolution());
3383
                        resolve_error(self,
3384 3385
                                      label.span,
                                      ResolutionError::UndeclaredLabel(&label.node.name.as_str()))
3386
                    }
3387
                    Some(DlDef(def @ Def::Label(_))) => {
3388
                        // Since this def is a label, it is never read.
C
corentih 已提交
3389 3390 3391 3392 3393 3394
                        self.record_def(expr.id,
                                        PathResolution {
                                            base_def: def,
                                            last_private: LastMod(AllPublic),
                                            depth: 0,
                                        })
3395 3396
                    }
                    Some(_) => {
C
corentih 已提交
3397
                        self.session.span_bug(expr.span, "label wasn't mapped to a label def!")
3398 3399 3400 3401
                    }
                }
            }

B
Brian Anderson 已提交
3402
            _ => {
3403
                intravisit::walk_expr(self, expr);
3404 3405 3406 3407
            }
        }
    }

E
Eduard Burtescu 已提交
3408
    fn record_candidate_traits_for_expr_if_necessary(&mut self, expr: &Expr) {
3409
        match expr.node {
3410
            ExprField(_, name) => {
3411 3412 3413 3414
                // FIXME(#6890): Even though you can't treat a method like a
                // field, we need to add any trait methods we find that match
                // the field name so that we can do some nice error reporting
                // later on in typeck.
3415
                let traits = self.get_traits_containing_item(name.node);
3416
                self.trait_map.insert(expr.id, traits);
3417
            }
3418
            ExprMethodCall(name, _, _) => {
C
corentih 已提交
3419
                debug!("(recording candidate traits for expr) recording traits for {}",
3420
                       expr.id);
3421
                let traits = self.get_traits_containing_item(name.node);
3422
                self.trait_map.insert(expr.id, traits);
3423
            }
3424
            _ => {
3425 3426 3427 3428 3429
                // Nothing to do.
            }
        }
    }

3430
    fn get_traits_containing_item(&mut self, name: Name) -> Vec<DefId> {
C
corentih 已提交
3431
        debug!("(getting traits containing item) looking for '{}'", name);
E
Eduard Burtescu 已提交
3432

C
corentih 已提交
3433
        fn add_trait_info(found_traits: &mut Vec<DefId>, trait_def_id: DefId, name: Name) {
3434
            debug!("(adding trait info) found trait {:?} for method '{}'",
C
corentih 已提交
3435 3436
                   trait_def_id,
                   name);
E
Eduard Burtescu 已提交
3437 3438
            found_traits.push(trait_def_id);
        }
3439

3440
        let mut found_traits = Vec::new();
3441
        let mut search_module = self.current_module;
E
Eduard Burtescu 已提交
3442 3443
        loop {
            // Look for the current trait.
3444 3445
            match self.current_trait_ref {
                Some((trait_def_id, _)) => {
3446
                    if self.trait_item_map.contains_key(&(name, trait_def_id)) {
3447
                        add_trait_info(&mut found_traits, trait_def_id, name);
3448 3449
                    }
                }
3450
                None => {} // Nothing to do.
E
Eduard Burtescu 已提交
3451
            }
3452

E
Eduard Burtescu 已提交
3453
            // Look for trait children.
3454
            build_reduced_graph::populate_module_if_necessary(self, &search_module);
3455

3456 3457
            search_module.for_each_child(|_, ns, name_binding| {
                if ns != TypeNS { return }
3458 3459
                let trait_def_id = match name_binding.def() {
                    Some(Def::Trait(trait_def_id)) => trait_def_id,
3460
                    Some(..) | None => return,
3461 3462 3463
                };
                if self.trait_item_map.contains_key(&(name, trait_def_id)) {
                    add_trait_info(&mut found_traits, trait_def_id, name);
3464 3465
                    let trait_name = self.get_trait_name(trait_def_id);
                    self.record_use(trait_name, TypeNS, name_binding);
3466
                }
3467
            });
3468

3469
            match search_module.parent_link {
E
Eduard Burtescu 已提交
3470 3471
                NoParentLink | ModuleParentLink(..) => break,
                BlockParentLink(parent_module, _) => {
3472
                    search_module = parent_module;
3473
                }
E
Eduard Burtescu 已提交
3474
            }
3475 3476
        }

E
Eduard Burtescu 已提交
3477
        found_traits
3478 3479
    }

3480 3481
    fn record_def(&mut self, node_id: NodeId, resolution: PathResolution) {
        debug!("(recording def) recording {:?} for {}", resolution, node_id);
C
corentih 已提交
3482 3483 3484 3485
        assert!(match resolution.last_private {
                    LastImport{..} => false,
                    _ => true,
                },
3486
                "Import should only be used for `use` directives");
3487

3488 3489
        if let Some(prev_res) = self.def_map.borrow_mut().insert(node_id, resolution) {
            let span = self.ast_map.opt_span(node_id).unwrap_or(codemap::DUMMY_SP);
C
corentih 已提交
3490 3491 3492 3493
            self.session.span_bug(span,
                                  &format!("path resolved multiple times ({:?} before, {:?} now)",
                                           prev_res,
                                           resolution));
3494
        }
3495 3496
    }

F
Felix S. Klock II 已提交
3497
    fn enforce_default_binding_mode(&mut self,
C
corentih 已提交
3498 3499 3500
                                    pat: &Pat,
                                    pat_binding_mode: BindingMode,
                                    descr: &str) {
3501
        match pat_binding_mode {
3502
            BindByValue(_) => {}
A
Alex Crichton 已提交
3503
            BindByRef(..) => {
3504 3505
                resolve_error(self,
                              pat.span,
3506
                              ResolutionError::CannotUseRefBindingModeWith(descr));
3507 3508 3509
            }
        }
    }
3510 3511
}

3512 3513 3514 3515 3516 3517 3518 3519 3520 3521

fn names_to_string(names: &[Name]) -> String {
    let mut first = true;
    let mut result = String::new();
    for name in names {
        if first {
            first = false
        } else {
            result.push_str("::")
        }
3522
        result.push_str(&name.as_str());
C
corentih 已提交
3523
    }
3524 3525 3526 3527
    result
}

fn path_names_to_string(path: &Path, depth: usize) -> String {
C
corentih 已提交
3528
    let names: Vec<ast::Name> = path.segments[..path.segments.len() - depth]
3529 3530 3531 3532 3533 3534 3535
                                    .iter()
                                    .map(|seg| seg.identifier.name)
                                    .collect();
    names_to_string(&names[..])
}

/// A somewhat inefficient routine to obtain the name of a module.
3536
fn module_to_string<'a>(module: Module<'a>) -> String {
3537 3538
    let mut names = Vec::new();

3539
    fn collect_mod<'a>(names: &mut Vec<ast::Name>, module: Module<'a>) {
3540 3541 3542 3543
        match module.parent_link {
            NoParentLink => {}
            ModuleParentLink(ref module, name) => {
                names.push(name);
3544
                collect_mod(names, module);
3545 3546 3547 3548
            }
            BlockParentLink(ref module, _) => {
                // danger, shouldn't be ident?
                names.push(special_idents::opaque.name);
3549
                collect_mod(names, module);
3550 3551 3552 3553 3554
            }
        }
    }
    collect_mod(&mut names, module);

3555
    if names.is_empty() {
3556 3557 3558 3559 3560
        return "???".to_string();
    }
    names_to_string(&names.into_iter().rev().collect::<Vec<ast::Name>>())
}

3561 3562
fn err_path_resolution() -> PathResolution {
    PathResolution {
3563
        base_def: Def::Err,
3564 3565 3566 3567 3568
        last_private: LastMod(AllPublic),
        depth: 0,
    }
}

3569

3570
pub struct CrateMap {
J
Jonathan S 已提交
3571
    pub def_map: RefCell<DefMap>,
3572
    pub freevars: FreevarMap,
3573
    pub export_map: ExportMap,
3574 3575
    pub trait_map: TraitMap,
    pub external_exports: ExternalExports,
C
corentih 已提交
3576
    pub glob_map: Option<GlobMap>,
3577 3578
}

N
Niko Matsakis 已提交
3579
#[derive(PartialEq,Copy, Clone)]
3580 3581
pub enum MakeGlobMap {
    Yes,
C
corentih 已提交
3582
    No,
3583 3584
}

3585
/// Entry point to crate resolution.
3586
pub fn resolve_crate<'a, 'tcx>(session: &'a Session,
3587
                               ast_map: &'a hir_map::Map<'tcx>,
3588 3589
                               make_glob_map: MakeGlobMap)
                               -> CrateMap {
3590 3591 3592 3593 3594 3595 3596 3597 3598
    // Currently, we ignore the name resolution data structures for
    // the purposes of dependency tracking. Instead we will run name
    // resolution and include its output in the hash of each item,
    // much like we do for macro expansion. In other words, the hash
    // reflects not just its contents but the results of name
    // resolution on those contents. Hopefully we'll push this back at
    // some point.
    let _task = ast_map.dep_graph.in_task(DepNode::Resolve);

3599
    let krate = ast_map.krate();
3600 3601
    let arenas = Resolver::arenas();
    let mut resolver = create_resolver(session, ast_map, krate, make_glob_map, &arenas, None);
3602 3603 3604 3605 3606

    resolver.resolve_crate(krate);

    check_unused::check_crate(&mut resolver, krate);

3607
    CrateMap {
3608 3609
        def_map: resolver.def_map,
        freevars: resolver.freevars,
3610
        export_map: resolver.export_map,
3611 3612
        trait_map: resolver.trait_map,
        external_exports: resolver.external_exports,
3613
        glob_map: if resolver.make_glob_map {
C
corentih 已提交
3614 3615 3616 3617
            Some(resolver.glob_map)
        } else {
            None
        },
3618
    }
3619
}
3620

3621 3622 3623 3624 3625 3626 3627 3628
/// Builds a name resolution walker to be used within this module,
/// or used externally, with an optional callback function.
///
/// The callback takes a &mut bool which allows callbacks to end a
/// walk when set to true, passing through the rest of the walk, while
/// preserving the ribs + current module. This allows resolve_path
/// calls to be made with the correct scope info. The node in the
/// callback corresponds to the current node in the walk.
G
Garming Sam 已提交
3629
pub fn create_resolver<'a, 'tcx>(session: &'a Session,
3630
                                 ast_map: &'a hir_map::Map<'tcx>,
G
Garming Sam 已提交
3631 3632
                                 krate: &'a Crate,
                                 make_glob_map: MakeGlobMap,
3633
                                 arenas: &'a ResolverArenas<'a>,
3634
                                 callback: Option<Box<Fn(hir_map::Node, &mut bool) -> bool>>)
G
Garming Sam 已提交
3635
                                 -> Resolver<'a, 'tcx> {
3636
    let mut resolver = Resolver::new(session, ast_map, make_glob_map, arenas);
G
Garming Sam 已提交
3637 3638 3639 3640 3641 3642 3643 3644 3645 3646

    resolver.callback = callback;

    build_reduced_graph::build_reduced_graph(&mut resolver, krate);

    resolve_imports::resolve_imports(&mut resolver);

    resolver
}

3647
__build_diagnostic_array! { librustc_resolve, DIAGNOSTICS }