lib.rs 150.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 822
    shadowed_traits: RefCell<Vec<&'a NameBinding<'a>>>,

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

826 827 828 829 830 831
    // 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>,

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

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

841 842 843 844 845
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 {
846
            parent_link: parent_link,
J
Jeffrey Seyfried 已提交
847
            def: def,
848
            is_public: is_public,
849
            is_extern_crate: false,
850
            children: RefCell::new(HashMap::new()),
851
            imports: RefCell::new(Vec::new()),
852
            anonymous_children: RefCell::new(NodeMap()),
853
            shadowed_traits: RefCell::new(Vec::new()),
854
            glob_count: Cell::new(0),
855 856
            pub_count: Cell::new(0),
            pub_glob_count: Cell::new(0),
857
            resolved_import_count: Cell::new(0),
858
            populated: Cell::new(!external),
859
        }
B
Brian Anderson 已提交
860 861
    }

862 863 864 865 866 867 868 869 870 871
    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) }
            })
872 873
    }

874
    // Define the name or return the existing binding if there is a collision.
875
    fn try_define_child(&self, name: Name, ns: Namespace, binding: &'a NameBinding<'a>)
876
                        -> Result<(), &'a NameBinding<'a>> {
877 878 879 880 881 882 883 884 885 886 887 888 889
        let mut children = self.children.borrow_mut();
        let resolution = children.entry((name, ns)).or_insert_with(Default::default);

        // FIXME #31379: We can use methods from imported traits shadowed by non-import items
        if let Some(old_binding) = resolution.binding {
            if !old_binding.is_import() && binding.is_import() {
                if let Some(Def::Trait(_)) = binding.def() {
                    self.shadowed_traits.borrow_mut().push(binding);
                }
            }
        }

        resolution.try_define(binding)
890 891
    }

892
    fn increment_outstanding_references_for(&self, name: Name, ns: Namespace) {
893 894
        let mut children = self.children.borrow_mut();
        children.entry((name, ns)).or_insert_with(Default::default).outstanding_references += 1;
895 896 897
    }

    fn decrement_outstanding_references_for(&self, name: Name, ns: Namespace) {
898
        match self.children.borrow_mut().get_mut(&(name, ns)).unwrap().outstanding_references {
899 900 901 902 903
            0 => panic!("No more outstanding references!"),
            ref mut outstanding_references => { *outstanding_references -= 1; }
        }
    }

904 905 906 907 908 909
    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));
        }
    }

910
    fn for_each_local_child<F: FnMut(Name, Namespace, &'a NameBinding<'a>)>(&self, mut f: F) {
911 912
        self.for_each_child(|name, ns, name_binding| {
            if !name_binding.is_import() && !name_binding.is_extern_crate() {
913 914
                f(name, ns, name_binding)
            }
915
        })
916 917
    }

918
    fn def_id(&self) -> Option<DefId> {
J
Jeffrey Seyfried 已提交
919
        self.def.as_ref().map(Def::def_id)
920 921 922
    }

    fn is_normal(&self) -> bool {
J
Jeffrey Seyfried 已提交
923
        match self.def {
924
            Some(Def::Mod(_)) | Some(Def::ForeignMod(_)) => true,
925 926 927 928 929
            _ => false,
        }
    }

    fn is_trait(&self) -> bool {
J
Jeffrey Seyfried 已提交
930
        match self.def {
931
            Some(Def::Trait(_)) => true,
932
            _ => false,
933
        }
B
Brian Anderson 已提交
934 935
    }

F
Felix S. Klock II 已提交
936
    fn all_imports_resolved(&self) -> bool {
937 938 939 940 941 942
        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()
        }
943 944
    }

V
Victor Berger 已提交
945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967
    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);
    }
}

968
impl<'a> fmt::Debug for ModuleS<'a> {
969
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
C
corentih 已提交
970
        write!(f,
971 972
               "{:?}, {}",
               self.def,
C
corentih 已提交
973 974 975 976 977
               if self.is_public {
                   "public"
               } else {
                   "private"
               })
978 979 980
    }
}

981
bitflags! {
J
Jorge Aparicio 已提交
982
    #[derive(Debug)]
983
    flags DefModifiers: u8 {
V
Vadim Petrochenkov 已提交
984 985
        // 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 已提交
986 987
        const PUBLIC     = 1 << 0,
        const IMPORTABLE = 1 << 1,
V
Vadim Petrochenkov 已提交
988
        // Variants are considered `PUBLIC`, but some of them live in private enums.
989 990
        // We need to track them to prohibit reexports like `pub use PrivEnum::Variant`.
        const PRIVATE_VARIANT = 1 << 2,
J
Jeffrey Seyfried 已提交
991
        const PRELUDE = 1 << 3,
992
        const GLOB_IMPORTED = 1 << 4,
993 994 995
    }
}

996
// Records a possibly-private value, type, or module definition.
997
#[derive(Debug)]
998
pub struct NameBinding<'a> {
999 1000
    modifiers: DefModifiers,
    kind: NameBindingKind<'a>,
1001
    span: Option<Span>,
1002 1003
}

1004 1005
#[derive(Debug)]
enum NameBindingKind<'a> {
1006
    Def(Def),
1007
    Module(Module<'a>),
1008 1009 1010 1011
    Import {
        binding: &'a NameBinding<'a>,
        id: NodeId,
    },
1012 1013
}

1014
impl<'a> NameBinding<'a> {
1015
    fn create_from_module(module: Module<'a>, span: Option<Span>) -> Self {
1016
        let modifiers = if module.is_public {
T
Fallout  
Tamir Duberstein 已提交
1017 1018 1019 1020
            DefModifiers::PUBLIC
        } else {
            DefModifiers::empty()
        } | DefModifiers::IMPORTABLE;
1021

1022
        NameBinding { modifiers: modifiers, kind: NameBindingKind::Module(module), span: span }
1023 1024
    }

1025
    fn module(&self) -> Option<Module<'a>> {
1026 1027 1028 1029
        match self.kind {
            NameBindingKind::Module(module) => Some(module),
            NameBindingKind::Def(_) => None,
            NameBindingKind::Import { binding, .. } => binding.module(),
1030 1031 1032
        }
    }

1033
    fn def(&self) -> Option<Def> {
1034 1035 1036 1037
        match self.kind {
            NameBindingKind::Def(def) => Some(def),
            NameBindingKind::Module(module) => module.def,
            NameBindingKind::Import { binding, .. } => binding.def(),
1038
        }
1039
    }
1040

1041
    fn defined_with(&self, modifiers: DefModifiers) -> bool {
1042
        self.modifiers.contains(modifiers)
1043 1044 1045 1046 1047 1048 1049 1050 1051 1052
    }

    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()) }))
    }
1053 1054 1055 1056

    fn is_extern_crate(&self) -> bool {
        self.module().map(|module| module.is_extern_crate).unwrap_or(false)
    }
1057 1058 1059 1060 1061 1062 1063

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

1066
/// Interns the names of the primitive types.
F
Felix S. Klock II 已提交
1067
struct PrimitiveTypeTable {
1068
    primitive_types: HashMap<Name, PrimTy>,
1069
}
1070

1071
impl PrimitiveTypeTable {
K
Kevin Butler 已提交
1072
    fn new() -> PrimitiveTypeTable {
C
corentih 已提交
1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089
        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 已提交
1090 1091 1092 1093

        table
    }

1094
    fn intern(&mut self, string: &str, primitive_type: PrimTy) {
1095
        self.primitive_types.insert(token::intern(string), primitive_type);
1096 1097 1098
    }
}

1099
/// The main resolver class.
C
corentih 已提交
1100
pub struct Resolver<'a, 'tcx: 'a> {
E
Eduard Burtescu 已提交
1101
    session: &'a Session,
1102

1103
    ast_map: &'a hir_map::Map<'tcx>,
1104

1105
    graph_root: Module<'a>,
1106

1107
    trait_item_map: FnvHashMap<(Name, DefId), DefId>,
1108

1109
    structs: FnvHashMap<DefId, Vec<Name>>,
1110

1111
    // The number of imports that are currently unresolved.
1112
    unresolved_imports: usize,
1113 1114

    // The module that represents the current item scope.
1115
    current_module: Module<'a>,
1116 1117

    // The current set of local scopes, for values.
1118
    // FIXME #4948: Reuse ribs to avoid allocation.
1119
    value_ribs: Vec<Rib<'a>>,
1120 1121

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

1124
    // The current set of local scopes, for labels.
1125
    label_ribs: Vec<Rib<'a>>,
1126

1127
    // The trait that the current context can refer to.
1128 1129 1130 1131
    current_trait_ref: Option<(DefId, TraitRef)>,

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

1133
    // The idents for the primitive types.
E
Eduard Burtescu 已提交
1134
    primitive_type_table: PrimitiveTypeTable,
1135

J
Jonathan S 已提交
1136
    def_map: RefCell<DefMap>,
1137 1138
    freevars: FreevarMap,
    freevars_seen: NodeMap<NodeMap<usize>>,
1139
    export_map: ExportMap,
1140
    trait_map: TraitMap,
1141
    external_exports: ExternalExports,
1142

1143 1144 1145 1146 1147
    // 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,

1148 1149 1150 1151 1152
    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,

1153
    used_imports: HashSet<(NodeId, Namespace)>,
1154
    used_crates: HashSet<CrateNum>,
G
Garming Sam 已提交
1155 1156

    // Callback function for intercepting walks
1157
    callback: Option<Box<Fn(hir_map::Node, &mut bool) -> bool>>,
G
Garming Sam 已提交
1158 1159 1160
    // The intention is that the callback modifies this flag.
    // Once set, the resolver falls out of the walk, preserving the ribs.
    resolved: bool,
1161 1162 1163 1164 1165 1166

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

pub struct ResolverArenas<'a> {
    modules: arena::TypedArena<ModuleS<'a>>,
1167
    name_bindings: arena::TypedArena<NameBinding<'a>>,
1168 1169
}

1170
#[derive(PartialEq)]
S
Steven Fackler 已提交
1171 1172
enum FallbackChecks {
    Everything,
C
corentih 已提交
1173
    OnlyTraitAndStatics,
S
Steven Fackler 已提交
1174 1175
}

1176 1177
impl<'a, 'tcx> Resolver<'a, 'tcx> {
    fn new(session: &'a Session,
1178
           ast_map: &'a hir_map::Map<'tcx>,
1179 1180
           make_glob_map: MakeGlobMap,
           arenas: &'a ResolverArenas<'a>)
C
corentih 已提交
1181
           -> Resolver<'a, 'tcx> {
1182
        let root_def_id = ast_map.local_def_id(CRATE_NODE_ID);
1183
        let graph_root = ModuleS::new(NoParentLink, Some(Def::Mod(root_def_id)), false, true);
1184
        let graph_root = arenas.modules.alloc(graph_root);
K
Kevin Butler 已提交
1185 1186 1187 1188

        Resolver {
            session: session,

1189 1190
            ast_map: ast_map,

K
Kevin Butler 已提交
1191 1192
            // The outermost module has def ID 0; this is not reflected in the
            // AST.
1193
            graph_root: graph_root,
K
Kevin Butler 已提交
1194

1195 1196
            trait_item_map: FnvHashMap(),
            structs: FnvHashMap(),
K
Kevin Butler 已提交
1197 1198 1199

            unresolved_imports: 0,

1200
            current_module: graph_root,
1201 1202 1203
            value_ribs: Vec::new(),
            type_ribs: Vec::new(),
            label_ribs: Vec::new(),
K
Kevin Butler 已提交
1204 1205 1206 1207 1208 1209

            current_trait_ref: None,
            current_self_type: None,

            primitive_type_table: PrimitiveTypeTable::new(),

1210
            def_map: RefCell::new(NodeMap()),
1211 1212
            freevars: NodeMap(),
            freevars_seen: NodeMap(),
1213 1214
            export_map: NodeMap(),
            trait_map: NodeMap(),
K
Kevin Butler 已提交
1215
            used_imports: HashSet::new(),
1216
            used_crates: HashSet::new(),
1217
            external_exports: DefIdSet(),
K
Kevin Butler 已提交
1218 1219

            emit_errors: true,
1220 1221
            make_glob_map: make_glob_map == MakeGlobMap::Yes,
            glob_map: HashMap::new(),
G
Garming Sam 已提交
1222 1223 1224

            callback: None,
            resolved: false,
1225 1226 1227 1228 1229 1230 1231 1232

            arenas: arenas,
        }
    }

