Skip to content

Commit

Permalink
Auto merge of rust-lang#43540 - petrochenkov:pathrelax, r=nikomatsakis
Browse files Browse the repository at this point in the history
syntax: Relax path grammar

TLDR: Accept the disambiguator `::` in "type" paths (`Type::<Args>`), accept the disambiguator `::` before parenthesized generic arguments (`Fn::(Args)`).

The "turbofish" disambiguator `::<>` in expression paths is a necessary evil required for path parsing to be both simple and to give reasonable results.
Since paths in expressions usually refer to values (but not necessarily, e.g. `Struct::<u8> { field: 0 }` is disambiguated, but refers to a type), people often consider `::<>` to be inherent to *values*, and not *expressions* and want to write disambiguated paths for values even in contexts where disambiguation is not strictly necessary, for example when a path is passed to a macro `m!(Vec::<i32>::new)`.
The problem is that currently, if the disambiguator is not *required*, then it's *prohibited*. This results in confusion - see rust-lang#41740, https://internals.rust-lang.org/t/macro-path-uses-novel-syntax/5561.

This PR makes the disambiguator *optional* instead of prohibited in contexts where it's not strictly required, so people can pass paths to macros in whatever form they consider natural (e.g. disambiguated form for value paths).
This PR also accepts the disambiguator in paths with parenthesized arguments (`Fn::(Args)`) for consistency and to simplify testing of stuff like rust-lang#41856 (comment).

Closes rust-lang#41740

cc @rust-lang/lang
r? @nikomatsakis
  • Loading branch information
bors committed Aug 21, 2017
2 parents 4fdb4be + 804459b commit 8df670b
Show file tree
Hide file tree
Showing 6 changed files with 49 additions and 44 deletions.
4 changes: 1 addition & 3 deletions src/libsyntax/ext/tt/macro_parser.rs
Original file line number Diff line number Diff line change
Expand Up @@ -599,9 +599,7 @@ fn parse_nt<'a>(p: &mut Parser<'a>, sp: Span, name: &str) -> Nonterminal {
panic!(FatalError)
}
},
"path" => {
token::NtPath(panictry!(p.parse_path(PathStyle::Type)))
},
"path" => token::NtPath(panictry!(p.parse_path_common(PathStyle::Type, false))),
"meta" => token::NtMeta(panictry!(p.parse_meta_item())),
"vis" => token::NtVis(panictry!(p.parse_visibility(true))),
// this is not supposed to happen, since it has been checked
Expand Down
39 changes: 18 additions & 21 deletions src/libsyntax/parse/parser.rs
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ pub enum PathStyle {
Expr,
/// In other contexts, notably in types, no ambiguity exists and paths can be written
/// without the disambiguator, e.g. `x<y>` - unambiguously a path.
/// Paths with disambiguators are rejected for now, but may be allowed in the future.
/// Paths with disambiguators are still accepted, `x::<Y>` - unambiguously a path too.
Type,
/// A path with generic arguments disallowed, e.g. `foo::bar::Baz`, used in imports,
/// visibilities or attributes.
Expand Down Expand Up @@ -1755,7 +1755,7 @@ impl<'a> Parser<'a> {
self.expect(&token::ModSep)?;

let qself = QSelf { ty, position: path.segments.len() };
self.parse_path_segments(&mut path.segments, style)?;
self.parse_path_segments(&mut path.segments, style, true)?;

Ok((qself, ast::Path { segments: path.segments, span: lo.to(self.prev_span) }))
}
Expand All @@ -1770,16 +1770,20 @@ impl<'a> Parser<'a> {
/// `a::b::C::<D>` (with disambiguator)
/// `Fn(Args)` (without disambiguator)
/// `Fn::(Args)` (with disambiguator)
pub fn parse_path(&mut self, style: PathStyle) -> PResult<'a, ast::Path>
{
pub fn parse_path(&mut self, style: PathStyle) -> PResult<'a, ast::Path> {
self.parse_path_common(style, true)
}

pub fn parse_path_common(&mut self, style: PathStyle, enable_warning: bool)
-> PResult<'a, ast::Path> {
maybe_whole!(self, NtPath, |x| x);

let lo = self.meta_var_span.unwrap_or(self.span);
let mut segments = Vec::new();
if self.eat(&token::ModSep) {
segments.push(PathSegment::crate_root(lo));
}
self.parse_path_segments(&mut segments, style)?;
self.parse_path_segments(&mut segments, style, enable_warning)?;

Ok(ast::Path { segments, span: lo.to(self.prev_span) })
}
Expand All @@ -1804,18 +1808,19 @@ impl<'a> Parser<'a> {
self.parse_path(style)
}

fn parse_path_segments(&mut self, segments: &mut Vec<PathSegment>, style: PathStyle)
-> PResult<'a, ()> {
fn parse_path_segments(&mut self, segments: &mut Vec<PathSegment>, style: PathStyle,
enable_warning: bool) -> PResult<'a, ()> {
loop {
segments.push(self.parse_path_segment(style)?);
segments.push(self.parse_path_segment(style, enable_warning)?);

if self.is_import_coupler() || !self.eat(&token::ModSep) {
return Ok(());
}
}
}

