lib.rs 132.8 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
#![deny(warnings)]
19

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

C
corentih 已提交
25 26 27 28
#[macro_use]
extern crate log;
#[macro_use]
extern crate syntax;
29 30
extern crate syntax_pos;
extern crate rustc_errors as errors;
31
extern crate arena;
C
corentih 已提交
32
#[macro_use]
33 34
extern crate rustc;

S
Steven Fackler 已提交
35 36 37 38
use self::Namespace::*;
use self::TypeParameters::*;
use self::RibKind::*;

39
use rustc::hir::map::{Definitions, DefCollector};
40
use rustc::hir::{self, PrimTy, TyBool, TyChar, TyFloat, TyInt, TyUint, TyStr};
41
use rustc::middle::cstore::CrateLoader;
42 43
use rustc::session::Session;
use rustc::lint;
44
use rustc::hir::def::*;
45
use rustc::hir::def_id::{CrateNum, CRATE_DEF_INDEX, LOCAL_CRATE, DefId};
46
use rustc::ty;
S
Seo Sanghyeon 已提交
47
use rustc::hir::{Freevar, FreevarMap, TraitCandidate, TraitMap, GlobMap};
48
use rustc::util::nodemap::{NodeMap, NodeSet, FxHashMap, FxHashSet};
49

J
Jeffrey Seyfried 已提交
50
use syntax::ext::hygiene::{Mark, SyntaxContext};
51
use syntax::ast::{self, Name, NodeId, Ident, SpannedIdent, FloatTy, IntTy, UintTy};
J
Jeffrey Seyfried 已提交
52
use syntax::ext::base::SyntaxExtension;
J
Jeffrey Seyfried 已提交
53
use syntax::ext::base::Determinacy::{Determined, Undetermined};
54
use syntax::symbol::{Symbol, keywords};
55
use syntax::util::lev_distance::find_best_match_for_name;
56

57
use syntax::visit::{self, FnKind, Visitor};
58
use syntax::attr;
59 60 61
use syntax::ast::{Arm, BindingMode, Block, Crate, Expr, ExprKind};
use syntax::ast::{FnDecl, ForeignItem, ForeignItemKind, Generics};
use syntax::ast::{Item, ItemKind, ImplItem, ImplItemKind};
62
use syntax::ast::{Local, Mutability, Pat, PatKind, Path};
J
Jeffrey Seyfried 已提交
63
use syntax::ast::{QSelf, TraitItemKind, TraitRef, Ty, TyKind};
64
use syntax::feature_gate::{emit_feature_err, GateIssue};
65

66
use syntax_pos::{Span, DUMMY_SP, MultiSpan};
67 68
use errors::DiagnosticBuilder;

69
use std::cell::{Cell, RefCell};
70
use std::fmt;
71
use std::mem::replace;
72
use std::rc::Rc;
73

74
use resolve_imports::{ImportDirective, ImportDirectiveSubclass, NameResolution, ImportResolver};
J
Jeffrey Seyfried 已提交
75
use macros::{InvocationData, LegacyBinding, LegacyScope};
76

77 78
// NB: This module needs to be declared first so diagnostics are
// registered before they are used.
J
Jeffrey Seyfried 已提交
79
mod diagnostics;
80

81
mod macros;
A
Alex Crichton 已提交
82
mod check_unused;
83
mod build_reduced_graph;
84
mod resolve_imports;
85

86 87 88
/// A free importable items suggested in case of resolution failure.
struct ImportSuggestion {
    path: Path,
89 90
}

91 92 93 94 95
/// A field or associated item from self type suggested in case of resolution failure.
enum AssocSuggestion {
    Field,
    MethodWithSelf,
    AssocItem,
96 97
}

J
Jeffrey Seyfried 已提交
98
enum ResolutionError<'a> {
99
    /// error E0401: can't use type parameters from outer function
100
    TypeParametersFromOuterFunction,
101
    /// error E0402: cannot use an outer type parameter in this context
102
    OuterTypeParameterContext,
103
    /// error E0403: the name is already used for a type parameter in this type parameter list
C
Chris Stankus 已提交
104
    NameAlreadyUsedInTypeParameterList(Name, &'a Span),
105
    /// error E0407: method is not a member of trait
106
    MethodNotMemberOfTrait(Name, &'a str),
107 108 109 110
    /// 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),
M
Manish Goregaokar 已提交
111 112
    /// error E0408: variable `{}` from pattern #{} is not bound in pattern #{}
    VariableNotBoundInPattern(Name, usize, usize),
113
    /// error E0409: variable is bound with different mode in pattern #{} than in pattern #1
M
Mikhail Modin 已提交
114
    VariableBoundWithDifferentMode(Name, usize, Span),
115
    /// error E0415: identifier is bound more than once in this parameter list
116
    IdentifierBoundMoreThanOnceInParameterList(&'a str),
117
    /// error E0416: identifier is bound more than once in the same pattern
118
    IdentifierBoundMoreThanOnceInSamePattern(&'a str),
119
    /// error E0426: use of undeclared label
120
    UndeclaredLabel(&'a str),
121
    /// error E0429: `self` imports are only allowed within a { } list
122
    SelfImportsOnlyAllowedWithin,
123
    /// error E0430: `self` import can only appear once in the list
124
    SelfImportCanOnlyAppearOnceInTheList,
125
    /// error E0431: `self` import can only appear in an import list with a non-empty prefix
126
    SelfImportOnlyInImportListWithNonEmptyPrefix,
127
    /// error E0432: unresolved import
128
    UnresolvedImport(Option<(&'a str, &'a str)>),
129
    /// error E0433: failed to resolve
130
    FailedToResolve(&'a str),
131
    /// error E0434: can't capture dynamic environment in a fn item
132
    CannotCaptureDynamicEnvironmentInFnItem,
133
    /// error E0435: attempt to use a non-constant value in a constant
134
    AttemptToUseNonConstantValueInConstant,
135
    /// error E0530: X bindings cannot shadow Ys
136
    BindingShadowsSomethingUnacceptable(&'a str, Name, &'a NameBinding<'a>),
137 138
}

139 140 141
fn resolve_error<'sess, 'a>(resolver: &'sess Resolver,
                            span: Span,
                            resolution_error: ResolutionError<'a>) {
N
Nick Cameron 已提交
142
    resolve_struct_error(resolver, span, resolution_error).emit();
N
Nick Cameron 已提交
143 144
}

145 146 147 148
fn resolve_struct_error<'sess, 'a>(resolver: &'sess Resolver,
                                   span: Span,
                                   resolution_error: ResolutionError<'a>)
                                   -> DiagnosticBuilder<'sess> {
N
Nick Cameron 已提交
149
    match resolution_error {
150
        ResolutionError::TypeParametersFromOuterFunction => {
151 152 153 154 155
            let mut err = struct_span_err!(resolver.session,
                                           span,
                                           E0401,
                                           "can't use type parameters from outer function; \
                                           try using a local type parameter instead");
156
            err.span_label(span, &format!("use of type variable from outer function"));
157
            err
C
corentih 已提交
158
        }
159
        ResolutionError::OuterTypeParameterContext => {
N
Nick Cameron 已提交
160 161 162 163
            struct_span_err!(resolver.session,
                             span,
                             E0402,
                             "cannot use an outer type parameter in this context")
C
corentih 已提交
164
        }
C
Chris Stankus 已提交
165 166 167 168 169 170 171 172 173 174
        ResolutionError::NameAlreadyUsedInTypeParameterList(name, first_use_span) => {
             let mut err = struct_span_err!(resolver.session,
                                            span,
                                            E0403,
                                            "the name `{}` is already used for a type parameter \
                                            in this type parameter list",
                                            name);
             err.span_label(span, &format!("already used"));
             err.span_label(first_use_span.clone(), &format!("first use of `{}`", name));
             err
C
corentih 已提交
175
        }
176
        ResolutionError::MethodNotMemberOfTrait(method, trait_) => {
C
crypto-universe 已提交
177 178 179 180 181 182
            let mut err = struct_span_err!(resolver.session,
                                           span,
                                           E0407,
                                           "method `{}` is not a member of trait `{}`",
                                           method,
                                           trait_);
183
            err.span_label(span, &format!("not a member of trait `{}`", trait_));
C
crypto-universe 已提交
184
            err
C
corentih 已提交
185
        }
186
        ResolutionError::TypeNotMemberOfTrait(type_, trait_) => {
S
Shyam Sundar B 已提交
187
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
188 189 190 191
                             span,
                             E0437,
                             "type `{}` is not a member of trait `{}`",
                             type_,
S
Shyam Sundar B 已提交
192
                             trait_);
193
            err.span_label(span, &format!("not a member of trait `{}`", trait_));
S
Shyam Sundar B 已提交
194
            err
C
corentih 已提交
195
        }
196
        ResolutionError::ConstNotMemberOfTrait(const_, trait_) => {
S
Shyam Sundar B 已提交
197
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
198 199 200 201
                             span,
                             E0438,
                             "const `{}` is not a member of trait `{}`",
                             const_,
S
Shyam Sundar B 已提交
202
                             trait_);
203
            err.span_label(span, &format!("not a member of trait `{}`", trait_));
S
Shyam Sundar B 已提交
204
            err
C
corentih 已提交
205
        }
M
Manish Goregaokar 已提交
206
        ResolutionError::VariableNotBoundInPattern(variable_name, from, to) => {
207
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
208 209
                             span,
                             E0408,
M
Manish Goregaokar 已提交
210
                             "variable `{}` from pattern #{} is not bound in pattern #{}",
N
Nick Cameron 已提交
211
                             variable_name,
M
Manish Goregaokar 已提交
212
                             from,
213 214 215
                             to);
            err.span_label(span, &format!("pattern doesn't bind `{}`", variable_name));
            err
C
corentih 已提交
216
        }
M
Mikhail Modin 已提交
217 218 219 220
        ResolutionError::VariableBoundWithDifferentMode(variable_name,
                                                        pattern_number,
                                                        first_binding_span) => {
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
221 222 223 224 225
                             span,
                             E0409,
                             "variable `{}` is bound with different mode in pattern #{} than in \
                              pattern #1",
                             variable_name,
M
Mikhail Modin 已提交
226 227 228 229
                             pattern_number);
            err.span_label(span, &format!("bound in different ways"));
            err.span_label(first_binding_span, &format!("first binding"));
            err
C
corentih 已提交
230
        }
231
        ResolutionError::IdentifierBoundMoreThanOnceInParameterList(identifier) => {
232
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
233 234 235
                             span,
                             E0415,
                             "identifier `{}` is bound more than once in this parameter list",
236
                             identifier);
237
            err.span_label(span, &format!("used as parameter more than once"));
238
            err
C
corentih 已提交
239
        }
240
        ResolutionError::IdentifierBoundMoreThanOnceInSamePattern(identifier) => {
241
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
242 243 244
                             span,
                             E0416,
                             "identifier `{}` is bound more than once in the same pattern",
245
                             identifier);
246
            err.span_label(span, &format!("used in a pattern more than once"));
247
            err
C
corentih 已提交
248
        }
249
        ResolutionError::UndeclaredLabel(name) => {
C
crypto-universe 已提交
250 251 252 253 254 255 256
            let mut err = struct_span_err!(resolver.session,
                                           span,
                                           E0426,
                                           "use of undeclared label `{}`",
                                           name);
            err.span_label(span, &format!("undeclared label `{}`",&name));
            err
C
corentih 已提交
257
        }
258
        ResolutionError::SelfImportsOnlyAllowedWithin => {
N
Nick Cameron 已提交
259 260 261 262 263
            struct_span_err!(resolver.session,
                             span,
                             E0429,
                             "{}",
                             "`self` imports are only allowed within a { } list")
C
corentih 已提交
264
        }
265
        ResolutionError::SelfImportCanOnlyAppearOnceInTheList => {
N
Nick Cameron 已提交
266 267 268 269
            struct_span_err!(resolver.session,
                             span,
                             E0430,
                             "`self` import can only appear once in the list")
C
corentih 已提交
270
        }
271
        ResolutionError::SelfImportOnlyInImportListWithNonEmptyPrefix => {
N
Nick Cameron 已提交
272 273 274 275 276
            struct_span_err!(resolver.session,
                             span,
                             E0431,
                             "`self` import can only appear in an import list with a \
                              non-empty prefix")
277
        }
278
        ResolutionError::UnresolvedImport(name) => {
279
            let msg = match name {
K
Knight 已提交
280
                Some((n, _)) => format!("unresolved import `{}`", n),
C
corentih 已提交
281
                None => "unresolved import".to_owned(),
282
            };
K
Knight 已提交
283 284 285 286 287
            let mut err = struct_span_err!(resolver.session, span, E0432, "{}", msg);
            if let Some((_, p)) = name {
                err.span_label(span, &p);
            }
            err
C
corentih 已提交
288
        }
289
        ResolutionError::FailedToResolve(msg) => {
J
Jonathan Turner 已提交
290 291
            let mut err = struct_span_err!(resolver.session, span, E0433,
                                           "failed to resolve. {}", msg);
292 293
            err.span_label(span, &msg);
            err
C
corentih 已提交
294
        }
295
        ResolutionError::CannotCaptureDynamicEnvironmentInFnItem => {
N
Nick Cameron 已提交
296 297 298 299 300 301
            struct_span_err!(resolver.session,
                             span,
                             E0434,
                             "{}",
                             "can't capture dynamic environment in a fn item; use the || { ... } \
                              closure form instead")
C
corentih 已提交
302 303
        }
        ResolutionError::AttemptToUseNonConstantValueInConstant => {
S
Shyam Sundar B 已提交
304
            let mut err = struct_span_err!(resolver.session,
N
Nick Cameron 已提交
305 306
                             span,
                             E0435,
S
Shyam Sundar B 已提交
307 308 309
                             "attempt to use a non-constant value in a constant");
            err.span_label(span, &format!("non-constant used with constant"));
            err
C
corentih 已提交
310
        }
311
        ResolutionError::BindingShadowsSomethingUnacceptable(what_binding, name, binding) => {
312
            let shadows_what = PathResolution::new(binding.def()).kind_name();
313 314
            let mut err = struct_span_err!(resolver.session,
                                           span,
315
                                           E0530,
316 317
                                           "{}s cannot shadow {}s", what_binding, shadows_what);
            err.span_label(span, &format!("cannot be named the same as a {}", shadows_what));
318 319 320
            let participle = if binding.is_import() { "imported" } else { "defined" };
            let msg = &format!("a {} `{}` is {} here", shadows_what, name, participle);
            err.span_label(binding.span, msg);
321 322
            err
        }
N
Nick Cameron 已提交
323
    }
324 325
}

N
Niko Matsakis 已提交
326
#[derive(Copy, Clone)]
327
struct BindingInfo {
328
    span: Span,
329
    binding_mode: BindingMode,
330 331 332
}

// Map from the name in a pattern to its binding mode.
333
type BindingMap = FxHashMap<Ident, BindingInfo>;
334

335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
enum PatternSource {
    Match,
    IfLet,
    WhileLet,
    Let,
    For,
    FnParam,
}

impl PatternSource {
    fn is_refutable(self) -> bool {
        match self {
            PatternSource::Match | PatternSource::IfLet | PatternSource::WhileLet => true,
            PatternSource::Let | PatternSource::For | PatternSource::FnParam  => false,
        }
    }
    fn descr(self) -> &'static str {
        match self {
            PatternSource::Match => "match binding",
            PatternSource::IfLet => "if let binding",
            PatternSource::WhileLet => "while let binding",
            PatternSource::Let => "let binding",
            PatternSource::For => "for binding",
            PatternSource::FnParam => "function parameter",
        }
    }
362 363
}

364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
enum PathSource<'a> {
    // Type paths `Path`.
    Type,
    // Trait paths in bounds or impls.
    Trait,
    // Expression paths `path`, with optional parent context.
    Expr(Option<&'a ExprKind>),
    // Paths in path patterns `Path`.
    Pat,
    // Paths in struct expressions and patterns `Path { .. }`.
    Struct,
    // Paths in tuple struct patterns `Path(..)`.
    TupleStruct,
    // `m::A::B` in `<T as m::A>::B::C`.
    TraitItem(Namespace),
    // Path in `pub(path)`
    Visibility,
    // Path in `use a::b::{...};`
    ImportPrefix,
}

impl<'a> PathSource<'a> {
    fn namespace(self) -> Namespace {
        match self {
            PathSource::Type | PathSource::Trait | PathSource::Struct |
            PathSource::Visibility | PathSource::ImportPrefix => TypeNS,
            PathSource::Expr(..) | PathSource::Pat | PathSource::TupleStruct => ValueNS,
            PathSource::TraitItem(ns) => ns,
        }
    }

    fn global_by_default(self) -> bool {
        match self {
            PathSource::Visibility | PathSource::ImportPrefix => true,
            PathSource::Type | PathSource::Expr(..) | PathSource::Pat |
            PathSource::Struct | PathSource::TupleStruct |
            PathSource::Trait | PathSource::TraitItem(..) => false,
        }
    }

    fn defer_to_typeck(self) -> bool {
        match self {
            PathSource::Type | PathSource::Expr(..) | PathSource::Pat |
            PathSource::Struct | PathSource::TupleStruct => true,
            PathSource::Trait | PathSource::TraitItem(..) |
            PathSource::Visibility | PathSource::ImportPrefix => false,
        }
    }

