-
Notifications
You must be signed in to change notification settings - Fork 12.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix ICE on recursively defined enum variant discriminant. #26515
Merged
Merged
Changes from 1 commit
Commits
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -13,68 +13,86 @@ | |
|
||
use ast_map; | ||
use session::Session; | ||
use middle::def::{DefStatic, DefConst, DefAssociatedConst, DefMap}; | ||
use middle::def::{DefStatic, DefConst, DefAssociatedConst, DefVariant, DefMap}; | ||
use util::nodemap::NodeMap; | ||
|
||
use syntax::{ast, ast_util}; | ||
use syntax::codemap::Span; | ||
use syntax::visit::Visitor; | ||
use syntax::visit; | ||
|
||
use std::cell::RefCell; | ||
|
||
struct CheckCrateVisitor<'a, 'ast: 'a> { | ||
sess: &'a Session, | ||
def_map: &'a DefMap, | ||
ast_map: &'a ast_map::Map<'ast> | ||
ast_map: &'a ast_map::Map<'ast>, | ||
discriminant_map: RefCell<NodeMap<Option<&'ast ast::Expr>>>, | ||
} | ||
|
||
impl<'v, 'a, 'ast> Visitor<'v> for CheckCrateVisitor<'a, 'ast> { | ||
fn visit_item(&mut self, it: &ast::Item) { | ||
impl<'a, 'ast: 'a> Visitor<'ast> for CheckCrateVisitor<'a, 'ast> { | ||
fn visit_item(&mut self, it: &'ast ast::Item) { | ||
match it.node { | ||
ast::ItemStatic(_, _, ref expr) | | ||
ast::ItemConst(_, ref expr) => { | ||
ast::ItemStatic(..) | | ||
ast::ItemConst(..) => { | ||
let mut recursion_visitor = | ||
CheckItemRecursionVisitor::new(self, &it.span); | ||
recursion_visitor.visit_item(it); | ||
visit::walk_expr(self, &*expr) | ||
}, | ||
_ => visit::walk_item(self, it) | ||
ast::ItemEnum(ref enum_def, ref generics) => { | ||
// We could process the whole enum, but handling the variants | ||
// with discriminant expressions one by one gives more specific, | ||
// less redundant output. | ||
for variant in &enum_def.variants { | ||
if let Some(_) = variant.node.disr_expr { | ||
let mut recursion_visitor = | ||
CheckItemRecursionVisitor::new(self, &variant.span); | ||
recursion_visitor.populate_enum_discriminants(enum_def); | ||
recursion_visitor.visit_variant(variant, generics); | ||
} | ||
} | ||
} | ||
_ => {} | ||
} | ||
visit::walk_item(self, it) | ||
} | ||
|
||
fn visit_trait_item(&mut self, ti: &ast::TraitItem) { | ||
fn visit_trait_item(&mut self, ti: &'ast ast::TraitItem) { | ||
match ti.node { | ||
ast::ConstTraitItem(_, ref default) => { | ||
if let Some(ref expr) = *default { | ||
if let Some(_) = *default { | ||
let mut recursion_visitor = | ||
CheckItemRecursionVisitor::new(self, &ti.span); | ||
recursion_visitor.visit_trait_item(ti); | ||
visit::walk_expr(self, &*expr) | ||
} | ||
} | ||
_ => visit::walk_trait_item(self, ti) | ||
_ => {} | ||
} | ||
visit::walk_trait_item(self, ti) | ||
} | ||
|
||
fn visit_impl_item(&mut self, ii: &ast::ImplItem) { | ||
fn visit_impl_item(&mut self, ii: &'ast ast::ImplItem) { | ||
match ii.node { | ||
ast::ConstImplItem(_, ref expr) => { | ||
ast::ConstImplItem(..) => { | ||
let mut recursion_visitor = | ||
CheckItemRecursionVisitor::new(self, &ii.span); | ||
recursion_visitor.visit_impl_item(ii); | ||
visit::walk_expr(self, &*expr) | ||
} | ||
_ => visit::walk_impl_item(self, ii) | ||
_ => {} | ||
} | ||
visit::walk_impl_item(self, ii) | ||
} | ||
} | ||
|
||
pub fn check_crate<'ast>(sess: &Session, | ||
krate: &ast::Crate, | ||
krate: &'ast ast::Crate, | ||
def_map: &DefMap, | ||
ast_map: &ast_map::Map<'ast>) { | ||
let mut visitor = CheckCrateVisitor { | ||
sess: sess, | ||
def_map: def_map, | ||
ast_map: ast_map | ||
ast_map: ast_map, | ||
discriminant_map: RefCell::new(NodeMap()), | ||
}; | ||
visit::walk_crate(&mut visitor, krate); | ||
sess.abort_if_errors(); | ||
|
@@ -85,53 +103,120 @@ struct CheckItemRecursionVisitor<'a, 'ast: 'a> { | |
sess: &'a Session, | ||
ast_map: &'a ast_map::Map<'ast>, | ||
def_map: &'a DefMap, | ||
idstack: Vec<ast::NodeId> | ||
discriminant_map: &'a RefCell<NodeMap<Option<&'ast ast::Expr>>>, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. comment please |
||
idstack: Vec<ast::NodeId>, | ||
} | ||
|
||
impl<'a, 'ast: 'a> CheckItemRecursionVisitor<'a, 'ast> { | ||
fn new(v: &CheckCrateVisitor<'a, 'ast>, span: &'a Span) | ||
fn new(v: &'a CheckCrateVisitor<'a, 'ast>, span: &'a Span) | ||
-> CheckItemRecursionVisitor<'a, 'ast> { | ||
CheckItemRecursionVisitor { | ||
root_span: span, | ||
sess: v.sess, | ||
ast_map: v.ast_map, | ||
def_map: v.def_map, | ||
idstack: Vec::new() | ||
discriminant_map: &v.discriminant_map, | ||
idstack: Vec::new(), | ||
} | ||
} | ||
fn with_item_id_pushed<F>(&mut self, id: ast::NodeId, f: F) | ||
where F: Fn(&mut Self) { | ||
if self.idstack.iter().any(|x| x == &(id)) { | ||
if self.idstack.iter().any(|x| *x == id) { | ||
span_err!(self.sess, *self.root_span, E0265, "recursive constant"); | ||
return; | ||
} | ||
self.idstack.push(id); | ||
f(self); | ||
self.idstack.pop(); | ||
} | ||
// If a variant has an expression specifying its discriminant, then it needs | ||
// to be checked just like a static or constant. However, if there are more | ||
// variants with no explicitly specified discriminant, those variants will | ||
// increment the same expression to get their values. | ||
// | ||
// So for every variant, we need to track whether there is an expression | ||
// somewhere in the enum definition that controls its discriminant. We do | ||
// this by starting from the end and searching backward. | ||
fn populate_enum_discriminants(&self, enum_definition: &'ast ast::EnumDef) { | ||
// Get the map, and return if we already processed this enum or if it | ||
// has no variants. | ||
let mut discriminant_map = self.discriminant_map.borrow_mut(); | ||
match enum_definition.variants.first() { | ||
None => { return; } | ||
Some(variant) if discriminant_map.contains_key(&variant.node.id) => { | ||
return; | ||
} | ||
_ => {} | ||
} | ||
|
||
// Go through all the variants. | ||
let mut variant_stack: Vec<ast::NodeId> = Vec::new(); | ||
for variant in enum_definition.variants.iter().rev() { | ||
variant_stack.push(variant.node.id); | ||
// When we find an expression, every variant currently on the stack | ||
// is affected by that expression. | ||
if let Some(ref expr) = variant.node.disr_expr { | ||
for id in &variant_stack { | ||
discriminant_map.insert(*id, Some(expr)); | ||
} | ||
variant_stack.clear() | ||
} | ||
} | ||
// If we are at the top, that always starts at 0, so any variant on the | ||
// stack has a default value and does not need to be checked. | ||
for id in &variant_stack { | ||
discriminant_map.insert(*id, None); | ||
} | ||
} | ||
} | ||
|
||
impl<'a, 'ast, 'v> Visitor<'v> for CheckItemRecursionVisitor<'a, 'ast> { | ||
fn visit_item(&mut self, it: &ast::Item) { | ||
impl<'a, 'ast: 'a> Visitor<'ast> for CheckItemRecursionVisitor<'a, 'ast> { | ||
fn visit_item(&mut self, it: &'ast ast::Item) { | ||
self.with_item_id_pushed(it.id, |v| visit::walk_item(v, it)); | ||
} | ||
|
||
fn visit_trait_item(&mut self, ti: &ast::TraitItem) { | ||
fn visit_enum_def(&mut self, enum_definition: &'ast ast::EnumDef, | ||
generics: &'ast ast::Generics) { | ||
self.populate_enum_discriminants(enum_definition); | ||
visit::walk_enum_def(self, enum_definition, generics); | ||
} | ||
|
||
fn visit_variant(&mut self, variant: &'ast ast::Variant, | ||
_: &'ast ast::Generics) { | ||
let variant_id = variant.node.id; | ||
let maybe_expr; | ||
if let Some(get_expr) = self.discriminant_map.borrow().get(&variant_id) { | ||
// This is necessary because we need to let the `discriminant_map` | ||
// borrow fall out of scope, so that we can reborrow farther down. | ||
maybe_expr = (*get_expr).clone(); | ||
} else { | ||
self.sess.span_bug(variant.span, | ||
"`check_static_recursion` attempted to visit \ | ||
variant with unknown discriminant") | ||
} | ||
// If `maybe_expr` is `None`, that's because no discriminant is | ||
// specified that affects this variant. Thus, no risk of recursion. | ||
if let Some(expr) = maybe_expr { | ||
self.with_item_id_pushed(expr.id, |v| visit::walk_expr(v, expr)); | ||
} | ||
} | ||
|
||
fn visit_trait_item(&mut self, ti: &'ast ast::TraitItem) { | ||
self.with_item_id_pushed(ti.id, |v| visit::walk_trait_item(v, ti)); | ||
} | ||
|
||
fn visit_impl_item(&mut self, ii: &ast::ImplItem) { | ||
fn visit_impl_item(&mut self, ii: &'ast ast::ImplItem) { | ||
self.with_item_id_pushed(ii.id, |v| visit::walk_impl_item(v, ii)); | ||
} | ||
|
||
fn visit_expr(&mut self, e: &ast::Expr) { | ||
fn visit_expr(&mut self, e: &'ast ast::Expr) { | ||
match e.node { | ||
ast::ExprPath(..) => { | ||
match self.def_map.borrow().get(&e.id).map(|d| d.base_def) { | ||
Some(DefStatic(def_id, _)) | | ||
Some(DefAssociatedConst(def_id, _)) | | ||
Some(DefConst(def_id)) if | ||
ast_util::is_local(def_id) => { | ||
Some(DefConst(def_id)) | ||
if ast_util::is_local(def_id) => { | ||
match self.ast_map.get(def_id.node) { | ||
ast_map::NodeItem(item) => | ||
self.visit_item(item), | ||
|
@@ -141,11 +226,28 @@ impl<'a, 'ast, 'v> Visitor<'v> for CheckItemRecursionVisitor<'a, 'ast> { | |
self.visit_impl_item(item), | ||
ast_map::NodeForeignItem(_) => {}, | ||
_ => { | ||
span_err!(self.sess, e.span, E0266, | ||
"expected item, found {}", | ||
self.ast_map.node_to_string(def_id.node)); | ||
return; | ||
}, | ||
self.sess.span_bug( | ||
e.span, | ||
&format!("expected item, found {}", | ||
self.ast_map.node_to_string(def_id.node))); | ||
} | ||
} | ||
} | ||
// For variants, we only want to check expressions that | ||
// affect the specific variant used, but we need to check | ||
// the whole enum definition to see what expression that | ||
// might be (if any). | ||
Some(DefVariant(enum_id, variant_id, false)) | ||
if ast_util::is_local(enum_id) => { | ||
if let ast::ItemEnum(ref enum_def, ref generics) = | ||
self.ast_map.expect_item(enum_id.local_id()).node { | ||
self.populate_enum_discriminants(enum_def); | ||
let variant = self.ast_map.expect_variant(variant_id.local_id()); | ||
self.visit_variant(variant, generics); | ||
} else { | ||
self.sess.span_bug(e.span, | ||
"`check_static_recursion` found \ | ||
non-enum in DefVariant"); | ||
} | ||
} | ||
_ => () | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
// Copyright 2015 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. | ||
|
||
enum X { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Please add a comment explaining what is being tested |
||
A = X::A as isize, //~ ERROR E0265 | ||
} | ||
|
||
enum Y { | ||
A = Y::B as isize, //~ ERROR E0265 | ||
B, | ||
} | ||
|
||
fn main() { } |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why is this change OK? It seems to me that if you have a const inside a function you'll miss it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh nvm, it just moved outside the match