Skip to content
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

feat: warn on unused imports #5847

Merged
merged 10 commits into from
Aug 28, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
62 changes: 59 additions & 3 deletions compiler/noirc_driver/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -131,6 +131,56 @@ pub struct CompileOptions {
pub skip_underconstrained_check: bool,
}

/// Similar to CompileOptions but with some extra fields specific for `check_crate`.
#[derive(Clone, Debug, Default)]
pub struct CheckOptions {
asterite marked this conversation as resolved.
Show resolved Hide resolved
pub expression_width: Option<ExpressionWidth>,
pub bounded_codegen: bool,
pub force_compile: bool,
pub show_ssa: bool,
pub emit_ssa: bool,
pub show_brillig: bool,
pub print_acir: bool,
pub benchmark_codegen: bool,
pub deny_warnings: bool,
pub silence_warnings: bool,
pub disable_macros: bool,
pub show_monomorphized: bool,
pub instrument_debug: bool,
pub force_brillig: bool,
pub debug_comptime_in_file: Option<String>,
pub show_artifact_paths: bool,
pub arithmetic_generics: bool,
pub skip_underconstrained_check: bool,
pub error_on_unused_imports: bool,
}

impl CheckOptions {
pub fn from_compile_options(options: &CompileOptions, error_on_unused_imports: bool) -> Self {
Self {
expression_width: options.expression_width,
bounded_codegen: options.bounded_codegen,
force_compile: options.force_compile,
show_ssa: options.show_ssa,
emit_ssa: options.emit_ssa,
show_brillig: options.show_brillig,
print_acir: options.print_acir,
benchmark_codegen: options.benchmark_codegen,
deny_warnings: options.deny_warnings,
silence_warnings: options.silence_warnings,
disable_macros: options.disable_macros,
show_monomorphized: options.show_monomorphized,
instrument_debug: options.instrument_debug,
force_brillig: options.force_brillig,
debug_comptime_in_file: options.debug_comptime_in_file.clone(),
show_artifact_paths: options.show_artifact_paths,
arithmetic_generics: options.arithmetic_generics,
skip_underconstrained_check: options.skip_underconstrained_check,
error_on_unused_imports,
}
}
}