    fn descr_expected(self) -> &'static str {
        match self {
            PathSource::Type => "type",
            PathSource::Trait => "trait",
            PathSource::Pat => "unit struct/variant or constant",
            PathSource::Struct => "struct, variant or union type",
            PathSource::TupleStruct => "tuple struct/variant",
            PathSource::Visibility => "module",
            PathSource::ImportPrefix => "module or enum",
            PathSource::TraitItem(ns) => match ns {
                TypeNS => "associated type",
                ValueNS => "method or associated constant",
                MacroNS => bug!("associated macro"),
            },
            PathSource::Expr(parent) => match parent {
                // "function" here means "anything callable" rather than `Def::Fn`,
                // this is not precise but usually more helpful than just "value".
                Some(&ExprKind::Call(..)) => "function",
                _ => "value",
            },
        }
    }

    fn is_expected(self, def: Def) -> bool {
        match self {
            PathSource::Type => match def {
                Def::Struct(..) | Def::Union(..) | Def::Enum(..) |
                Def::Trait(..) | Def::TyAlias(..) | Def::AssociatedTy(..) |
                Def::PrimTy(..) | Def::TyParam(..) | Def::SelfTy(..) => true,
                _ => false,
            },
            PathSource::Trait => match def {
                Def::Trait(..) => true,
                _ => false,
            },
            PathSource::Expr(..) => match def {
                Def::StructCtor(_, CtorKind::Const) | Def::StructCtor(_, CtorKind::Fn) |
                Def::VariantCtor(_, CtorKind::Const) | Def::VariantCtor(_, CtorKind::Fn) |
                Def::Const(..) | Def::Static(..) | Def::Local(..) | Def::Upvar(..) |
                Def::Fn(..) | Def::Method(..) | Def::AssociatedConst(..) => true,
                _ => false,
            },
            PathSource::Pat => match def {
                Def::StructCtor(_, CtorKind::Const) |
                Def::VariantCtor(_, CtorKind::Const) |
                Def::Const(..) | Def::AssociatedConst(..) => true,
                _ => false,
            },
            PathSource::TupleStruct => match def {
                Def::StructCtor(_, CtorKind::Fn) | Def::VariantCtor(_, CtorKind::Fn) => true,
                _ => false,
            },
            PathSource::Struct => match def {
                Def::Struct(..) | Def::Union(..) | Def::Variant(..) |
                Def::TyAlias(..) | Def::AssociatedTy(..) | Def::SelfTy(..) => true,
                _ => false,
            },
            PathSource::TraitItem(ns) => match def {
                Def::AssociatedConst(..) | Def::Method(..) if ns == ValueNS => true,
                Def::AssociatedTy(..) if ns == TypeNS => true,
                _ => false,
            },
            PathSource::ImportPrefix => match def {
                Def::Mod(..) | Def::Enum(..) => true,
                _ => false,
            },
            PathSource::Visibility => match def {
                Def::Mod(..) => true,
                _ => false,
            },
        }
    }

    fn error_code(self, has_unexpected_resolution: bool) -> &'static str {
        __diagnostic_used!(E0404);
        __diagnostic_used!(E0405);
        __diagnostic_used!(E0412);
        __diagnostic_used!(E0422);
        __diagnostic_used!(E0423);
        __diagnostic_used!(E0425);
        __diagnostic_used!(E0531);
        __diagnostic_used!(E0532);
        __diagnostic_used!(E0573);
        __diagnostic_used!(E0574);
        __diagnostic_used!(E0575);
        __diagnostic_used!(E0576);
        __diagnostic_used!(E0577);
        __diagnostic_used!(E0578);
        match (self, has_unexpected_resolution) {
            (PathSource::Trait, true) => "E0404",
            (PathSource::Trait, false) => "E0405",
            (PathSource::Type, true) => "E0573",
            (PathSource::Type, false) => "E0412",
            (PathSource::Struct, true) => "E0574",
            (PathSource::Struct, false) => "E0422",
            (PathSource::Expr(..), true) => "E0423",
            (PathSource::Expr(..), false) => "E0425",
            (PathSource::Pat, true) | (PathSource::TupleStruct, true) => "E0532",
            (PathSource::Pat, false) | (PathSource::TupleStruct, false) => "E0531",
            (PathSource::TraitItem(..), true) => "E0575",
            (PathSource::TraitItem(..), false) => "E0576",
            (PathSource::Visibility, true) | (PathSource::ImportPrefix, true) => "E0577",
            (PathSource::Visibility, false) | (PathSource::ImportPrefix, false) => "E0578",
        }
    }
}

N
Niko Matsakis 已提交
521
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
G
Garming Sam 已提交
522
pub enum Namespace {
523
    TypeNS,
C
corentih 已提交
524
    ValueNS,
525
    MacroNS,
526 527
}

J
Jeffrey Seyfried 已提交
528 529 530 531
#[derive(Clone, Default, Debug)]
pub struct PerNS<T> {
    value_ns: T,
    type_ns: T,
532
    macro_ns: Option<T>,
J
Jeffrey Seyfried 已提交
533 534 535 536 537 538 539 540
}

impl<T> ::std::ops::Index<Namespace> for PerNS<T> {
    type Output = T;
    fn index(&self, ns: Namespace) -> &T {
        match ns {
            ValueNS => &self.value_ns,
            TypeNS => &self.type_ns,
541
            MacroNS => self.macro_ns.as_ref().unwrap(),
J
Jeffrey Seyfried 已提交
542 543 544 545 546 547 548 549 550
        }
    }
}

impl<T> ::std::ops::IndexMut<Namespace> for PerNS<T> {
    fn index_mut(&mut self, ns: Namespace) -> &mut T {
        match ns {
            ValueNS => &mut self.value_ns,
            TypeNS => &mut self.type_ns,
551
            MacroNS => self.macro_ns.as_mut().unwrap(),
J
Jeffrey Seyfried 已提交
552 553 554 555
        }
    }
}

556 557
impl<'a, 'tcx> Visitor<'tcx> for Resolver<'a> {
    fn visit_item(&mut self, item: &'tcx Item) {
A
Alex Crichton 已提交
558
        self.resolve_item(item);
559
    }
560
    fn visit_arm(&mut self, arm: &'tcx Arm) {
A
Alex Crichton 已提交
561
        self.resolve_arm(arm);
562
    }
563
    fn visit_block(&mut self, block: &'tcx Block) {
A
Alex Crichton 已提交
564
        self.resolve_block(block);
565
    }
566
    fn visit_expr(&mut self, expr: &'tcx Expr) {
567
        self.resolve_expr(expr, None);
568
    }
569
    fn visit_local(&mut self, local: &'tcx Local) {
A
Alex Crichton 已提交
570
        self.resolve_local(local);
571
    }
572
    fn visit_ty(&mut self, ty: &'tcx Ty) {
573 574
        if let TyKind::Path(ref qself, ref path) = ty.node {
            self.smart_resolve_path(ty.id, qself.as_ref(), path, PathSource::Type);
575 576 577 578 579
        } else if let TyKind::ImplicitSelf = ty.node {
            let self_ty = keywords::SelfType.ident();
            let def = self.resolve_ident_in_lexical_scope(self_ty, TypeNS, Some(ty.span))
                          .map_or(Def::Err, |d| d.def());
            self.record_def(ty.id, PathResolution::new(def));
580 581 582 583 584 585
        } else if let TyKind::Array(ref element, ref length) = ty.node {
            self.visit_ty(element);
            self.with_constant_rib(|this| {
                this.visit_expr(length);
            });
            return;
586 587
        }
        visit::walk_ty(self, ty);
588
    }
589 590 591
    fn visit_poly_trait_ref(&mut self,
                            tref: &'tcx ast::PolyTraitRef,
                            m: &'tcx ast::TraitBoundModifier) {
592 593
        self.smart_resolve_path(tref.trait_ref.ref_id, None,
                                &tref.trait_ref.path, PathSource::Trait);
594
        visit::walk_poly_trait_ref(self, tref, m);
595
    }
C
corentih 已提交
596
    fn visit_variant(&mut self,
597 598
                     variant: &'tcx ast::Variant,
                     generics: &'tcx Generics,
C
corentih 已提交
599
                     item_id: ast::NodeId) {
600 601 602
        if let Some(ref dis_expr) = variant.node.disr_expr {
            // resolve the discriminator expr as a constant
            self.with_constant_rib(|this| {
603
                this.visit_expr(dis_expr);
604 605 606
            });
        }

607
        // `visit::walk_variant` without the discriminant expression.
C
corentih 已提交
608 609 610 611 612
        self.visit_variant_data(&variant.node.data,
                                variant.node.name,
                                generics,
                                item_id,
                                variant.span);
613
    }
614
    fn visit_foreign_item(&mut self, foreign_item: &'tcx ForeignItem) {
615
        let type_parameters = match foreign_item.node {
616
            ForeignItemKind::Fn(_, ref generics) => {
617
                HasTypeParameters(generics, ItemRibKind)
618
            }
619
            ForeignItemKind::Static(..) => NoTypeParameters,
620 621
        };
        self.with_type_parameter_rib(type_parameters, |this| {
622
            visit::walk_foreign_item(this, foreign_item);
623 624 625
        });
    }
    fn visit_fn(&mut self,
626 627
                function_kind: FnKind<'tcx>,
                declaration: &'tcx FnDecl,
628 629 630
                _: Span,
                node_id: NodeId) {
        let rib_kind = match function_kind {
V
Vadim Petrochenkov 已提交
631
            FnKind::ItemFn(_, generics, ..) => {
632 633 634
                self.visit_generics(generics);
                ItemRibKind
            }
635
            FnKind::Method(_, sig, _, _) => {
636
                self.visit_generics(&sig.generics);
V
Vadim Petrochenkov 已提交
637
                MethodRibKind(!sig.decl.has_self())
638
            }
639
            FnKind::Closure(_) => ClosureRibKind(node_id),
640
        };
641 642

        // Create a value rib for the function.
J
Jeffrey Seyfried 已提交
643
        self.ribs[ValueNS].push(Rib::new(rib_kind));
644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672

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

        // Add each argument to the rib.
        let mut bindings_list = FxHashMap();
        for argument in &declaration.inputs {
            self.resolve_pattern(&argument.pat, PatternSource::FnParam, &mut bindings_list);

            self.visit_ty(&argument.ty);

            debug!("(resolving function) recorded argument");
        }
        visit::walk_fn_ret_ty(self, &declaration.output);

        // Resolve the function body.
        match function_kind {
            FnKind::ItemFn(.., body) |
            FnKind::Method(.., body) => {
                self.visit_block(body);
            }
            FnKind::Closure(body) => {
                self.visit_expr(body);
            }
        };

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

        self.label_ribs.pop();
J
Jeffrey Seyfried 已提交
673
        self.ribs[ValueNS].pop();
674
    }
675
}
676

677
pub type ErrorMessage = Option<(Span, String)>;
678

N
Niko Matsakis 已提交
679
#[derive(Copy, Clone)]
680
enum TypeParameters<'a, 'b> {
681
    NoTypeParameters,
C
corentih 已提交
682
    HasTypeParameters(// Type parameters.
683
                      &'b Generics,
684

C
corentih 已提交
685
                      // The kind of the rib used for type parameters.
686
                      RibKind<'a>),
687 688
}

689
// The rib kind controls the translation of local
690
// definitions (`Def::Local`) to upvars (`Def::Upvar`).
N
Niko Matsakis 已提交
691
#[derive(Copy, Clone, Debug)]
692
enum RibKind<'a> {
693 694
    // No translation needs to be applied.
    NormalRibKind,
695

696 697
    // We passed through a closure scope at the given node ID.
    // Translate upvars as appropriate.
698
    ClosureRibKind(NodeId /* func id */),
699

700
    // We passed through an impl or trait and are now in one of its
701
    // methods. Allow references to ty params that impl or trait
702 703
    // binds. Disallow any other upvars (including other ty params that are
    // upvars).
704 705 706
    //
    // The boolean value represents the fact that this method is static or not.
    MethodRibKind(bool),
707

708 709
    // We passed through an item scope. Disallow upvars.
    ItemRibKind,
710 711

    // We're in a constant item. Can't refer to dynamic stuff.
C
corentih 已提交
712
    ConstantItemRibKind,
713

714 715
    // We passed through a module.
    ModuleRibKind(Module<'a>),
716 717

    // We passed through a `macro_rules!` statement with the given expansion
718
    MacroDefinition(Mark),
719 720
}

721
/// One local scope.
J
Jorge Aparicio 已提交
722
#[derive(Debug)]
723
struct Rib<'a> {
724
    bindings: FxHashMap<Ident, Def>,
725
    kind: RibKind<'a>,
B
Brian Anderson 已提交
726
}
727

728 729
impl<'a> Rib<'a> {
    fn new(kind: RibKind<'a>) -> Rib<'a> {
730
        Rib {
731
            bindings: FxHashMap(),
C
corentih 已提交
732
            kind: kind,
733
        }
734 735 736
    }
}

737
/// A definition along with the index of the rib it was found on
738
#[derive(Copy, Clone, Debug)]
739 740
struct LocalDef {
    ribs: Option<(Namespace, usize)>,
C
corentih 已提交
741
    def: Def,
742 743
}

744 745
enum LexicalScopeBinding<'a> {
    Item(&'a NameBinding<'a>),
J
Jeffrey Seyfried 已提交
746
    Def(Def),
747 748
}

749
impl<'a> LexicalScopeBinding<'a> {
750
    fn item(self) -> Option<&'a NameBinding<'a>> {
751
        match self {
752
            LexicalScopeBinding::Item(binding) => Some(binding),
753 754 755
            _ => None,
        }
    }
756 757 758 759 760 761 762

    fn def(self) -> Def {
        match self {
            LexicalScopeBinding::Item(binding) => binding.def(),
            LexicalScopeBinding::Def(def) => def,
        }
    }
763 764
}

J
Jeffrey Seyfried 已提交
765 766 767 768 769 770 771 772
#[derive(Clone)]
enum PathResult<'a> {
    Module(Module<'a>),
    NonModule(PathResolution),
    Indeterminate,
    Failed(String, bool /* is the error from the last segment? */),
}

J
Jeffrey Seyfried 已提交
773 774 775
enum ModuleKind {
    Block(NodeId),
    Def(Def, Name),
776 777
}

778
/// One node in the tree of modules.
779
pub struct ModuleData<'a> {
J
Jeffrey Seyfried 已提交
780 781
    parent: Option<Module<'a>>,
    kind: ModuleKind,
782

783 784
    // The def id of the closest normal module (`mod`) ancestor (including this module).
    normal_ancestor_id: DefId,
785

786 787
    resolutions: RefCell<FxHashMap<(Ident, Namespace), &'a RefCell<NameResolution<'a>>>>,
    legacy_macro_resolutions: RefCell<Vec<(Mark, Ident, Span)>>,
788
    macro_resolutions: RefCell<Vec<(Box<[Ident]>, Span)>>,
789

790 791 792
    // Macro invocations that can expand into items in this module.
    unresolved_invocations: RefCell<FxHashSet<Mark>>,

793
    no_implicit_prelude: bool,
794

795
    glob_importers: RefCell<Vec<&'a ImportDirective<'a>>>,
796
    globs: RefCell<Vec<&'a ImportDirective<'a>>>,
797

J
Jeffrey Seyfried 已提交
798
    // Used to memoize the traits in this module for faster searches through all traits in scope.
799
    traits: RefCell<Option<Box<[(Ident, &'a NameBinding<'a>)]>>>,
J
Jeffrey Seyfried 已提交
800

801 802 803
    // Whether this module is populated. If not populated, any attempt to
    // access the children must be preceded with a
    // `populate_module_if_necessary` call.
804
    populated: Cell<bool>,
805 806
}

807
pub type Module<'a> = &'a ModuleData<'a>;
808

809
impl<'a> ModuleData<'a> {
810
    fn new(parent: Option<Module<'a>>, kind: ModuleKind, normal_ancestor_id: DefId) -> Self {
811
        ModuleData {
J
Jeffrey Seyfried 已提交
812 813
            parent: parent,
            kind: kind,
814
            normal_ancestor_id: normal_ancestor_id,
815
            resolutions: RefCell::new(FxHashMap()),
816
            legacy_macro_resolutions: RefCell::new(Vec::new()),
817
            macro_resolutions: RefCell::new(Vec::new()),
818
            unresolved_invocations: RefCell::new(FxHashSet()),
819
            no_implicit_prelude: false,
820
            glob_importers: RefCell::new(Vec::new()),
821
            globs: RefCell::new((Vec::new())),
J
Jeffrey Seyfried 已提交
822
            traits: RefCell::new(None),
823
            populated: Cell::new(normal_ancestor_id.is_local()),
824
        }
B
Brian Anderson 已提交
825 826
    }

827 828 829
    fn for_each_child<F: FnMut(Ident, Namespace, &'a NameBinding<'a>)>(&self, mut f: F) {
        for (&(ident, ns), name_resolution) in self.resolutions.borrow().iter() {
            name_resolution.borrow().binding.map(|binding| f(ident, ns, binding));
830 831 832
        }
    }

J
Jeffrey Seyfried 已提交
833 834 835 836 837 838 839
    fn def(&self) -> Option<Def> {
        match self.kind {
            ModuleKind::Def(def, _) => Some(def),
            _ => None,
        }
    }

840
    fn def_id(&self) -> Option<DefId> {
J
Jeffrey Seyfried 已提交
841
        self.def().as_ref().map(Def::def_id)
842 843
    }

844
    // `self` resolves to the first module ancestor that `is_normal`.
845
    fn is_normal(&self) -> bool {
J
Jeffrey Seyfried 已提交
846 847
        match self.kind {
            ModuleKind::Def(Def::Mod(_), _) => true,
848 849 850 851 852
            _ => false,
        }
    }

    fn is_trait(&self) -> bool {
J
Jeffrey Seyfried 已提交
853 854
        match self.kind {
            ModuleKind::Def(Def::Trait(_), _) => true,
855
            _ => false,
856
        }
B
Brian Anderson 已提交
857
    }
858 859

    fn is_local(&self) -> bool {
860
        self.normal_ancestor_id.is_local()
861
    }
V
Victor Berger 已提交
862 863
}

864
impl<'a> fmt::Debug for ModuleData<'a> {
865
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
J
Jeffrey Seyfried 已提交
866
        write!(f, "{:?}", self.def())
867 868 869
    }
}

870
// Records a possibly-private value, type, or module definition.
871
#[derive(Clone, Debug)]
872
pub struct NameBinding<'a> {
873
    kind: NameBindingKind<'a>,
874
    expansion: Mark,
875
    span: Span,
876
    vis: ty::Visibility,
877 878
}