    fn arenas() -> ResolverArenas<'a> {
        ResolverArenas {
            modules: arena::TypedArena::new(),
1233
            name_bindings: arena::TypedArena::new(),
K
Kevin Butler 已提交
1234 1235
        }
    }
1236

1237 1238 1239 1240 1241 1242 1243 1244
    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))
    }

1245 1246 1247 1248
    fn new_name_binding(&self, name_binding: NameBinding<'a>) -> &'a NameBinding<'a> {
        self.arenas.name_bindings.alloc(name_binding)
    }

1249 1250 1251 1252 1253 1254
    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)
    }

1255 1256 1257 1258
    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 }
    }

1259
    #[inline]
1260 1261 1262 1263 1264 1265
    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);
        }

1266 1267 1268 1269 1270
        let import_id = match binding.kind {
            NameBindingKind::Import { id, .. } => id,
            _ => return,
        };

1271 1272
        self.used_imports.insert((import_id, ns));

1273 1274 1275 1276
        if !self.make_glob_map {
            return;
        }
        if self.glob_map.contains_key(&import_id) {
1277
            self.glob_map.get_mut(&import_id).unwrap().insert(name);
1278 1279 1280 1281 1282 1283 1284 1285 1286
            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 {
1287 1288
        if let Some(node_id) = self.ast_map.as_local_node_id(did) {
            self.ast_map.expect_item(node_id).name
1289
        } else {
1290
            self.session.cstore.item_name(did)
1291 1292 1293
        }
    }

1294
    /// Resolves the given module path from the given root `module_`.
F
Felix S. Klock II 已提交
1295
    fn resolve_module_path_from_root(&mut self,
1296
                                     module_: Module<'a>,
1297
                                     module_path: &[Name],
1298
                                     index: usize,
1299 1300
                                     span: Span,
                                     lp: LastPrivate)
1301
                                     -> ResolveResult<(Module<'a>, LastPrivate)> {
1302
        fn search_parent_externals<'a>(needle: Name, module: Module<'a>) -> Option<Module<'a>> {
1303 1304
            match module.resolve_name(needle, TypeNS, false) {
                Success(binding) if binding.is_extern_crate() => Some(module),
1305
                _ => match module.parent_link {
1306
                    ModuleParentLink(ref parent, _) => {
1307
                        search_parent_externals(needle, parent)
1308
                    }
C
corentih 已提交
1309 1310
                    _ => None,
                },
1311
            }
1312 1313
        }

1314
        let mut search_module = module_;
1315
        let mut index = index;
A
Alex Crichton 已提交
1316
        let module_path_len = module_path.len();
1317
        let mut closest_private = lp;
1318 1319 1320 1321 1322

        // 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 已提交
1323
            let name = module_path[index];
1324
            match self.resolve_name_in_module(search_module, name, TypeNS, false, true) {
1325
                Failed(None) => {
1326
                    let segment_name = name.as_str();
1327
                    let module_name = module_to_string(search_module);
1328
                    let mut span = span;
1329
                    let msg = if "???" == &module_name {
1330
                        span.hi = span.lo + Pos::from_usize(segment_name.len());
1331

C
corentih 已提交
1332
                        match search_parent_externals(name, &self.current_module) {
1333
                            Some(module) => {
1334 1335
                                let path_str = names_to_string(module_path);
                                let target_mod_str = module_to_string(&*module);
C
corentih 已提交
1336
                                let current_mod_str = module_to_string(&*self.current_module);
1337 1338 1339 1340 1341 1342 1343

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

1344
                                format!("Did you mean `{}{}`?", prefix, path_str)
C
corentih 已提交
1345 1346
                            }
                            None => format!("Maybe a missing `extern crate {}`?", segment_name),
1347
                        }
1348
                    } else {
C
corentih 已提交
1349
                        format!("Could not find `{}` in `{}`", segment_name, module_name)
1350
                    };
1351

1352
                    return Failed(Some((span, msg)));
1353
                }
1354
                Failed(err) => return Failed(err),
B
Brian Anderson 已提交
1355
                Indeterminate => {
C
corentih 已提交
1356 1357 1358
                    debug!("(resolving module path for import) module resolution is \
                            indeterminate: {}",
                           name);
B
Brian Anderson 已提交
1359
                    return Indeterminate;
1360
                }
1361
                Success(binding) => {
1362 1363
                    // Check to see whether there are type bindings, and, if
                    // so, whether there is a module within.
J
Jeffrey Seyfried 已提交
1364
                    if let Some(module_def) = binding.module() {
1365
                        search_module = module_def;
1366

1367 1368
                        // Keep track of the closest private module used
                        // when resolving this import chain.
1369
                        if !binding.is_public() {
1370
                            if let Some(did) = search_module.def_id() {
1371
                                closest_private = LastMod(DependsOn(did));
1372 1373
                            }
                        }
1374 1375 1376
                    } else {
                        let msg = format!("Not a module `{}`", name);
                        return Failed(Some((span, msg)));
1377 1378 1379 1380
                    }
                }
            }

T
Tim Chevalier 已提交
1381
            index += 1;
1382 1383
        }

1384
        return Success((search_module, closest_private));
1385 1386
    }

1387 1388
    /// Attempts to resolve the module part of an import directive or path
    /// rooted at the given module.
1389 1390 1391
    ///
    /// On success, returns the resolved module, and the closest *private*
    /// module found to the destination when resolving this path.
F
Felix S. Klock II 已提交
1392
    fn resolve_module_path(&mut self,
1393
                           module_: Module<'a>,
1394
                           module_path: &[Name],
1395
                           use_lexical_scope: UseLexicalScopeFlag,
J
Jeffrey Seyfried 已提交
1396
                           span: Span)
1397
                           -> ResolveResult<(Module<'a>, LastPrivate)> {
1398 1399 1400
        if module_path.len() == 0 {
            return Success((self.graph_root, LastMod(AllPublic))) // Use the crate root
        }
1401

1402
        debug!("(resolving module path for import) processing `{}` rooted at `{}`",
1403 1404
               names_to_string(module_path),
               module_to_string(&*module_));
1405

1406
        // Resolve the module prefix, if any.
1407
        let module_prefix_result = self.resolve_module_prefix(module_, module_path);
1408

1409 1410
        let search_module;
        let start_index;
1411
        let last_private;
1412
        match module_prefix_result {
1413
            Failed(None) => {
1414
                let mpath = names_to_string(module_path);
1415
                let mpath = &mpath[..];
1416
                match mpath.rfind(':') {
C
Fix ICE  
Corey Richardson 已提交
1417
                    Some(idx) => {
1418
                        let msg = format!("Could not find `{}` in `{}`",
C
corentih 已提交
1419 1420 1421 1422
                                          // idx +- 1 to account for the
                                          // colons on either side
                                          &mpath[idx + 1..],
                                          &mpath[..idx - 1]);
1423
                        return Failed(Some((span, msg)));
C
corentih 已提交
1424
                    }
1425
                    None => {
C
corentih 已提交
1426
                        return Failed(None);
1427
                    }
1428
                }
1429
            }
1430
            Failed(err) => return Failed(err),
B
Brian Anderson 已提交
1431
            Indeterminate => {
C
corentih 已提交
1432
                debug!("(resolving module path for import) indeterminate; bailing");
B
Brian Anderson 已提交
1433
                return Indeterminate;
1434
            }
1435 1436 1437 1438 1439 1440 1441 1442
            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.
1443
                        search_module = self.graph_root;
1444
                        start_index = 0;
1445
                        last_private = LastMod(AllPublic);
1446 1447 1448 1449 1450
                    }
                    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.
1451 1452 1453 1454
                        match self.resolve_item_in_lexical_scope(module_,
                                                                 module_path[0],
                                                                 TypeNS,
                                                                 true) {
1455
                            Failed(err) => return Failed(err),
1456
                            Indeterminate => {
C
corentih 已提交
1457
                                debug!("(resolving module path for import) indeterminate; bailing");
1458 1459
                                return Indeterminate;
                            }
1460
                            Success(binding) => match binding.module() {
1461 1462 1463 1464 1465 1466
                                Some(containing_module) => {
                                    search_module = containing_module;
                                    start_index = 1;
                                    last_private = LastMod(AllPublic);
                                }
                                None => return Failed(None),
1467 1468 1469 1470 1471
                            }
                        }
                    }
                }
            }
E
Eduard Burtescu 已提交
1472
            Success(PrefixFound(ref containing_module, index)) => {
1473
                search_module = containing_module;
1474
                start_index = index;
1475
                last_private = LastMod(DependsOn(containing_module.def_id()
1476
                                                                  .unwrap()));
1477 1478 1479
            }
        }

1480 1481 1482 1483
        self.resolve_module_path_from_root(search_module,
                                           module_path,
                                           start_index,
                                           span,
1484
                                           last_private)
1485 1486
    }

1487 1488
    /// Invariant: This must only be called during main resolution, not during
    /// import resolution.
F
Felix S. Klock II 已提交
1489
    fn resolve_item_in_lexical_scope(&mut self,
1490
                                     module_: Module<'a>,
1491
                                     name: Name,
1492 1493
                                     namespace: Namespace,
                                     record_used: bool)
1494
                                     -> ResolveResult<&'a NameBinding<'a>> {
C
corentih 已提交
1495
        debug!("(resolving item in lexical scope) resolving `{}` in namespace {:?} in `{}`",
1496
               name,
1497
               namespace,
1498
               module_to_string(&*module_));
1499

1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511
        // 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.
1512

1513 1514
                    debug!("(resolving item in lexical scope) indeterminate higher scope; bailing");
                    return Indeterminate;
1515
                }
1516
                Success(binding) => {
1517 1518
                    // We found the module.
                    debug!("(resolving item in lexical scope) found name in module, done");
1519
                    return Success(binding);
1520 1521 1522 1523
                }
            }

            // Go to the next parent.
1524
            match search_module.parent_link {
B
Brian Anderson 已提交
1525
                NoParentLink => {
1526
                    // No more parents. This module was unresolved.
1527
                    debug!("(resolving item in lexical scope) unresolved module: no parent module");
1528
                    return Failed(None);
1529
                }
1530
                ModuleParentLink(parent_module_node, _) => {
1531 1532 1533 1534
                    if search_module.is_normal() {
                        // We stop the search here.
                        debug!("(resolving item in lexical scope) unresolved module: not \
                                searching through module parents");
1535
                            return Failed(None);
1536
                    } else {
1537
                        search_module = parent_module_node;
1538 1539
                    }
                }
1540 1541
                BlockParentLink(parent_module_node, _) => {
                    search_module = parent_module_node;
1542 1543 1544 1545 1546
                }
            }
        }
    }

1547
    /// Returns the nearest normal module parent of the given module.
1548
    fn get_nearest_normal_module_parent(&mut self, module_: Module<'a>) -> Option<Module<'a>> {
1549 1550
        let mut module_ = module_;
        loop {
1551
            match module_.parent_link {
1552 1553 1554
                NoParentLink => return None,
                ModuleParentLink(new_module, _) |
                BlockParentLink(new_module, _) => {
1555
                    let new_module = new_module;
1556 1557
                    if new_module.is_normal() {
                        return Some(new_module);
1558
                    }
1559
                    module_ = new_module;
1560 1561 1562 1563 1564
                }
            }
        }
    }

1565 1566
    /// Returns the nearest normal module parent of the given module, or the
    /// module itself if it is a normal module.
1567
    fn get_nearest_normal_module_parent_or_self(&mut self, module_: Module<'a>) -> Module<'a> {
1568 1569 1570
        if module_.is_normal() {
            return module_;
        }
1571
        match self.get_nearest_normal_module_parent(module_) {
1572 1573
            None => module_,
            Some(new_module) => new_module,
1574 1575 1576
        }
    }

1577
    /// Resolves a "module prefix". A module prefix is one or both of (a) `self::`;
1578
    /// (b) some chain of `super::`.
1579
    /// grammar: (SELF MOD_SEP ) ? (SUPER MOD_SEP) *
F
Felix S. Klock II 已提交
1580
    fn resolve_module_prefix(&mut self,
1581
                             module_: Module<'a>,
1582
                             module_path: &[Name])
1583
                             -> ResolveResult<ModulePrefixResult<'a>> {
1584 1585
        // Start at the current module if we see `self` or `super`, or at the
        // top of the crate otherwise.
1586 1587 1588 1589 1590 1591
        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_);
1592 1593

        // Now loop through all the `super`s we find.
1594
        while i < module_path.len() && "super" == module_path[i].as_str() {
1595
            debug!("(resolving module prefix) resolving `super` at {}",
1596
                   module_to_string(&*containing_module));
1597
            match self.get_nearest_normal_module_parent(containing_module) {
1598
                None => return Failed(None),
1599 1600 1601
                Some(new_module) => {
                    containing_module = new_module;
                    i += 1;
1602 1603 1604 1605
                }
            }
        }

1606
        debug!("(resolving module prefix) finished resolving prefix at {}",
1607
               module_to_string(&*containing_module));
1608 1609

        return Success(PrefixFound(containing_module, i));
1610 1611
    }