pub fn parse_expression_width(input: &str) -> Result<ExpressionWidth, std::io::Error> {
use std::io::{Error, ErrorKind};
let width = input
Expand Down Expand Up @@ -278,7 +328,7 @@ pub fn add_dep(
pub fn check_crate(
context: &mut Context,
crate_id: CrateId,
options: &CompileOptions,
options: &CheckOptions,
) -> CompilationResult<()> {
let macros: &[&dyn MacroProcessor] =
if options.disable_macros { &[] } else { &[&aztec_macros::AztecMacro] };
Expand All @@ -289,6 +339,7 @@ pub fn check_crate(
context,
options.debug_comptime_in_file.as_deref(),
options.arithmetic_generics,
options.error_on_unused_imports,
macros,
);
errors.extend(diagnostics.into_iter().map(|(error, file_id)| {
Expand Down Expand Up @@ -322,7 +373,10 @@ pub fn compile_main(
options: &CompileOptions,
cached_program: Option<CompiledProgram>,
) -> CompilationResult<CompiledProgram> {
let (_, mut warnings) = check_crate(context, crate_id, options)?;
let error_on_unused_imports = true;
let check_options = CheckOptions::from_compile_options(options, error_on_unused_imports);

let (_, mut warnings) = check_crate(context, crate_id, &check_options)?;

let main = context.get_main_function(&crate_id).ok_or_else(|| {
// TODO(#2155): This error might be a better to exist in Nargo
Expand Down Expand Up @@ -357,7 +411,9 @@ pub fn compile_contract(
crate_id: CrateId,
options: &CompileOptions,
) -> CompilationResult<CompiledContract> {
let (_, warnings) = check_crate(context, crate_id, options)?;
let error_on_unused_imports = true;
let check_options = CheckOptions::from_compile_options(options, error_on_unused_imports);
let (_, warnings) = check_crate(context, crate_id, &check_options)?;

// TODO: We probably want to error if contracts is empty
let contracts = context.get_all_contracts(&crate_id);
Expand Down
14 changes: 5 additions & 9 deletions compiler/noirc_frontend/src/elaborator/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ use crate::{
UnresolvedTypeAlias,
},
def_map::DefMaps,
resolution::{errors::ResolverError, path_resolver::PathResolver},
resolution::errors::ResolverError,
scope::ScopeForest as GenericScopeForest,
type_check::{generics::TraitGenerics, TypeCheckError},
},
Expand All @@ -36,7 +36,7 @@ use crate::{
hir::{
def_collector::{dc_crate::CollectedItems, errors::DefCollectorErrorKind},
def_map::{LocalModuleId, ModuleDefId, ModuleId, MAIN_FUNCTION},
resolution::{import::PathResolution, path_resolver::StandardPathResolver},
resolution::import::PathResolution,
Context,
},
hir_def::function::{FuncMeta, HirFunction},
Expand Down Expand Up @@ -630,10 +630,8 @@ impl<'context> Elaborator<'context> {
}
}

pub fn resolve_module_by_path(&self, path: Path) -> Option<ModuleId> {
let path_resolver = StandardPathResolver::new(self.module_id());

match path_resolver.resolve(self.def_maps, path.clone(), &mut None) {
pub fn resolve_module_by_path(&mut self, path: Path) -> Option<ModuleId> {
match self.resolve_path(path.clone()) {
Ok(PathResolution { module_def_id: ModuleDefId::ModuleId(module_id), error }) => {
if error.is_some() {
None
Expand All @@ -646,9 +644,7 @@ impl<'context> Elaborator<'context> {
}

fn resolve_trait_by_path(&mut self, path: Path) -> Option<TraitId> {
let path_resolver = StandardPathResolver::new(self.module_id());

let error = match path_resolver.resolve(self.def_maps, path.clone(), &mut None) {
let error = match self.resolve_path(path.clone()) {
Ok(PathResolution { module_def_id: ModuleDefId::TraitId(trait_id), error }) => {
if let Some(error) = error {
self.push_err(error);
Expand Down
43 changes: 29 additions & 14 deletions compiler/noirc_frontend/src/elaborator/scope.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ use noirc_errors::{Location, Spanned};

use crate::ast::{PathKind, ERROR_IDENT};
use crate::hir::def_map::{LocalModuleId, ModuleId};
use crate::hir::resolution::import::{PathResolution, PathResolutionResult};
use crate::hir::resolution::path_resolver::{PathResolver, StandardPathResolver};
use crate::hir::scope::{Scope as GenericScope, ScopeTree as GenericScopeTree};
use crate::macros_api::Ident;
Expand Down Expand Up @@ -29,7 +30,7 @@ type ScopeTree = GenericScopeTree<String, ResolverMeta>;
impl<'context> Elaborator<'context> {
pub(super) fn lookup<T: TryFromModuleDefId>(&mut self, path: Path) -> Result<T, ResolverError> {
let span = path.span();
let id = self.resolve_path(path)?;
let id = self.resolve_path_or_error(path)?;
T::try_from(id).ok_or_else(|| ResolverError::Expected {
expected: T::description(),
got: id.as_str().to_owned(),
Expand All @@ -42,15 +43,37 @@ impl<'context> Elaborator<'context> {
ModuleId { krate: self.crate_id, local_id: self.local_module }
}

pub(super) fn resolve_path(&mut self, path: Path) -> Result<ModuleDefId, ResolverError> {
pub(super) fn resolve_path_or_error(
&mut self,
path: Path,
) -> Result<ModuleDefId, ResolverError> {
let path_resolution = self.resolve_path(path)?;

if let Some(error) = path_resolution.error {
self.push_err(error);
}

Ok(path_resolution.module_def_id)
}

pub(super) fn resolve_path(&mut self, path: Path) -> PathResolutionResult {
let mut module_id = self.module_id();
let mut path = path;

if path.kind == PathKind::Plain {
let def_map = self.def_maps.get_mut(&self.crate_id).unwrap();
let module_data = &mut def_map.modules[module_id.local_id.0];
module_data.use_import(&path.segments[0].ident);
}

if path.kind == PathKind::Plain && path.first_name() == SELF_TYPE_NAME {
if let Some(Type::Struct(struct_type, _)) = &self.self_type {
let struct_type = struct_type.borrow();
if path.segments.len() == 1 {
return Ok(ModuleDefId::TypeId(struct_type.id));
return Ok(PathResolution {
module_def_id: ModuleDefId::TypeId(struct_type.id),
error: None,
});
}

module_id = struct_type.id.module_id();
Expand All @@ -65,11 +88,7 @@ impl<'context> Elaborator<'context> {
self.resolve_path_in_module(path, module_id)
}

fn resolve_path_in_module(
&mut self,
path: Path,
module_id: ModuleId,
) -> Result<ModuleDefId, ResolverError> {
fn resolve_path_in_module(&mut self, path: Path, module_id: ModuleId) -> PathResolutionResult {
let resolver = StandardPathResolver::new(module_id);
let path_resolution;

Expand Down Expand Up @@ -99,11 +118,7 @@ impl<'context> Elaborator<'context> {
path_resolution = resolver.resolve(self.def_maps, path, &mut None)?;
}

if let Some(error) = path_resolution.error {
self.push_err(error);
}

Ok(path_resolution.module_def_id)
Ok(path_resolution)
}

pub(super) fn get_struct(&self, type_id: StructId) -> Shared<StructType> {
Expand Down Expand Up @@ -150,7 +165,7 @@ impl<'context> Elaborator<'context> {

pub(super) fn lookup_global(&mut self, path: Path) -> Result<DefinitionId, ResolverError> {
let span = path.span();
let id = self.resolve_path(path)?;
let id = self.resolve_path_or_error(path)?;

if let Some(function) = TryFromModuleDefId::try_from(id) {
return Ok(self.interner.function_definition_id(function));
Expand Down
2 changes: 1 addition & 1 deletion compiler/noirc_frontend/src/elaborator/types.rs
Original file line number Diff line number Diff line change
Expand Up @@ -421,7 +421,7 @@ impl<'context> Elaborator<'context> {
}

// If we cannot find a local generic of the same name, try to look up a global
match self.resolve_path(path.clone()) {
match self.resolve_path_or_error(path.clone()) {
Ok(ModuleDefId::GlobalId(id)) => {
if let Some(current_item) = self.current_item {
self.interner.add_global_dependency(current_item, id);
Expand Down
22 changes: 22 additions & 0 deletions compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -245,13 +245,15 @@
/// Collect all of the definitions in a given crate into a CrateDefMap
/// Modules which are not a part of the module hierarchy starting with
/// the root module, will be ignored.
#[allow(clippy::too_many_arguments)]
pub fn collect_crate_and_dependencies(
mut def_map: CrateDefMap,
context: &mut Context,
ast: SortedModule,
root_file_id: FileId,
debug_comptime_in_file: Option<&str>,
enable_arithmetic_generics: bool,
error_on_unused_imports: bool,
macro_processors: &[&dyn MacroProcessor],
) -> Vec<(CompilationError, FileId)> {
let mut errors: Vec<(CompilationError, FileId)> = vec![];
Expand All @@ -265,11 +267,13 @@
let crate_graph = &context.crate_graph[crate_id];

for dep in crate_graph.dependencies.clone() {
let error_on_unused_imports = false;
errors.extend(CrateDefMap::collect_defs(
dep.crate_id,
context,
debug_comptime_in_file,
enable_arithmetic_generics,
error_on_unused_imports,
macro_processors,
));

Expand All @@ -282,8 +286,8 @@
def_map.extern_prelude.insert(dep.as_name(), module_id);

let location = dep_def_map[dep_def_root].location;
let attriutes = ModuleAttributes { name: dep.as_name(), location, parent: None };

Check warning on line 289 in compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs

View workflow job for this annotation

GitHub Actions / Code

Unknown word (attriutes)
context.def_interner.add_module_attributes(module_id, attriutes);

Check warning on line 290 in compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs

View workflow job for this annotation

GitHub Actions / Code

Unknown word (attriutes)
}

// At this point, all dependencies are resolved and type checked.
Expand Down Expand Up @@ -413,8 +417,26 @@
);
}

if error_on_unused_imports {
Self::check_unused_imports(context, crate_id, &mut errors);
}

errors
}

fn check_unused_imports(
context: &Context,
crate_id: CrateId,
errors: &mut Vec<(CompilationError, FileId)>,
) {
errors.extend(context.def_maps[&crate_id].modules().iter().flat_map(|(_, module)| {
module.unused_imports().iter().map(|ident| {
let ident = ident.clone();
let error = CompilationError::ResolverError(ResolverError::UnusedImport { ident });
(error, module.location.file)
})
}));
}
}

fn add_import_reference(
Expand Down
3 changes: 3 additions & 0 deletions compiler/noirc_frontend/src/hir/def_map/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,7 @@ impl CrateDefMap {
context: &mut Context,
debug_comptime_in_file: Option<&str>,
enable_arithmetic_generics: bool,
error_on_unused_imports: bool,
macro_processors: &[&dyn MacroProcessor],
) -> Vec<(CompilationError, FileId)> {
// Check if this Crate has already been compiled
Expand Down Expand Up @@ -127,12 +128,14 @@ impl CrateDefMap {
root_file_id,
debug_comptime_in_file,
enable_arithmetic_generics,
error_on_unused_imports,
macro_processors,
));

errors.extend(
parsing_errors.iter().map(|e| (e.clone().into(), root_file_id)).collect::<Vec<_>>(),
);

errors
}

Expand Down
22 changes: 21 additions & 1 deletion compiler/noirc_frontend/src/hir/def_map/module_data.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use std::collections::HashMap;
use std::collections::{HashMap, HashSet};

use noirc_errors::Location;

Expand All @@ -24,6 +24,10 @@ pub struct ModuleData {

/// True if this module is a `contract Foo { ... }` module containing contract functions
pub is_contract: bool,

/// List of all unused imports. Each time something is imported into this module it's added
/// to this set. When it's used, it's removed. At the end of the program only unused imports remain.
unused_imports: HashSet<Ident>,
}

impl ModuleData {
Expand All @@ -35,6 +39,7 @@ impl ModuleData {
definitions: ItemScope::default(),
location,
is_contract,
unused_imports: HashSet::new(),
}
}

Expand Down Expand Up @@ -121,6 +126,11 @@ impl ModuleData {
id: ModuleDefId,
is_prelude: bool,
) -> Result<(), (Ident, Ident)> {
// Empty spans could come from implicitly injected imports, and we don't want to track those
if name.span().start() < name.span().end() {
self.unused_imports.insert(name.clone());
}

self.scope.add_item_to_namespace(name, ItemVisibility::Public, id, None, is_prelude)
}

Expand All @@ -137,4 +147,14 @@ impl ModuleData {
pub fn value_definitions(&self) -> impl Iterator<Item = ModuleDefId> + '_ {
self.definitions.values().values().flat_map(|a| a.values().map(|(id, _, _)| *id))
}

/// Marks an ident as being used by an import.
pub fn use_import(&mut self, ident: &Ident) {
self.unused_imports.remove(ident);
}

/// Returns the list of all unused imports at this moment.
pub fn unused_imports(&self) -> &HashSet<Ident> {
&self.unused_imports
}
}
11 changes: 11 additions & 0 deletions compiler/noirc_frontend/src/hir/resolution/errors.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,8 @@ pub enum ResolverError {
DuplicateDefinition { name: String, first_span: Span, second_span: Span },
#[error("Unused variable")]
UnusedVariable { ident: Ident },
#[error("Unused import")]
UnusedImport { ident: Ident },
#[error("Could not find variable in this scope")]
VariableNotDeclared { name: String, span: Span },
#[error("path is not an identifier")]
Expand Down Expand Up @@ -152,6 +154,15 @@ impl<'a> From<&'a ResolverError> for Diagnostic {
ident.span(),
)
}
ResolverError::UnusedImport { ident } => {
let name = &ident.0.contents;

Diagnostic::simple_warning(
format!("unused import {name}"),
"unused import ".to_string(),
ident.span(),
)
}
ResolverError::VariableNotDeclared { name, span } => Diagnostic::simple_error(
format!("cannot find `{name}` in this scope "),
"not found in this scope".to_string(),
Expand Down
Loading
Loading