879
pub trait ToNameBinding<'a> {
J
Jeffrey Seyfried 已提交
880
    fn to_name_binding(self, arenas: &'a ResolverArenas<'a>) -> &'a NameBinding<'a>;
881 882
}

J
Jeffrey Seyfried 已提交
883 884
impl<'a> ToNameBinding<'a> for &'a NameBinding<'a> {
    fn to_name_binding(self, _: &'a ResolverArenas<'a>) -> &'a NameBinding<'a> {
885 886 887 888
        self
    }
}

889
#[derive(Clone, Debug)]
890
enum NameBindingKind<'a> {
891
    Def(Def),
892
    Module(Module<'a>),
893 894
    Import {
        binding: &'a NameBinding<'a>,
895
        directive: &'a ImportDirective<'a>,
896
        used: Cell<bool>,
J
Jeffrey Seyfried 已提交
897
        legacy_self_import: bool,
898
    },
899 900 901
    Ambiguity {
        b1: &'a NameBinding<'a>,
        b2: &'a NameBinding<'a>,
902
        legacy: bool,
903
    }
904 905
}

906 907
struct PrivacyError<'a>(Span, Name, &'a NameBinding<'a>);

J
Jeffrey Seyfried 已提交
908 909 910
struct AmbiguityError<'a> {
    span: Span,
    name: Name,
911
    lexical: bool,
J
Jeffrey Seyfried 已提交
912 913
    b1: &'a NameBinding<'a>,
    b2: &'a NameBinding<'a>,
914
    legacy: bool,
J
Jeffrey Seyfried 已提交
915 916
}

917
impl<'a> NameBinding<'a> {
J
Jeffrey Seyfried 已提交
918
    fn module(&self) -> Option<Module<'a>> {
919
        match self.kind {
J
Jeffrey Seyfried 已提交
920
            NameBindingKind::Module(module) => Some(module),
921
            NameBindingKind::Import { binding, .. } => binding.module(),
922
            NameBindingKind::Ambiguity { legacy: true, b1, .. } => b1.module(),
J
Jeffrey Seyfried 已提交
923
            _ => None,
924 925 926
        }
    }

927
    fn def(&self) -> Def {
928
        match self.kind {
929
            NameBindingKind::Def(def) => def,
J
Jeffrey Seyfried 已提交
930
            NameBindingKind::Module(module) => module.def().unwrap(),
931
            NameBindingKind::Import { binding, .. } => binding.def(),
932
            NameBindingKind::Ambiguity { legacy: true, b1, .. } => b1.def(),
933
            NameBindingKind::Ambiguity { .. } => Def::Err,
934
        }
935
    }
936

J
Jeffrey Seyfried 已提交
937 938 939 940 941 942 943 944
    fn get_macro(&self, resolver: &mut Resolver<'a>) -> Rc<SyntaxExtension> {
        match self.kind {
            NameBindingKind::Import { binding, .. } => binding.get_macro(resolver),
            NameBindingKind::Ambiguity { b1, .. } => b1.get_macro(resolver),
            _ => resolver.get_macro(self.def()),
        }
    }

945 946 947 948 949 950 951
    // We sometimes need to treat variants as `pub` for backwards compatibility
    fn pseudo_vis(&self) -> ty::Visibility {
        if self.is_variant() { ty::Visibility::Public } else { self.vis }
    }

    fn is_variant(&self) -> bool {
        match self.kind {
952 953
            NameBindingKind::Def(Def::Variant(..)) |
            NameBindingKind::Def(Def::VariantCtor(..)) => true,
954 955
            _ => false,
        }
956 957
    }

958
    fn is_extern_crate(&self) -> bool {
959 960 961 962 963 964 965 966
        match self.kind {
            NameBindingKind::Import {
                directive: &ImportDirective {
                    subclass: ImportDirectiveSubclass::ExternCrate, ..
                }, ..
            } => true,
            _ => false,
        }
967
    }
968 969 970 971 972 973 974

    fn is_import(&self) -> bool {
        match self.kind {
            NameBindingKind::Import { .. } => true,
            _ => false,
        }
    }
975 976 977 978

    fn is_glob_import(&self) -> bool {
        match self.kind {
            NameBindingKind::Import { directive, .. } => directive.is_glob(),
979
            NameBindingKind::Ambiguity { b1, .. } => b1.is_glob_import(),
980 981 982 983 984
            _ => false,
        }
    }

    fn is_importable(&self) -> bool {
985
        match self.def() {
986 987 988 989
            Def::AssociatedConst(..) | Def::Method(..) | Def::AssociatedTy(..) => false,
            _ => true,
        }
    }
990 991
}

992
/// Interns the names of the primitive types.
F
Felix S. Klock II 已提交
993
struct PrimitiveTypeTable {
994
    primitive_types: FxHashMap<Name, PrimTy>,
995
}
996

997
impl PrimitiveTypeTable {
K
Kevin Butler 已提交
998
    fn new() -> PrimitiveTypeTable {
999
        let mut table = PrimitiveTypeTable { primitive_types: FxHashMap() };
C
corentih 已提交
1000 1001 1002

        table.intern("bool", TyBool);
        table.intern("char", TyChar);
1003 1004
        table.intern("f32", TyFloat(FloatTy::F32));
        table.intern("f64", TyFloat(FloatTy::F64));
1005 1006 1007 1008 1009
        table.intern("isize", TyInt(IntTy::Is));
        table.intern("i8", TyInt(IntTy::I8));
        table.intern("i16", TyInt(IntTy::I16));
        table.intern("i32", TyInt(IntTy::I32));
        table.intern("i64", TyInt(IntTy::I64));
1010
        table.intern("i128", TyInt(IntTy::I128));
C
corentih 已提交
1011
        table.intern("str", TyStr);
1012 1013 1014 1015 1016
        table.intern("usize", TyUint(UintTy::Us));
        table.intern("u8", TyUint(UintTy::U8));
        table.intern("u16", TyUint(UintTy::U16));
        table.intern("u32", TyUint(UintTy::U32));
        table.intern("u64", TyUint(UintTy::U64));
1017
        table.intern("u128", TyUint(UintTy::U128));
K
Kevin Butler 已提交
1018 1019 1020
        table
    }

1021
    fn intern(&mut self, string: &str, primitive_type: PrimTy) {
1022
        self.primitive_types.insert(Symbol::intern(string), primitive_type);
1023 1024 1025
    }
}

1026
/// The main resolver class.
1027
pub struct Resolver<'a> {
E
Eduard Burtescu 已提交
1028
    session: &'a Session,
1029

1030
    pub definitions: Definitions,
1031

1032 1033
    // Maps the node id of a statement to the expansions of the `macro_rules!`s
    // immediately above the statement (if appropriate).
1034
    macros_at_scope: FxHashMap<NodeId, Vec<Mark>>,
1035

1036
    graph_root: Module<'a>,
1037

1038 1039
    prelude: Option<Module<'a>>,

1040
    trait_item_map: FxHashMap<(DefId, Name, Namespace), (Def, bool /* has self */)>,
1041

V
Vadim Petrochenkov 已提交
1042 1043
    // Names of fields of an item `DefId` accessible with dot syntax.
    // Used for hints during error reporting.
1044
    field_names: FxHashMap<DefId, Vec<Name>>,
1045

1046 1047 1048 1049
    // All imports known to succeed or fail.
    determined_imports: Vec<&'a ImportDirective<'a>>,

    // All non-determined imports.
1050
    indeterminate_imports: Vec<&'a ImportDirective<'a>>,
1051 1052

    // The module that represents the current item scope.
1053
    current_module: Module<'a>,
1054

J
Jeffrey Seyfried 已提交
1055
    // The current set of local scopes for types and values.
1056
    // FIXME #4948: Reuse ribs to avoid allocation.
J
Jeffrey Seyfried 已提交
1057
    ribs: PerNS<Vec<Rib<'a>>>,
1058

1059
    // The current set of local scopes, for labels.
1060
    label_ribs: Vec<Rib<'a>>,
1061

1062
    // The trait that the current context can refer to.
1063 1064 1065 1066
    current_trait_ref: Option<(DefId, TraitRef)>,

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

1068
    // The idents for the primitive types.
E
Eduard Burtescu 已提交
1069
    primitive_type_table: PrimitiveTypeTable,
1070

1071
    def_map: DefMap,
1072
    pub freevars: FreevarMap,
1073
    freevars_seen: NodeMap<NodeMap<usize>>,
1074 1075
    pub export_map: ExportMap,
    pub trait_map: TraitMap,
1076

1077
    // A map from nodes to anonymous modules.
1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090
    // Anonymous modules 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`.
1091 1092
    block_map: NodeMap<Module<'a>>,
    module_map: FxHashMap<DefId, Module<'a>>,
1093
    extern_crate_roots: FxHashMap<(CrateNum, bool /* MacrosOnly? */), Module<'a>>,
1094

1095
    pub make_glob_map: bool,
1096 1097
    // Maps imports to the names of items actually imported (this actually maps
    // all imports, but only glob imports are actually interesting).
1098
    pub glob_map: GlobMap,
1099

1100 1101
    used_imports: FxHashSet<(NodeId, Namespace)>,
    used_crates: FxHashSet<CrateNum>,
1102
    pub maybe_unused_trait_imports: NodeSet,
G
Garming Sam 已提交
1103

1104
    privacy_errors: Vec<PrivacyError<'a>>,
J
Jeffrey Seyfried 已提交
1105
    ambiguity_errors: Vec<AmbiguityError<'a>>,
1106
    disallowed_shadowing: Vec<&'a LegacyBinding<'a>>,
1107 1108

    arenas: &'a ResolverArenas<'a>,
1109
    dummy_binding: &'a NameBinding<'a>,
1110
    use_extern_macros: bool, // true if `#![feature(use_extern_macros)]`
1111

1112
    pub exported_macros: Vec<ast::MacroDef>,
1113
    crate_loader: &'a mut CrateLoader,
1114
    macro_names: FxHashSet<Name>,
1115
    builtin_macros: FxHashMap<Name, &'a NameBinding<'a>>,
1116
    lexical_macro_resolutions: Vec<(Name, &'a Cell<LegacyScope<'a>>)>,
J
Jeffrey Seyfried 已提交
1117 1118
    macro_map: FxHashMap<DefId, Rc<SyntaxExtension>>,
    macro_exports: Vec<Export>,
1119
    pub whitelisted_legacy_custom_derives: Vec<Name>,
1120 1121

    // Maps the `Mark` of an expansion to its containing module or block.
1122
    invocations: FxHashMap<Mark, &'a InvocationData<'a>>,
1123 1124 1125

    // Avoid duplicated errors for "name already defined".
    name_already_seen: FxHashMap<Name, Span>,
1126 1127
}

1128
pub struct ResolverArenas<'a> {
1129
    modules: arena::TypedArena<ModuleData<'a>>,
1130
    local_modules: RefCell<Vec<Module<'a>>>,
1131
    name_bindings: arena::TypedArena<NameBinding<'a>>,
1132
    import_directives: arena::TypedArena<ImportDirective<'a>>,
1133
    name_resolutions: arena::TypedArena<RefCell<NameResolution<'a>>>,
1134
    invocation_data: arena::TypedArena<InvocationData<'a>>,
J
Jeffrey Seyfried 已提交
1135
    legacy_bindings: arena::TypedArena<LegacyBinding<'a>>,
1136 1137 1138
}

impl<'a> ResolverArenas<'a> {
1139
    fn alloc_module(&'a self, module: ModuleData<'a>) -> Module<'a> {
1140 1141 1142 1143 1144 1145 1146 1147
        let module = self.modules.alloc(module);
        if module.def_id().map(|def_id| def_id.is_local()).unwrap_or(true) {
            self.local_modules.borrow_mut().push(module);
        }
        module
    }
    fn local_modules(&'a self) -> ::std::cell::Ref<'a, Vec<Module<'a>>> {
        self.local_modules.borrow()
1148 1149 1150 1151
    }
    fn alloc_name_binding(&'a self, name_binding: NameBinding<'a>) -> &'a NameBinding<'a> {
        self.name_bindings.alloc(name_binding)
    }
1152 1153
    fn alloc_import_directive(&'a self, import_directive: ImportDirective<'a>)
                              -> &'a ImportDirective {
1154 1155
        self.import_directives.alloc(import_directive)
    }
1156 1157 1158
    fn alloc_name_resolution(&'a self) -> &'a RefCell<NameResolution<'a>> {
        self.name_resolutions.alloc(Default::default())
    }
1159 1160 1161
    fn alloc_invocation_data(&'a self, expansion_data: InvocationData<'a>)
                             -> &'a InvocationData<'a> {
        self.invocation_data.alloc(expansion_data)
J
Jeffrey Seyfried 已提交
1162
    }
J
Jeffrey Seyfried 已提交
1163 1164 1165
    fn alloc_legacy_binding(&'a self, binding: LegacyBinding<'a>) -> &'a LegacyBinding<'a> {
        self.legacy_bindings.alloc(binding)
    }
1166 1167
}

1168 1169 1170 1171 1172 1173
impl<'a, 'b: 'a> ty::DefIdTree for &'a Resolver<'b> {
    fn parent(self, id: DefId) -> Option<DefId> {
        match id.krate {
            LOCAL_CRATE => self.definitions.def_key(id.index).parent,
            _ => self.session.cstore.def_key(id).parent,
        }.map(|index| DefId { index: index, ..id })
1174 1175 1176
    }
}

1177
impl<'a> hir::lowering::Resolver for Resolver<'a> {
J
Jeffrey Seyfried 已提交
1178
    fn resolve_hir_path(&mut self, path: &mut hir::Path, is_value: bool) {
1179
        let namespace = if is_value { ValueNS } else { TypeNS };
1180
        let hir::Path { ref segments, span, ref mut def } = *path;
J
Jeffrey Seyfried 已提交
1181
        let path: Vec<_> = segments.iter().map(|seg| Ident::with_empty_ctxt(seg.name)).collect();
1182
        match self.resolve_path(&path, Some(namespace), Some(span)) {
J
Jeffrey Seyfried 已提交
1183 1184
            PathResult::Module(module) => *def = module.def().unwrap(),
            PathResult::NonModule(path_res) if path_res.depth == 0 => *def = path_res.base_def,
1185
            PathResult::NonModule(..) => match self.resolve_path(&path, None, Some(span)) {
J
Jeffrey Seyfried 已提交
1186 1187 1188 1189 1190 1191 1192 1193
                PathResult::Failed(msg, _) => {
                    resolve_error(self, span, ResolutionError::FailedToResolve(&msg));
                }
                _ => {}
            },
            PathResult::Indeterminate => unreachable!(),
            PathResult::Failed(msg, _) => {
                resolve_error(self, span, ResolutionError::FailedToResolve(&msg));
1194 1195 1196 1197
            }
        }
    }

1198 1199 1200 1201
    fn get_resolution(&mut self, id: NodeId) -> Option<PathResolution> {
        self.def_map.get(&id).cloned()
    }

1202 1203
    fn definitions(&mut self) -> &mut Definitions {
        &mut self.definitions
1204 1205 1206
    }
}

1207
impl<'a> Resolver<'a> {
1208
    pub fn new(session: &'a Session,
1209
               krate: &Crate,
1210
               make_glob_map: MakeGlobMap,
1211
               crate_loader: &'a mut CrateLoader,
1212
               arenas: &'a ResolverArenas<'a>)
1213
               -> Resolver<'a> {
1214 1215
        let root_def_id = DefId::local(CRATE_DEF_INDEX);
        let root_module_kind = ModuleKind::Def(Def::Mod(root_def_id), keywords::Invalid.name());
1216
        let graph_root = arenas.alloc_module(ModuleData {
1217
            no_implicit_prelude: attr::contains_name(&krate.attrs, "no_implicit_prelude"),
1218
            ..ModuleData::new(None, root_module_kind, root_def_id)
1219
        });
1220 1221
        let mut module_map = FxHashMap();
        module_map.insert(DefId::local(CRATE_DEF_INDEX), graph_root);
K
Kevin Butler 已提交
1222

1223 1224 1225
        let mut definitions = Definitions::new();
        DefCollector::new(&mut definitions).collect_root();

1226
        let mut invocations = FxHashMap();
1227 1228
        invocations.insert(Mark::root(),
                           arenas.alloc_invocation_data(InvocationData::root(graph_root)));
1229

K
Kevin Butler 已提交
1230 1231 1232
        Resolver {
            session: session,

1233
            definitions: definitions,
1234
            macros_at_scope: FxHashMap(),
1235

K
Kevin Butler 已提交
1236 1237
            // The outermost module has def ID 0; this is not reflected in the
            // AST.
1238
            graph_root: graph_root,
1239
            prelude: None,
K
Kevin Butler 已提交
1240

1241 1242
            trait_item_map: FxHashMap(),
            field_names: FxHashMap(),
K
Kevin Butler 已提交
1243

1244
            determined_imports: Vec::new(),
1245
            indeterminate_imports: Vec::new(),
K
Kevin Butler 已提交
1246

1247
            current_module: graph_root,
J
Jeffrey Seyfried 已提交
1248 1249 1250
            ribs: PerNS {
                value_ns: vec![Rib::new(ModuleRibKind(graph_root))],
                type_ns: vec![Rib::new(ModuleRibKind(graph_root))],
1251
                macro_ns: None,
J
Jeffrey Seyfried 已提交
1252
            },
1253
            label_ribs: Vec::new(),
K
Kevin Butler 已提交
1254 1255 1256 1257 1258 1259

            current_trait_ref: None,
            current_self_type: None,

            primitive_type_table: PrimitiveTypeTable::new(),

1260
            def_map: NodeMap(),
1261 1262
            freevars: NodeMap(),
            freevars_seen: NodeMap(),
1263 1264
            export_map: NodeMap(),
            trait_map: NodeMap(),
1265
            module_map: module_map,
1266
            block_map: NodeMap(),
1267
            extern_crate_roots: FxHashMap(),
K
Kevin Butler 已提交
1268

1269
            make_glob_map: make_glob_map == MakeGlobMap::Yes,
1270
            glob_map: NodeMap(),
G
Garming Sam 已提交
1271

1272 1273
            used_imports: FxHashSet(),
            used_crates: FxHashSet(),
S
Seo Sanghyeon 已提交
1274 1275
            maybe_unused_trait_imports: NodeSet(),

1276
            privacy_errors: Vec::new(),
1277
            ambiguity_errors: Vec::new(),
1278
            disallowed_shadowing: Vec::new(),
1279 1280

            arenas: arenas,
1281 1282
            dummy_binding: arenas.alloc_name_binding(NameBinding {
                kind: NameBindingKind::Def(Def::Err),
1283
                expansion: Mark::root(),
1284 1285 1286
                span: DUMMY_SP,
                vis: ty::Visibility::Public,
            }),
1287
            use_extern_macros: session.features.borrow().use_extern_macros,
1288

1289
            exported_macros: Vec::new(),
1290
            crate_loader: crate_loader,
1291 1292
            macro_names: FxHashSet(),
            builtin_macros: FxHashMap(),
J
Jeffrey Seyfried 已提交
1293
            lexical_macro_resolutions: Vec::new(),
J
Jeffrey Seyfried 已提交
1294 1295
            macro_map: FxHashMap(),
            macro_exports: Vec::new(),
1296
            invocations: invocations,
1297
            name_already_seen: FxHashMap(),
1298
            whitelisted_legacy_custom_derives: Vec::new(),
1299 1300 1301
        }
    }