1612
    /// Attempts to resolve the supplied name in the given module for the
J
Jeffrey Seyfried 已提交
1613
    /// given namespace. If successful, returns the binding corresponding to
1614
    /// the name.
F
Felix S. Klock II 已提交
1615
    fn resolve_name_in_module(&mut self,
1616
                              module: Module<'a>,
1617
                              name: Name,
1618
                              namespace: Namespace,
1619 1620
                              allow_private_imports: bool,
                              record_used: bool)
1621
                              -> ResolveResult<&'a NameBinding<'a>> {
1622
        debug!("(resolving name in module) resolving `{}` in `{}`", name, module_to_string(module));
1623

1624 1625 1626 1627
        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);
1628
            }
1629 1630
            Success(binding)
        })
1631 1632
    }

1633
    fn report_unresolved_imports(&mut self, module_: Module<'a>) {
1634
        let index = module_.resolved_import_count.get();
1635 1636
        let imports = module_.imports.borrow();
        let import_count = imports.len();
1637
        if index != import_count {
1638 1639 1640
            resolve_error(self,
                          (*imports)[index].span,
                          ResolutionError::UnresolvedImport(None));
1641 1642 1643
        }

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

1646
        module_.for_each_local_child(|_, _, child_node| {
1647
            match child_node.module() {
1648 1649 1650 1651 1652
                None => {
                    // Continue.
                }
                Some(child_module) => {
                    self.report_unresolved_imports(child_module);
1653 1654
                }
            }
1655
        });
1656

1657
        for (_, module_) in module_.anonymous_children.borrow().iter() {
1658
            self.report_unresolved_imports(module_);
1659 1660 1661 1662 1663
        }
    }

    // AST resolution
    //
1664
    // We maintain a list of value ribs and type ribs.
1665 1666 1667 1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679
    //
    // 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 已提交
1680 1681
    fn with_scope<F>(&mut self, name: Option<Name>, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1682
    {
1683
        let orig_module = self.current_module;
1684 1685

        // Move down in the graph.
1686
        match name {
B
Brian Anderson 已提交
1687
            None => {
1688 1689
                // Nothing to do.
            }
B
Brian Anderson 已提交
1690
            Some(name) => {
1691
                build_reduced_graph::populate_module_if_necessary(self, &orig_module);
1692

1693 1694 1695 1696 1697 1698 1699 1700 1701
                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;
1702 1703 1704 1705 1706 1707
                        }
                    }
                }
            }
        }

A
Alex Crichton 已提交
1708
        f(self);
1709 1710 1711 1712

        self.current_module = orig_module;
    }

S
Seo Sanghyeon 已提交
1713 1714
    /// Searches the current set of local scopes for labels.
    /// Stops after meeting a closure.
1715 1716 1717 1718 1719 1720 1721 1722
    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 已提交
1723
                    return None;
1724 1725 1726
                }
            }
            let result = rib.bindings.get(&name).cloned();
S
Seo Sanghyeon 已提交
1727
            if result.is_some() {
C
corentih 已提交
1728
                return result;
1729 1730 1731 1732 1733
            }
        }
        None
    }

1734
    fn resolve_crate(&mut self, krate: &hir::Crate) {
1735
        debug!("(resolving crate) starting");
1736

1737
        intravisit::walk_crate(self, krate);
1738 1739
    }

W
we 已提交
1740 1741
    fn check_if_primitive_type_name(&self, name: Name, span: Span) {
        if let Some(_) = self.primitive_type_table.primitive_types.get(&name) {
C
corentih 已提交
1742 1743 1744 1745
            span_err!(self.session,
                      span,
                      E0317,
                      "user-defined types or type parameters cannot shadow the primitive types");
W
we 已提交
1746 1747 1748
        }
    }

1749
    fn resolve_item(&mut self, item: &Item) {
V
Vadim Petrochenkov 已提交
1750
        let name = item.name;
1751

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

1754
        match item.node {
1755 1756 1757
            ItemEnum(_, ref generics) |
            ItemTy(_, ref generics) |
            ItemStruct(_, ref generics) => {
W
we 已提交
1758 1759
                self.check_if_primitive_type_name(name, item.span);

C
corentih 已提交
1760
                self.with_type_parameter_rib(HasTypeParameters(generics, TypeSpace, ItemRibKind),
1761
                                             |this| intravisit::walk_item(this, item));
1762
            }
1763
            ItemFn(_, _, _, _, ref generics, _) => {
C
corentih 已提交
1764
                self.with_type_parameter_rib(HasTypeParameters(generics, FnSpace, ItemRibKind),
1765
                                             |this| intravisit::walk_item(this, item));
1766 1767
            }

F
Flavio Percoco 已提交
1768
            ItemDefaultImpl(_, ref trait_ref) => {
1769
                self.with_optional_trait_ref(Some(trait_ref), |_, _| {});
1770
            }
C
corentih 已提交
1771
            ItemImpl(_, _, ref generics, ref opt_trait_ref, ref self_type, ref impl_items) => {
1772
                self.resolve_implementation(generics,
1773
                                            opt_trait_ref,
1774
                                            &**self_type,
1775
                                            item.id,
1776
                                            impl_items);
1777 1778
            }

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

1782 1783 1784 1785 1786
                // Create a new rib for the trait-wide type parameters.
                self.with_type_parameter_rib(HasTypeParameters(generics,
                                                               TypeSpace,
                                                               ItemRibKind),
                                             |this| {
1787
                    let local_def_id = this.ast_map.local_def_id(item.id);
1788
                    this.with_self_rib(Def::SelfTy(Some(local_def_id), None), |this| {
1789
                        this.visit_generics(generics);
1790
                        walk_list!(this, visit_ty_param_bound, bounds);
1791 1792

                        for trait_item in trait_items {
1793
                            match trait_item.node {
1794
                                hir::ConstTraitItem(_, ref default) => {
1795 1796 1797 1798 1799
                                    // 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| {
1800
                                            intravisit::walk_trait_item(this, trait_item)
1801 1802
                                        });
                                    } else {
1803
                                        intravisit::walk_trait_item(this, trait_item)
1804 1805
                                    }
                                }
1806
                                hir::MethodTraitItem(ref sig, _) => {
1807 1808 1809 1810 1811
                                    let type_parameters =
                                        HasTypeParameters(&sig.generics,
                                                          FnSpace,
                                                          MethodRibKind);
                                    this.with_type_parameter_rib(type_parameters, |this| {
1812
                                        intravisit::walk_trait_item(this, trait_item)
1813
                                    });
1814
                                }
1815
                                hir::TypeTraitItem(..) => {
V
Vadim Petrochenkov 已提交
1816
                                    this.check_if_primitive_type_name(trait_item.name,
1817
                                                                      trait_item.span);
1818
                                    this.with_type_parameter_rib(NoTypeParameters, |this| {
1819
                                        intravisit::walk_trait_item(this, trait_item)
1820
                                    });
1821 1822 1823 1824
                                }
                            };
                        }
                    });
1825
                });
1826 1827
            }

1828
            ItemMod(_) | ItemForeignMod(_) => {
1829
                self.with_scope(Some(name), |this| {
1830
                    intravisit::walk_item(this, item);
1831
                });
1832 1833
            }

1834
            ItemConst(..) | ItemStatic(..) => {
A
Alex Crichton 已提交
1835
                self.with_constant_rib(|this| {
1836
                    intravisit::walk_item(this, item);
1837
                });
1838
            }
1839

W
we 已提交
1840 1841
            ItemUse(ref view_path) => {
                // check for imports shadowing primitive types
1842
                let check_rename = |this: &Self, id, name| {
1843
                    match this.def_map.borrow().get(&id).map(|d| d.full_def()) {
1844 1845
                        Some(Def::Enum(..)) | Some(Def::TyAlias(..)) | Some(Def::Struct(..)) |
                        Some(Def::Trait(..)) | None => {
1846
                            this.check_if_primitive_type_name(name, item.span);
W
we 已提交
1847 1848 1849
                        }
                        _ => {}
                    }
1850 1851 1852
                };

                match view_path.node {
1853 1854
                    hir::ViewPathSimple(name, _) => {
                        check_rename(self, item.id, name);
1855
                    }
1856
                    hir::ViewPathList(ref prefix, ref items) => {
1857
                        for item in items {
1858 1859
                            if let Some(name) = item.node.rename() {
                                check_rename(self, item.node.id(), name);
1860 1861 1862 1863 1864 1865 1866 1867
                            }
                        }

                        // 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 已提交
1868 1869
                                Some((def, lp)) =>
                                    self.record_def(item.id, PathResolution::new(def, lp, 0)),
1870 1871 1872 1873 1874
                                None => {
                                    resolve_error(self,
                                                  prefix.span,
                                                  ResolutionError::FailedToResolve(
                                                      &path_names_to_string(prefix, 0)));
1875
                                    self.record_def(item.id, err_path_resolution());
1876
                                }
1877 1878 1879 1880
                            }
                        }
                    }
                    _ => {}
W
we 已提交
1881 1882 1883
                }
            }

1884
            ItemExternCrate(_) => {
1885
                // do nothing, these are just around to be encoded
1886
            }
1887 1888 1889
        }
    }

1890
    fn with_type_parameter_rib<'b, F>(&'b mut self, type_parameters: TypeParameters<'a, 'b>, f: F)
C
corentih 已提交
1891
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1892
    {
1893
        match type_parameters {
1894
            HasTypeParameters(generics, space, rib_kind) => {
1895
                let mut function_type_rib = Rib::new(rib_kind);
1896
                let mut seen_bindings = HashSet::new();
D
Daniel Micay 已提交
1897
                for (index, type_parameter) in generics.ty_params.iter().enumerate() {
1898
                    let name = type_parameter.name;
1899
                    debug!("with_type_parameter_rib: {}", type_parameter.id);
1900

1901
                    if seen_bindings.contains(&name) {
1902 1903
                        resolve_error(self,
                                      type_parameter.span,
C
corentih 已提交
1904
                                      ResolutionError::NameAlreadyUsedInTypeParameterList(name));
1905
                    }
1906
                    seen_bindings.insert(name);
1907

1908
                    // plain insert (no renaming)
C
corentih 已提交
1909 1910
                    function_type_rib.bindings
                                     .insert(name,
1911
                                             DlDef(Def::TyParam(space,
C
corentih 已提交
1912 1913 1914 1915
                                                              index as u32,
                                                              self.ast_map
                                                                  .local_def_id(type_parameter.id),
                                                              name)));
1916
                }
1917
                self.type_ribs.push(function_type_rib);
1918 1919
            }

B
Brian Anderson 已提交
1920
            NoTypeParameters => {
1921 1922 1923 1924
                // Nothing to do.
            }
        }

A
Alex Crichton 已提交
1925
        f(self);
1926

1927
        match type_parameters {
C
corentih 已提交
1928 1929 1930 1931 1932 1933
            HasTypeParameters(..) => {
                if !self.resolved {
                    self.type_ribs.pop();
                }
            }
            NoTypeParameters => {}
1934 1935 1936
        }
    }

