rust/src/librustc/middle/dead.rs

637 lines
23 KiB
Rust
Raw Normal View History

2013-12-08 01:55:27 -06:00
// Copyright 2013 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.
// This implements the dead-code warning pass. It follows middle::reachable
// closely. The idea is that all reachable symbols are live, codes called
// from live codes are live, and everything else is dead.
2015-06-09 18:40:45 -05:00
use ast_map;
use middle::{def, pat_util, privacy, ty};
2014-09-20 06:26:10 -05:00
use lint;
use util::nodemap::NodeSet;
2013-12-08 01:55:27 -06:00
use std::collections::HashSet;
2015-06-09 18:40:45 -05:00
use syntax::{ast, codemap};
use syntax::ast_util::{local_def, is_local};
2015-01-03 21:42:21 -06:00
use syntax::attr::{self, AttrMetaMethods};
use syntax::visit::{self, Visitor};
2013-12-08 01:55:27 -06:00
// Any local node that may call something in its body block should be
// explored. For example, if it's a live NodeItem that is a
2013-12-08 01:55:27 -06:00
// function, then we should explore its block to check for codes that
// may need to be marked as live.
2014-03-05 21:07:47 -06:00
fn should_explore(tcx: &ty::ctxt, def_id: ast::DefId) -> bool {
2013-12-08 01:55:27 -06:00
if !is_local(def_id) {
return false;
}
2013-12-27 18:09:29 -06:00
match tcx.map.find(def_id.node) {
Some(ast_map::NodeItem(..))
| Some(ast_map::NodeImplItem(..))
| Some(ast_map::NodeForeignItem(..))
| Some(ast_map::NodeTraitItem(..)) => true,
2013-12-08 01:55:27 -06:00
_ => false
}
}
struct MarkSymbolVisitor<'a, 'tcx: 'a> {
2014-03-05 21:07:47 -06:00
worklist: Vec<ast::NodeId>,
tcx: &'a ty::ctxt<'tcx>,
live_symbols: Box<HashSet<ast::NodeId>>,
2014-09-20 06:26:10 -05:00
struct_has_extern_repr: bool,
ignore_non_const_paths: bool,
inherited_pub_visibility: bool,
ignore_variant_stack: Vec<ast::NodeId>,
}
impl<'a, 'tcx> MarkSymbolVisitor<'a, 'tcx> {
fn new(tcx: &'a ty::ctxt<'tcx>,
worklist: Vec<ast::NodeId>) -> MarkSymbolVisitor<'a, 'tcx> {
2013-12-08 01:55:27 -06:00
MarkSymbolVisitor {
worklist: worklist,
tcx: tcx,
2014-04-25 03:08:02 -05:00
live_symbols: box HashSet::new(),
2014-09-20 06:26:10 -05:00
struct_has_extern_repr: false,
ignore_non_const_paths: false,
inherited_pub_visibility: false,
ignore_variant_stack: vec![],
2013-12-08 01:55:27 -06:00
}
}
fn check_def_id(&mut self, def_id: ast::DefId) {
if should_explore(self.tcx, def_id) {
self.worklist.push(def_id.node);
}
self.live_symbols.insert(def_id.node);
}
fn lookup_and_handle_definition(&mut self, id: &ast::NodeId) {
2014-11-06 11:25:16 -06:00
self.tcx.def_map.borrow().get(id).map(|def| {
match def.full_def() {
def::DefConst(_) | def::DefAssociatedConst(..) => {
self.check_def_id(def.def_id())
}
_ if self.ignore_non_const_paths => (),
def::DefPrimTy(_) => (),
def::DefVariant(enum_id, variant_id, _) => {
2014-09-20 06:26:10 -05:00
self.check_def_id(enum_id);
if !self.ignore_variant_stack.contains(&variant_id.node) {
self.check_def_id(variant_id);
}
2014-09-20 06:26:10 -05:00
}
_ => {
self.check_def_id(def.def_id());
}
}
});
}
fn lookup_and_handle_method(&mut self, id: ast::NodeId,
span: codemap::Span) {
let method_call = ty::MethodCall::expr(id);
2014-11-06 11:25:16 -06:00
match self.tcx.method_map.borrow().get(&method_call) {
Some(method) => {
match method.origin {
ty::MethodStatic(def_id) => {
match self.tcx.provided_source(def_id) {
Some(p_did) => self.check_def_id(p_did),
None => self.check_def_id(def_id)
}
}
ty::MethodStaticClosure(_) => {}
ty::MethodTypeParam(ty::MethodParam {
ref trait_ref,
method_num: index,
..
}) |
ty::MethodTraitObject(ty::MethodObject {
ref trait_ref,
method_num: index,
..
}) => {
let trait_item = self.tcx.trait_item(trait_ref.def_id, index);
self.check_def_id(trait_item.def_id());
}
2013-12-08 01:55:27 -06:00
}
}
None => {
self.tcx.sess.span_bug(span,
"method call expression not \
in method map?!")
}
2013-12-08 01:55:27 -06:00
}
}
fn handle_field_access(&mut self, lhs: &ast::Expr, name: ast::Name) {
match self.tcx.expr_ty_adjusted(lhs).sty {
ty::TyStruct(id, _) => {
let fields = self.tcx.lookup_struct_fields(id);
2014-06-05 17:00:29 -05:00
let field_id = fields.iter()
.find(|field| field.name == name).unwrap().id;
2014-06-05 17:00:29 -05:00
self.live_symbols.insert(field_id.node);
},
_ => ()
}
}
fn handle_tup_field_access(&mut self, lhs: &ast::Expr, idx: usize) {
match self.tcx.expr_ty_adjusted(lhs).sty {
ty::TyStruct(id, _) => {
let fields = self.tcx.lookup_struct_fields(id);
let field_id = fields[idx].id;
self.live_symbols.insert(field_id.node);
},
_ => ()
}
}
fn handle_field_pattern_match(&mut self, lhs: &ast::Pat,
pats: &[codemap::Spanned<ast::FieldPat>]) {
let id = match self.tcx.def_map.borrow().get(&lhs.id).unwrap().full_def() {
def::DefVariant(_, id, _) => id,
_ => {
match self.tcx.node_id_to_type(lhs.id).ty_to_def_id() {
None => {
self.tcx.sess.span_bug(lhs.span,
"struct pattern wasn't of a \
type with a def ID?!")
}
Some(def_id) => def_id,
2014-06-05 17:00:29 -05:00
}
}
};
let fields = self.tcx.lookup_struct_fields(id);
2015-01-31 11:20:46 -06:00
for pat in pats {
if let ast::PatWild(ast::PatWildSingle) = pat.node.pat.node {
continue;
}
let field_id = fields.iter()
.find(|field| field.name == pat.node.ident.name).unwrap().id;
self.live_symbols.insert(field_id.node);
2014-06-05 17:00:29 -05:00
}
}
2013-12-08 01:55:27 -06:00
fn mark_live_symbols(&mut self) {
let mut scanned = HashSet::new();
while !self.worklist.is_empty() {
let id = self.worklist.pop().unwrap();
2013-12-08 01:55:27 -06:00
if scanned.contains(&id) {
continue
}
scanned.insert(id);
2013-12-27 18:09:29 -06:00
match self.tcx.map.find(id) {
Some(ref node) => {
2013-12-08 01:55:27 -06:00
self.live_symbols.insert(id);
self.visit_node(node);
}
None => (),
}
}
}
fn visit_node(&mut self, node: &ast_map::Node) {
let had_extern_repr = self.struct_has_extern_repr;
self.struct_has_extern_repr = false;
let had_inherited_pub_visibility = self.inherited_pub_visibility;
self.inherited_pub_visibility = false;
2013-12-08 01:55:27 -06:00
match *node {
ast_map::NodeItem(item) => {
2013-12-08 01:55:27 -06:00
match item.node {
ast::ItemStruct(..) => {
self.struct_has_extern_repr = item.attrs.iter().any(|attr| {
attr::find_repr_attrs(self.tcx.sess.diagnostic(), attr)
.contains(&attr::ReprExtern)
});
visit::walk_item(self, &*item);
2014-06-05 17:00:29 -05:00
}
ast::ItemEnum(..) => {
self.inherited_pub_visibility = item.vis == ast::Public;
visit::walk_item(self, &*item);
}
ast::ItemFn(..)
| ast::ItemTy(..)
rustc: Add `const` globals to the language This change is an implementation of [RFC 69][rfc] which adds a third kind of global to the language, `const`. This global is most similar to what the old `static` was, and if you're unsure about what to use then you should use a `const`. The semantics of these three kinds of globals are: * A `const` does not represent a memory location, but only a value. Constants are translated as rvalues, which means that their values are directly inlined at usage location (similar to a #define in C/C++). Constant values are, well, constant, and can not be modified. Any "modification" is actually a modification to a local value on the stack rather than the actual constant itself. Almost all values are allowed inside constants, whether they have interior mutability or not. There are a few minor restrictions listed in the RFC, but they should in general not come up too often. * A `static` now always represents a memory location (unconditionally). Any references to the same `static` are actually a reference to the same memory location. Only values whose types ascribe to `Sync` are allowed in a `static`. This restriction is in place because many threads may access a `static` concurrently. Lifting this restriction (and allowing unsafe access) is a future extension not implemented at this time. * A `static mut` continues to always represent a memory location. All references to a `static mut` continue to be `unsafe`. This is a large breaking change, and many programs will need to be updated accordingly. A summary of the breaking changes is: * Statics may no longer be used in patterns. Statics now always represent a memory location, which can sometimes be modified. To fix code, repurpose the matched-on-`static` to a `const`. static FOO: uint = 4; match n { FOO => { /* ... */ } _ => { /* ... */ } } change this code to: const FOO: uint = 4; match n { FOO => { /* ... */ } _ => { /* ... */ } } * Statics may no longer refer to other statics by value. Due to statics being able to change at runtime, allowing them to reference one another could possibly lead to confusing semantics. If you are in this situation, use a constant initializer instead. Note, however, that statics may reference other statics by address, however. * Statics may no longer be used in constant expressions, such as array lengths. This is due to the same restrictions as listed above. Use a `const` instead. [breaking-change] [rfc]: https://github.com/rust-lang/rfcs/pull/246
2014-10-06 10:17:01 -05:00
| ast::ItemStatic(..)
| ast::ItemConst(..) => {
visit::walk_item(self, &*item);
2013-12-08 01:55:27 -06:00
}
_ => ()
}
}
ast_map::NodeTraitItem(trait_item) => {
visit::walk_trait_item(self, trait_item);
2013-12-08 01:55:27 -06:00
}
ast_map::NodeImplItem(impl_item) => {
visit::walk_impl_item(self, impl_item);
2013-12-08 01:55:27 -06:00
}
ast_map::NodeForeignItem(foreign_item) => {
visit::walk_foreign_item(self, &*foreign_item);
}
2013-12-08 01:55:27 -06:00
_ => ()
}
self.struct_has_extern_repr = had_extern_repr;
self.inherited_pub_visibility = had_inherited_pub_visibility;
2013-12-08 01:55:27 -06:00
}
}
impl<'a, 'tcx, 'v> Visitor<'v> for MarkSymbolVisitor<'a, 'tcx> {
fn visit_struct_def(&mut self, def: &ast::StructDef, _: ast::Ident,
_: &ast::Generics, _: ast::NodeId) {
let has_extern_repr = self.struct_has_extern_repr;
let inherited_pub_visibility = self.inherited_pub_visibility;
let live_fields = def.fields.iter().filter(|f| {
has_extern_repr || inherited_pub_visibility || match f.node.kind {
ast::NamedField(_, ast::Public) => true,
_ => false
}
});
self.live_symbols.extend(live_fields.map(|f| f.node.id));
visit::walk_struct_def(self, def);
}
2013-12-08 01:55:27 -06:00
fn visit_expr(&mut self, expr: &ast::Expr) {
2013-12-08 01:55:27 -06:00
match expr.node {
ast::ExprMethodCall(..) => {
self.lookup_and_handle_method(expr.id, expr.span);
2013-12-08 01:55:27 -06:00
}
ast::ExprField(ref lhs, ref ident) => {
self.handle_field_access(&**lhs, ident.node.name);
2014-06-05 17:00:29 -05:00
}
ast::ExprTupField(ref lhs, idx) => {
self.handle_tup_field_access(&**lhs, idx.node);
}
2013-12-08 01:55:27 -06:00
_ => ()
}
visit::walk_expr(self, expr);
2013-12-08 01:55:27 -06:00
}
fn visit_arm(&mut self, arm: &ast::Arm) {
if arm.pats.len() == 1 {
let pat = &*arm.pats[0];
let variants = pat_util::necessary_variants(&self.tcx.def_map, pat);
// Inside the body, ignore constructions of variants
// necessary for the pattern to match. Those construction sites
// can't be reached unless the variant is constructed elsewhere.
let len = self.ignore_variant_stack.len();
self.ignore_variant_stack.push_all(&*variants);
visit::walk_arm(self, arm);
self.ignore_variant_stack.truncate(len);
} else {
visit::walk_arm(self, arm);
}
}
fn visit_pat(&mut self, pat: &ast::Pat) {
2014-09-20 06:26:10 -05:00
let def_map = &self.tcx.def_map;
2014-06-05 17:00:29 -05:00
match pat.node {
ast::PatStruct(_, ref fields, _) => {
self.handle_field_pattern_match(pat, fields);
2014-06-05 17:00:29 -05:00
}
2014-09-20 06:26:10 -05:00
_ if pat_util::pat_is_const(def_map, pat) => {
// it might be the only use of a const
self.lookup_and_handle_definition(&pat.id)
}
2014-06-05 17:00:29 -05:00
_ => ()
}
self.ignore_non_const_paths = true;
visit::walk_pat(self, pat);
self.ignore_non_const_paths = false;
2014-06-05 17:00:29 -05:00
}
fn visit_path(&mut self, path: &ast::Path, id: ast::NodeId) {
self.lookup_and_handle_definition(&id);
visit::walk_path(self, path);
}
fn visit_item(&mut self, _: &ast::Item) {
2013-12-08 01:55:27 -06:00
// Do not recurse into items. These items will be added to the
// worklist and recursed into manually if necessary.
}
}
fn has_allow_dead_code_or_lang_attr(attrs: &[ast::Attribute]) -> bool {
if attr::contains_name(attrs, "lang") {
return true;
}
let dead_code = lint::builtin::DEAD_CODE.name_lower();
for attr in lint::gather_attrs(attrs) {
match attr {
Ok((ref name, lint::Allow, _))
if &name[..] == dead_code => return true,
_ => (),
}
}
false
}
// This visitor seeds items that
// 1) We want to explicitly consider as live:
// * Item annotated with #[allow(dead_code)]
// - This is done so that if we want to suppress warnings for a
// group of dead functions, we only have to annotate the "root".
// For example, if both `f` and `g` are dead and `f` calls `g`,
// then annotating `f` with `#[allow(dead_code)]` will suppress
// warning for both `f` and `g`.
// * Item annotated with #[lang=".."]
// - This is because lang items are always callable from elsewhere.
// or
// 2) We are not sure to be live or not
// * Implementation of a trait method
struct LifeSeeder {
2014-09-20 06:26:10 -05:00
worklist: Vec<ast::NodeId>
2013-12-08 01:55:27 -06:00
}
impl<'v> Visitor<'v> for LifeSeeder {
fn visit_item(&mut self, item: &ast::Item) {
let allow_dead_code = has_allow_dead_code_or_lang_attr(&item.attrs);
2014-09-20 06:26:10 -05:00
if allow_dead_code {
self.worklist.push(item.id);
}
2013-12-08 01:55:27 -06:00
match item.node {
2014-09-20 06:26:10 -05:00
ast::ItemEnum(ref enum_def, _) if allow_dead_code => {
self.worklist.extend(enum_def.variants.iter().map(|variant| variant.node.id));
}
ast::ItemTrait(_, _, _, ref trait_items) => {
for trait_item in trait_items {
match trait_item.node {
ast::ConstTraitItem(_, Some(_)) |
ast::MethodTraitItem(_, Some(_)) => {
if has_allow_dead_code_or_lang_attr(&trait_item.attrs) {
self.worklist.push(trait_item.id);
}
}
_ => {}
}
}
}
ast::ItemImpl(_, _, _, ref opt_trait, _, ref impl_items) => {
2015-01-31 11:20:46 -06:00
for impl_item in impl_items {
match impl_item.node {
ast::ConstImplItem(..) |
ast::MethodImplItem(..) => {
if opt_trait.is_some() ||
has_allow_dead_code_or_lang_attr(&impl_item.attrs) {
self.worklist.push(impl_item.id);
}
}
ast::TypeImplItem(_) => {}
ast::MacImplItem(_) => panic!("unexpanded macro")
}
2013-12-08 01:55:27 -06:00
}
}
_ => ()
}
visit::walk_item(self, item);
}
2013-12-08 01:55:27 -06:00
}
2014-03-05 21:07:47 -06:00
fn create_and_seed_worklist(tcx: &ty::ctxt,
2013-12-08 01:55:27 -06:00
exported_items: &privacy::ExportedItems,
reachable_symbols: &NodeSet,
krate: &ast::Crate) -> Vec<ast::NodeId> {
let mut worklist = Vec::new();
2013-12-08 01:55:27 -06:00
// Preferably, we would only need to seed the worklist with reachable
// symbols. However, since the set of reachable symbols differs
// depending on whether a crate is built as bin or lib, and we want
// the warning to be consistent, we also seed the worklist with
// exported symbols.
2015-01-31 11:20:46 -06:00
for id in exported_items {
worklist.push(*id);
2013-12-08 01:55:27 -06:00
}
2015-01-31 11:20:46 -06:00
for id in reachable_symbols {
// Reachable variants can be dead, because we warn about
// variants never constructed, not variants never used.
if let Some(ast_map::NodeVariant(..)) = tcx.map.find(*id) {
continue;
}
worklist.push(*id);
2013-12-08 01:55:27 -06:00
}
// Seed entry point
match *tcx.sess.entry_fn.borrow() {
2013-12-08 01:55:27 -06:00
Some((id, _)) => worklist.push(id),
None => ()
}
// Seed implemented trait items
let mut life_seeder = LifeSeeder {
2013-12-08 01:55:27 -06:00
worklist: worklist
};
visit::walk_crate(&mut life_seeder, krate);
2013-12-08 01:55:27 -06:00
return life_seeder.worklist;
2013-12-08 01:55:27 -06:00
}
2014-03-05 21:07:47 -06:00
fn find_live(tcx: &ty::ctxt,
2013-12-08 01:55:27 -06:00
exported_items: &privacy::ExportedItems,
reachable_symbols: &NodeSet,
krate: &ast::Crate)
-> Box<HashSet<ast::NodeId>> {
2013-12-08 01:55:27 -06:00
let worklist = create_and_seed_worklist(tcx, exported_items,
reachable_symbols, krate);
let mut symbol_visitor = MarkSymbolVisitor::new(tcx, worklist);
2013-12-08 01:55:27 -06:00
symbol_visitor.mark_live_symbols();
symbol_visitor.live_symbols
}
fn get_struct_ctor_id(item: &ast::Item) -> Option<ast::NodeId> {
2013-12-08 01:55:27 -06:00
match item.node {
2014-09-07 12:09:06 -05:00
ast::ItemStruct(ref struct_def, _) => struct_def.ctor_id,
2013-12-08 01:55:27 -06:00
_ => None
}
}
struct DeadVisitor<'a, 'tcx: 'a> {
tcx: &'a ty::ctxt<'tcx>,
live_symbols: Box<HashSet<ast::NodeId>>,
2013-12-08 01:55:27 -06:00
}
impl<'a, 'tcx> DeadVisitor<'a, 'tcx> {
2014-09-20 06:26:10 -05:00
fn should_warn_about_item(&mut self, item: &ast::Item) -> bool {
let should_warn = match item.node {
ast::ItemStatic(..)
| ast::ItemConst(..)
2014-09-20 06:26:10 -05:00
| ast::ItemFn(..)
| ast::ItemEnum(..)
| ast::ItemStruct(..) => true,
_ => false
};
let ctor_id = get_struct_ctor_id(item);
should_warn && !self.symbol_is_live(item.id, ctor_id)
}
2014-06-05 17:00:29 -05:00
fn should_warn_about_field(&mut self, node: &ast::StructField_) -> bool {
let is_named = node.ident().is_some();
let field_type = self.tcx.node_id_to_type(node.id);
let is_marker_field = match field_type.ty_to_def_id() {
2014-06-05 17:00:29 -05:00
Some(def_id) => self.tcx.lang_items.items().any(|(_, item)| *item == Some(def_id)),
_ => false
};
is_named
&& !self.symbol_is_live(node.id, None)
&& !is_marker_field
&& !has_allow_dead_code_or_lang_attr(&node.attrs)
2014-06-05 17:00:29 -05:00
}
2014-09-20 06:26:10 -05:00
fn should_warn_about_variant(&mut self, variant: &ast::Variant_) -> bool {
!self.symbol_is_live(variant.id, None)
&& !has_allow_dead_code_or_lang_attr(&variant.attrs)
2014-09-20 06:26:10 -05:00
}
2013-12-08 01:55:27 -06:00
// id := node id of an item's definition.
// ctor_id := `Some` if the item is a struct_ctor (tuple struct),
// `None` otherwise.
// If the item is a struct_ctor, then either its `id` or
// `ctor_id` (unwrapped) is in the live_symbols set. More specifically,
// DefMap maps the ExprPath of a struct_ctor to the node referred by
// `ctor_id`. On the other hand, in a statement like
// `type <ident> <generics> = <ty>;` where <ty> refers to a struct_ctor,
// DefMap maps <ty> to `id` instead.
fn symbol_is_live(&mut self, id: ast::NodeId,
ctor_id: Option<ast::NodeId>) -> bool {
if self.live_symbols.contains(&id)
|| ctor_id.map_or(false,
|ctor| self.live_symbols.contains(&ctor)) {
2013-12-08 01:55:27 -06:00
return true;
}
// If it's a type whose items are live, then it's live, too.
2013-12-08 01:55:27 -06:00
// This is done to handle the case where, for example, the static
// method of a private type is used, but the type itself is never
// called directly.
let impl_items = self.tcx.impl_items.borrow();
2014-11-06 11:25:16 -06:00
match self.tcx.inherent_impls.borrow().get(&local_def(id)) {
2013-12-08 01:55:27 -06:00
None => (),
2014-04-22 11:06:43 -05:00
Some(impl_list) => {
for impl_did in impl_list.iter() {
for item_did in impl_items.get(impl_did).unwrap().iter() {
if self.live_symbols.contains(&item_did.def_id()
.node) {
2013-12-08 01:55:27 -06:00
return true;
}
}
}
}
}
false
}
fn warn_dead_code(&mut self,
id: ast::NodeId,
span: codemap::Span,
name: ast::Name,
node_type: &str) {
let name = name.as_str();
if !name.starts_with("_") {
self.tcx
.sess
.add_lint(lint::builtin::DEAD_CODE,
id,
span,
format!("{} is never used: `{}`", node_type, name));
}
}
2013-12-08 01:55:27 -06:00
}
impl<'a, 'tcx, 'v> Visitor<'v> for DeadVisitor<'a, 'tcx> {
fn visit_item(&mut self, item: &ast::Item) {
2014-09-20 06:26:10 -05:00
if self.should_warn_about_item(item) {
self.warn_dead_code(
item.id,
item.span,
item.ident.name,
item.node.descriptive_variant()
);
2014-09-20 06:26:10 -05:00
} else {
match item.node {
ast::ItemEnum(ref enum_def, _) => {
2015-01-31 11:20:46 -06:00
for variant in &enum_def.variants {
2014-09-20 06:26:10 -05:00
if self.should_warn_about_variant(&variant.node) {
self.warn_dead_code(variant.node.id, variant.span,
variant.node.name.name, "variant");
2014-09-20 06:26:10 -05:00
}
}
},
_ => ()
}
2013-12-08 01:55:27 -06:00
}
visit::walk_item(self, item);
2013-12-08 01:55:27 -06:00
}
fn visit_foreign_item(&mut self, fi: &ast::ForeignItem) {
if !self.symbol_is_live(fi.id, None) {
self.warn_dead_code(fi.id, fi.span, fi.ident.name, fi.node.descriptive_variant());
}
visit::walk_foreign_item(self, fi);
}
fn visit_struct_field(&mut self, field: &ast::StructField) {
2014-06-05 17:00:29 -05:00
if self.should_warn_about_field(&field.node) {
self.warn_dead_code(field.node.id, field.span,
field.node.ident().unwrap().name, "struct field");
2014-06-05 17:00:29 -05:00
}
visit::walk_struct_field(self, field);
2014-06-05 17:00:29 -05:00
}
fn visit_impl_item(&mut self, impl_item: &ast::ImplItem) {
match impl_item.node {
ast::ConstImplItem(_, ref expr) => {
if !self.symbol_is_live(impl_item.id, None) {
self.warn_dead_code(impl_item.id, impl_item.span,
impl_item.ident.name, "associated const");
}
visit::walk_expr(self, expr)
}
ast::MethodImplItem(_, ref body) => {
if !self.symbol_is_live(impl_item.id, None) {
self.warn_dead_code(impl_item.id, impl_item.span,
impl_item.ident.name, "method");
}
visit::walk_block(self, body)
}
ast::TypeImplItem(..) |
ast::MacImplItem(..) => {}
}
}
// Overwrite so that we don't warn the trait item itself.
fn visit_trait_item(&mut self, trait_item: &ast::TraitItem) {
match trait_item.node {
ast::ConstTraitItem(_, Some(ref expr)) => {
visit::walk_expr(self, expr)
}
ast::MethodTraitItem(_, Some(ref body)) => {
visit::walk_block(self, body)
}
ast::ConstTraitItem(_, None) |
ast::MethodTraitItem(_, None) |
ast::TypeTraitItem(..) => {}
2013-12-08 01:55:27 -06:00
}
}
}
2014-03-05 21:07:47 -06:00
pub fn check_crate(tcx: &ty::ctxt,
2013-12-08 01:55:27 -06:00
exported_items: &privacy::ExportedItems,
2014-09-07 12:09:06 -05:00
reachable_symbols: &NodeSet) {
let krate = tcx.map.krate();
let live_symbols = find_live(tcx, exported_items,
reachable_symbols, krate);
2013-12-08 01:55:27 -06:00
let mut visitor = DeadVisitor { tcx: tcx, live_symbols: live_symbols };
visit::walk_crate(&mut visitor, krate);
2013-12-08 01:55:27 -06:00
}