1302
    pub fn arenas() -> ResolverArenas<'a> {
1303 1304
        ResolverArenas {
            modules: arena::TypedArena::new(),
1305
            local_modules: RefCell::new(Vec::new()),
1306
            name_bindings: arena::TypedArena::new(),
1307
            import_directives: arena::TypedArena::new(),
1308
            name_resolutions: arena::TypedArena::new(),
1309
            invocation_data: arena::TypedArena::new(),
J
Jeffrey Seyfried 已提交
1310
            legacy_bindings: arena::TypedArena::new(),
K
Kevin Butler 已提交
1311 1312
        }
    }
1313

J
Jeffrey Seyfried 已提交
1314 1315 1316 1317
    fn per_ns<T, F: FnMut(&mut Self, Namespace) -> T>(&mut self, mut f: F) -> PerNS<T> {
        PerNS {
            type_ns: f(self, TypeNS),
            value_ns: f(self, ValueNS),
1318 1319 1320 1321
            macro_ns: match self.use_extern_macros {
                true => Some(f(self, MacroNS)),
                false => None,
            },
J
Jeffrey Seyfried 已提交
1322 1323 1324
        }
    }

1325 1326
    /// Entry point to crate resolution.
    pub fn resolve_crate(&mut self, krate: &Crate) {
1327
        ImportResolver { resolver: self }.finalize_imports();
1328
        self.current_module = self.graph_root;
1329
        self.finalize_current_module_macro_resolutions();
1330 1331 1332
        visit::walk_crate(self, krate);

        check_unused::check_crate(self, krate);
1333
        self.report_errors();
1334
        self.crate_loader.postprocess(krate);
1335 1336
    }

1337 1338 1339
    fn new_module(&self, parent: Module<'a>, kind: ModuleKind, normal_ancestor_id: DefId)
                  -> Module<'a> {
        self.arenas.alloc_module(ModuleData::new(Some(parent), kind, normal_ancestor_id))
1340 1341
    }

1342
    fn record_use(&mut self, ident: Ident, ns: Namespace, binding: &'a NameBinding<'a>, span: Span)
1343
                  -> bool /* true if an error was reported */ {
1344
        // track extern crates for unused_extern_crate lint
1345
        if let Some(DefId { krate, .. }) = binding.module().and_then(ModuleData::def_id) {
1346 1347 1348
            self.used_crates.insert(krate);
        }

1349
        match binding.kind {
J
Jeffrey Seyfried 已提交
1350 1351
            NameBindingKind::Import { directive, binding, ref used, legacy_self_import }
                    if !used.get() => {
1352
                used.set(true);
J
Jeffrey Seyfried 已提交
1353 1354 1355 1356
                if legacy_self_import {
                    self.warn_legacy_self_import(directive);
                    return false;
                }
1357
                self.used_imports.insert((directive.id, ns));
1358 1359
                self.add_to_glob_map(directive.id, ident);
                self.record_use(ident, ns, binding, span)
1360 1361
            }
            NameBindingKind::Import { .. } => false,
1362
            NameBindingKind::Ambiguity { b1, b2, legacy } => {
1363
                self.ambiguity_errors.push(AmbiguityError {
1364
                    span: span, name: ident.name, lexical: false, b1: b1, b2: b2, legacy: legacy,
1365
                });
1366
                if legacy {
A
Alex Crichton 已提交
1367
                    self.record_use(ident, ns, b1, span);
1368 1369
                }
                !legacy
1370 1371
            }
            _ => false
1372
        }
1373
    }
1374

1375
    fn add_to_glob_map(&mut self, id: NodeId, ident: Ident) {
1376
        if self.make_glob_map {
1377
            self.glob_map.entry(id).or_insert_with(FxHashSet).insert(ident.name);
1378
        }
1379 1380
    }

1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394
    /// This resolves the identifier `ident` in the namespace `ns` in the current lexical scope.
    /// More specifically, we proceed up the hierarchy of scopes and return the binding for
    /// `ident` in the first scope that defines it (or None if no scopes define it).
    ///
    /// A block's items are above its local variables in the scope hierarchy, regardless of where
    /// the items are defined in the block. For example,
    /// ```rust
    /// fn f() {
    ///    g(); // Since there are no local variables in scope yet, this resolves to the item.
    ///    let g = || {};
    ///    fn g() {}
    ///    g(); // This resolves to the local variable `g` since it shadows the item.
    /// }
    /// ```
1395
    ///
1396 1397
    /// Invariant: This must only be called during main resolution, not during
    /// import resolution.
1398
    fn resolve_ident_in_lexical_scope(&mut self,
1399
                                      mut ident: Ident,
1400
                                      ns: Namespace,
1401
                                      record_used: Option<Span>)
1402
                                      -> Option<LexicalScopeBinding<'a>> {
1403
        if ns == TypeNS {
1404
            ident = ident.unhygienize();
1405
        }
1406

1407
        // Walk backwards up the ribs in scope.
J
Jeffrey Seyfried 已提交
1408 1409
        for i in (0 .. self.ribs[ns].len()).rev() {
            if let Some(def) = self.ribs[ns][i].bindings.get(&ident).cloned() {
1410
                // The ident resolves to a type parameter or local variable.
1411 1412 1413
                return Some(LexicalScopeBinding::Def(
                    self.adjust_local_def(LocalDef { ribs: Some((ns, i)), def: def }, record_used)
                ));
1414 1415
            }

J
Jeffrey Seyfried 已提交
1416
            if let ModuleRibKind(module) = self.ribs[ns][i].kind {
1417
                let item = self.resolve_ident_in_module(module, ident, ns, false, record_used);
J
Jeffrey Seyfried 已提交
1418
                if let Ok(binding) = item {
1419 1420
                    // The ident resolves to an item.
                    return Some(LexicalScopeBinding::Item(binding));
1421
                }
1422

J
Jeffrey Seyfried 已提交
1423
                if let ModuleKind::Block(..) = module.kind { // We can see through blocks
1424
                } else if !module.no_implicit_prelude {
J
Jeffrey Seyfried 已提交
1425
                    return self.prelude.and_then(|prelude| {
1426
                        self.resolve_ident_in_module(prelude, ident, ns, false, None).ok()
J
Jeffrey Seyfried 已提交
1427 1428 1429
                    }).map(LexicalScopeBinding::Item)
                } else {
                    return None;
1430
                }
1431
            }
1432

J
Jeffrey Seyfried 已提交
1433
            if let MacroDefinition(mac) = self.ribs[ns][i].kind {
1434 1435
                // If an invocation of this macro created `ident`, give up on `ident`
                // and switch to `ident`'s source from the macro definition.
1436 1437 1438
                let (source_ctxt, source_macro) = ident.ctxt.source();
                if source_macro == mac {
                    ident.ctxt = source_ctxt;
1439 1440
                }
            }
1441
        }
1442

1443 1444 1445
        None
    }

1446 1447 1448 1449 1450 1451 1452 1453
    fn resolve_crate_var(&mut self, mut crate_var_ctxt: SyntaxContext) -> Module<'a> {
        while crate_var_ctxt.source().0 != SyntaxContext::empty() {
            crate_var_ctxt = crate_var_ctxt.source().0;
        }
        let module = self.invocations[&crate_var_ctxt.source().1].module.get();
        if module.is_local() { self.graph_root } else { module }
    }

1454 1455
    // AST resolution
    //
1456
    // We maintain a list of value ribs and type ribs.
1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471
    //
    // 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.

1472
    fn with_scope<F>(&mut self, id: NodeId, f: F)
C
corentih 已提交
1473
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1474
    {
1475
        let id = self.definitions.local_def_id(id);
1476 1477
        let module = self.module_map.get(&id).cloned(); // clones a reference
        if let Some(module) = module {
1478
            // Move down in the graph.
1479
            let orig_module = replace(&mut self.current_module, module);
J
Jeffrey Seyfried 已提交
1480 1481
            self.ribs[ValueNS].push(Rib::new(ModuleRibKind(module)));
            self.ribs[TypeNS].push(Rib::new(ModuleRibKind(module)));
1482

1483
            self.finalize_current_module_macro_resolutions();
1484
            f(self);
1485

1486
            self.current_module = orig_module;
J
Jeffrey Seyfried 已提交
1487 1488
            self.ribs[ValueNS].pop();
            self.ribs[TypeNS].pop();
1489 1490 1491
        } else {
            f(self);
        }
1492 1493
    }

S
Seo Sanghyeon 已提交
1494 1495
    /// Searches the current set of local scopes for labels.
    /// Stops after meeting a closure.
1496
    fn search_label(&self, mut ident: Ident) -> Option<Def> {
1497 1498 1499 1500 1501
        for rib in self.label_ribs.iter().rev() {
            match rib.kind {
                NormalRibKind => {
                    // Continue
                }
1502 1503 1504
                MacroDefinition(mac) => {
                    // If an invocation of this macro created `ident`, give up on `ident`
                    // and switch to `ident`'s source from the macro definition.
1505 1506 1507
                    let (source_ctxt, source_macro) = ident.ctxt.source();
                    if source_macro == mac {
                        ident.ctxt = source_ctxt;
1508 1509
                    }
                }
1510 1511
                _ => {
                    // Do not resolve labels across function boundary
C
corentih 已提交
1512
                    return None;
1513 1514
                }
            }
1515
            let result = rib.bindings.get(&ident).cloned();
S
Seo Sanghyeon 已提交
1516
            if result.is_some() {
C
corentih 已提交
1517
                return result;
1518 1519 1520 1521 1522
            }
        }
        None
    }

1523
    fn resolve_item(&mut self, item: &Item) {
1524
        let name = item.ident.name;
1525

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

1528
        match item.node {
1529 1530
            ItemKind::Enum(_, ref generics) |
            ItemKind::Ty(_, ref generics) |
1531
            ItemKind::Struct(_, ref generics) |
1532
            ItemKind::Union(_, ref generics) |
V
Vadim Petrochenkov 已提交
1533
            ItemKind::Fn(.., ref generics, _) => {
1534
                self.with_type_parameter_rib(HasTypeParameters(generics, ItemRibKind),
1535
                                             |this| visit::walk_item(this, item));
1536 1537
            }

1538
            ItemKind::DefaultImpl(_, ref trait_ref) => {
1539
                self.with_optional_trait_ref(Some(trait_ref), |_, _| {});
1540
            }
V
Vadim Petrochenkov 已提交
1541
            ItemKind::Impl(.., ref generics, ref opt_trait_ref, ref self_type, ref impl_items) =>
1542
                self.resolve_implementation(generics,
1543
                                            opt_trait_ref,
J
Jonas Schievink 已提交
1544
                                            &self_type,
1545
                                            item.id,
1546
                                            impl_items),
1547

1548
            ItemKind::Trait(_, ref generics, ref bounds, ref trait_items) => {
1549
                // Create a new rib for the trait-wide type parameters.
1550
                self.with_type_parameter_rib(HasTypeParameters(generics, ItemRibKind), |this| {
1551
                    let local_def_id = this.definitions.local_def_id(item.id);
1552
                    this.with_self_rib(Def::SelfTy(Some(local_def_id), None), |this| {
1553
                        this.visit_generics(generics);
1554
                        walk_list!(this, visit_ty_param_bound, bounds);
1555 1556

                        for trait_item in trait_items {
1557
                            match trait_item.node {
1558
                                TraitItemKind::Const(_, ref default) => {
1559 1560 1561 1562 1563
                                    // 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| {
1564
                                            visit::walk_trait_item(this, trait_item)
1565 1566
                                        });
                                    } else {
1567
                                        visit::walk_trait_item(this, trait_item)
1568 1569
                                    }
                                }
1570
                                TraitItemKind::Method(ref sig, _) => {
1571 1572
                                    let type_parameters =
                                        HasTypeParameters(&sig.generics,
V
Vadim Petrochenkov 已提交
1573
                                                          MethodRibKind(!sig.decl.has_self()));
1574
                                    this.with_type_parameter_rib(type_parameters, |this| {
1575
                                        visit::walk_trait_item(this, trait_item)
1576
                                    });
1577
                                }
1578
                                TraitItemKind::Type(..) => {
1579
                                    this.with_type_parameter_rib(NoTypeParameters, |this| {
1580
                                        visit::walk_trait_item(this, trait_item)
1581
                                    });
1582
                                }
1583
                                TraitItemKind::Macro(_) => panic!("unexpanded macro in resolve!"),
1584 1585 1586
                            };
                        }
                    });
1587
                });
1588 1589
            }

1590
            ItemKind::Mod(_) | ItemKind::ForeignMod(_) => {
1591
                self.with_scope(item.id, |this| {
1592
                    visit::walk_item(this, item);
1593
                });
1594 1595
            }

1596
            ItemKind::Const(..) | ItemKind::Static(..) => {
A
Alex Crichton 已提交
1597
                self.with_constant_rib(|this| {
1598
                    visit::walk_item(this, item);
1599
                });
1600
            }
1601

1602
            ItemKind::Use(ref view_path) => {
1603
                match view_path.node {
1604 1605 1606
                    ast::ViewPathList(ref prefix, ref items) if items.is_empty() => {
                        // Resolve prefix of an import with empty braces (issue #28388).
                        self.smart_resolve_path(item.id, None, prefix, PathSource::ImportPrefix);
1607 1608
                    }
                    _ => {}
W
we 已提交
1609 1610 1611
                }
            }

1612
            ItemKind::ExternCrate(_) => {
1613
                // do nothing, these are just around to be encoded
1614
            }
1615 1616

            ItemKind::Mac(_) => panic!("unexpanded macro in resolve!"),
1617 1618 1619
        }
    }

1620
    fn with_type_parameter_rib<'b, F>(&'b mut self, type_parameters: TypeParameters<'a, 'b>, f: F)
C
corentih 已提交
1621
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1622
    {
1623
        match type_parameters {
1624
            HasTypeParameters(generics, rib_kind) => {
1625
                let mut function_type_rib = Rib::new(rib_kind);
1626
                let mut seen_bindings = FxHashMap();
1627
                for type_parameter in &generics.ty_params {
1628
                    let name = type_parameter.ident.name;
1629
                    debug!("with_type_parameter_rib: {}", type_parameter.id);
1630

C
Chris Stankus 已提交
1631 1632
                    if seen_bindings.contains_key(&name) {
                        let span = seen_bindings.get(&name).unwrap();
1633 1634
                        resolve_error(self,
                                      type_parameter.span,
C
Chris Stankus 已提交
1635 1636
                                      ResolutionError::NameAlreadyUsedInTypeParameterList(name,
                                                                                          span));
1637
                    }
C
Chris Stankus 已提交
1638
                    seen_bindings.entry(name).or_insert(type_parameter.span);
1639

1640
                    // plain insert (no renaming)
1641
                    let def_id = self.definitions.local_def_id(type_parameter.id);
1642
                    let def = Def::TyParam(def_id);
1643
                    function_type_rib.bindings.insert(Ident::with_empty_ctxt(name), def);
1644
                    self.record_def(type_parameter.id, PathResolution::new(def));
1645
                }
J
Jeffrey Seyfried 已提交
1646
                self.ribs[TypeNS].push(function_type_rib);
1647 1648
            }

B
Brian Anderson 已提交
1649
            NoTypeParameters => {
1650 1651 1652 1653
                // Nothing to do.
            }
        }

A
Alex Crichton 已提交
1654
        f(self);
1655

J
Jeffrey Seyfried 已提交
1656
        if let HasTypeParameters(..) = type_parameters {
J
Jeffrey Seyfried 已提交
1657
            self.ribs[TypeNS].pop();
1658 1659 1660
        }
    }