C
corentih 已提交
1937 1938
    fn with_label_rib<F>(&mut self, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1939
    {
1940
        self.label_ribs.push(Rib::new(NormalRibKind));
A
Alex Crichton 已提交
1941
        f(self);
G
Garming Sam 已提交
1942 1943 1944
        if !self.resolved {
            self.label_ribs.pop();
        }
1945
    }
1946

C
corentih 已提交
1947 1948
    fn with_constant_rib<F>(&mut self, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1949
    {
1950 1951
        self.value_ribs.push(Rib::new(ConstantItemRibKind));
        self.type_ribs.push(Rib::new(ConstantItemRibKind));
A
Alex Crichton 已提交
1952
        f(self);
G
Garming Sam 已提交
1953 1954 1955 1956
        if !self.resolved {
            self.type_ribs.pop();
            self.value_ribs.pop();
        }
1957 1958
    }

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

1963
        // Create a label rib for the function.
1964
        self.label_ribs.push(Rib::new(rib_kind));
1965

1966 1967 1968
        // Add each argument to the rib.
        let mut bindings_list = HashMap::new();
        for argument in &declaration.inputs {
C
corentih 已提交
1969
            self.resolve_pattern(&*argument.pat, ArgumentIrrefutableMode, &mut bindings_list);
1970

1971
            self.visit_ty(&*argument.ty);
1972

1973 1974
            debug!("(resolving function) recorded argument");
        }
1975
        intravisit::walk_fn_ret_ty(self, &declaration.output);
1976

1977
        // Resolve the function body.
1978
        self.visit_block(block);
1979

1980
        debug!("(resolving function) leaving function");
1981

G
Garming Sam 已提交
1982 1983 1984 1985
        if !self.resolved {
            self.label_ribs.pop();
            self.value_ribs.pop();
        }
1986 1987
    }

F
Felix S. Klock II 已提交
1988
    fn resolve_trait_reference(&mut self,
N
Nick Cameron 已提交
1989
                               id: NodeId,
1990
                               trait_path: &Path,
1991
                               path_depth: usize)
1992 1993
                               -> Result<PathResolution, ()> {
        if let Some(path_res) = self.resolve_path(id, trait_path, path_depth, TypeNS, true) {
1994
            if let Def::Trait(_) = path_res.base_def {
1995 1996 1997
                debug!("(resolving trait) found trait def: {:?}", path_res);
                Ok(path_res)
            } else {
N
Nick Cameron 已提交
1998 1999 2000 2001 2002
                let mut err =
                    resolve_struct_error(self,
                                  trait_path.span,
                                  ResolutionError::IsNotATrait(&*path_names_to_string(trait_path,
                                                                                      path_depth)));
2003 2004

                // If it's a typedef, give a note
2005
                if let Def::TyAlias(..) = path_res.base_def {
N
Nick Cameron 已提交
2006 2007
                    err.span_note(trait_path.span,
                                  "`type` aliases cannot be used for traits");
2008
                }
N
Nick Cameron 已提交
2009
                err.emit();
2010 2011
                Err(())
            }
2012
        } else {
2013 2014
            resolve_error(self,
                          trait_path.span,
C
corentih 已提交
2015 2016
                          ResolutionError::UndeclaredTraitName(&*path_names_to_string(trait_path,
                                                                                      path_depth)));
2017
            Err(())
2018 2019 2020
        }
    }

2021
    fn resolve_generics(&mut self, generics: &Generics) {
2022
        for type_parameter in generics.ty_params.iter() {
2023
            self.check_if_primitive_type_name(type_parameter.name, type_parameter.span);
2024 2025
        }
        for predicate in &generics.where_clause.predicates {
2026
            match predicate {
2027 2028 2029
                &hir::WherePredicate::BoundPredicate(_) |
                &hir::WherePredicate::RegionPredicate(_) => {}
                &hir::WherePredicate::EqPredicate(ref eq_pred) => {
2030
                    let path_res = self.resolve_path(eq_pred.id, &eq_pred.path, 0, TypeNS, true);
2031
                    if let Some(PathResolution { base_def: Def::TyParam(..), .. }) = path_res {
2032 2033
                        self.record_def(eq_pred.id, path_res.unwrap());
                    } else {
2034 2035
                        resolve_error(self,
                                      eq_pred.span,
2036
                                      ResolutionError::UndeclaredAssociatedType);
2037
                        self.record_def(eq_pred.id, err_path_resolution());
2038 2039
                    }
                }
2040 2041
            }
        }
2042
        intravisit::walk_generics(self, generics);
2043 2044
    }

2045 2046
    fn with_current_self_type<T, F>(&mut self, self_type: &Ty, f: F) -> T
        where F: FnOnce(&mut Resolver) -> T
J
Jorge Aparicio 已提交
2047
    {
2048 2049 2050 2051 2052 2053 2054
        // 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 已提交
2055
    fn with_optional_trait_ref<T, F>(&mut self, opt_trait_ref: Option<&TraitRef>, f: F) -> T
2056
        where F: FnOnce(&mut Resolver, Option<DefId>) -> T
J
Jorge Aparicio 已提交
2057
    {
2058
        let mut new_val = None;
2059
        let mut new_id = None;
E
Eduard Burtescu 已提交
2060
        if let Some(trait_ref) = opt_trait_ref {
2061
            if let Ok(path_res) = self.resolve_trait_reference(trait_ref.ref_id,
C
corentih 已提交
2062 2063
                                                               &trait_ref.path,
                                                               0) {
2064 2065 2066 2067
                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());
2068 2069
            } else {
                self.record_def(trait_ref.ref_id, err_path_resolution());
2070
            }
2071
            intravisit::walk_trait_ref(self, trait_ref);
2072
        }
2073
        let original_trait_ref = replace(&mut self.current_trait_ref, new_val);
2074
        let result = f(self, new_id);
2075 2076 2077 2078
        self.current_trait_ref = original_trait_ref;
        result
    }

2079 2080 2081 2082 2083 2084 2085 2086 2087 2088
    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 已提交
2089 2090 2091
        if !self.resolved {
            self.type_ribs.pop();
        }
2092 2093
    }

F
Felix S. Klock II 已提交
2094
    fn resolve_implementation(&mut self,
2095 2096 2097
                              generics: &Generics,
                              opt_trait_reference: &Option<TraitRef>,
                              self_type: &Ty,
2098
                              item_id: NodeId,
2099
                              impl_items: &[ImplItem]) {
2100
        // If applicable, create a rib for the type parameters.
2101
        self.with_type_parameter_rib(HasTypeParameters(generics,
2102
                                                       TypeSpace,
2103
                                                       ItemRibKind),
2104
                                     |this| {
2105
            // Resolve the type parameters.
2106
            this.visit_generics(generics);
2107

2108
            // Resolve the trait reference, if necessary.
2109
            this.with_optional_trait_ref(opt_trait_reference.as_ref(), |this, trait_id| {
2110
                // Resolve the self type.
2111
                this.visit_ty(self_type);
2112

2113
                this.with_self_rib(Def::SelfTy(trait_id, Some((item_id, self_type.id))), |this| {
2114 2115 2116
                    this.with_current_self_type(self_type, |this| {
                        for impl_item in impl_items {
                            match impl_item.node {
2117
                                hir::ImplItemKind::Const(..) => {
2118
                                    // If this is a trait impl, ensure the const
2119
                                    // exists in trait
V
Vadim Petrochenkov 已提交
2120
                                    this.check_trait_item(impl_item.name,
2121 2122
                                                          impl_item.span,
                                        |n, s| ResolutionError::ConstNotMemberOfTrait(n, s));
2123
                                    this.with_constant_rib(|this| {
2124
                                        intravisit::walk_impl_item(this, impl_item);
2125 2126
                                    });
                                }
2127
                                hir::ImplItemKind::Method(ref sig, _) => {
2128 2129
                                    // If this is a trait impl, ensure the method
                                    // exists in trait
V
Vadim Petrochenkov 已提交
2130
                                    this.check_trait_item(impl_item.name,
2131 2132
                                                          impl_item.span,
                                        |n, s| ResolutionError::MethodNotMemberOfTrait(n, s));
2133 2134 2135 2136 2137 2138 2139 2140

                                    // 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| {
2141
                                        intravisit::walk_impl_item(this, impl_item);
2142 2143
                                    });
                                }
2144
                                hir::ImplItemKind::Type(ref ty) => {
2145
                                    // If this is a trait impl, ensure the type
2146
                                    // exists in trait
V
Vadim Petrochenkov 已提交
2147
                                    this.check_trait_item(impl_item.name,
2148 2149
                                                          impl_item.span,
                                        |n, s| ResolutionError::TypeNotMemberOfTrait(n, s));
2150

2151 2152
                                    this.visit_ty(ty);
                                }
2153
                            }
2154
                        }
2155
                    });
2156 2157
                });
            });
2158
        });
2159 2160
    }

2161
    fn check_trait_item<F>(&self, name: Name, span: Span, err: F)
C
corentih 已提交
2162 2163 2164 2165
        where F: FnOnce(Name, &str) -> ResolutionError
    {
        // If there is a TraitRef in scope for an impl, then the method must be in the
        // trait.
2166
        if let Some((did, ref trait_ref)) = self.current_trait_ref {
2167
            if !self.trait_item_map.contains_key(&(name, did)) {
2168
                let path_str = path_names_to_string(&trait_ref.path, 0);
C
corentih 已提交
2169
                resolve_error(self, span, err(name, &*path_str));
2170 2171 2172 2173
            }
        }
    }

E
Eduard Burtescu 已提交
2174
    fn resolve_local(&mut self, local: &Local) {
2175
        // Resolve the type.
2176
        walk_list!(self, visit_ty, &local.ty);
2177

2178
        // Resolve the initializer.
2179
        walk_list!(self, visit_expr, &local.init);
2180 2181

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

J
John Clements 已提交
2185 2186 2187 2188
    // 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 已提交
2189
    fn binding_mode_map(&mut self, pat: &Pat) -> BindingMap {
2190
        let mut result = HashMap::new();
2191 2192
        pat_bindings(&self.def_map, pat, |binding_mode, _id, sp, path1| {
            let name = path1.node;
C
corentih 已提交
2193 2194 2195 2196 2197
            result.insert(name,
                          BindingInfo {
                              span: sp,
                              binding_mode: binding_mode,
                          });
2198
        });
2199
        return result;
2200 2201
    }

J
John Clements 已提交
2202 2203
    // 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 已提交
2204
    fn check_consistent_bindings(&mut self, arm: &Arm) {
2205
        if arm.pats.is_empty() {
C
corentih 已提交
2206
            return;
2207
        }
2208
        let map_0 = self.binding_mode_map(&*arm.pats[0]);
D
Daniel Micay 已提交
2209
        for (i, p) in arm.pats.iter().enumerate() {
2210
            let map_i = self.binding_mode_map(&**p);
2211

2212
            for (&key, &binding_0) in &map_0 {
2213
                match map_i.get(&key) {
C
corentih 已提交
2214
                    None => {
2215
                        resolve_error(self,
C
corentih 已提交
2216 2217 2218 2219 2220 2221 2222 2223 2224 2225
                                      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));
                        }
2226
                    }
2227 2228 2229
                }
            }

2230
            for (&key, &binding) in &map_i {
2231
                if !map_0.contains_key(&key) {
2232 2233
                    resolve_error(self,
                                  binding.span,
C
corentih 已提交
2234
                                  ResolutionError::VariableNotBoundInParentPattern(key, i + 1));
2235 2236 2237
                }
            }
        }
2238 2239
    }

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

2243
        let mut bindings_list = HashMap::new();
2244
        for pattern in &arm.pats {
2245
            self.resolve_pattern(&**pattern, RefutableMode, &mut bindings_list);
2246 2247
        }

2248 2249 2250 2251
        // This has to happen *after* we determine which
        // pat_idents are variants
        self.check_consistent_bindings(arm);

2252
        walk_list!(self, visit_expr, &arm.guard);
2253
        self.visit_expr(&*arm.body);
2254

G
Garming Sam 已提交
2255 2256 2257
        if !self.resolved {
            self.value_ribs.pop();
        }
2258 2259
    }

E
Eduard Burtescu 已提交
2260
    fn resolve_block(&mut self, block: &Block) {
2261
        debug!("(resolving block) entering block");
2262
        // Move down in the graph, if there's an anonymous module rooted here.
2263
        let orig_module = self.current_module;
2264 2265 2266 2267 2268 2269 2270 2271 2272 2273
        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));
2274 2275 2276
        }

        // Descend into the block.
2277
        intravisit::walk_block(self, block);
