// Copyright (C) 2021-2025 Free Software Foundation, Inc. // This file is part of GCC. // GCC is free software; you can redistribute it and/or modify it under // the terms of the GNU General Public License as published by the Free // Software Foundation; either version 3, or (at your option) any later // version. // GCC is distributed in the hope that it will be useful, but WITHOUT ANY // WARRANTY; without even the implied warranty of MERCHANTABILITY or // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License // for more details. // You should have received a copy of the GNU General Public License // along with GCC; see the file COPYING3. If not see // <http://www.gnu.org/licenses/>. #ifndef RUST_HIR_SCAN_DEADCODE #define RUST_HIR_SCAN_DEADCODE #include "rust-hir-full-decls.h" #include "rust-hir-map.h" #include "rust-lint-marklive.h" #include "rust-name-resolver.h" #include "rust-diagnostics.h" namespace Rust { namespace Analysis { // Scan item symbols and warn the symbol if it is not in the live_symbols set. // There are three kinds of item we should handle in this pass. // 1. Function item // 2. The function item in the impl block without trait // 3. StructStruct, e.g., `Struct Foo{one: 1, two: 2}`. Furthermore, the unused // struct fields will be warned too. // 4. TupleStruct, e.g., `Struct Foo(i32, i32)` class ScanDeadcode : public MarkLiveBase { using Rust::Analysis::MarkLiveBase::visit; public: static void Scan (HIR::Crate &crate) { std::set<HirId> live_symbols = Analysis::MarkLive::Analysis (crate); ScanDeadcode sdc (live_symbols); for (auto &it : crate.get_items ()) it.get ()->accept_vis (sdc); }; void visit (HIR::Function &function) override { HirId hirId = function.get_mappings ().get_hirid (); if (should_warn (hirId) && !function.get_visibility ().is_public ()) { if (mappings.is_impl_item (hirId)) { HIR::ImplBlock *implBlock = mappings.lookup_associated_impl (hirId); if (!implBlock->has_trait_ref ()) { rust_warning_at ( function.get_function_name ().get_locus (), 0, "associated function is never used: %qs", function.get_function_name ().as_string ().c_str ()); } } else { rust_warning_at ( function.get_function_name ().get_locus (), 0, "function is never used: %qs", function.get_function_name ().as_string ().c_str ()); } } } void visit (HIR::StructStruct &stct) override { HirId hirId = stct.get_mappings ().get_hirid (); if (should_warn (hirId) && !stct.get_visibility ().is_public ()) { bool name_starts_underscore = stct.get_identifier ().as_string ().at (0) == '_'; if (!name_starts_underscore) rust_warning_at (stct.get_locus (), 0, "struct is never constructed: %qs", stct.get_identifier ().as_string ().c_str ()); } else { // only warn the unused fields when in unwarned struct. for (auto &field : stct.get_fields ()) { HirId field_hir_id = field.get_mappings ().get_hirid (); if (should_warn (field_hir_id) && !field.get_visibility ().is_public () && field.get_field_name ().as_string ().at (0) != '_') { rust_warning_at (field.get_locus (), 0, "field is never read: %qs", field.get_field_name ().as_string ().c_str ()); } } } } void visit (HIR::TupleStruct &stct) override { // only warn tuple struct unconstructed, and ignoring unused field HirId hirId = stct.get_mappings ().get_hirid (); if (should_warn (hirId) && !stct.get_visibility ().is_public ()) { rust_warning_at (stct.get_locus (), 0, "struct is never constructed: %qs", stct.get_identifier ().as_string ().c_str ()); } } void visit (HIR::ImplBlock &blc) override { if (blc.has_impl_items ()) { for (auto &implItem : blc.get_impl_items ()) { implItem->accept_vis (*this); } } } void visit (HIR::Module &mod) override { for (auto &item : mod.get_items ()) item->accept_vis (*this); } private: std::set<HirId> live_symbols; Resolver::Resolver *resolver; Analysis::Mappings &mappings; ScanDeadcode (std::set<HirId> &live_symbols) : live_symbols (live_symbols), resolver (Resolver::Resolver::get ()), mappings (Analysis::Mappings::get ()){}; bool should_warn (HirId hirId) { // TODO: There are more condition to check if should warn, i.e visibility, // attributes. return live_symbols.find (hirId) == live_symbols.end (); } }; } // namespace Analysis } // namespace Rust #endif