C
corentih 已提交
1661 1662
    fn with_label_rib<F>(&mut self, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1663
    {
1664
        self.label_ribs.push(Rib::new(NormalRibKind));
A
Alex Crichton 已提交
1665
        f(self);
J
Jeffrey Seyfried 已提交
1666
        self.label_ribs.pop();
1667
    }
1668

C
corentih 已提交
1669 1670
    fn with_constant_rib<F>(&mut self, f: F)
        where F: FnOnce(&mut Resolver)
J
Jorge Aparicio 已提交
1671
    {
J
Jeffrey Seyfried 已提交
1672 1673
        self.ribs[ValueNS].push(Rib::new(ConstantItemRibKind));
        self.ribs[TypeNS].push(Rib::new(ConstantItemRibKind));
A
Alex Crichton 已提交
1674
        f(self);
J
Jeffrey Seyfried 已提交
1675 1676
        self.ribs[TypeNS].pop();
        self.ribs[ValueNS].pop();
1677 1678
    }

1679 1680
    fn with_current_self_type<T, F>(&mut self, self_type: &Ty, f: F) -> T
        where F: FnOnce(&mut Resolver) -> T
J
Jorge Aparicio 已提交
1681
    {
1682 1683 1684 1685 1686 1687 1688
        // 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
    }

1689
    fn with_optional_trait_ref<T, F>(&mut self, opt_trait_ref: Option<&TraitRef>, f: F) -> T
1690
        where F: FnOnce(&mut Resolver, Option<DefId>) -> T
J
Jorge Aparicio 已提交
1691
    {
1692
        let mut new_val = None;
1693
        let mut new_id = None;
E
Eduard Burtescu 已提交
1694
        if let Some(trait_ref) = opt_trait_ref {
1695 1696 1697 1698 1699
            let def = self.smart_resolve_path(trait_ref.ref_id, None,
                                              &trait_ref.path, PathSource::Trait).base_def;
            if def != Def::Err {
                new_val = Some((def.def_id(), trait_ref.clone()));
                new_id = Some(def.def_id());
1700
            }
1701
            visit::walk_trait_ref(self, trait_ref);
1702
        }
1703
        let original_trait_ref = replace(&mut self.current_trait_ref, new_val);
1704
        let result = f(self, new_id);
1705 1706 1707 1708
        self.current_trait_ref = original_trait_ref;
        result
    }

1709 1710 1711 1712 1713 1714
    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....)
1715
        self_type_rib.bindings.insert(keywords::SelfType.ident(), self_def);
J
Jeffrey Seyfried 已提交
1716
        self.ribs[TypeNS].push(self_type_rib);
1717
        f(self);
J
Jeffrey Seyfried 已提交
1718
        self.ribs[TypeNS].pop();
1719 1720
    }

F
Felix S. Klock II 已提交
1721
    fn resolve_implementation(&mut self,
1722 1723 1724
                              generics: &Generics,
                              opt_trait_reference: &Option<TraitRef>,
                              self_type: &Ty,
1725
                              item_id: NodeId,
1726
                              impl_items: &[ImplItem]) {
1727
        // If applicable, create a rib for the type parameters.
1728
        self.with_type_parameter_rib(HasTypeParameters(generics, ItemRibKind), |this| {
1729
            // Resolve the type parameters.
1730
            this.visit_generics(generics);
1731

1732
            // Resolve the trait reference, if necessary.
1733
            this.with_optional_trait_ref(opt_trait_reference.as_ref(), |this, trait_id| {
1734
                // Resolve the self type.
1735
                this.visit_ty(self_type);
1736

1737 1738
                let item_def_id = this.definitions.local_def_id(item_id);
                this.with_self_rib(Def::SelfTy(trait_id, Some(item_def_id)), |this| {
1739 1740
                    this.with_current_self_type(self_type, |this| {
                        for impl_item in impl_items {
1741
                            this.resolve_visibility(&impl_item.vis);
1742
                            match impl_item.node {
1743
                                ImplItemKind::Const(..) => {
1744
                                    // If this is a trait impl, ensure the const
1745
                                    // exists in trait
1746
                                    this.check_trait_item(impl_item.ident.name,
1747
                                                          ValueNS,
1748 1749
                                                          impl_item.span,
                                        |n, s| ResolutionError::ConstNotMemberOfTrait(n, s));
1750
                                    visit::walk_impl_item(this, impl_item);
1751
                                }
1752
                                ImplItemKind::Method(ref sig, _) => {
1753 1754
                                    // If this is a trait impl, ensure the method
                                    // exists in trait
1755
                                    this.check_trait_item(impl_item.ident.name,
1756
                                                          ValueNS,
1757 1758
                                                          impl_item.span,
                                        |n, s| ResolutionError::MethodNotMemberOfTrait(n, s));
1759 1760 1761 1762 1763

                                    // We also need a new scope for the method-
                                    // specific type parameters.
                                    let type_parameters =
                                        HasTypeParameters(&sig.generics,
V
Vadim Petrochenkov 已提交
1764
                                                          MethodRibKind(!sig.decl.has_self()));
1765
                                    this.with_type_parameter_rib(type_parameters, |this| {
1766
                                        visit::walk_impl_item(this, impl_item);
1767 1768
                                    });
                                }
1769
                                ImplItemKind::Type(ref ty) => {
1770
                                    // If this is a trait impl, ensure the type
1771
                                    // exists in trait
1772
                                    this.check_trait_item(impl_item.ident.name,
1773
                                                          TypeNS,
1774 1775
                                                          impl_item.span,
                                        |n, s| ResolutionError::TypeNotMemberOfTrait(n, s));
1776

1777 1778
                                    this.visit_ty(ty);
                                }
1779
                                ImplItemKind::Macro(_) => panic!("unexpanded macro in resolve!"),
1780
                            }
1781
                        }
1782
                    });
1783
                });
1784
            });
1785
        });
1786 1787
    }

1788
    fn check_trait_item<F>(&self, name: Name, ns: Namespace, span: Span, err: F)
C
corentih 已提交
1789 1790 1791 1792
        where F: FnOnce(Name, &str) -> ResolutionError
    {
        // If there is a TraitRef in scope for an impl, then the method must be in the
        // trait.
1793
        if let Some((did, ref trait_ref)) = self.current_trait_ref {
1794 1795
            if !self.trait_item_map.contains_key(&(did, name, ns)) {
                let path_str = path_names_to_string(&trait_ref.path);
J
Jonas Schievink 已提交
1796
                resolve_error(self, span, err(name, &path_str));
1797 1798 1799 1800
            }
        }
    }

E
Eduard Burtescu 已提交
1801
    fn resolve_local(&mut self, local: &Local) {
1802
        // Resolve the type.
1803
        walk_list!(self, visit_ty, &local.ty);
1804

1805
        // Resolve the initializer.
1806
        walk_list!(self, visit_expr, &local.init);
1807 1808

        // Resolve the pattern.
1809
        self.resolve_pattern(&local.pat, PatternSource::Let, &mut FxHashMap());
1810 1811
    }

J
John Clements 已提交
1812 1813 1814 1815
    // 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 已提交
1816
    fn binding_mode_map(&mut self, pat: &Pat) -> BindingMap {
1817
        let mut binding_map = FxHashMap();
1818 1819 1820 1821 1822 1823 1824 1825

        pat.walk(&mut |pat| {
            if let PatKind::Ident(binding_mode, ident, ref sub_pat) = pat.node {
                if sub_pat.is_some() || match self.def_map.get(&pat.id) {
                    Some(&PathResolution { base_def: Def::Local(..), .. }) => true,
                    _ => false,
                } {
                    let binding_info = BindingInfo { span: ident.span, binding_mode: binding_mode };
1826
                    binding_map.insert(ident.node, binding_info);
1827 1828 1829
                }
            }
            true
1830
        });
1831 1832

        binding_map
1833 1834
    }

J
John Clements 已提交
1835 1836
    // 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 已提交
1837
    fn check_consistent_bindings(&mut self, arm: &Arm) {
1838
        if arm.pats.is_empty() {
C
corentih 已提交
1839
            return;
1840
        }
J
Jonas Schievink 已提交
1841
        let map_0 = self.binding_mode_map(&arm.pats[0]);
D
Daniel Micay 已提交
1842
        for (i, p) in arm.pats.iter().enumerate() {
J
Jonas Schievink 已提交
1843
            let map_i = self.binding_mode_map(&p);
1844

1845
            for (&key, &binding_0) in &map_0 {
1846
                match map_i.get(&key) {
C
corentih 已提交
1847
                    None => {
1848 1849
                        let error = ResolutionError::VariableNotBoundInPattern(key.name, 1, i + 1);
                        resolve_error(self, p.span, error);
C
corentih 已提交
1850 1851 1852 1853 1854
                    }
                    Some(binding_i) => {
                        if binding_0.binding_mode != binding_i.binding_mode {
                            resolve_error(self,
                                          binding_i.span,
M
Mikhail Modin 已提交
1855 1856 1857 1858
                                          ResolutionError::VariableBoundWithDifferentMode(
                                              key.name,
                                              i + 1,
                                              binding_0.span));
C
corentih 已提交
1859
                        }
1860
                    }
1861 1862 1863
                }
            }

1864
            for (&key, &binding) in &map_i {
1865
                if !map_0.contains_key(&key) {
1866 1867
                    resolve_error(self,
                                  binding.span,
1868
                                  ResolutionError::VariableNotBoundInPattern(key.name, i + 1, 1));
1869 1870 1871
                }
            }
        }
1872 1873
    }

F
Felix S. Klock II 已提交
1874
    fn resolve_arm(&mut self, arm: &Arm) {
J
Jeffrey Seyfried 已提交
1875
        self.ribs[ValueNS].push(Rib::new(NormalRibKind));
1876

1877
        let mut bindings_list = FxHashMap();
1878
        for pattern in &arm.pats {
1879
            self.resolve_pattern(&pattern, PatternSource::Match, &mut bindings_list);
1880 1881
        }

1882 1883 1884 1885
        // This has to happen *after* we determine which
        // pat_idents are variants
        self.check_consistent_bindings(arm);

1886
        walk_list!(self, visit_expr, &arm.guard);
J
Jonas Schievink 已提交
1887
        self.visit_expr(&arm.body);
1888

J
Jeffrey Seyfried 已提交
1889
        self.ribs[ValueNS].pop();
1890 1891
    }

E
Eduard Burtescu 已提交
1892
    fn resolve_block(&mut self, block: &Block) {
1893
        debug!("(resolving block) entering block");
1894
        // Move down in the graph, if there's an anonymous module rooted here.
1895
        let orig_module = self.current_module;
1896
        let anonymous_module = self.block_map.get(&block.id).cloned(); // clones a reference
1897

1898
        let mut num_macro_definition_ribs = 0;
1899 1900
        if let Some(anonymous_module) = anonymous_module {
            debug!("(resolving block) found anonymous module, moving down");
J
Jeffrey Seyfried 已提交
1901 1902
            self.ribs[ValueNS].push(Rib::new(ModuleRibKind(anonymous_module)));
            self.ribs[TypeNS].push(Rib::new(ModuleRibKind(anonymous_module)));
1903
            self.current_module = anonymous_module;
1904
            self.finalize_current_module_macro_resolutions();
1905
        } else {
J
Jeffrey Seyfried 已提交
1906
            self.ribs[ValueNS].push(Rib::new(NormalRibKind));
1907 1908 1909
        }

        // Descend into the block.
1910 1911
        for stmt in &block.stmts {
            if let Some(marks) = self.macros_at_scope.remove(&stmt.id) {
1912
                num_macro_definition_ribs += marks.len() as u32;
1913
                for mark in marks {
J
Jeffrey Seyfried 已提交
1914
                    self.ribs[ValueNS].push(Rib::new(MacroDefinition(mark)));
1915
                    self.label_ribs.push(Rib::new(MacroDefinition(mark)));
1916 1917 1918 1919 1920
                }
            }

            self.visit_stmt(stmt);
        }
1921 1922

        // Move back up.
J
Jeffrey Seyfried 已提交
1923
        self.current_module = orig_module;
1924
        for _ in 0 .. num_macro_definition_ribs {
J
Jeffrey Seyfried 已提交
1925
            self.ribs[ValueNS].pop();
1926
            self.label_ribs.pop();
1927
        }
J
Jeffrey Seyfried 已提交
1928
        self.ribs[ValueNS].pop();
J
Jeffrey Seyfried 已提交
1929
        if let Some(_) = anonymous_module {
J
Jeffrey Seyfried 已提交
1930
            self.ribs[TypeNS].pop();
G
Garming Sam 已提交
1931
        }
1932
        debug!("(resolving block) leaving block");
1933 1934
    }

1935
    fn fresh_binding(&mut self,
J
Jeffrey Seyfried 已提交
1936
                     ident: &SpannedIdent,
1937 1938 1939
                     pat_id: NodeId,
                     outer_pat_id: NodeId,
                     pat_src: PatternSource,
1940
                     bindings: &mut FxHashMap<Ident, NodeId>)
1941 1942
                     -> PathResolution {
        // Add the binding to the local ribs, if it
1943 1944
        // doesn't already exist in the bindings map. (We
        // must not add it if it's in the bindings map
1945 1946
        // because that breaks the assumptions later
        // passes make about or-patterns.)
1947
        let mut def = Def::Local(self.definitions.local_def_id(pat_id));
1948
        match bindings.get(&ident.node).cloned() {
1949 1950 1951 1952 1953 1954 1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967
            Some(id) if id == outer_pat_id => {
                // `Variant(a, a)`, error
                resolve_error(
                    self,
                    ident.span,
                    ResolutionError::IdentifierBoundMoreThanOnceInSamePattern(
                        &ident.node.name.as_str())
                );
            }
            Some(..) if pat_src == PatternSource::FnParam => {
                // `fn f(a: u8, a: u8)`, error
                resolve_error(
                    self,
                    ident.span,
                    ResolutionError::IdentifierBoundMoreThanOnceInParameterList(
                        &ident.node.name.as_str())
                );
            }
            Some(..) if pat_src == PatternSource::Match => {
1968 1969
                // `Variant1(a) | Variant2(a)`, ok
                // Reuse definition from the first `a`.
J
Jeffrey Seyfried 已提交
1970
                def = self.ribs[ValueNS].last_mut().unwrap().bindings[&ident.node];
1971 1972 1973 1974 1975 1976
            }
            Some(..) => {
                span_bug!(ident.span, "two bindings with the same name from \
                                       unexpected pattern source {:?}", pat_src);
            }
            None => {
1977
                // A completely fresh binding, add to the lists if it's valid.
1978
                if ident.node.name != keywords::Invalid.name() {
1979
                    bindings.insert(ident.node, outer_pat_id);
J
Jeffrey Seyfried 已提交
1980
                    self.ribs[ValueNS].last_mut().unwrap().bindings.insert(ident.node, def);
1981
                }
1982
            }
1983
        }
1984

1985
        PathResolution::new(def)
1986
    }
1987