2278 2279

        // Move back up.
G
Garming Sam 已提交
2280
        if !self.resolved {
2281
            self.current_module = orig_module;
G
Garming Sam 已提交
2282
            self.value_ribs.pop();
2283 2284 2285
            if let Some(_) = anonymous_module {
                self.type_ribs.pop();
            }
G
Garming Sam 已提交
2286
        }
2287
        debug!("(resolving block) leaving block");
2288 2289
    }

F
Felix S. Klock II 已提交
2290
    fn resolve_type(&mut self, ty: &Ty) {
2291
        match ty.node {
2292
            TyPath(ref maybe_qself, ref path) => {
C
corentih 已提交
2293 2294 2295 2296 2297 2298 2299 2300
                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.
2301
                        intravisit::walk_ty(self, ty);
C
corentih 已提交
2302 2303 2304 2305
                        return;
                    }
                    ResolveAttempt(resolution) => resolution,
                };
2306 2307

                // This is a path in the type namespace. Walk through scopes
2308
                // looking for it.
2309
                match resolution {
B
Brian Anderson 已提交
2310
                    Some(def) => {
2311
                        // Write the result into the def map.
C
corentih 已提交
2312
                        debug!("(resolving type) writing resolution for `{}` (id {}) = {:?}",
2313
                               path_names_to_string(path, 0),
C
corentih 已提交
2314 2315
                               ty.id,
                               def);
2316
                        self.record_def(ty.id, def);
2317
                    }
B
Brian Anderson 已提交
2318
                    None => {
2319 2320
                        self.record_def(ty.id, err_path_resolution());

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

2324 2325 2326 2327
                        let kind = if maybe_qself.is_some() {
                            "associated type"
                        } else {
                            "type name"
2328
                        };
2329

2330
                        let self_type_name = special_idents::type_self.name;
C
corentih 已提交
2331 2332 2333 2334
                        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 已提交
2335
                        if is_invalid_self_type_name {
2336 2337
                            resolve_error(self,
                                          ty.span,
2338
                                          ResolutionError::SelfUsedOutsideImplOrTrait);
2339
                        } else {
2340 2341
                            resolve_error(self,
                                          ty.span,
2342
                                          ResolutionError::UseOfUndeclared(
2343 2344 2345 2346
                                                                    kind,
                                                                    &*path_names_to_string(path,
                                                                                           0))
                                         );
G
Guillaume Gomez 已提交
2347
                        }
2348 2349
                    }
                }
2350
            }
2351
            _ => {}
2352
        }
2353
        // Resolve embedded types.
2354
        intravisit::walk_ty(self, ty);
2355 2356
    }

F
Felix S. Klock II 已提交
2357
    fn resolve_pattern(&mut self,
E
Eduard Burtescu 已提交
2358
                       pattern: &Pat,
2359 2360 2361
                       mode: PatternBindingMode,
                       // Maps idents to the node ID for the (outermost)
                       // pattern that binds them
2362
                       bindings_list: &mut HashMap<Name, NodeId>) {
2363
        let pat_id = pattern.id;
2364
        walk_pat(pattern, |pattern| {
2365
            match pattern.node {
2366 2367
                PatIdent(binding_mode, ref path1, ref at_rhs) => {
                    // The meaning of PatIdent with no type parameters
2368 2369 2370 2371
                    // 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
2372 2373 2374 2375
                    // 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();
2376

2377
                    let ident = path1.node;
2378
                    let renamed = ident.name;
2379

2380 2381
                    match self.resolve_bare_identifier_pattern(ident.unhygienic_name,
                                                               pattern.span) {
2382
                        FoundStructOrEnumVariant(def, lp) if const_ok => {
C
corentih 已提交
2383
                            debug!("(resolving pattern) resolving `{}` to struct or enum variant",
2384
                                   renamed);
2385

C
corentih 已提交
2386 2387 2388 2389 2390 2391 2392 2393 2394
                            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,
                                            });
2395
                        }
A
Alex Crichton 已提交
2396
                        FoundStructOrEnumVariant(..) => {
2397
                            resolve_error(
2398
                                self,
2399
                                pattern.span,
2400
                                ResolutionError::DeclarationShadowsEnumVariantOrUnitLikeStruct(
2401 2402
                                    renamed)
                            );
2403
                            self.record_def(pattern.id, err_path_resolution());
2404
                        }
2405
                        FoundConst(def, lp, _) if const_ok => {
C
corentih 已提交
2406 2407 2408 2409 2410 2411 2412 2413 2414
                            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,
                                            });
2415
                        }
2416
                        FoundConst(def, _, name) => {
2417
                            resolve_error(
2418 2419
                                self,
                                pattern.span,
M
Manish Goregaokar 已提交
2420 2421
                                ResolutionError::OnlyIrrefutablePatternsAllowedHere(def.def_id(),
                                                                                    name)
2422
                            );
2423
                            self.record_def(pattern.id, err_path_resolution());
2424
                        }
2425
                        BareIdentifierPatternUnresolved => {
C
corentih 已提交
2426
                            debug!("(resolving pattern) binding `{}`", renamed);
2427

2428
                            let def_id = self.ast_map.local_def_id(pattern.id);
2429
                            let def = Def::Local(def_id, pattern.id);
2430 2431 2432 2433 2434

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

C
corentih 已提交
2435 2436 2437 2438 2439 2440
                            self.record_def(pattern.id,
                                            PathResolution {
                                                base_def: def,
                                                last_private: LastMod(AllPublic),
                                                depth: 0,
                                            });
2441 2442 2443 2444 2445 2446

                            // 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.)
2447 2448
                            if !bindings_list.contains_key(&renamed) {
                                let this = &mut *self;
2449 2450
                                let last_rib = this.value_ribs.last_mut().unwrap();
                                last_rib.bindings.insert(renamed, DlDef(def));
2451
                                bindings_list.insert(renamed, pat_id);
2452
                            } else if mode == ArgumentIrrefutableMode &&
C
corentih 已提交
2453
                               bindings_list.contains_key(&renamed) {
2454 2455
                                // Forbid duplicate bindings in the same
                                // parameter list.
2456
                                resolve_error(
2457 2458
                                    self,
                                    pattern.span,
2459
                                    ResolutionError::IdentifierBoundMoreThanOnceInParameterList(
2460
                                        &ident.name.as_str())
2461
                                );
C
corentih 已提交
2462
                            } else if bindings_list.get(&renamed) == Some(&pat_id) {
2463 2464
                                // Then this is a duplicate variable in the
                                // same disjunction, which is an error.
2465
                                resolve_error(
2466 2467
                                    self,
                                    pattern.span,
2468
                                    ResolutionError::IdentifierBoundMoreThanOnceInSamePattern(
2469
                                        &ident.name.as_str())
2470
                                );
2471
                            }
2472 2473
                            // Else, not bound in the same pattern: do
                            // nothing.
2474 2475 2476 2477
                        }
                    }
                }

2478
                PatEnum(ref path, _) => {
2479
                    // This must be an enum variant, struct or const.
C
corentih 已提交
2480 2481 2482 2483 2484 2485 2486 2487 2488 2489 2490 2491 2492 2493 2494 2495 2496 2497
                    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,
                    };
2498
                    if let Some(path_res) = resolution {
2499
                        match path_res.base_def {
2500
                            Def::Struct(..) if path_res.depth == 0 => {
2501 2502
                                self.record_def(pattern.id, path_res);
                            }
2503
                            Def::Variant(..) | Def::Const(..) => {
2504 2505
                                self.record_def(pattern.id, path_res);
                            }
2506
                            Def::Static(..) => {
2507 2508
                                resolve_error(&self,
                                              path.span,
2509
                                              ResolutionError::StaticVariableReference);
2510
                                self.record_def(pattern.id, err_path_resolution());
2511
                            }
2512 2513 2514 2515
                            _ => {
                                // If anything ends up here entirely resolved,
                                // it's an error. If anything ends up here
                                // partially resolved, that's OK, because it may
2516
                                // be a `T::CONST` that typeck will resolve.
2517
                                if path_res.depth == 0 {
2518
                                    resolve_error(
2519
                                        self,
2520
                                        path.span,
2521
                                        ResolutionError::NotAnEnumVariantStructOrConst(
2522 2523 2524 2525 2526 2527
                                            &path.segments
                                                 .last()
                                                 .unwrap()
                                                 .identifier
                                                 .name
                                                 .as_str())
2528
                                    );
2529
                                    self.record_def(pattern.id, err_path_resolution());
2530
                                } else {
C
corentih 已提交
2531 2532 2533 2534 2535
                                    let const_name = path.segments
                                                         .last()
                                                         .unwrap()
                                                         .identifier
                                                         .name;
2536 2537
                                    let traits = self.get_traits_containing_item(const_name);
                                    self.trait_map.insert(pattern.id, traits);
2538 2539 2540 2541 2542
                                    self.record_def(pattern.id, path_res);
                                }
                            }
                        }
                    } else {
2543
                        resolve_error(
2544 2545
                            self,
                            path.span,
2546
                            ResolutionError::UnresolvedEnumVariantStructOrConst(
2547
                                &path.segments.last().unwrap().identifier.name.as_str())
2548
                        );
2549
                        self.record_def(pattern.id, err_path_resolution());
2550
                    }
2551
                    intravisit::walk_path(self, path);
2552 2553 2554 2555
                }

                PatQPath(ref qself, ref path) => {
                    // Associated constants only.
C
corentih 已提交
2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570 2571
                    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);
2572
                            intravisit::walk_pat(self, pattern);
C
corentih 已提交
2573 2574 2575 2576
                            return true;
                        }
                        ResolveAttempt(resolution) => resolution,
                    };
2577 2578 2579 2580
                    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.
2581
                            Def::AssociatedConst(..) => {
2582 2583
                                self.record_def(pattern.id, path_res);
                            }
2584
                            _ => {
2585
                                resolve_error(
2586 2587
                                    self,
                                    path.span,
2588
                                    ResolutionError::NotAnAssociatedConst(
2589
                                        &path.segments.last().unwrap().identifier.name.as_str()
2590 2591
                                    )
                                );
2592
                                self.record_def(pattern.id, err_path_resolution());
2593
                            }
2594
                        }
2595
                    } else {
C
corentih 已提交
2596 2597 2598 2599 2600 2601 2602 2603
                        resolve_error(self,
                                      path.span,
                                      ResolutionError::UnresolvedAssociatedConst(&path.segments
                                                                                      .last()
                                                                                      .unwrap()
                                                                                      .identifier
                                                                                      .name
                                                                                      .as_str()));
2604
                        self.record_def(pattern.id, err_path_resolution());
2605
                    }
2606
                    intravisit::walk_pat(self, pattern);
2607 2608
                }

2609
                PatStruct(ref path, _, _) => {
2610
                    match self.resolve_path(pat_id, path, 0, TypeNS, false) {
2611
                        Some(definition) => {
2612 2613
                            self.record_def(pattern.id, definition);
                        }
2614
                        result => {
C
corentih 已提交
2615
                            debug!("(resolving pattern) didn't find struct def: {:?}", result);
2616 2617 2618
                            resolve_error(
                                self,
                                path.span,
2619
                                ResolutionError::DoesNotNameAStruct(
2620 2621
                                    &*path_names_to_string(path, 0))
                            );
2622
                            self.record_def(pattern.id, err_path_resolution());
2623 2624
                        }
                    }
2625
                    intravisit::walk_path(self, path);
2626 2627 2628
                }

                PatLit(_) | PatRange(..) => {
2629
                    intravisit::walk_pat(self, pattern);
2630 2631
                }

2632
                _ => {
2633 2634 2635
                    // Nothing to do.
                }
            }
2636
            true
2637
        });
2638 2639
    }

C
corentih 已提交
2640 2641 2642
    fn resolve_bare_identifier_pattern(&mut self,
                                       name: Name,
                                       span: Span)
