Skip to content

Commit 0f9ba99

Browse files
committed
Auto merge of #33161 - jseyfried:parse_tuple_struct_field_vis, r=nikomatsakis
Parse `pub(restricted)` visibilities on tuple struct fields Parse `pub(restricted)` on tuple struct fields (cc #32409). r? @nikomatsakis
2 parents 009a649 + 78a8127 commit 0f9ba99

File tree

5 files changed

+108
-7
lines changed

5 files changed

+108
-7
lines changed

src/libsyntax/parse/parser.rs

+38-7
Original file line numberDiff line numberDiff line change
@@ -5218,8 +5218,25 @@ impl<'a> Parser<'a> {
52185218
|p| {
52195219
let attrs = p.parse_outer_attributes()?;
52205220
let lo = p.span.lo;
5221-
let vis = p.parse_visibility(false)?;
5222-
let ty = p.parse_ty_sum()?;
5221+
let mut vis = p.parse_visibility(false)?;
5222+
let ty_is_interpolated =
5223+
p.token.is_interpolated() || p.look_ahead(1, |t| t.is_interpolated());
5224+
let mut ty = p.parse_ty_sum()?;
5225+
5226+
// Handle `pub(path) type`, in which `vis` will be `pub` and `ty` will be `(path)`.
5227+
if vis == Visibility::Public && !ty_is_interpolated &&
5228+
p.token != token::Comma && p.token != token::CloseDelim(token::Paren) {
5229+
ty = if let TyKind::Paren(ref path_ty) = ty.node {
5230+
if let TyKind::Path(None, ref path) = path_ty.node {
5231+
vis = Visibility::Restricted { path: P(path.clone()), id: path_ty.id };
5232+
Some(p.parse_ty_sum()?)
5233+
} else {
5234+
None
5235+
}
5236+
} else {
5237+
None
5238+
}.unwrap_or(ty);
5239+
}
52235240
Ok(StructField {
52245241
span: mk_sp(lo, p.span.hi),
52255242
vis: vis,
@@ -5263,15 +5280,29 @@ impl<'a> Parser<'a> {
52635280
self.parse_single_struct_field(vis, attrs)
52645281
}
52655282

5266-
fn parse_visibility(&mut self, allow_restricted: bool) -> PResult<'a, Visibility> {
5283+
// If `allow_path` is false, just parse the `pub` in `pub(path)` (but still parse `pub(crate)`)
5284+
fn parse_visibility(&mut self, allow_path: bool) -> PResult<'a, Visibility> {
5285+
let pub_crate = |this: &mut Self| {
5286+
let span = this.last_span;
5287+
this.expect(&token::CloseDelim(token::Paren))?;
5288+
Ok(Visibility::Crate(span))
5289+
};
5290+
52675291
if !self.eat_keyword(keywords::Pub) {
52685292
Ok(Visibility::Inherited)
5269-
} else if !allow_restricted || !self.eat(&token::OpenDelim(token::Paren)) {
5293+
} else if !allow_path {
5294+
// Look ahead to avoid eating the `(` in `pub(path)` while still parsing `pub(crate)`
5295+
if self.token == token::OpenDelim(token::Paren) &&
5296+
self.look_ahead(1, |t| t.is_keyword(keywords::Crate)) {
5297+
self.bump(); self.bump();
5298+
pub_crate(self)
5299+
} else {
5300+
Ok(Visibility::Public)
5301+
}
5302+
} else if !self.eat(&token::OpenDelim(token::Paren)) {
52705303
Ok(Visibility::Public)
52715304
} else if self.eat_keyword(keywords::Crate) {
5272-
let span = self.last_span;
5273-
self.expect(&token::CloseDelim(token::Paren))?;
5274-
Ok(Visibility::Crate(span))
5305+
pub_crate(self)
52755306
} else {
52765307
let path = self.parse_path(PathStyle::Mod)?;
52775308
self.expect(&token::CloseDelim(token::Paren))?;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(pub_restricted, type_macros)]
12+
13+
mod foo {
14+
type T = ();
15+
struct S1(pub(foo) (), pub(T), pub(crate) (), pub(((), T)));
16+
struct S2(pub((foo)) ()); //~ ERROR expected one of `+` or `,`, found `(`
17+
//~| ERROR expected one of `+`, `;`, or `where`, found `(`
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(pub_restricted, type_macros)]
12+
13+
macro_rules! define_struct {
14+
($t:ty) => {
15+
struct S1(pub $t);
16+
struct S2(pub (foo) ());
17+
struct S3(pub $t ()); //~ ERROR expected one of `+` or `,`, found `(`
18+
//~| ERROR expected one of `+`, `;`, or `where`, found `(`
19+
}
20+
}
21+
22+
mod foo {
23+
define_struct! { (foo) }
24+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(pub_restricted, type_macros)]
12+
13+
macro_rules! define_struct {
14+
($t:ty) => {
15+
struct S1(pub($t));
16+
struct S2(pub (foo) ());
17+
struct S3(pub($t) ()); //~ ERROR expected one of `+` or `,`, found `(`
18+
//~| ERROR expected one of `+`, `;`, or `where`, found `(`
19+
}
20+
}
21+
22+
mod foo {
23+
define_struct! { foo }
24+
}

src/test/compile-fail/privacy/restricted/ty-params.rs

+4
Original file line numberDiff line numberDiff line change
@@ -17,4 +17,8 @@ macro_rules! m {
1717
struct S<T>(T);
1818
m!{ S<u8> } //~ ERROR type or lifetime parameters in visibility path
1919

20+
mod foo {
21+
struct S(pub(foo<T>) ()); //~ ERROR type or lifetime parameters in visibility path
22+
}
23+
2024
fn main() {}

0 commit comments

Comments
 (0)