1988 1989 1990 1991 1992
    fn resolve_pattern(&mut self,
                       pat: &Pat,
                       pat_src: PatternSource,
                       // Maps idents to the node ID for the
                       // outermost pattern that binds them.
1993
                       bindings: &mut FxHashMap<Ident, NodeId>) {
1994
        // Visit all direct subpatterns of this pattern.
1995 1996 1997 1998 1999 2000
        let outer_pat_id = pat.id;
        pat.walk(&mut |pat| {
            match pat.node {
                PatKind::Ident(bmode, ref ident, ref opt_pat) => {
                    // First try to resolve the identifier as some existing
                    // entity, then fall back to a fresh binding.
2001
                    let binding = self.resolve_ident_in_lexical_scope(ident.node, ValueNS, None)
2002
                                      .and_then(LexicalScopeBinding::item);
2003
                    let resolution = binding.map(NameBinding::def).and_then(|def| {
2004 2005
                        let always_binding = !pat_src.is_refutable() || opt_pat.is_some() ||
                                             bmode != BindingMode::ByValue(Mutability::Immutable);
2006
                        match def {
2007 2008 2009 2010
                            Def::StructCtor(_, CtorKind::Const) |
                            Def::VariantCtor(_, CtorKind::Const) |
                            Def::Const(..) if !always_binding => {
                                // A unit struct/variant or constant pattern.
2011
                                self.record_use(ident.node, ValueNS, binding.unwrap(), ident.span);
2012
                                Some(PathResolution::new(def))
2013
                            }
2014
                            Def::StructCtor(..) | Def::VariantCtor(..) |
2015
                            Def::Const(..) | Def::Static(..) => {
2016
                                // A fresh binding that shadows something unacceptable.
2017
                                resolve_error(
2018
                                    self,
2019 2020
                                    ident.span,
                                    ResolutionError::BindingShadowsSomethingUnacceptable(
2021
                                        pat_src.descr(), ident.node.name, binding.unwrap())
2022
                                );
2023
                                None
2024
                            }
2025
                            Def::Local(..) | Def::Upvar(..) | Def::Fn(..) | Def::Err => {
2026 2027
                                // These entities are explicitly allowed
                                // to be shadowed by fresh bindings.
2028
                                None
2029 2030 2031
                            }
                            def => {
                                span_bug!(ident.span, "unexpected definition for an \
2032
                                                       identifier in pattern: {:?}", def);
2033
                            }
2034
                        }
2035
                    }).unwrap_or_else(|| {
2036
                        self.fresh_binding(ident, pat.id, outer_pat_id, pat_src, bindings)
2037
                    });
2038 2039

                    self.record_def(pat.id, resolution);
2040 2041
                }

2042
                PatKind::TupleStruct(ref path, ..) => {
2043
                    self.smart_resolve_path(pat.id, None, path, PathSource::TupleStruct);
2044 2045
                }

2046
                PatKind::Path(ref qself, ref path) => {
2047
                    self.smart_resolve_path(pat.id, qself.as_ref(), path, PathSource::Pat);
2048 2049
                }

V
Vadim Petrochenkov 已提交
2050
                PatKind::Struct(ref path, ..) => {
2051
                    self.smart_resolve_path(pat.id, None, path, PathSource::Struct);
2052
                }
2053 2054

                _ => {}
2055
            }
2056
            true
2057
        });
2058

2059
        visit::walk_pat(self, pat);
2060 2061
    }

2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073 2074 2075 2076 2077
    // High-level and context dependent path resolution routine.
    // Resolves the path and records the resolution into definition map.
    // If resolution fails tries several techniques to find likely
    // resolution candidates, suggest imports or other help, and report
    // errors in user friendly way.
    fn smart_resolve_path(&mut self,
                          id: NodeId,
                          qself: Option<&QSelf>,
                          path: &Path,
                          source: PathSource)
                          -> PathResolution {
        let segments = &path.segments.iter().map(|seg| seg.identifier).collect::<Vec<_>>();
        self.smart_resolve_path_fragment(id, qself, segments, path.span, source)
    }

    fn smart_resolve_path_fragment(&mut self,
2078
                                   id: NodeId,
2079 2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092
                                   qself: Option<&QSelf>,
                                   path: &[Ident],
                                   span: Span,
                                   source: PathSource)
                                   -> PathResolution {
        let ns = source.namespace();
        let is_expected = &|def| source.is_expected(def);

        // Base error is amended with one short label and possibly some longer helps/notes.
        let report_errors = |this: &mut Self, def: Option<Def>| {
            // Make the base error.
            let expected = source.descr_expected();
            let path_str = names_to_string(path);
            let code = source.error_code(def.is_some());
2093 2094 2095
            let (base_msg, fallback_label) = if let Some(def) = def {
                (format!("expected {}, found {} `{}`", expected, def.kind_name(), path_str),
                 format!("not a {}", expected))
2096
            } else {
2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111
                let item_str = path[path.len() - 1];
                let (mod_prefix, mod_str) = if path.len() == 1 {
                    (format!(""), format!("this scope"))
                } else if path.len() == 2 && path[0].name == keywords::CrateRoot.name() {
                    (format!(""), format!("the crate root"))
                } else {
                    let mod_path = &path[..path.len() - 1];
                    let mod_prefix = match this.resolve_path(mod_path, Some(TypeNS), None) {
                        PathResult::Module(module) => module.def(),
                        _ => None,
                    }.map_or(format!(""), |def| format!("{} ", def.kind_name()));
                    (mod_prefix, format!("`{}`", names_to_string(mod_path)))
                };
                (format!("cannot find {} `{}` in {}{}", expected, item_str, mod_prefix, mod_str),
                 format!("not found in {}", mod_str))
2112 2113 2114 2115 2116 2117 2118 2119 2120 2121 2122 2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134 2135 2136 2137 2138 2139 2140 2141 2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172 2173 2174 2175 2176 2177 2178 2179 2180 2181 2182 2183 2184 2185 2186 2187 2188 2189 2190 2191 2192 2193
            };
            let mut err = this.session.struct_span_err_with_code(span, &base_msg, code);

            // Emit special messages for unresolved `Self` and `self`.
            if is_self_type(path, ns) {
                __diagnostic_used!(E0411);
                err.code("E0411".into());
                err.span_label(span, &format!("`Self` is only available in traits and impls"));
                return err;
            }
            if is_self_value(path, ns) {
                __diagnostic_used!(E0424);
                err.code("E0424".into());
                err.span_label(span, &format!("`self` value is only available in \
                                               methods with `self` parameter"));
                return err;
            }

            // Try to lookup the name in more relaxed fashion for better error reporting.
            let name = path.last().unwrap().name;
            let candidates = this.lookup_import_candidates(name, ns, is_expected);
            if !candidates.is_empty() {
                // Report import candidates as help and proceed searching for labels.
                show_candidates(&mut err, &candidates, def.is_some());
            }
            if path.len() == 1 && this.self_type_is_available() {
                if let Some(candidate) = this.lookup_assoc_candidate(name, ns, is_expected) {
                    let self_is_available = this.self_value_is_available(path[0].ctxt);
                    match candidate {
                        AssocSuggestion::Field => {
                            err.span_label(span, &format!("did you mean `self.{}`?", path_str));
                            if !self_is_available {
                                err.span_label(span, &format!("`self` value is only available in \
                                                               methods with `self` parameter"));
                            }
                        }
                        AssocSuggestion::MethodWithSelf if self_is_available => {
                            err.span_label(span, &format!("did you mean `self.{}(...)`?",
                                                           path_str));
                        }
                        AssocSuggestion::MethodWithSelf | AssocSuggestion::AssocItem => {
                            err.span_label(span, &format!("did you mean `Self::{}`?", path_str));
                        }
                    }
                    return err;
                }
            }

            // Try context dependent help if relaxed lookup didn't work.
            if let Some(def) = def {
                match (def, source) {
                    (Def::Macro(..), _) => {
                        err.span_label(span, &format!("did you mean `{}!(...)`?", path_str));
                        return err;
                    }
                    (Def::TyAlias(..), PathSource::Trait) => {
                        err.span_label(span, &format!("type aliases cannot be used for traits"));
                        return err;
                    }
                    (Def::Mod(..), PathSource::Expr(Some(parent))) => match *parent {
                        ExprKind::Field(_, ident) => {
                            err.span_label(span, &format!("did you mean `{}::{}`?",
                                                           path_str, ident.node));
                            return err;
                        }
                        ExprKind::MethodCall(ident, ..) => {
                            err.span_label(span, &format!("did you mean `{}::{}(...)`?",
                                                           path_str, ident.node));
                            return err;
                        }
                        _ => {}
                    },
                    _ if ns == ValueNS && is_struct_like(def) => {
                        err.span_label(span, &format!("did you mean `{} {{ /* fields */ }}`?",
                                                       path_str));
                        return err;
                    }
                    _ => {}
                }
            }

            // Try Levenshtein if nothing else worked.
2194 2195 2196
            if let Some(candidate) = this.lookup_typo_candidate(path, ns, is_expected) {
                err.span_label(span, &format!("did you mean `{}`?", candidate));
                return err;
2197 2198
            }

2199 2200
            // Fallback label.
            err.span_label(span, &fallback_label);
2201 2202 2203 2204 2205 2206 2207 2208 2209 2210 2211 2212 2213 2214 2215 2216 2217 2218 2219 2220 2221 2222 2223 2224 2225 2226 2227 2228 2229 2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284 2285 2286 2287 2288 2289 2290
            err
        };
        let report_errors = |this: &mut Self, def: Option<Def>| {
            report_errors(this, def).emit();
            err_path_resolution()
        };

        let resolution = match self.resolve_qpath_anywhere(id, qself, path, ns, span,
                                                           source.defer_to_typeck(),
                                                           source.global_by_default()) {
            Some(resolution) if resolution.depth == 0 => {
                if is_expected(resolution.base_def) || resolution.base_def == Def::Err {
                    resolution
                } else {
                    report_errors(self, Some(resolution.base_def))
                }
            }
            Some(resolution) if source.defer_to_typeck() => {
                // Not fully resolved associated item `T::A::B` or `<T as Tr>::A::B`
                // or `<T>::A::B`. If `B` should be resolved in value namespace then
                // it needs to be added to the trait map.
                if ns == ValueNS {
                    let item_name = path.last().unwrap().name;
                    let traits = self.get_traits_containing_item(item_name, ns);
                    self.trait_map.insert(id, traits);
                }
                resolution
            }
            _ => report_errors(self, None)
        };

        if let PathSource::TraitItem(..) = source {} else {
            // Avoid recording definition of `A::B` in `<T as A>::B::C`.
            self.record_def(id, resolution);
        }
        resolution
    }

    fn self_type_is_available(&mut self) -> bool {
        let binding = self.resolve_ident_in_lexical_scope(keywords::SelfType.ident(), TypeNS, None);
        if let Some(LexicalScopeBinding::Def(def)) = binding { def != Def::Err } else { false }
    }

    fn self_value_is_available(&mut self, ctxt: SyntaxContext) -> bool {
        let ident = Ident { name: keywords::SelfValue.name(), ctxt: ctxt };
        let binding = self.resolve_ident_in_lexical_scope(ident, ValueNS, None);
        if let Some(LexicalScopeBinding::Def(def)) = binding { def != Def::Err } else { false }
    }

    // Resolve in alternative namespaces if resolution in the primary namespace fails.
    fn resolve_qpath_anywhere(&mut self,
                              id: NodeId,
                              qself: Option<&QSelf>,
                              path: &[Ident],
                              primary_ns: Namespace,
                              span: Span,
                              defer_to_typeck: bool,
                              global_by_default: bool)
                              -> Option<PathResolution> {
        let mut fin_res = None;
        // FIXME: can't resolve paths in macro namespace yet, macros are
        // processed by the little special hack below.
        for (i, ns) in [primary_ns, TypeNS, ValueNS, /*MacroNS*/].iter().cloned().enumerate() {
            if i == 0 || ns != primary_ns {
                match self.resolve_qpath(id, qself, path, ns, span, global_by_default) {
                    // If defer_to_typeck, then resolution > no resolution,
                    // otherwise full resolution > partial resolution > no resolution.
                    Some(res) if res.depth == 0 || defer_to_typeck => return Some(res),
                    res => if fin_res.is_none() { fin_res = res },
                };
            }
        }
        if primary_ns != MacroNS && path.len() == 1 &&
                self.macro_names.contains(&path[0].name) {
            // Return some dummy definition, it's enough for error reporting.
            return Some(PathResolution::new(Def::Macro(DefId::local(CRATE_DEF_INDEX))));
        }
        fin_res
    }

    /// Handles paths that may refer to associated items.
    fn resolve_qpath(&mut self,
                     id: NodeId,
                     qself: Option<&QSelf>,
                     path: &[Ident],
                     ns: Namespace,
                     span: Span,
                     global_by_default: bool)
                     -> Option<PathResolution> {
        if let Some(qself) = qself {
J
Jeffrey Seyfried 已提交
2291 2292 2293
            if qself.position == 0 {
                // FIXME: Create some fake resolution that can't possibly be a type.
                return Some(PathResolution {
2294
                    base_def: Def::Mod(DefId::local(CRATE_DEF_INDEX)),
J
Jeffrey Seyfried 已提交
2295 2296
                    depth: path.len(),
                });
2297
            }
2298 2299 2300 2301 2302 2303 2304 2305
            // Make sure `A::B` in `<T as A>::B::C` is a trait item.
            let ns = if qself.position + 1 == path.len() { ns } else { TypeNS };
            let mut res = self.smart_resolve_path_fragment(id, None, &path[..qself.position + 1],
                                                           span, PathSource::TraitItem(ns));
            if res.base_def != Def::Err {
                res.depth += path.len() - qself.position - 1;
            }
            return Some(res);
2306 2307
        }

2308
        let result = match self.resolve_path(&path, Some(ns), Some(span)) {
2309
            PathResult::NonModule(path_res) => path_res,
J
Jeffrey Seyfried 已提交
2310 2311
            PathResult::Module(module) if !module.is_normal() => {
                PathResolution::new(module.def().unwrap())
V
Cleanup  
Vadim Petrochenkov 已提交
2312
            }
2313 2314 2315 2316 2317 2318
            // In `a(::assoc_item)*` `a` cannot be a module. If `a` does resolve to a module we
            // don't report an error right away, but try to fallback to a primitive type.
            // So, we are still able to successfully resolve something like
            //
            // use std::u8; // bring module u8 in scope
            // fn f() -> u8 { // OK, resolves to primitive u8, not to std::u8
V
Cleanup  
Vadim Petrochenkov 已提交
2319 2320
            //     u8::max_value() // OK, resolves to associated function <u8>::max_value,
            //                     // not to non-existent std::u8::max_value
2321 2322 2323 2324
            // }
            //
            // Such behavior is required for backward compatibility.
            // The same fallback is used when `a` resolves to nothing.
2325
            PathResult::Module(..) | PathResult::Failed(..)
2326
                    if (ns == TypeNS || path.len() > 1) &&
2327
                       self.primitive_type_table.primitive_types.contains_key(&path[0].name) => {
2328 2329 2330
                let prim = self.primitive_type_table.primitive_types[&path[0].name];
                match prim {
                    TyUint(UintTy::U128) | TyInt(IntTy::I128) => {
S
Simonas Kazlauskas 已提交
2331 2332
                        if !self.session.features.borrow().i128_type {
                            emit_feature_err(&self.session.parse_sess,
2333 2334 2335 2336 2337 2338 2339
                                                "i128_type", span, GateIssue::Language,
                                                "128-bit type is unstable");

                        }
                    }
                    _ => {}
                }
J
Jeffrey Seyfried 已提交
2340
                PathResolution {
2341
                    base_def: Def::PrimTy(prim),
2342
                    depth: path.len() - 1,
J
Jeffrey Seyfried 已提交
2343 2344 2345 2346 2347 2348 2349
                }
            }
            PathResult::Module(module) => PathResolution::new(module.def().unwrap()),
            PathResult::Failed(msg, false) => {
                resolve_error(self, span, ResolutionError::FailedToResolve(&msg));
                err_path_resolution()
            }
2350 2351
            PathResult::Failed(..) => return None,
            PathResult::Indeterminate => bug!("indetermined path result in resolve_qpath"),
J
Jeffrey Seyfried 已提交
2352 2353
        };

2354 2355 2356 2357 2358 2359 2360 2361 2362 2363 2364 2365
        if path.len() > 1 && !global_by_default && result.base_def != Def::Err &&
           path[0].name != keywords::CrateRoot.name() && path[0].name != "$crate" {
            let unqualified_result = {
                match self.resolve_path(&[*path.last().unwrap()], Some(ns), None) {
                    PathResult::NonModule(path_res) => path_res.base_def,
                    PathResult::Module(module) => module.def().unwrap(),
                    _ => return Some(result),
                }
            };
            if result.base_def == unqualified_result {
                let lint = lint::builtin::UNUSED_QUALIFICATIONS;
                self.session.add_lint(lint, id, span, "unnecessary qualification".to_string());
N
Nick Cameron 已提交
2366
            }
2367
        }
N
Nick Cameron 已提交
2368

J
Jeffrey Seyfried 已提交
2369
        Some(result)
2370 2371
    }

J
Jeffrey Seyfried 已提交
2372 2373 2374 2375 2376
    fn resolve_path(&mut self,
                    path: &[Ident],
                    opt_ns: Option<Namespace>, // `None` indicates a module path
                    record_used: Option<Span>)
                    -> PathResult<'a> {
2377 2378
        let mut module = None;
        let mut allow_super = true;
J
Jeffrey Seyfried 已提交
2379 2380 2381 2382 2383

        for (i, &ident) in path.iter().enumerate() {
            let is_last = i == path.len() - 1;
            let ns = if is_last { opt_ns.unwrap_or(TypeNS) } else { TypeNS };

2384
            if i == 0 && ns == TypeNS && ident.name == keywords::SelfValue.name() {
2385
                module = Some(self.module_map[&self.current_module.normal_ancestor_id]);
J
Jeffrey Seyfried 已提交
2386 2387 2388
                continue
            } else if allow_super && ns == TypeNS && ident.name == keywords::Super.name() {
                let current_module = if i == 0 { self.current_module } else { module.unwrap() };
2389
                let self_module = self.module_map[&current_module.normal_ancestor_id];
J
Jeffrey Seyfried 已提交
2390
                if let Some(parent) = self_module.parent {
2391
                    module = Some(self.module_map[&parent.normal_ancestor_id]);
J
Jeffrey Seyfried 已提交
2392 2393 2394 2395 2396 2397 2398 2399
                    continue
                } else {
                    let msg = "There are too many initial `super`s.".to_string();
                    return PathResult::Failed(msg, false);
                }
            }
            allow_super = false;

2400 2401 2402 2403 2404 2405 2406 2407
            if i == 0 && ns == TypeNS && ident.name == keywords::CrateRoot.name() {
                module = Some(self.graph_root);
                continue
            } else if i == 0 && ns == TypeNS && ident.name == "$crate" {
                module = Some(self.resolve_crate_var(ident.ctxt));
                continue
            }

J
Jeffrey Seyfried 已提交
2408
            let binding = if let Some(module) = module {
2409
                self.resolve_ident_in_module(module, ident, ns, false, record_used)
2410
            } else if opt_ns == Some(MacroNS) {
2411
                self.resolve_lexical_macro_path_segment(ident, ns, record_used)
J
Jeffrey Seyfried 已提交
2412 2413 2414
            } else {
                match self.resolve_ident_in_lexical_scope(ident, ns, record_used) {
                    Some(LexicalScopeBinding::Item(binding)) => Ok(binding),
2415 2416
                    Some(LexicalScopeBinding::Def(def))
                            if opt_ns == Some(TypeNS) || opt_ns == Some(ValueNS) => {
J
Jeffrey Seyfried 已提交
2417 2418 2419 2420 2421 2422 2423 2424 2425 2426 2427
                        return PathResult::NonModule(PathResolution {
                            base_def: def,
                            depth: path.len() - 1,
                        });
                    }
                    _ => Err(if record_used.is_some() { Determined } else { Undetermined }),
                }
            };

            match binding {
                Ok(binding) => {
2428 2429
                    let def = binding.def();
                    let maybe_assoc = opt_ns != Some(MacroNS) && PathSource::Type.is_expected(def);
J
Jeffrey Seyfried 已提交
2430
                    if let Some(next_module) = binding.module() {
J
Jeffrey Seyfried 已提交
2431
                        module = Some(next_module);
2432
                    } else if def == Def::Err {
J
Jeffrey Seyfried 已提交
2433
                        return PathResult::NonModule(err_path_resolution());
2434
                    } else if opt_ns.is_some() && (is_last || maybe_assoc) {
J
Jeffrey Seyfried 已提交
2435
                        return PathResult::NonModule(PathResolution {
2436
                            base_def: def,
J
Jeffrey Seyfried 已提交
2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448 2449 2450 2451 2452
                            depth: path.len() - i - 1,
                        });
                    } else {
                        return PathResult::Failed(format!("Not a module `{}`", ident), is_last);
                    }
                }
                Err(Undetermined) => return PathResult::Indeterminate,
                Err(Determined) => {
                    if let Some(module) = module {
                        if opt_ns.is_some() && !module.is_normal() {
                            return PathResult::NonModule(PathResolution {
                                base_def: module.def().unwrap(),
                                depth: path.len() - i,
                            });
                        }
                    }
2453
                    let msg = if module.and_then(ModuleData::def) == self.graph_root.def() {
J
Jeffrey Seyfried 已提交
2454 2455
                        let is_mod = |def| match def { Def::Mod(..) => true, _ => false };
                        let mut candidates =
2456 2457
                            self.lookup_import_candidates(ident.name, TypeNS, is_mod);
                        candidates.sort_by_key(|c| (c.path.segments.len(), c.path.to_string()));
J
Jeffrey Seyfried 已提交
2458
                        if let Some(candidate) = candidates.get(0) {
2459
                            format!("Did you mean `{}`?", candidate.path)
J
Jeffrey Seyfried 已提交
2460 2461 2462 2463 2464 2465 2466 2467 2468 2469 2470
                        } else {
                            format!("Maybe a missing `extern crate {};`?", ident)
                        }
                    } else if i == 0 {
                        format!("Use of undeclared type or module `{}`", ident)
                    } else {
                        format!("Could not find `{}` in `{}`", ident, path[i - 1])
                    };
                    return PathResult::Failed(msg, is_last);
                }
            }
2471 2472
        }

2473
        PathResult::Module(module.unwrap_or(self.graph_root))
2474 2475 2476
    }

    // Resolve a local definition, potentially adjusting for closures.