E
Eduard Burtescu 已提交
2643
                                       -> BareIdentifierPatternResolution {
2644
        let module = self.current_module;
2645
        match self.resolve_item_in_lexical_scope(module, name, ValueNS, true) {
2646
            Success(binding) => {
C
corentih 已提交
2647 2648
                debug!("(resolve bare identifier pattern) succeeded in finding {} at {:?}",
                       name,
2649
                       binding);
J
Jeffrey Seyfried 已提交
2650
                match binding.def() {
B
Brian Anderson 已提交
2651
                    None => {
C
corentih 已提交
2652 2653
                        panic!("resolved name in the value namespace to a set of name bindings \
                                with no def?!");
2654
                    }
2655 2656 2657
                    // 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.
2658
                    Some(def @ Def::Variant(..)) | Some(def @ Def::Struct(..)) => {
2659 2660
                        return FoundStructOrEnumVariant(def, LastMod(AllPublic));
                    }
2661
                    Some(def @ Def::Const(..)) | Some(def @ Def::AssociatedConst(..)) => {
2662
                        return FoundConst(def, LastMod(AllPublic), name);
2663
                    }
2664
                    Some(Def::Static(..)) => {
2665 2666
                        resolve_error(self, span, ResolutionError::StaticVariableReference);
                        return BareIdentifierPatternUnresolved;
2667
                    }
2668
                    _ => return BareIdentifierPatternUnresolved
2669 2670 2671
                }
            }

2672
            Indeterminate => return BareIdentifierPatternUnresolved,
2673 2674 2675
            Failed(err) => {
                match err {
                    Some((span, msg)) => {
2676
                        resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
2677
                    }
C
corentih 已提交
2678
                    None => (),
2679
                }
2680

C
corentih 已提交
2681
                debug!("(resolve bare identifier pattern) failed to find {}", name);
2682
                return BareIdentifierPatternUnresolved;
2683 2684 2685 2686
            }
        }
    }

2687 2688 2689
    /// Handles paths that may refer to associated items
    fn resolve_possibly_assoc_item(&mut self,
                                   id: NodeId,
2690
                                   maybe_qself: Option<&hir::QSelf>,
2691 2692 2693
                                   path: &Path,
                                   namespace: Namespace,
                                   check_ribs: bool)
C
corentih 已提交
2694
                                   -> AssocItemResolveResult {
2695 2696
        let max_assoc_types;

2697
        match maybe_qself {
2698 2699 2700 2701 2702 2703 2704 2705 2706 2707 2708
            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();
            }
2709 2710 2711 2712 2713 2714 2715 2716 2717 2718
        }

        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 已提交
2719
                resolution = this.resolve_path(id, path, depth, TypeNS, true);
2720 2721
            });
        }
2722
        if let Some(Def::Mod(_)) = resolution.map(|r| r.base_def) {
2723 2724 2725 2726 2727 2728
            // A module is not a valid type or value.
            resolution = None;
        }
        ResolveAttempt(resolution)
    }

2729 2730
    /// If `check_ribs` is true, checks the local definitions first; i.e.
    /// doesn't skip straight to the containing module.
2731 2732
    /// Skips `path_depth` trailing segments, which is also reflected in the
    /// returned value. See `middle::def::PathResolution` for more info.
G
Garming Sam 已提交
2733 2734 2735 2736 2737
    pub fn resolve_path(&mut self,
                        id: NodeId,
                        path: &Path,
                        path_depth: usize,
                        namespace: Namespace,
C
corentih 已提交
2738 2739
                        check_ribs: bool)
                        -> Option<PathResolution> {
2740
        let span = path.span;
C
corentih 已提交
2741
        let segments = &path.segments[..path.segments.len() - path_depth];
2742

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

2745
        if path.global {
2746
            let def = self.resolve_crate_relative_path(span, segments, namespace);
2747
            return def.map(mk_res);
2748 2749
        }

2750
        // Try to find a path to an item in a module.
2751
        let last_ident = segments.last().unwrap().identifier;
N
Nick Cameron 已提交
2752
        if segments.len() <= 1 {
2753
            let unqualified_def = self.resolve_identifier(last_ident, namespace, check_ribs, true);
C
corentih 已提交
2754 2755 2756 2757
            return unqualified_def.and_then(|def| self.adjust_local_def(def, span))
                                  .map(|def| {
                                      PathResolution::new(def, LastMod(AllPublic), path_depth)
                                  });
N
Nick Cameron 已提交
2758
        }
2759

2760
        let unqualified_def = self.resolve_identifier(last_ident, namespace, check_ribs, false);
N
Nick Cameron 已提交
2761 2762
        let def = self.resolve_module_relative_path(span, segments, namespace);
        match (def, unqualified_def) {
2763
            (Some((ref d, _)), Some(ref ud)) if *d == ud.def => {
N
Nick Cameron 已提交
2764 2765
                self.session
                    .add_lint(lint::builtin::UNUSED_QUALIFICATIONS,
C
corentih 已提交
2766 2767
                              id,
                              span,
N
Nick Cameron 已提交
2768 2769 2770
                              "unnecessary qualification".to_string());
            }
            _ => {}
2771
        }
N
Nick Cameron 已提交
2772 2773

        def.map(mk_res)
2774 2775
    }

2776
    // Resolve a single identifier
F
Felix S. Klock II 已提交
2777
    fn resolve_identifier(&mut self,
2778
                          identifier: hir::Ident,
2779
                          namespace: Namespace,
2780 2781
                          check_ribs: bool,
                          record_used: bool)
2782
                          -> Option<LocalDef> {
2783
        if identifier.name == special_idents::invalid.name {
2784
            return Some(LocalDef::from_def(Def::Err));
2785 2786
        }

2787 2788 2789 2790
        // 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
2791
                                        .get(&identifier.unhygienic_name) {
2792
                return Some(LocalDef::from_def(Def::PrimTy(prim_ty)));
2793 2794 2795
            }
        }

2796
        if check_ribs {
C
corentih 已提交
2797
            if let Some(def) = self.resolve_identifier_in_local_ribs(identifier, namespace) {
2798
                return Some(def);
2799 2800 2801
            }
        }

2802 2803
        // Check the items.
        let module = self.current_module;
2804
        let name = identifier.unhygienic_name;
2805
        match self.resolve_item_in_lexical_scope(module, name, namespace, record_used) {
2806
            Success(binding) => binding.def().map(LocalDef::from_def),
2807 2808 2809 2810 2811 2812
            Failed(Some((span, msg))) => {
                resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
                None
            }
            _ => None,
        }
2813 2814 2815
    }

    // Resolve a local definition, potentially adjusting for closures.