fn parse_path_segment(&mut self, style: PathStyle) -> PResult<'a, PathSegment> {
fn parse_path_segment(&mut self, style: PathStyle, enable_warning: bool)
-> PResult<'a, PathSegment> {
let ident_span = self.span;
let ident = self.parse_path_segment_ident()?;

Expand All @@ -1835,17 +1840,9 @@ impl<'a> Parser<'a> {
&& self.look_ahead(1, |t| is_args_start(t)) {
// Generic arguments are found - `<`, `(`, `::<` or `::(`.
let lo = self.span;
if self.eat(&token::ModSep) {
// These errors are not strictly necessary and may be removed in the future.
if style == PathStyle::Type {
let mut err = self.diagnostic().struct_span_err(self.prev_span,
"unnecessary path disambiguator");
err.span_label(self.prev_span, "try removing `::`");
err.emit();
} else if self.token == token::OpenDelim(token::Paren) {
self.diagnostic().span_err(self.prev_span,
"`::` is not supported before parenthesized generic arguments")
}
if self.eat(&token::ModSep) && style == PathStyle::Type && enable_warning {
self.diagnostic().struct_span_warn(self.prev_span, "unnecessary path disambiguator")
.span_label(self.prev_span, "try removing `::`").emit();
}

let parameters = if self.eat_lt() {
Expand Down Expand Up @@ -2382,7 +2379,7 @@ impl<'a> Parser<'a> {

// Assuming we have just parsed `.`, continue parsing into an expression.
fn parse_dot_suffix(&mut self, self_arg: P<Expr>, lo: Span) -> PResult<'a, P<Expr>> {
let segment = self.parse_path_segment(PathStyle::Expr)?;
let segment = self.parse_path_segment(PathStyle::Expr, true)?;
Ok(match self.token {
token::OpenDelim(token::Paren) => {
// Method call `expr.f()`
Expand Down
8 changes: 2 additions & 6 deletions src/test/compile-fail/issue-32995.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,15 +19,11 @@ fn main() {
//~^ ERROR parenthesized parameters may only be used with a trait
//~| WARN previously accepted

macro_rules! pathexpr {
($p:path) => { $p }
}

let p = pathexpr!(::std::str()::from_utf8)(b"foo").unwrap();
let p = ::std::str::()::from_utf8(b"foo").unwrap();
//~^ ERROR parenthesized parameters may only be used with a trait
//~| WARN previously accepted

let p = pathexpr!(::std::str::from_utf8())(b"foo").unwrap();
let p = ::std::str::from_utf8::()(b"foo").unwrap();
//~^ ERROR parenthesized parameters may only be used with a trait
//~| WARN previously accepted

Expand Down
28 changes: 21 additions & 7 deletions src/test/compile-fail/issue-36116.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,16 +8,30 @@
// option. This file may not be copied, modified, or distributed
// except according to those terms.

// Unnecessary path disambiguator is ok

#![feature(rustc_attrs)]
#![allow(unused)]

macro_rules! m {
($p: path) => {
let _ = $p(0);
let _: $p;
}
}

struct Foo<T> {
_a: T,
}

fn main() {
let f = Some(Foo { _a: 42 }).map(|a| a as Foo::<i32>);
//~^ ERROR unnecessary path disambiguator
//~| NOTE try removing `::`
struct S<T>(T);

fn f() {
let f = Some(Foo { _a: 42 }).map(|a| a as Foo::<i32>); //~ WARN unnecessary path disambiguator
let g: Foo::<i32> = Foo { _a: 42 }; //~ WARN unnecessary path disambiguator

let g: Foo::<i32> = Foo { _a: 42 };
//~^ ERROR unnecessary path disambiguator
//~| NOTE try removing `::`
m!(S::<u8>); // OK, no warning
}

#[rustc_error]
fn main() {} //~ ERROR compilation successful
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,7 @@
// option. This file may not be copied, modified, or distributed
// except according to those terms.

// compile-flags: -Z parse-only

// Test that parentheses form doesn't work in expression paths.
// Test that parentheses form parses in expression paths.

struct Bar<A,R> {
f: A, r: R
Expand All @@ -21,10 +19,10 @@ impl<A,B> Bar<A,B> {
}

fn bar() {
let b = Box::Bar::<isize,usize>::new(); // OK
let b = Bar::<isize, usize>::new(); // OK

let b = Box::Bar::()::new();
//~^ ERROR `::` is not supported before parenthesized generic arguments
let b = Bar::(isize, usize)::new(); // OK too (for the parser)
//~^ ERROR parenthesized parameters may only be used with a trait
}

fn main() { }
fn main() {}
2 changes: 2 additions & 0 deletions src/test/parse-fail/type-parameters-in-field-exprs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,6 @@ fn main() {
//~^ ERROR field expressions may not have generic arguments
f.x::<>;
//~^ ERROR field expressions may not have generic arguments
f.x::();
//~^ ERROR field expressions may not have generic arguments
}

0 comments on commit 8df670b

Please sign in to comment.