Skip to content

Commit

Permalink
[WIP] High priority resolutions for associated variants
Browse files Browse the repository at this point in the history
  • Loading branch information
petrochenkov committed Jan 10, 2019
1 parent 797eff9 commit dcaebe0
Show file tree
Hide file tree
Showing 10 changed files with 94 additions and 56 deletions.
7 changes: 7 additions & 0 deletions src/librustc/lint/builtin.rs
Original file line number Diff line number Diff line change
Expand Up @@ -149,6 +149,12 @@ declare_lint! {
"safe access to extern statics was erroneously allowed"
}

declare_lint! {
pub TYPE_VS_VARIANT_AMBIGUITY,
Forbid,
"type vs variant ambiguity"
}

declare_lint! {
pub SAFE_PACKED_BORROWS,
Warn,
Expand Down Expand Up @@ -400,6 +406,7 @@ impl LintPass for HardwiredLints {
CONST_ERR,
RENAMED_AND_REMOVED_LINTS,
SAFE_EXTERN_STATICS,
TYPE_VS_VARIANT_AMBIGUITY,
SAFE_PACKED_BORROWS,
PATTERNS_IN_FNS_WITHOUT_BODY,
LEGACY_DIRECTORY_OWNERSHIP,
Expand Down
23 changes: 17 additions & 6 deletions src/librustc_typeck/astconv.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1288,7 +1288,8 @@ impl<'o, 'gcx: 'tcx, 'tcx> dyn AstConv<'gcx, 'tcx> + 'o {
span: Span,
ty: Ty<'tcx>,
ty_path_def: Def,
item_segment: &hir::PathSegment)
item_segment: &hir::PathSegment,
permit_variants: bool)
-> (Ty<'tcx>, Def)
{
let tcx = self.tcx();
Expand All @@ -1305,11 +1306,21 @@ impl<'o, 'gcx: 'tcx, 'tcx> dyn AstConv<'gcx, 'tcx> + 'o {
tcx.hygienic_eq(assoc_name, vd.ident, adt_def.did)
});
if let Some(variant_def) = variant_def {
check_type_alias_enum_variants_enabled(tcx, span);
if permit_variants {
check_type_alias_enum_variants_enabled(tcx, span);

let def = Def::Variant(variant_def.did);
tcx.check_stability(def.def_id(), Some(ref_id), span);
return (ty, def);
let def = Def::Variant(variant_def.did);
tcx.check_stability(def.def_id(), Some(ref_id), span);
return (ty, def);
} else {
use rustc::lint::builtin::TYPE_VS_VARIANT_AMBIGUITY;
tcx.lint_node(
TYPE_VS_VARIANT_AMBIGUITY,
ref_id,
span,
"type vs variant ambiguity",
);
}
}
},
_ => (),
Expand Down Expand Up @@ -1773,7 +1784,7 @@ impl<'o, 'gcx: 'tcx, 'tcx> dyn AstConv<'gcx, 'tcx> + 'o {
} else {
Def::Err
};
self.associated_path_def_to_ty(ast_ty.id, ast_ty.span, ty, def, segment).0
self.associated_path_def_to_ty(ast_ty.id, ast_ty.span, ty, def, segment, false).0
}
hir::TyKind::Array(ref ty, ref length) => {
let length_def_id = tcx.hir().local_def_id(length.id);
Expand Down
2 changes: 1 addition & 1 deletion src/librustc_typeck/check/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4623,7 +4623,7 @@ impl<'a, 'gcx, 'tcx> FnCtxt<'a, 'gcx, 'tcx> {
Def::Err
};
let (ty, def) = AstConv::associated_path_def_to_ty(self, node_id, path_span,
ty, def, segment);
ty, def, segment, true);

// Write back the new resolution.
let hir_id = self.tcx.hir().node_to_hir_id(node_id);
Expand Down
30 changes: 15 additions & 15 deletions src/librustc_typeck/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -111,7 +111,7 @@ use rustc::infer::InferOk;
use rustc::lint;
use rustc::middle;
use rustc::session;
use rustc::session::config::nightly_options;
// use rustc::session::config::nightly_options;
use rustc::traits::{ObligationCause, ObligationCauseCode, TraitEngine, TraitEngineExt};
use rustc::ty::subst::Substs;
use rustc::ty::{self, Ty, TyCtxt};
Expand All @@ -130,20 +130,20 @@ pub struct TypeAndSubsts<'tcx> {
ty: Ty<'tcx>,
}

fn check_type_alias_enum_variants_enabled<'a, 'gcx, 'tcx>(tcx: TyCtxt<'a, 'gcx, 'tcx>,
span: Span) {
if !tcx.features().type_alias_enum_variants {
let mut err = tcx.sess.struct_span_err(
span,
"enum variants on type aliases are experimental"
);
if nightly_options::is_nightly_build() {
help!(&mut err,
"add `#![feature(type_alias_enum_variants)]` to the \
crate attributes to enable");
}
err.emit();
}
fn check_type_alias_enum_variants_enabled<'a, 'gcx, 'tcx>(_tcx: TyCtxt<'a, 'gcx, 'tcx>,
_span: Span) {
// if !tcx.features().type_alias_enum_variants {
// let mut err = tcx.sess.struct_span_err(
// span,
// "enum variants on type aliases are experimental"
// );
// if nightly_options::is_nightly_build() {
// help!(&mut err,
// "add `#![feature(type_alias_enum_variants)]` to the \
// crate attributes to enable");
// }
// err.emit();
// }
}

fn require_c_abi_if_variadic(tcx: TyCtxt,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@
// option. This file may not be copied, modified, or distributed
// except according to those terms.

// compile-pass

enum Foo {
Bar(i32),
Baz { i: i32 },
Expand Down

This file was deleted.

13 changes: 13 additions & 0 deletions src/test/ui/type-alias-enum-variants-priority-2.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
#![feature(type_alias_enum_variants)]

enum E {
V(u8)
}

impl E {
fn V() {}
}

fn main() {
<E>::V(); //~ ERROR this function takes 1 parameter but 0 parameters were supplied
}
12 changes: 12 additions & 0 deletions src/test/ui/type-alias-enum-variants-priority-2.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
error[E0061]: this function takes 1 parameter but 0 parameters were supplied
--> $DIR/type-alias-enum-variants-priority-2.rs:12:5
|
LL | V(u8)
| ----- defined here
...
LL | <E>::V(); //~ ERROR this function takes 1 parameter but 0 parameters were supplied
| ^^^^^^^^ expected 1 parameter

error: aborting due to previous error

For more information about this error, try `rustc --explain E0061`.
17 changes: 17 additions & 0 deletions src/test/ui/type-alias-enum-variants-priority.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
#![feature(type_alias_enum_variants)]

enum E {
V(u8)
}

trait Tr {
type V;
fn f() -> Self::V;
}

impl Tr for E {
type V = u8;
fn f() -> Self::V { loop {} } //~ ERROR type vs variant ambiguity
}

fn main() {}
10 changes: 10 additions & 0 deletions src/test/ui/type-alias-enum-variants-priority.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
error: type vs variant ambiguity
--> $DIR/type-alias-enum-variants-priority.rs:14:15
|
LL | fn f() -> Self::V { loop {} } //~ ERROR type vs variant ambiguity
| ^^^^^^^
|
= note: #[forbid(type_vs_variant_ambiguity)] on by default

error: aborting due to previous error

0 comments on commit dcaebe0

Please sign in to comment.