2816
    fn adjust_local_def(&mut self, local_def: LocalDef, span: Span) -> Option<Def> {
2817
        let ribs = match local_def.ribs {
C
corentih 已提交
2818 2819 2820
            Some((TypeNS, i)) => &self.type_ribs[i + 1..],
            Some((ValueNS, i)) => &self.value_ribs[i + 1..],
            _ => &[] as &[_],
2821 2822 2823
        };
        let mut def = local_def.def;
        match def {
2824
            Def::Upvar(..) => {
C
corentih 已提交
2825
                self.session.span_bug(span, &format!("unexpected {:?} in bindings", def))
2826
            }
2827
            Def::Local(_, node_id) => {
2828 2829
                for rib in ribs {
                    match rib.kind {
2830
                        NormalRibKind | AnonymousModuleRibKind(..) => {
2831 2832 2833 2834 2835 2836
                            // 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 已提交
2837 2838 2839
                            let seen = self.freevars_seen
                                           .entry(function_id)
                                           .or_insert_with(|| NodeMap());
2840
                            if let Some(&index) = seen.get(&node_id) {
2841
                                def = Def::Upvar(node_def_id, node_id, index, function_id);
2842 2843
                                continue;
                            }
C
corentih 已提交
2844 2845 2846
                            let vec = self.freevars
                                          .entry(function_id)
                                          .or_insert_with(|| vec![]);
2847
                            let depth = vec.len();
C
corentih 已提交
2848 2849 2850 2851
                            vec.push(Freevar {
                                def: prev_def,
                                span: span,
                            });
2852

2853
                            def = Def::Upvar(node_def_id, node_id, depth, function_id);
2854 2855 2856 2857 2858 2859
                            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 已提交
2860 2861 2862
                            resolve_error(self,
                                          span,
                                          ResolutionError::CannotCaptureDynamicEnvironmentInFnItem);
2863 2864 2865 2866
                            return None;
                        }
                        ConstantItemRibKind => {
                            // Still doesn't deal with upvars
C
corentih 已提交
2867 2868 2869
                            resolve_error(self,
                                          span,
                                          ResolutionError::AttemptToUseNonConstantValueInConstant);
2870 2871 2872 2873 2874
                            return None;
                        }
                    }
                }
            }
2875
            Def::TyParam(..) | Def::SelfTy(..) => {
2876 2877
                for rib in ribs {
                    match rib.kind {
2878 2879
                        NormalRibKind | MethodRibKind | ClosureRibKind(..) |
                        AnonymousModuleRibKind(..) => {
2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901
                            // 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);
2902 2903
    }

2904
    // resolve a "module-relative" path, e.g. a::b::c
F
Felix S. Klock II 已提交
2905
    fn resolve_module_relative_path(&mut self,
2906
                                    span: Span,
2907
                                    segments: &[hir::PathSegment],
2908 2909
                                    namespace: Namespace)
                                    -> Option<(Def, LastPrivate)> {
C
corentih 已提交
2910 2911 2912 2913 2914 2915
        let module_path = segments.split_last()
                                  .unwrap()
                                  .1
                                  .iter()
                                  .map(|ps| ps.identifier.name)
                                  .collect::<Vec<_>>();
2916

2917
        let containing_module;
2918
        let last_private;
2919
        let current_module = self.current_module;
J
Jeffrey Seyfried 已提交
2920
        match self.resolve_module_path(current_module, &module_path, UseLexicalScope, span) {
2921 2922 2923 2924
            Failed(err) => {
                let (span, msg) = match err {
                    Some((span, msg)) => (span, msg),
                    None => {
2925
                        let msg = format!("Use of undeclared type or module `{}`",
2926
                                          names_to_string(&module_path));
2927
                        (span, msg)
2928 2929
                    }
                };
2930

2931
                resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
2932
                return None;
2933
            }
2934
            Indeterminate => return None,
2935
            Success((resulting_module, resulting_last_private)) => {
2936
                containing_module = resulting_module;
2937
                last_private = resulting_last_private;
2938 2939 2940
            }
        }

2941
        let name = segments.last().unwrap().identifier.name;
2942 2943
        let result = self.resolve_name_in_module(containing_module, name, namespace, false, true);
        let def = match result {
2944
            Success(binding) => {
2945
                let (def, lp) = binding.def_and_lp();
2946
                (def, last_private.or(lp))
2947
            }
2948
            _ => return None,
2949 2950
        };
        return Some(def);
2951 2952
    }

2953 2954
    /// Invariant: This must be called only during main resolution, not during
    /// import resolution.
F
Felix S. Klock II 已提交
2955
    fn resolve_crate_relative_path(&mut self,
2956
                                   span: Span,
2957
                                   segments: &[hir::PathSegment],
2958
                                   namespace: Namespace)
C
corentih 已提交
2959 2960 2961 2962 2963 2964 2965
                                   -> Option<(Def, LastPrivate)> {
        let module_path = segments.split_last()
                                  .unwrap()
                                  .1
                                  .iter()
                                  .map(|ps| ps.identifier.name)
                                  .collect::<Vec<_>>();
2966

2967
        let root_module = self.graph_root;
2968

2969
        let containing_module;
2970
        let last_private;
2971
        match self.resolve_module_path_from_root(root_module,
2972
                                                 &module_path,
2973
                                                 0,
2974
                                                 span,
2975
                                                 LastMod(AllPublic)) {
2976 2977 2978 2979 2980
            Failed(err) => {
                let (span, msg) = match err {
                    Some((span, msg)) => (span, msg),
                    None => {
                        let msg = format!("Use of undeclared module `::{}`",
2981
                                          names_to_string(&module_path));
2982
                        (span, msg)
2983 2984 2985
                    }
                };

2986
                resolve_error(self, span, ResolutionError::FailedToResolve(&*msg));
B
Brian Anderson 已提交
2987
                return None;
2988 2989
            }

2990
            Indeterminate => return None,
2991

2992
            Success((resulting_module, resulting_last_private)) => {
2993
                containing_module = resulting_module;
2994
                last_private = resulting_last_private;
2995 2996 2997
            }
        }

2998
        let name = segments.last().unwrap().identifier.name;
2999
        match self.resolve_name_in_module(containing_module, name, namespace, false, true) {
3000
            Success(binding) => {
3001 3002
                let (def, lp) = binding.def_and_lp();
                Some((def, last_private.or(lp)))
3003
            }
3004
            _ => None,
3005 3006 3007
        }
    }

F
Felix S. Klock II 已提交
3008
    fn resolve_identifier_in_local_ribs(&mut self,
3009
                                        ident: hir::Ident,
3010 3011
                                        namespace: Namespace)
                                        -> Option<LocalDef> {
3012
        // Check the local set of ribs.
3013
        let name = match namespace { ValueNS => ident.name, TypeNS => ident.unhygienic_name };
3014

3015 3016
        for i in (0 .. self.get_ribs(namespace).len()).rev() {
            if let Some(def_like) = self.get_ribs(namespace)[i].bindings.get(&name).cloned() {
3017 3018 3019
                match def_like {
                    DlDef(def) => {
                        debug!("(resolving path in local ribs) resolved `{}` to {:?} at {}",
C
corentih 已提交
3020 3021 3022
                               name,
                               def,
                               i);
3023 3024
                        return Some(LocalDef {
                            ribs: Some((namespace, i)),
C
corentih 已提交
3025
                            def: def,
3026 3027 3028 3029
                        });
                    }
                    def_like => {
                        debug!("(resolving path in local ribs) resolved `{}` to pseudo-def {:?}",
C
corentih 已提交
3030 3031
                               name,
                               def_like);
3032 3033 3034
                        return None;
                    }
                }
3035
            }
3036 3037

            if let AnonymousModuleRibKind(module) = self.get_ribs(namespace)[i].kind {
3038 3039 3040 3041 3042
                if let Success(binding) = self.resolve_name_in_module(module,
                                                                      ident.unhygienic_name,
                                                                      namespace,
                                                                      true,
                                                                      true) {
J
Jeffrey Seyfried 已提交
3043
                    if let Some(def) = binding.def() {
3044 3045 3046 3047
                        return Some(LocalDef::from_def(def));
                    }
                }
            }
3048
        }
3049 3050

        None
3051 3052
    }

C
corentih 已提交
3053 3054
    fn with_no_errors<T, F>(&mut self, f: F) -> T
        where F: FnOnce(&mut Resolver) -> T
J
Jorge Aparicio 已提交
3055
    {
3056
        self.emit_errors = false;
A
Alex Crichton 已提交
3057
        let rs = f(self);
3058 3059 3060 3061
        self.emit_errors = true;
        rs
    }

3062
    fn find_fallback_in_self_type(&mut self, name: Name) -> FallbackSuggestion {
C
corentih 已提交
3063 3064 3065
        fn extract_path_and_node_id(t: &Ty,
                                    allow: FallbackChecks)
                                    -> Option<(Path, NodeId, FallbackChecks)> {
3066
            match t.node {
3067
                TyPath(None, ref path) => Some((path.clone(), t.id, allow)),
3068 3069
                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),
3070 3071 3072 3073 3074 3075 3076
                // 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,
            }
        }

3077 3078 3079 3080 3081
        fn get_module<'a, 'tcx>(this: &mut Resolver<'a, 'tcx>,
                                span: Span,
                                name_path: &[ast::Name])
                                -> Option<Module<'a>> {
            let root = this.current_module;
3082
            let last_name = name_path.last().unwrap();
3083

3084
            if name_path.len() == 1 {
3085
                match this.primitive_type_table.primitive_types.get(last_name) {
3086
                    Some(_) => None,
3087
                    None => this.current_module.resolve_name(*last_name, TypeNS, true).success()
3088
                                               .and_then(NameBinding::module)
3089 3090
                }
            } else {
J
Jeffrey Seyfried 已提交
3091
                match this.resolve_module_path(root, &name_path, UseLexicalScope, span) {
3092
                    Success((module, _)) => Some(module),
C
corentih 已提交
3093
                    _ => None,
3094 3095 3096 3097
                }
            }
        }

3098
        fn is_static_method(this: &Resolver, did: DefId) -> bool {
3099 3100
            if let Some(node_id) = this.ast_map.as_local_node_id(did) {
                let sig = match this.ast_map.get(node_id) {
3101 3102
                    hir_map::NodeTraitItem(trait_item) => match trait_item.node {
                        hir::MethodTraitItem(ref sig, _) => sig,
C
corentih 已提交
3103
                        _ => return false,
3104
                    },
3105
                    hir_map::NodeImplItem(impl_item) => match impl_item.node {
3106
                        hir::ImplItemKind::Method(ref sig, _) => sig,
C
corentih 已提交
3107
                        _ => return false,
3108
                    },
C
corentih 已提交
3109
                    _ => return false,
3110
                };
3111
                sig.explicit_self.node == hir::SelfStatic
3112
            } else {
3113
                this.session.cstore.is_static_method(did)
3114 3115 3116
            }
        }

3117 3118 3119 3120
        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,
3121 3122 3123 3124
            },
            None => return NoSuggestion,
        };

3125 3126
        if allowed == Everything {
            // Look for a field with the same name in the current self_type.
3127
            match self.def_map.borrow().get(&node_id).map(|d| d.full_def()) {
3128 3129 3130 3131
                Some(Def::Enum(did)) |
                Some(Def::TyAlias(did)) |
                Some(Def::Struct(did)) |
                Some(Def::Variant(_, did)) => match self.structs.get(&did) {
3132 3133 3134 3135 3136
                    None => {}
                    Some(fields) => {
                        if fields.iter().any(|&field_name| name == field_name) {
                            return Field;
                        }
3137
                    }
3138 3139 3140
                },
                _ => {} // Self type didn't resolve properly
            }
3141 3142
        }

3143
        let name_path = path.segments.iter().map(|seg| seg.identifier.name).collect::<Vec<_>>();
3144 3145

        // Look for a method in the current self type's impl module.
3146
        if let Some(module) = get_module(self, path.span, &name_path) {
3147
            if let Success(binding) = module.resolve_name(name, ValueNS, true) {
3148
                if let Some(Def::Method(did)) = binding.def() {
3149
                    if is_static_method(self, did) {
C
corentih 已提交
3150
                        return StaticMethod(path_names_to_string(&path, 0));
3151 3152 3153 3154 3155
                    }
                    if self.current_trait_ref.is_some() {
                        return TraitItem;
                    } else if allowed == Everything {
                        return Method;
3156 3157
                    }
                }
3158
            }
3159 3160 3161
        }

        // Look for a method in the current trait.
3162 3163 3164
        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) {
3165
                    return TraitMethod(path_names_to_string(&trait_ref.path, 0));
3166 3167
                } else {
                    return TraitItem;
3168 3169 3170 3171 3172 3173 3174
                }
            }
        }

        NoSuggestion
    }

3175
    fn find_best_match(&mut self, name: &str) -> SuggestionType {
3176
        if let Some(macro_name) = self.session.available_macros
3177
                                  .borrow().iter().find(|n| n.as_str() == name) {
3178 3179 3180
            return SuggestionType::Macro(format!("{}!", macro_name));
        }

3181 3182 3183 3184
        let names = self.value_ribs
                    .iter()
                    .rev()
                    .flat_map(|rib| rib.bindings.keys());
3185

3186 3187 3188
        if let Some(found) = find_best_match_for_name(names, name, None) {
            if name != &*found {
                return SuggestionType::Function(found);
3189
            }
3190
        } SuggestionType::NotFound
3191 3192
    }

E
Eduard Burtescu 已提交
3193
    fn resolve_expr(&mut self, expr: &Expr) {
P
Patrick Walton 已提交
3194 3195
        // First, record candidate traits for this expression if it could
        // result in the invocation of a method call.
3196 3197 3198

        self.record_candidate_traits_for_expr_if_necessary(expr);

3199
        // Next, resolve the node.
3200
        match expr.node {
3201
            ExprPath(ref maybe_qself, ref path) => {
C
corentih 已提交
3202 3203 3204 3205 3206 3207 3208 3209 3210 3211
                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);
3212
                        intravisit::walk_expr(self, expr);
C
corentih 已提交
3213 3214 3215 3216
                        return;
                    }
                    ResolveAttempt(resolution) => resolution,
                };
3217

3218 3219
                // This is a local path in the value namespace. Walk through
                // scopes looking for it.
3220
                if let Some(path_res) = resolution {
3221
                    // Check if struct variant
3222
                    let is_struct_variant = if let Def::Variant(_, variant_id) = path_res.base_def {
3223 3224 3225 3226 3227 3228
                        self.structs.contains_key(&variant_id)
                    } else {
                        false
                    };
                    if is_struct_variant {
                        let _ = self.structs.contains_key(&path_res.base_def.def_id());
3229
                        let path_name = path_names_to_string(path, 0);
3230

N
Nick Cameron 已提交
3231 3232 3233
                        let mut err = resolve_struct_error(self,
                                        expr.span,
                                        ResolutionError::StructVariantUsedAsFunction(&*path_name));
3234

C
corentih 已提交
3235
                        let msg = format!("did you mean to write: `{} {{ /* fields */ }}`?",
3236 3237
                                          path_name);
                        if self.emit_errors {
N
Nick Cameron 已提交
3238
                            err.fileline_help(expr.span, &msg);
3239
                        } else {
N
Nick Cameron 已提交
3240
                            err.span_help(expr.span, &msg);
3241
                        }
N
Nick Cameron 已提交
3242
                        err.emit();
3243
                        self.record_def(expr.id, err_path_resolution());
3244
                    } else {
3245
                        // Write the result into the def map.
3246
                        debug!("(resolving expr) resolved `{}`",
3247
                               path_names_to_string(path, 0));
3248

3249 3250
                        // Partial resolutions will need the set of traits in scope,
                        // so they can be completed during typeck.
3251
                        if path_res.depth != 0 {
3252
                            let method_name = path.segments.last().unwrap().identifier.name;
3253
                            let traits = self.get_traits_containing_item(method_name);
3254 3255 3256
                            self.trait_map.insert(expr.id, traits);
                        }

3257
                        self.record_def(expr.id, path_res);
3258
                    }
3259 3260 3261 3262 3263
                } 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.)
3264
                    let path_name = path_names_to_string(path, 0);
3265 3266 3267
                    let type_res = self.with_no_errors(|this| {
                        this.resolve_path(expr.id, path, 0, TypeNS, false)
                    });
3268 3269

                    self.record_def(expr.id, err_path_resolution());
3270
                    match type_res.map(|r| r.base_def) {
3271
                        Some(Def::Struct(..)) => {
N
Nick Cameron 已提交
3272 3273 3274
                            let mut err = resolve_struct_error(self,
                                expr.span,
                                ResolutionError::StructVariantUsedAsFunction(&*path_name));
3275

C
corentih 已提交
3276 3277 3278
                            let msg = format!("did you mean to write: `{} {{ /* fields */ }}`?",
                                              path_name);
                            if self.emit_errors {
N
Nick Cameron 已提交
3279
                                err.fileline_help(expr.span, &msg);
C
corentih 已提交
3280
                            } else {
N
Nick Cameron 已提交
3281
                                err.span_help(expr.span, &msg);
3282
                            }
N
Nick Cameron 已提交
3283
                            err.emit();
C
corentih 已提交
3284
                        }
3285 3286 3287 3288 3289 3290 3291 3292 3293 3294 3295 3296 3297
                        _ => {
                            // 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
                            });
3298

3299
                            if method_scope && special_names::self_.as_str() == &path_name[..] {
C
corentih 已提交
3300 3301 3302
                                resolve_error(self,
                                              expr.span,
                                              ResolutionError::SelfNotAvailableInStaticMethod);
3303 3304 3305 3306 3307 3308
                            } 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
3309
                                        match self.find_best_match(&path_name) {
3310 3311 3312 3313 3314 3315
                                            SuggestionType::Macro(s) => {
                                                format!("the macro `{}`", s)
                                            }
                                            SuggestionType::Function(s) => format!("`{}`", s),
                                            SuggestionType::NotFound => "".to_string(),
                                        }
3316 3317 3318
                                    }
                                    Field => format!("`self.{}`", path_name),
                                    Method |
C
corentih 已提交
3319
                                    TraitItem => format!("to call `self.{}`", path_name),
3320 3321
                                    TraitMethod(path_str) |
                                    StaticMethod(path_str) =>
C
corentih 已提交
3322
                                        format!("to call `{}::{}`", path_str, path_name),