2477
    fn adjust_local_def(&mut self, local_def: LocalDef, record_used: Option<Span>) -> Def {
2478
        let ribs = match local_def.ribs {
J
Jeffrey Seyfried 已提交
2479 2480
            Some((ns, i)) => &self.ribs[ns][i + 1..],
            None => &[] as &[_],
2481 2482 2483
        };
        let mut def = local_def.def;
        match def {
2484
            Def::Upvar(..) => {
2485
                span_bug!(record_used.unwrap_or(DUMMY_SP), "unexpected {:?} in bindings", def)
2486
            }
2487
            Def::Local(def_id) => {
2488 2489
                for rib in ribs {
                    match rib.kind {
2490
                        NormalRibKind | ModuleRibKind(..) | MacroDefinition(..) => {
2491 2492 2493 2494
                            // Nothing to do. Continue.
                        }
                        ClosureRibKind(function_id) => {
                            let prev_def = def;
2495
                            let node_id = self.definitions.as_local_node_id(def_id).unwrap();
2496

C
corentih 已提交
2497 2498 2499
                            let seen = self.freevars_seen
                                           .entry(function_id)
                                           .or_insert_with(|| NodeMap());
2500
                            if let Some(&index) = seen.get(&node_id) {
2501
                                def = Def::Upvar(def_id, index, function_id);
2502 2503
                                continue;
                            }
C
corentih 已提交
2504 2505 2506
                            let vec = self.freevars
                                          .entry(function_id)
                                          .or_insert_with(|| vec![]);
2507
                            let depth = vec.len();
2508
                            def = Def::Upvar(def_id, depth, function_id);
2509 2510 2511 2512 2513 2514 2515 2516

                            if let Some(span) = record_used {
                                vec.push(Freevar {
                                    def: prev_def,
                                    span: span,
                                });
                                seen.insert(node_id, depth);
                            }
2517
                        }
2518
                        ItemRibKind | MethodRibKind(_) => {
2519 2520 2521
                            // This was an attempt to access an upvar inside a
                            // named function item. This is not allowed, so we
                            // report an error.
2522 2523 2524 2525
                            if let Some(span) = record_used {
                                resolve_error(self, span,
                                        ResolutionError::CannotCaptureDynamicEnvironmentInFnItem);
                            }
2526
                            return Def::Err;
2527 2528 2529
                        }
                        ConstantItemRibKind => {
                            // Still doesn't deal with upvars
2530 2531 2532 2533
                            if let Some(span) = record_used {
                                resolve_error(self, span,
                                        ResolutionError::AttemptToUseNonConstantValueInConstant);
                            }
2534
                            return Def::Err;
2535 2536 2537 2538
                        }
                    }
                }
            }
2539
            Def::TyParam(..) | Def::SelfTy(..) => {
2540 2541
                for rib in ribs {
                    match rib.kind {
2542
                        NormalRibKind | MethodRibKind(_) | ClosureRibKind(..) |
2543
                        ModuleRibKind(..) | MacroDefinition(..) => {
2544 2545 2546 2547 2548
                            // Nothing to do. Continue.
                        }
                        ItemRibKind => {
                            // This was an attempt to use a type parameter outside
                            // its scope.
2549 2550 2551 2552
                            if let Some(span) = record_used {
                                resolve_error(self, span,
                                              ResolutionError::TypeParametersFromOuterFunction);
                            }
2553
                            return Def::Err;
2554 2555 2556
                        }
                        ConstantItemRibKind => {
                            // see #9186
2557 2558 2559 2560
                            if let Some(span) = record_used {
                                resolve_error(self, span,
                                              ResolutionError::OuterTypeParameterContext);
                            }
2561
                            return Def::Err;
2562 2563 2564 2565 2566 2567
                        }
                    }
                }
            }
            _ => {}
        }
2568
        return def;
2569 2570
    }

2571 2572
    // Calls `f` with a `Resolver` whose current lexical scope is `module`'s lexical scope,
    // i.e. the module's items and the prelude (unless the module is `#[no_implicit_prelude]`).
J
Jeffrey Seyfried 已提交
2573
    // FIXME #34673: This needs testing.
