c2cab1fd58
Add AST validation pass and move some checks to it The purpose of this pass is to catch constructions that fit into AST data structures, but not permitted by the language. As an example, `impl`s don't have visibilities, but for convenience and uniformity with other items they are represented with a structure `Item` which has `Visibility` field. This pass is intended to run after expansion of macros and syntax extensions (and before lowering to HIR), so it can catch erroneous constructions that were generated by them. This pass allows to remove ad hoc semantic checks from the parser, which can be overruled by syntax extensions and occasionally macros. The checks can be put here if they are simple, local, don't require results of any complex analysis like name resolution or type checking and maybe don't logically fall into other passes. I expect most of errors generated by this pass to be non-fatal and allowing the compilation to proceed. I intend to move some more checks to this pass later and maybe extend it with new checks, like, for example, identifier validity. Given that syntax extensions are going to be stabilized in the measurable future, it's important that they would not be able to subvert usual language rules. In this patch I've added two new checks - a check for labels named `'static` and a check for lifetimes and labels named `'_`. The first one gives a hard error, the second one - a future compatibility warning. Fixes https://github.com/rust-lang/rust/issues/33059 ([breaking-change]) cc https://github.com/rust-lang/rfcs/pull/1177 r? @nrc
207 lines
4.2 KiB
Rust
207 lines
4.2 KiB
Rust
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
|
// 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.
|
|
|
|
#![allow(non_snake_case)]
|
|
|
|
register_long_diagnostics! {
|
|
|
|
E0445: r##"
|
|
A private trait was used on a public type parameter bound. Erroneous code
|
|
examples:
|
|
|
|
```compile_fail
|
|
#![deny(private_in_public)]
|
|
|
|
trait Foo {
|
|
fn dummy(&self) { }
|
|
}
|
|
|
|
pub trait Bar : Foo {} // error: private trait in public interface
|
|
pub struct Bar2<T: Foo>(pub T); // same error
|
|
pub fn foo<T: Foo> (t: T) {} // same error
|
|
```
|
|
|
|
To solve this error, please ensure that the trait is also public. The trait
|
|
can be made inaccessible if necessary by placing it into a private inner
|
|
module, but it still has to be marked with `pub`. Example:
|
|
|
|
```ignore
|
|
pub trait Foo { // we set the Foo trait public
|
|
fn dummy(&self) { }
|
|
}
|
|
|
|
pub trait Bar : Foo {} // ok!
|
|
pub struct Bar2<T: Foo>(pub T); // ok!
|
|
pub fn foo<T: Foo> (t: T) {} // ok!
|
|
```
|
|
"##,
|
|
|
|
E0446: r##"
|
|
A private type was used in a public type signature. Erroneous code example:
|
|
|
|
```compile_fail
|
|
#![deny(private_in_public)]
|
|
|
|
mod Foo {
|
|
struct Bar(u32);
|
|
|
|
pub fn bar() -> Bar { // error: private type in public interface
|
|
Bar(0)
|
|
}
|
|
}
|
|
```
|
|
|
|
To solve this error, please ensure that the type is also public. The type
|
|
can be made inaccessible if necessary by placing it into a private inner
|
|
module, but it still has to be marked with `pub`.
|
|
Example:
|
|
|
|
```
|
|
mod Foo {
|
|
pub struct Bar(u32); // we set the Bar type public
|
|
|
|
pub fn bar() -> Bar { // ok!
|
|
Bar(0)
|
|
}
|
|
}
|
|
```
|
|
"##,
|
|
|
|
E0447: r##"
|
|
The `pub` keyword was used inside a function. Erroneous code example:
|
|
|
|
```ignore
|
|
fn foo() {
|
|
pub struct Bar; // error: visibility has no effect inside functions
|
|
}
|
|
```
|
|
|
|
Since we cannot access items defined inside a function, the visibility of its
|
|
items does not impact outer code. So using the `pub` keyword in this context
|
|
is invalid.
|
|
"##,
|
|
|
|
E0448: r##"
|
|
The `pub` keyword was used inside a public enum. Erroneous code example:
|
|
|
|
```compile_fail
|
|
pub enum Foo {
|
|
pub Bar, // error: unnecessary `pub` visibility
|
|
}
|
|
```
|
|
|
|
Since the enum is already public, adding `pub` on one its elements is
|
|
unnecessary. Example:
|
|
|
|
```compile_fail
|
|
enum Foo {
|
|
pub Bar, // not ok!
|
|
}
|
|
```
|
|
|
|
This is the correct syntax:
|
|
|
|
```ignore
|
|
pub enum Foo {
|
|
Bar, // ok!
|
|
}
|
|
```
|
|
"##,
|
|
|
|
E0450: r##"
|
|
A tuple constructor was invoked while some of its fields are private. Erroneous
|
|
code example:
|
|
|
|
```compile_fail
|
|
mod Bar {
|
|
pub struct Foo(isize);
|
|
}
|
|
|
|
let f = Bar::Foo(0); // error: cannot invoke tuple struct constructor with
|
|
// private fields
|
|
```
|
|
|
|
To solve this issue, please ensure that all of the fields of the tuple struct
|
|
are public. Alternatively, provide a `new()` method to the tuple struct to
|
|
construct it from a given inner value. Example:
|
|
|
|
```
|
|
mod Bar {
|
|
pub struct Foo(pub isize); // we set its field to public
|
|
}
|
|
|
|
let f = Bar::Foo(0); // ok!
|
|
|
|
// or:
|
|
mod bar {
|
|
pub struct Foo(isize);
|
|
|
|
impl Foo {
|
|
pub fn new(x: isize) -> Foo {
|
|
Foo(x)
|
|
}
|
|
}
|
|
}
|
|
|
|
let f = bar::Foo::new(1);
|
|
```
|
|
"##,
|
|
|
|
E0451: r##"
|
|
A struct constructor with private fields was invoked. Erroneous code example:
|
|
|
|
```compile_fail
|
|
mod Bar {
|
|
pub struct Foo {
|
|
pub a: isize,
|
|
b: isize,
|
|
}
|
|
}
|
|
|
|
let f = Bar::Foo{ a: 0, b: 0 }; // error: field `b` of struct `Bar::Foo`
|
|
// is private
|
|
```
|
|
|
|
To fix this error, please ensure that all the fields of the struct are public,
|
|
or implement a function for easy instantiation. Examples:
|
|
|
|
```
|
|
mod Bar {
|
|
pub struct Foo {
|
|
pub a: isize,
|
|
pub b: isize, // we set `b` field public
|
|
}
|
|
}
|
|
|
|
let f = Bar::Foo{ a: 0, b: 0 }; // ok!
|
|
```
|
|
|
|
Or:
|
|
|
|
```
|
|
mod Bar {
|
|
pub struct Foo {
|
|
pub a: isize,
|
|
b: isize, // still private
|
|
}
|
|
|
|
impl Foo {
|
|
pub fn new() -> Foo { // we create a method to instantiate `Foo`
|
|
Foo { a: 0, b: 0 }
|
|
}
|
|
}
|
|
}
|
|
|
|
let f = Bar::Foo::new(); // ok!
|
|
```
|
|
"##,
|
|
|
|
}
|