3323 3324
                                };

3325
                                let mut context =  UnresolvedNameContext::Other;
3326
                                if !msg.is_empty() {
3327 3328 3329 3330 3331 3332 3333
                                    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<_>>();
3334
                                    let current_module = self.current_module;
3335 3336

                                    match self.resolve_module_path(current_module,
J
Jeffrey Seyfried 已提交
3337 3338 3339
                                                                   &name_path[..],
                                                                   UseLexicalScope,
                                                                   expr.span) {
3340 3341 3342 3343 3344
                                        Success(_) => {
                                            context = UnresolvedNameContext::PathIsMod(expr.id);
                                        },
                                        _ => {},
                                    };
3345
                                }
3346

3347 3348
                                resolve_error(self,
                                              expr.span,
3349 3350
                                              ResolutionError::UnresolvedName(
                                                  &*path_name, &*msg, context));
3351
                            }
V
Vincent Belliard 已提交
3352
                        }
3353 3354 3355
                    }
                }

3356
                intravisit::walk_expr(self, expr);
3357 3358
            }

3359
            ExprStruct(ref path, _, _) => {
3360 3361 3362
                // 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.
3363
                match self.resolve_path(expr.id, path, 0, TypeNS, false) {
3364
                    Some(definition) => self.record_def(expr.id, definition),
3365 3366
                    None => {
                        debug!("(resolving expression) didn't find struct def",);
3367

3368 3369
                        resolve_error(self,
                                      path.span,
3370
                                      ResolutionError::DoesNotNameAStruct(
3371 3372
                                                                &*path_names_to_string(path, 0))
                                     );
3373
                        self.record_def(expr.id, err_path_resolution());
3374 3375 3376
                    }
                }

3377
                intravisit::walk_expr(self, expr);
3378 3379
            }

P
Pythoner6 已提交
3380
            ExprLoop(_, Some(label)) | ExprWhile(_, _, Some(label)) => {
3381
                self.with_label_rib(|this| {
3382
                    let def_like = DlDef(Def::Label(expr.id));
3383

3384
                    {
3385
                        let rib = this.label_ribs.last_mut().unwrap();
3386
                        rib.bindings.insert(label.name, def_like);
3387
                    }
3388

3389
                    intravisit::walk_expr(this, expr);
3390
                })
3391 3392
            }

3393
            ExprBreak(Some(label)) | ExprAgain(Some(label)) => {
3394
                match self.search_label(label.node.name) {
3395
                    None => {
3396
                        self.record_def(expr.id, err_path_resolution());
3397
                        resolve_error(self,
3398 3399
                                      label.span,
                                      ResolutionError::UndeclaredLabel(&label.node.name.as_str()))
3400
                    }
3401
                    Some(DlDef(def @ Def::Label(_))) => {
3402
                        // Since this def is a label, it is never read.
C
corentih 已提交
3403 3404 3405 3406 3407 3408
                        self.record_def(expr.id,
                                        PathResolution {
                                            base_def: def,
                                            last_private: LastMod(AllPublic),
                                            depth: 0,
                                        })
3409 3410
                    }
                    Some(_) => {
C
corentih 已提交
3411
                        self.session.span_bug(expr.span, "label wasn't mapped to a label def!")
3412 3413 3414 3415
                    }
                }
            }

B
Brian Anderson 已提交
3416
            _ => {
3417
                intravisit::walk_expr(self, expr);
3418 3419 3420 3421
            }
        }
    }

E
Eduard Burtescu 已提交
3422
    fn record_candidate_traits_for_expr_if_necessary(&mut self, expr: &Expr) {
3423
        match expr.node {
3424
            ExprField(_, name) => {
3425 3426 3427 3428
                // 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.
3429
                let traits = self.get_traits_containing_item(name.node);
3430
                self.trait_map.insert(expr.id, traits);
3431
            }
3432
            ExprMethodCall(name, _, _) => {
C
corentih 已提交
3433
                debug!("(recording candidate traits for expr) recording traits for {}",
3434
                       expr.id);
3435
                let traits = self.get_traits_containing_item(name.node);
3436
                self.trait_map.insert(expr.id, traits);
3437
            }
3438
            _ => {
3439 3440 3441 3442 3443
                // Nothing to do.
            }
        }
    }

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

C
corentih 已提交
3447
        fn add_trait_info(found_traits: &mut Vec<DefId>, trait_def_id: DefId, name: Name) {
3448
            debug!("(adding trait info) found trait {:?} for method '{}'",
C
corentih 已提交
3449 3450
                   trait_def_id,
                   name);
E
Eduard Burtescu 已提交
3451 3452
            found_traits.push(trait_def_id);
        }
3453

3454
        let mut found_traits = Vec::new();
3455
        let mut search_module = self.current_module;
E
Eduard Burtescu 已提交
3456 3457
        loop {
            // Look for the current trait.
3458 3459
            match self.current_trait_ref {
                Some((trait_def_id, _)) => {
3460
                    if self.trait_item_map.contains_key(&(name, trait_def_id)) {
3461
                        add_trait_info(&mut found_traits, trait_def_id, name);
3462 3463
                    }
                }
3464
                None => {} // Nothing to do.
E
Eduard Burtescu 已提交
3465
            }
3466

E
Eduard Burtescu 已提交
3467
            // Look for trait children.
3468
            build_reduced_graph::populate_module_if_necessary(self, &search_module);
3469

3470 3471
            search_module.for_each_child(|_, ns, name_binding| {
                if ns != TypeNS { return }
3472 3473
                let trait_def_id = match name_binding.def() {
                    Some(Def::Trait(trait_def_id)) => trait_def_id,
3474
                    Some(..) | None => return,
3475 3476 3477
                };
                if self.trait_item_map.contains_key(&(name, trait_def_id)) {
                    add_trait_info(&mut found_traits, trait_def_id, name);
3478 3479
                    let trait_name = self.get_trait_name(trait_def_id);
                    self.record_use(trait_name, TypeNS, name_binding);
3480
                }
3481
            });
3482

3483 3484 3485 3486 3487 3488 3489 3490 3491 3492
            // Look for shadowed traits.
            for binding in search_module.shadowed_traits.borrow().iter() {
                let did = binding.def().unwrap().def_id();
                if self.trait_item_map.contains_key(&(name, did)) {
                    add_trait_info(&mut found_traits, did, name);
                    let trait_name = self.get_trait_name(did);
                    self.record_use(trait_name, TypeNS, binding);
                }
            }

3493
            match search_module.parent_link {
E
Eduard Burtescu 已提交
3494 3495
                NoParentLink | ModuleParentLink(..) => break,
                BlockParentLink(parent_module, _) => {
3496
                    search_module = parent_module;
3497
                }
E
Eduard Burtescu 已提交
3498
            }
3499 3500
        }

E
Eduard Burtescu 已提交
3501
        found_traits
3502 3503
    }

3504 3505
    fn record_def(&mut self, node_id: NodeId, resolution: PathResolution) {
        debug!("(recording def) recording {:?} for {}", resolution, node_id);
C
corentih 已提交
3506 3507 3508 3509
        assert!(match resolution.last_private {
                    LastImport{..} => false,
                    _ => true,
                },
3510
                "Import should only be used for `use` directives");
3511

3512 3513
        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 已提交
3514 3515 3516 3517
            self.session.span_bug(span,
                                  &format!("path resolved multiple times ({:?} before, {:?} now)",
                                           prev_res,
                                           resolution));
3518
        }
3519 3520
    }

F
Felix S. Klock II 已提交
3521
    fn enforce_default_binding_mode(&mut self,
C
corentih 已提交
3522 3523 3524
                                    pat: &Pat,
                                    pat_binding_mode: BindingMode,
                                    descr: &str) {
3525
        match pat_binding_mode {
3526
            BindByValue(_) => {}
A
Alex Crichton 已提交
3527
            BindByRef(..) => {
3528 3529
                resolve_error(self,
                              pat.span,
3530
                              ResolutionError::CannotUseRefBindingModeWith(descr));
3531 3532 3533
            }
        }
    }
3534 3535
}

3536 3537 3538 3539 3540 3541 3542 3543 3544 3545

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("::")
        }
3546
        result.push_str(&name.as_str());
C
corentih 已提交
3547
    }
3548 3549 3550 3551
    result
}

fn path_names_to_string(path: &Path, depth: usize) -> String {
C
corentih 已提交
3552
    let names: Vec<ast::Name> = path.segments[..path.segments.len() - depth]
3553 3554 3555 3556 3557 3558 3559
                                    .iter()
                                    .map(|seg| seg.identifier.name)
                                    .collect();
    names_to_string(&names[..])
}

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

3563
    fn collect_mod<'a>(names: &mut Vec<ast::Name>, module: Module<'a>) {
3564 3565 3566 3567
        match module.parent_link {
            NoParentLink => {}
            ModuleParentLink(ref module, name) => {
                names.push(name);
3568
                collect_mod(names, module);
3569 3570 3571 3572
            }
            BlockParentLink(ref module, _) => {
                // danger, shouldn't be ident?
                names.push(special_idents::opaque.name);
3573
                collect_mod(names, module);
3574 3575 3576 3577 3578
            }
        }
    }
    collect_mod(&mut names, module);

3579
    if names.is_empty() {
3580 3581 3582 3583 3584
        return "???".to_string();
    }
    names_to_string(&names.into_iter().rev().collect::<Vec<ast::Name>>())
}

3585 3586
fn err_path_resolution() -> PathResolution {
    PathResolution {
3587
        base_def: Def::Err,
3588 3589 3590 3591 3592
        last_private: LastMod(AllPublic),
        depth: 0,
    }
}

3593

3594
pub struct CrateMap {
J
Jonathan S 已提交
3595
    pub def_map: RefCell<DefMap>,
3596
    pub freevars: FreevarMap,
3597
    pub export_map: ExportMap,
3598 3599
    pub trait_map: TraitMap,
    pub external_exports: ExternalExports,
C
corentih 已提交
3600
    pub glob_map: Option<GlobMap>,
3601 3602
}

N
Niko Matsakis 已提交
3603
#[derive(PartialEq,Copy, Clone)]
3604 3605
pub enum MakeGlobMap {
    Yes,
C
corentih 已提交
3606
    No,
3607 3608
}

3609
/// Entry point to crate resolution.
3610
pub fn resolve_crate<'a, 'tcx>(session: &'a Session,
3611
                               ast_map: &'a hir_map::Map<'tcx>,
3612 3613
                               make_glob_map: MakeGlobMap)
                               -> CrateMap {
3614 3615 3616 3617 3618 3619 3620 3621 3622
    // 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);

3623
    let krate = ast_map.krate();
3624 3625
    let arenas = Resolver::arenas();
    let mut resolver = create_resolver(session, ast_map, krate, make_glob_map, &arenas, None);
3626 3627 3628 3629 3630

    resolver.resolve_crate(krate);

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

3631
    CrateMap {
3632 3633
        def_map: resolver.def_map,
        freevars: resolver.freevars,
3634
        export_map: resolver.export_map,
3635 3636
        trait_map: resolver.trait_map,
        external_exports: resolver.external_exports,
3637
        glob_map: if resolver.make_glob_map {
C
corentih 已提交
3638 3639 3640 3641
            Some(resolver.glob_map)
        } else {
            None
        },
3642
    }
3643
}
3644

3645 3646 3647 3648 3649 3650 3651 3652
/// 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 已提交
3653
pub fn create_resolver<'a, 'tcx>(session: &'a Session,
3654
                                 ast_map: &'a hir_map::Map<'tcx>,
G
Garming Sam 已提交
3655 3656
                                 krate: &'a Crate,
                                 make_glob_map: MakeGlobMap,
3657
                                 arenas: &'a ResolverArenas<'a>,
3658
                                 callback: Option<Box<Fn(hir_map::Node, &mut bool) -> bool>>)
G
Garming Sam 已提交
3659
                                 -> Resolver<'a, 'tcx> {
3660
    let mut resolver = Resolver::new(session, ast_map, make_glob_map, arenas);
G
Garming Sam 已提交
3661 3662 3663 3664 3665 3666 3667 3668 3669 3670

    resolver.callback = callback;

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

    resolve_imports::resolve_imports(&mut resolver);

    resolver
}

3671
__build_diagnostic_array! { librustc_resolve, DIAGNOSTICS }