2574 2575 2576 2577
    pub fn with_module_lexical_scope<T, F>(&mut self, module: Module<'a>, f: F) -> T
        where F: FnOnce(&mut Resolver<'a>) -> T,
    {
        self.with_empty_ribs(|this| {
J
Jeffrey Seyfried 已提交
2578 2579
            this.ribs[ValueNS].push(Rib::new(ModuleRibKind(module)));
            this.ribs[TypeNS].push(Rib::new(ModuleRibKind(module)));
2580 2581 2582 2583 2584 2585 2586
            f(this)
        })
    }

    fn with_empty_ribs<T, F>(&mut self, f: F) -> T
        where F: FnOnce(&mut Resolver<'a>) -> T,
    {
J
Jeffrey Seyfried 已提交
2587
        let ribs = replace(&mut self.ribs, PerNS::<Vec<Rib>>::default());
2588 2589 2590
        let label_ribs = replace(&mut self.label_ribs, Vec::new());

        let result = f(self);
J
Jeffrey Seyfried 已提交
2591
        self.ribs = ribs;
2592 2593 2594 2595
        self.label_ribs = label_ribs;
        result
    }

2596 2597 2598 2599 2600 2601 2602
    fn lookup_assoc_candidate<FilterFn>(&mut self,
                                        name: Name,
                                        ns: Namespace,
                                        filter_fn: FilterFn)
                                        -> Option<AssocSuggestion>
        where FilterFn: Fn(Def) -> bool
    {
2603
        fn extract_node_id(t: &Ty) -> Option<NodeId> {
2604
            match t.node {
2605 2606
                TyKind::Path(None, _) => Some(t.id),
                TyKind::Rptr(_, ref mut_ty) => extract_node_id(&mut_ty.ty),
2607 2608 2609 2610 2611 2612 2613
                // 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,
            }
        }

2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624
        // Fields are generally expected in the same contexts as locals.
        if filter_fn(Def::Local(DefId::local(CRATE_DEF_INDEX))) {
            if let Some(node_id) = self.current_self_type.as_ref().and_then(extract_node_id) {
                // Look for a field with the same name in the current self_type.
                if let Some(resolution) = self.def_map.get(&node_id) {
                    match resolution.base_def {
                        Def::Struct(did) | Def::Union(did) if resolution.depth == 0 => {
                            if let Some(field_names) = self.field_names.get(&did) {
                                if field_names.iter().any(|&field_name| name == field_name) {
                                    return Some(AssocSuggestion::Field);
                                }
2625
                            }
2626
                        }
2627
                        _ => {}
2628
                    }
2629
                }
2630
            }
2631 2632
        }

2633 2634 2635 2636 2637 2638 2639 2640 2641
        // Look for associated items in the current trait.
        if let Some((trait_did, _)) = self.current_trait_ref {
            if let Some(&(def, has_self)) = self.trait_item_map.get(&(trait_did, name, ns)) {
                if filter_fn(def) {
                    return Some(if has_self {
                        AssocSuggestion::MethodWithSelf
                    } else {
                        AssocSuggestion::AssocItem
                    });
2642 2643 2644 2645
                }
            }
        }

2646
        None
2647 2648
    }

2649
    fn lookup_typo_candidate<FilterFn>(&mut self,
2650
                                       path: &[Ident],
2651 2652
                                       ns: Namespace,
                                       filter_fn: FilterFn)
2653
                                       -> Option<String>
2654 2655
        where FilterFn: Fn(Def) -> bool
    {
2656 2657 2658 2659 2660 2661 2662 2663 2664 2665 2666 2667 2668 2669 2670 2671 2672 2673 2674 2675 2676 2677 2678 2679 2680 2681 2682 2683 2684 2685 2686 2687 2688 2689 2690 2691 2692 2693 2694 2695 2696 2697 2698 2699 2700 2701 2702 2703 2704 2705 2706 2707 2708 2709 2710 2711 2712 2713 2714 2715
        let add_module_candidates = |module: Module, names: &mut Vec<Name>| {
            for (&(ident, _), resolution) in module.resolutions.borrow().iter() {
                if let Some(binding) = resolution.borrow().binding {
                    if filter_fn(binding.def()) {
                        names.push(ident.name);
                    }
                }
            }
        };

        let mut names = Vec::new();
        let prefix_str = if path.len() == 1 {
            // Search in lexical scope.
            // Walk backwards up the ribs in scope and collect candidates.
            for rib in self.ribs[ns].iter().rev() {
                // Locals and type parameters
                for (ident, def) in &rib.bindings {
                    if filter_fn(*def) {
                        names.push(ident.name);
                    }
                }
                // Items in scope
                if let ModuleRibKind(module) = rib.kind {
                    // Items from this module
                    add_module_candidates(module, &mut names);

                    if let ModuleKind::Block(..) = module.kind {
                        // We can see through blocks
                    } else {
                        // Items from the prelude
                        if let Some(prelude) = self.prelude {
                            if !module.no_implicit_prelude {
                                add_module_candidates(prelude, &mut names);
                            }
                        }
                        break;
                    }
                }
            }
            // Add primitive types to the mix
            if filter_fn(Def::PrimTy(TyBool)) {
                for (name, _) in &self.primitive_type_table.primitive_types {
                    names.push(*name);
                }
            }
            String::new()
        } else {
            // Search in module.
            let mod_path = &path[..path.len() - 1];
            if let PathResult::Module(module) = self.resolve_path(mod_path, Some(TypeNS), None) {
                add_module_candidates(module, &mut names);
            }
            names_to_string(mod_path) + "::"
        };

        let name = path[path.len() - 1].name;
        // Make sure error reporting is deterministic.
        names.sort_by_key(|name| name.as_str());
        match find_best_match_for_name(names.iter(), &name.as_str(), None) {
            Some(found) if found != name => Some(format!("{}{}", prefix_str, found)),
2716
            _ => None,
2717
        }
2718 2719
    }

J
Jeffrey Seyfried 已提交
2720
    fn resolve_labeled_block(&mut self, label: Option<SpannedIdent>, id: NodeId, block: &Block) {
2721
        if let Some(label) = label {
2722
            let def = Def::Label(id);
2723
            self.with_label_rib(|this| {
J
Jeffrey Seyfried 已提交
2724
                this.label_ribs.last_mut().unwrap().bindings.insert(label.node, def);
2725 2726 2727 2728 2729 2730 2731
                this.visit_block(block);
            });
        } else {
            self.visit_block(block);
        }
    }

2732
    fn resolve_expr(&mut self, expr: &Expr, parent: Option<&ExprKind>) {
P
Patrick Walton 已提交
2733 2734
        // First, record candidate traits for this expression if it could
        // result in the invocation of a method call.
2735 2736 2737

        self.record_candidate_traits_for_expr_if_necessary(expr);

2738
        // Next, resolve the node.
2739
        match expr.node {
2740 2741
            ExprKind::Path(ref qself, ref path) => {
                self.smart_resolve_path(expr.id, qself.as_ref(), path, PathSource::Expr(parent));
2742
                visit::walk_expr(self, expr);
2743 2744
            }

V
Vadim Petrochenkov 已提交
2745
            ExprKind::Struct(ref path, ..) => {
2746
                self.smart_resolve_path(expr.id, None, path, PathSource::Struct);
2747
                visit::walk_expr(self, expr);
2748 2749
            }

2750
            ExprKind::Break(Some(label), _) | ExprKind::Continue(Some(label)) => {
2751
                match self.search_label(label.node) {
2752
                    None => {
2753
                        self.record_def(expr.id, err_path_resolution());
2754
                        resolve_error(self,
2755
                                      label.span,
2756
                                      ResolutionError::UndeclaredLabel(&label.node.name.as_str()));
2757
                    }
2758
                    Some(def @ Def::Label(_)) => {
2759
                        // Since this def is a label, it is never read.
2760
                        self.record_def(expr.id, PathResolution::new(def));
2761 2762
                    }
                    Some(_) => {
2763
                        span_bug!(expr.span, "label wasn't mapped to a label def!");
2764 2765
                    }
                }
2766 2767 2768

                // visit `break` argument if any
                visit::walk_expr(self, expr);
2769
            }
2770 2771 2772 2773

            ExprKind::IfLet(ref pattern, ref subexpression, ref if_block, ref optional_else) => {
                self.visit_expr(subexpression);

J
Jeffrey Seyfried 已提交
2774
                self.ribs[ValueNS].push(Rib::new(NormalRibKind));
2775
                self.resolve_pattern(pattern, PatternSource::IfLet, &mut FxHashMap());
2776
                self.visit_block(if_block);
J
Jeffrey Seyfried 已提交
2777
                self.ribs[ValueNS].pop();
2778 2779 2780 2781

                optional_else.as_ref().map(|expr| self.visit_expr(expr));
            }

J
Jeffrey Seyfried 已提交
2782 2783 2784 2785 2786 2787 2788
            ExprKind::Loop(ref block, label) => self.resolve_labeled_block(label, expr.id, &block),

            ExprKind::While(ref subexpression, ref block, label) => {
                self.visit_expr(subexpression);
                self.resolve_labeled_block(label, expr.id, &block);
            }

2789 2790
            ExprKind::WhileLet(ref pattern, ref subexpression, ref block, label) => {
                self.visit_expr(subexpression);
J
Jeffrey Seyfried 已提交
2791
                self.ribs[ValueNS].push(Rib::new(NormalRibKind));
2792
                self.resolve_pattern(pattern, PatternSource::WhileLet, &mut FxHashMap());
2793

J
Jeffrey Seyfried 已提交
2794
                self.resolve_labeled_block(label, expr.id, block);
2795

J
Jeffrey Seyfried 已提交
2796
                self.ribs[ValueNS].pop();
2797 2798 2799 2800
            }

            ExprKind::ForLoop(ref pattern, ref subexpression, ref block, label) => {
                self.visit_expr(subexpression);
J
Jeffrey Seyfried 已提交
2801
                self.ribs[ValueNS].push(Rib::new(NormalRibKind));
2802
                self.resolve_pattern(pattern, PatternSource::For, &mut FxHashMap());
2803

J
Jeffrey Seyfried 已提交
2804
                self.resolve_labeled_block(label, expr.id, block);
2805

J
Jeffrey Seyfried 已提交
2806
                self.ribs[ValueNS].pop();
2807 2808
            }

2809
            // Equivalent to `visit::walk_expr` + passing some context to children.
2810
            ExprKind::Field(ref subexpression, _) => {
2811
                self.resolve_expr(subexpression, Some(&expr.node));
2812
            }
2813
            ExprKind::MethodCall(_, ref types, ref arguments) => {
2814
                let mut arguments = arguments.iter();
2815
                self.resolve_expr(arguments.next().unwrap(), Some(&expr.node));
2816 2817 2818 2819 2820 2821 2822
                for argument in arguments {
                    self.resolve_expr(argument, None);
                }
                for ty in types.iter() {
                    self.visit_ty(ty);
                }
            }
2823 2824 2825 2826 2827 2828 2829

            ExprKind::Repeat(ref element, ref count) => {
                self.visit_expr(element);
                self.with_constant_rib(|this| {
                    this.visit_expr(count);
                });
            }
2830 2831 2832 2833 2834 2835
            ExprKind::Call(ref callee, ref arguments) => {
                self.resolve_expr(callee, Some(&expr.node));
                for argument in arguments {
                    self.resolve_expr(argument, None);
                }
            }
2836

B
Brian Anderson 已提交
2837
            _ => {
2838
                visit::walk_expr(self, expr);
2839 2840 2841 2842
            }
        }
    }

E
Eduard Burtescu 已提交
2843
    fn record_candidate_traits_for_expr_if_necessary(&mut self, expr: &Expr) {
2844
        match expr.node {
2845
            ExprKind::Field(_, name) => {
2846 2847 2848 2849
                // 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.
2850
                let traits = self.get_traits_containing_item(name.node.name, ValueNS);
2851
                self.trait_map.insert(expr.id, traits);
2852
            }
V
Vadim Petrochenkov 已提交
2853
            ExprKind::MethodCall(name, ..) => {
C
corentih 已提交
2854
                debug!("(recording candidate traits for expr) recording traits for {}",
2855
                       expr.id);
2856
                let traits = self.get_traits_containing_item(name.node.name, ValueNS);
2857
                self.trait_map.insert(expr.id, traits);
2858
            }
2859
            _ => {
2860 2861 2862 2863 2864
                // Nothing to do.
            }
        }
    }

2865
    fn get_traits_containing_item(&mut self, name: Name, ns: Namespace) -> Vec<TraitCandidate> {
C
corentih 已提交
2866
        debug!("(getting traits containing item) looking for '{}'", name);
E
Eduard Burtescu 已提交
2867

2868
        let mut found_traits = Vec::new();
J
Jeffrey Seyfried 已提交
2869 2870
        // Look for the current trait.
        if let Some((trait_def_id, _)) = self.current_trait_ref {
2871
            if self.trait_item_map.contains_key(&(trait_def_id, name, ns)) {
2872
                found_traits.push(TraitCandidate { def_id: trait_def_id, import_id: None });
E
Eduard Burtescu 已提交
2873
            }
J
Jeffrey Seyfried 已提交
2874
        }
2875

J
Jeffrey Seyfried 已提交
2876 2877
        let mut search_module = self.current_module;
        loop {
2878
            self.get_traits_in_module_containing_item(name, ns, search_module, &mut found_traits);
2879 2880 2881 2882 2883
            match search_module.kind {
                ModuleKind::Block(..) => search_module = search_module.parent.unwrap(),
                _ => break,
            }
        }
2884

2885 2886
        if let Some(prelude) = self.prelude {
            if !search_module.no_implicit_prelude {
2887
                self.get_traits_in_module_containing_item(name, ns, prelude, &mut found_traits);
E
Eduard Burtescu 已提交
2888
            }
2889 2890
        }

E
Eduard Burtescu 已提交
2891
        found_traits
2892 2893
    }

2894 2895
    fn get_traits_in_module_containing_item(&mut self,
                                            name: Name,
2896
                                            ns: Namespace,
2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907 2908 2909 2910 2911 2912
                                            module: Module,
                                            found_traits: &mut Vec<TraitCandidate>) {
        let mut traits = module.traits.borrow_mut();
        if traits.is_none() {
            let mut collected_traits = Vec::new();
            module.for_each_child(|name, ns, binding| {
                if ns != TypeNS { return }
                if let Def::Trait(_) = binding.def() {
                    collected_traits.push((name, binding));
                }
            });
            *traits = Some(collected_traits.into_boxed_slice());
        }

        for &(trait_name, binding) in traits.as_ref().unwrap().iter() {
            let trait_def_id = binding.def().def_id();
2913
            if self.trait_item_map.contains_key(&(trait_def_id, name, ns)) {
2914 2915 2916 2917 2918 2919 2920 2921 2922 2923 2924 2925 2926
                let import_id = match binding.kind {
                    NameBindingKind::Import { directive, .. } => {
                        self.maybe_unused_trait_imports.insert(directive.id);
                        self.add_to_glob_map(directive.id, trait_name);
                        Some(directive.id)
                    }
                    _ => None,
                };
                found_traits.push(TraitCandidate { def_id: trait_def_id, import_id: import_id });
            }
        }
    }

2927 2928 2929 2930 2931 2932 2933
    /// When name resolution fails, this method can be used to look up candidate
    /// entities with the expected name. It allows filtering them using the
    /// supplied predicate (which should be used to only accept the types of
    /// definitions expected e.g. traits). The lookup spans across all crates.
    ///
    /// NOTE: The method does not look into imports, but this is not a problem,
    /// since we report the definitions (thus, the de-aliased imports).
2934 2935 2936 2937 2938 2939 2940 2941
    fn lookup_import_candidates<FilterFn>(&mut self,
                                          lookup_name: Name,
                                          namespace: Namespace,
                                          filter_fn: FilterFn)
                                          -> Vec<ImportSuggestion>
        where FilterFn: Fn(Def) -> bool
    {
        let mut candidates = Vec::new();
2942
        let mut worklist = Vec::new();
2943
        let mut seen_modules = FxHashSet();
2944 2945 2946 2947 2948
        worklist.push((self.graph_root, Vec::new(), false));

        while let Some((in_module,
                        path_segments,
                        in_module_is_extern)) = worklist.pop() {
2949
            self.populate_module_if_necessary(in_module);
2950

2951
            in_module.for_each_child(|ident, ns, name_binding| {
2952 2953

                // avoid imports entirely
2954
                if name_binding.is_import() && !name_binding.is_extern_crate() { return; }
2955 2956
                // avoid non-importable candidates as well
                if !name_binding.is_importable() { return; }
2957 2958

                // collect results based on the filter function
2959
                if ident.name == lookup_name && ns == namespace {
2960
                    if filter_fn(name_binding.def()) {
2961
                        // create the path
2962
                        let span = name_binding.span;
2963
                        let mut segms = path_segments.clone();
J
Jeffrey Seyfried 已提交
2964
                        segms.push(ident.into());
2965 2966 2967 2968 2969 2970 2971 2972 2973 2974 2975
                        let path = Path {
                            span: span,
                            segments: segms,
                        };
                        // the entity is accessible in the following cases:
                        // 1. if it's defined in the same crate, it's always
                        // accessible (since private entities can be made public)
                        // 2. if it's defined in another crate, it's accessible
                        // only if both the module is public and the entity is
                        // declared as public (due to pruning, we don't explore
                        // outside crate private modules => no need to check this)
2976
                        if !in_module_is_extern || name_binding.vis == ty::Visibility::Public {
2977
                            candidates.push(ImportSuggestion { path: path });
2978 2979 2980 2981 2982
                        }
                    }
                }

                // collect submodules to explore
J
Jeffrey Seyfried 已提交
2983
                if let Some(module) = name_binding.module() {
2984
                    // form the path
2985
                    let mut path_segments = path_segments.clone();
J
Jeffrey Seyfried 已提交
2986
                    path_segments.push(ident.into());
2987

2988
                    if !in_module_is_extern || name_binding.vis == ty::Visibility::Public {
2989
                        // add the module to the lookup
2990
                        let is_extern = in_module_is_extern || name_binding.is_extern_crate();
2991
                        if seen_modules.insert(module.def_id().unwrap()) {
2992 2993
                            worklist.push((module, path_segments, is_extern));
                        }
2994 2995 2996 2997 2998
                    }
                }
            })
        }

2999
        candidates
3000 3001
    }

3002 3003
    fn record_def(&mut self, node_id: NodeId, resolution: PathResolution) {
        debug!("(recording def) recording {:?} for {}", resolution, node_id);
3004
        assert!(resolution.depth == 0 || resolution.base_def != Def::Err);
3005
        if let Some(prev_res) = self.def_map.insert(node_id, resolution) {
3006
            panic!("path resolved multiple times ({:?} before, {:?} now)", prev_res, resolution);
3007
        }
3008 3009
    }

3010
    fn resolve_visibility(&mut self, vis: &ast::Visibility) -> ty::Visibility {
3011 3012 3013
        match *vis {
            ast::Visibility::Public => ty::Visibility::Public,
            ast::Visibility::Crate(..) => ty::Visibility::Restricted(DefId::local(CRATE_DEF_INDEX)),
3014
            ast::Visibility::Inherited => {
3015
                ty::Visibility::Restricted(self.current_module.normal_ancestor_id)
3016
            }
3017 3018 3019 3020 3021 3022 3023 3024 3025 3026 3027 3028 3029 3030
            ast::Visibility::Restricted { ref path, id } => {
                let def = self.smart_resolve_path(id, None, path, PathSource::Visibility).base_def;
                if def == Def::Err {
                    ty::Visibility::Public
                } else {
                    let vis = ty::Visibility::Restricted(def.def_id());
                    if self.is_accessible(vis) {
                        vis
                    } else {
                        self.session.span_err(path.span, "visibilities can only be restricted \
                                                          to ancestor modules");
                        ty::Visibility::Public
                    }
                }
3031 3032 3033 3034
            }
        }
    }

3035
    fn is_accessible(&self, vis: ty::Visibility) -> bool {
3036
        vis.is_accessible_from(self.current_module.normal_ancestor_id, self)
3037 3038
    }

3039
    fn is_accessible_from(&self, vis: ty::Visibility, module: Module<'a>) -> bool {
3040
        vis.is_accessible_from(module.normal_ancestor_id, self)
3041 3042
    }

3043 3044
    fn report_errors(&mut self) {
        self.report_shadowing_errors();
3045
        let mut reported_spans = FxHashSet();
3046

3047
        for &AmbiguityError { span, name, b1, b2, lexical, legacy } in &self.ambiguity_errors {
3048
            if !reported_spans.insert(span) { continue }
3049 3050 3051
            let participle = |binding: &NameBinding| {
                if binding.is_import() { "imported" } else { "defined" }
            };
3052 3053
            let msg1 = format!("`{}` could refer to the name {} here", name, participle(b1));
            let msg2 = format!("`{}` could also refer to the name {} here", name, participle(b2));
3054 3055 3056 3057 3058 3059 3060 3061 3062 3063 3064 3065 3066 3067 3068 3069 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079
            let note = if !lexical && b1.is_glob_import() {
                format!("consider adding an explicit import of `{}` to disambiguate", name)
            } else if let Def::Macro(..) = b1.def() {
                format!("macro-expanded {} do not shadow",
                        if b1.is_import() { "macro imports" } else { "macros" })
            } else {
                format!("macro-expanded {} do not shadow when used in a macro invocation path",
                        if b1.is_import() { "imports" } else { "items" })
            };
            if legacy {
                let id = match b2.kind {
                    NameBindingKind::Import { directive, .. } => directive.id,
                    _ => unreachable!(),
                };
                let mut span = MultiSpan::from_span(span);
                span.push_span_label(b1.span, msg1);
                span.push_span_label(b2.span, msg2);
                let msg = format!("`{}` is ambiguous", name);
                self.session.add_lint(lint::builtin::LEGACY_IMPORTS, id, span, msg);
            } else {
                self.session.struct_span_err(span, &format!("`{}` is ambiguous", name))
                    .span_note(b1.span, &msg1)
                    .span_note(b2.span, &msg2)
                    .note(&note)
                    .emit();
            }
3080 3081
        }

3082 3083 3084 3085
        for &PrivacyError(span, name, binding) in &self.privacy_errors {
            if !reported_spans.insert(span) { continue }
            if binding.is_extern_crate() {
                // Warn when using an inaccessible extern crate.
3086 3087 3088 3089
                let node_id = match binding.kind {
                    NameBindingKind::Import { directive, .. } => directive.id,
                    _ => unreachable!(),
                };
3090 3091 3092
                let msg = format!("extern crate `{}` is private", name);
                self.session.add_lint(lint::builtin::INACCESSIBLE_EXTERN_CRATE, node_id, span, msg);
            } else {
3093
                let def = binding.def();
3094 3095 3096 3097
                self.session.span_err(span, &format!("{} `{}` is private", def.kind_name(), name));
            }
        }
    }
3098

3099
    fn report_shadowing_errors(&mut self) {
J
Jeffrey Seyfried 已提交
3100
        for (name, scope) in replace(&mut self.lexical_macro_resolutions, Vec::new()) {
3101
            self.resolve_legacy_scope(scope, name, true);
J
Jeffrey Seyfried 已提交
3102 3103
        }

3104
        let mut reported_errors = FxHashSet();
3105
        for binding in replace(&mut self.disallowed_shadowing, Vec::new()) {
3106
            if self.resolve_legacy_scope(&binding.parent, binding.name, false).is_some() &&
3107 3108 3109
               reported_errors.insert((binding.name, binding.span)) {
                let msg = format!("`{}` is already in scope", binding.name);
                self.session.struct_span_err(binding.span, &msg)
3110 3111
                    .note("macro-expanded `macro_rules!`s may not shadow \
                           existing macros (see RFC 1560)")
3112 3113 3114 3115 3116
                    .emit();
            }
        }
    }

3117
    fn report_conflict(&mut self,
3118
                       parent: Module,
3119
                       ident: Ident,
3120 3121 3122 3123
                       ns: Namespace,
                       binding: &NameBinding,
                       old_binding: &NameBinding) {
        // Error on the second of two conflicting names
3124
        if old_binding.span.lo > binding.span.lo {
3125
            return self.report_conflict(parent, ident, ns, old_binding, binding);
3126 3127
        }

J
Jeffrey Seyfried 已提交
3128 3129 3130 3131
        let container = match parent.kind {
            ModuleKind::Def(Def::Mod(_), _) => "module",
            ModuleKind::Def(Def::Trait(_), _) => "trait",
            ModuleKind::Block(..) => "block",
3132 3133 3134
            _ => "enum",
        };

3135
        let (participle, noun) = match old_binding.is_import() {
3136 3137 3138 3139
            true => ("imported", "import"),
            false => ("defined", "definition"),
        };

3140
        let (name, span) = (ident.name, binding.span);
3141 3142 3143 3144 3145 3146 3147

        if let Some(s) = self.name_already_seen.get(&name) {
            if s == &span {
                return;
            }
        }

3148 3149 3150
        let msg = {
            let kind = match (ns, old_binding.module()) {
                (ValueNS, _) => "a value",
3151
                (MacroNS, _) => "a macro",
3152
                (TypeNS, _) if old_binding.is_extern_crate() => "an extern crate",
J
Jeffrey Seyfried 已提交
3153 3154
                (TypeNS, Some(module)) if module.is_normal() => "a module",
                (TypeNS, Some(module)) if module.is_trait() => "a trait",
3155 3156 3157 3158 3159 3160 3161
                (TypeNS, _) => "a type",
            };
            format!("{} named `{}` has already been {} in this {}",
                    kind, name, participle, container)
        };

        let mut err = match (old_binding.is_extern_crate(), binding.is_extern_crate()) {
3162 3163 3164 3165
            (true, true) => struct_span_err!(self.session, span, E0259, "{}", msg),
            (true, _) | (_, true) => match binding.is_import() && old_binding.is_import() {
                true => struct_span_err!(self.session, span, E0254, "{}", msg),
                false => struct_span_err!(self.session, span, E0260, "{}", msg),
M
Mohit Agarwal 已提交
3166
            },
3167
            _ => match (old_binding.is_import(), binding.is_import()) {
3168 3169 3170
                (false, false) => struct_span_err!(self.session, span, E0428, "{}", msg),
                (true, true) => struct_span_err!(self.session, span, E0252, "{}", msg),
                _ => struct_span_err!(self.session, span, E0255, "{}", msg),
3171 3172 3173
            },
        };

3174
        err.span_label(span, &format!("`{}` already {}", name, participle));
3175
        if old_binding.span != syntax_pos::DUMMY_SP {
3176
            err.span_label(old_binding.span, &format!("previous {} of `{}` here", noun, name));
3177 3178
        }
        err.emit();
3179
        self.name_already_seen.insert(name, span);
3180
    }
J
Jeffrey Seyfried 已提交
3181 3182 3183 3184 3185 3186

    fn warn_legacy_self_import(&self, directive: &'a ImportDirective<'a>) {
        let (id, span) = (directive.id, directive.span);
        let msg = "`self` no longer imports values".to_string();
        self.session.add_lint(lint::builtin::LEGACY_IMPORTS, id, span, msg);
    }
3187
}
3188

3189 3190 3191 3192 3193 3194 3195 3196 3197 3198 3199 3200 3201 3202 3203 3204
fn is_struct_like(def: Def) -> bool {
    match def {
        Def::VariantCtor(_, CtorKind::Fictive) => true,
        _ => PathSource::Struct.is_expected(def),
    }
}

fn is_self_type(path: &[Ident], namespace: Namespace) -> bool {
    namespace == TypeNS && path.len() == 1 && path[0].name == keywords::SelfType.name()
}

fn is_self_value(path: &[Ident], namespace: Namespace) -> bool {
    namespace == ValueNS && path.len() == 1 && path[0].name == keywords::SelfValue.name()
}

fn names_to_string(idents: &[Ident]) -> String {
3205
    let mut result = String::new();
3206
    for (i, ident) in idents.iter().filter(|i| i.name != keywords::CrateRoot.name()).enumerate() {
3207 3208 3209
        if i > 0 {
            result.push_str("::");
        }
3210
        result.push_str(&ident.name.as_str());
C
corentih 已提交
3211
    }
3212 3213 3214
    result
}

3215 3216
fn path_names_to_string(path: &Path) -> String {
    names_to_string(&path.segments.iter().map(|seg| seg.identifier).collect::<Vec<_>>())
3217 3218
}

3219 3220 3221 3222
/// When an entity with a given name is not available in scope, we search for
/// entities with that name in all crates. This method allows outputting the
/// results of this search in a programmer-friendly way
fn show_candidates(session: &mut DiagnosticBuilder,
3223 3224 3225 3226 3227 3228 3229 3230 3231 3232 3233 3234 3235 3236 3237 3238
                   candidates: &[ImportSuggestion],
                   better: bool) {
    // don't show more than MAX_CANDIDATES results, so
    // we're consistent with the trait suggestions
    const MAX_CANDIDATES: usize = 5;

    // we want consistent results across executions, but candidates are produced
    // by iterating through a hash map, so make sure they are ordered:
    let mut path_strings: Vec<_> =
        candidates.into_iter().map(|c| path_names_to_string(&c.path)).collect();
    path_strings.sort();

    let better = if better { "better " } else { "" };
    let msg_diff = match path_strings.len() {
        1 => " is found in another module, you can import it",
        _ => "s are found in other modules, you can import them",
3239
    };
3240 3241 3242 3243 3244 3245 3246 3247 3248 3249 3250 3251 3252 3253 3254
    session.help(&format!("possible {}candidate{} into scope:", better, msg_diff));

    let count = path_strings.len() as isize - MAX_CANDIDATES as isize + 1;
    for (idx, path_string) in path_strings.iter().enumerate() {
        if idx == MAX_CANDIDATES - 1 && count > 1 {
            session.help(
                &format!("  and {} other candidates", count).to_string(),
            );
            break;
        } else {
            session.help(
                &format!("  `use {};`", path_string).to_string(),
            );
        }
    }
3255 3256
}

3257
/// A somewhat inefficient routine to obtain the name of a module.
3258
fn module_to_string(module: Module) -> String {
3259 3260
    let mut names = Vec::new();

3261
    fn collect_mod(names: &mut Vec<Ident>, module: Module) {
J
Jeffrey Seyfried 已提交
3262 3263
        if let ModuleKind::Def(_, name) = module.kind {
            if let Some(parent) = module.parent {
3264
                names.push(Ident::with_empty_ctxt(name));
J
Jeffrey Seyfried 已提交
3265
                collect_mod(names, parent);
3266
            }
J
Jeffrey Seyfried 已提交
3267 3268
        } else {
            // danger, shouldn't be ident?
3269
            names.push(Ident::from_str("<opaque>"));
J
Jeffrey Seyfried 已提交
3270
            collect_mod(names, module.parent.unwrap());
3271 3272 3273 3274
        }
    }
    collect_mod(&mut names, module);

3275
    if names.is_empty() {
3276 3277
        return "???".to_string();
    }
3278
    names_to_string(&names.into_iter().rev().collect::<Vec<_>>())
3279 3280
}

3281
fn err_path_resolution() -> PathResolution {
3282
    PathResolution::new(Def::Err)
3283 3284
}

N
Niko Matsakis 已提交
3285
#[derive(PartialEq,Copy, Clone)]
3286 3287
pub enum MakeGlobMap {
    Yes,
C
corentih 已提交
3288
    No,
3289 3290
}

3291
__build_diagnostic_array! { librustc_resolve, DIAGNOSTICS }