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: mod.nr entrypoint #5039

Merged
merged 51 commits into from
Jul 1, 2024
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
0af1f6f
wip
michaeljklein May 16, 2024
0a682aa
feat: add skip_prelude hidden cli option, skip inject_prelude when true
michaeljklein May 16, 2024
0f682a3
cargo fmt / clippy
michaeljklein May 16, 2024
a5c4c90
add missing parameter
michaeljklein May 16, 2024
ea02811
wip debugging
michaeljklein May 16, 2024
8ef80ae
Merge branch 'michaeljklein/no-stdlib-cli' into michaeljklein/mod-nr-…
michaeljklein May 16, 2024
fd76bd8
wip debugging: find_module doesn't appear to get called for import?
michaeljklein May 16, 2024
e35a630
debugging (def maps)
michaeljklein May 16, 2024
f214832
wip debugging
michaeljklein May 16, 2024
1acbca1
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
michaeljklein Jun 10, 2024
fa75c8b
debugging cleanup
michaeljklein Jun 10, 2024
414e765
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
michaeljklein Jun 10, 2024
ef4b60c
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
TomAFrench Jun 11, 2024
3357176
chore: fmt
TomAFrench Jun 11, 2024
8407360
chore: fix compilation
TomAFrench Jun 11, 2024
c48932f
add missing "mod foo;" declarations: tests passing
michaeljklein Jun 11, 2024
3ff4884
nargo fmt, add error for overlapping paths, restrict 'main' in path, …
michaeljklein Jun 11, 2024
e6dbe62
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
michaeljklein Jun 20, 2024
98cc4dd
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
TomAFrench Jun 25, 2024
239ecc1
Update compiler/fm/src/file_map.rs
michaeljklein Jun 27, 2024
ccd8576
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
michaeljklein Jun 27, 2024
5acff7f
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
asterite Jun 28, 2024
71ed978
Make things compile again
asterite Jun 28, 2024
0f565f1
Remove the `skip_prelude` option
asterite Jun 28, 2024
d99d7f1
No need for name_to_id to be pub
asterite Jun 28, 2024
ff6f7e0
Pass reference to Ident, only clone when needed
asterite Jun 28, 2024
5559bc9
No need to create files when dealing with FileManager
asterite Jun 28, 2024
e13f443
Simplify tests a bit
asterite Jun 28, 2024
70e880f
Add a few more tests, and fix the overlapping_lib_and_mod program tha…
asterite Jun 28, 2024
80291e2
Remove maybe wrong compile failure case
asterite Jun 28, 2024
e1601bb
More tests and some code cleanup
asterite Jun 28, 2024
032e973
Undo unrelated changes
asterite Jun 28, 2024
7232716
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
asterite Jun 28, 2024
8e86139
Get the logic right
asterite Jun 28, 2024
7dfb94d
Simplify a bit more
asterite Jun 28, 2024
4046586
clippy
asterite Jun 28, 2024
bb5e890
Document `mod.nr`
asterite Jul 1, 2024
b60b49c
chore: shorten test names
TomAFrench Jul 1, 2024
a71f6e5
Delete test_programs/compile_failure/invalid_mod_mod_path/Prover.toml
TomAFrench Jul 1, 2024
9bf5a0e
Delete test_programs/compile_failure/overlapping_mod/Prover.toml
TomAFrench Jul 1, 2024
461a724
Delete test_programs/compile_success_empty/mod_nr_entrypoint/Prover.toml
TomAFrench Jul 1, 2024
9a5c713
Delete test_programs/compile_failure/invalid_main_sub_lib/Prover.toml
TomAFrench Jul 1, 2024
dff7ade
chore: add failing test case
TomAFrench Jul 1, 2024
454daa9
Update compiler/noirc_frontend/src/hir/def_collector/dc_mod.rs
TomAFrench Jul 1, 2024
46712a1
chore: rename tests
TomAFrench Jul 1, 2024
d94ace3
Merge branch 'master' into michaeljklein/mod-nr-entrypoint
asterite Jul 1, 2024
621bdf5
Restore program that fails to compile
asterite Jul 1, 2024
9656af7
Tests that are easier to write and read
asterite Jul 1, 2024
7665a5b
Merge match branches
asterite Jul 1, 2024
254a967
clippy
asterite Jul 1, 2024
d98751e
clippy
asterite Jul 1, 2024
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
3 changes: 2 additions & 1 deletion compiler/fm/src/file_map.rs
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,8 @@ impl From<&PathBuf> for PathString {
#[derive(Debug, Clone)]
pub struct FileMap {
files: SimpleFiles<PathString, String>,
name_to_id: HashMap<PathString, FileId>,
// TODO remove before PR
asterite marked this conversation as resolved.
Show resolved Hide resolved
pub name_to_id: HashMap<PathString, FileId>,
michaeljklein marked this conversation as resolved.
Show resolved Hide resolved
}

// XXX: Note that we derive Default here due to ModuleOrigin requiring us to set a FileId
Expand Down
10 changes: 9 additions & 1 deletion compiler/noirc_driver/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,10 @@ pub struct CompileOptions {
#[arg(long, hide = true)]
pub use_elaborator: bool,

/// Skip loading the prelude (stdlib)
#[arg(long, hide = true)]
pub skip_prelude: bool,

/// Outputs the paths to any modified artifacts
#[arg(long, hide = true)]
pub show_artifact_paths: bool,
Expand Down Expand Up @@ -258,12 +262,14 @@ pub fn check_crate(
deny_warnings: bool,
disable_macros: bool,
use_elaborator: bool,
skip_prelude: bool,
) -> CompilationResult<()> {
let macros: &[&dyn MacroProcessor] =
if disable_macros { &[] } else { &[&aztec_macros::AztecMacro as &dyn MacroProcessor] };

let mut errors = vec![];
let diagnostics = CrateDefMap::collect_defs(crate_id, context, use_elaborator, macros);
let diagnostics =
CrateDefMap::collect_defs(crate_id, context, use_elaborator, skip_prelude, macros);
errors.extend(diagnostics.into_iter().map(|(error, file_id)| {
let diagnostic = CustomDiagnostic::from(&error);
diagnostic.in_file(file_id)
Expand Down Expand Up @@ -301,6 +307,7 @@ pub fn compile_main(
options.deny_warnings,
options.disable_macros,
options.use_elaborator,
options.skip_prelude,
)?;

let main = context.get_main_function(&crate_id).ok_or_else(|| {
Expand Down Expand Up @@ -342,6 +349,7 @@ pub fn compile_contract(
options.deny_warnings,
options.disable_macros,
options.use_elaborator,
options.skip_prelude,
)?;

// TODO: We probably want to error if contracts is empty
Expand Down
2 changes: 1 addition & 1 deletion compiler/noirc_driver/tests/stdlib_warnings.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ fn stdlib_does_not_produce_constant_warnings() -> Result<(), ErrorsAndWarnings>
let root_crate_id = prepare_crate(&mut context, file_name);

let ((), warnings) =
noirc_driver::check_crate(&mut context, root_crate_id, false, false, false)?;
noirc_driver::check_crate(&mut context, root_crate_id, false, false, false, false)?;

assert_eq!(warnings, Vec::new(), "stdlib is producing warnings");

Expand Down
15 changes: 12 additions & 3 deletions compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -257,6 +257,7 @@ impl DefCollector {
ast: SortedModule,
root_file_id: FileId,
use_elaborator: bool,
skip_prelude: bool,
macro_processors: &[&dyn MacroProcessor],
) -> Vec<(CompilationError, FileId)> {
let mut errors: Vec<(CompilationError, FileId)> = vec![];
Expand All @@ -274,6 +275,7 @@ impl DefCollector {
dep.crate_id,
context,
use_elaborator,
skip_prelude,
macro_processors,
));

Expand Down Expand Up @@ -308,9 +310,16 @@ impl DefCollector {
// Add the current crate to the collection of DefMaps
context.def_maps.insert(crate_id, def_collector.def_map);

inject_prelude(crate_id, context, crate_root, &mut def_collector.imports);
for submodule in submodules {
inject_prelude(crate_id, context, LocalModuleId(submodule), &mut def_collector.imports);
if !skip_prelude {
inject_prelude(crate_id, context, crate_root, &mut def_collector.imports);
for submodule in submodules {
inject_prelude(
crate_id,
context,
LocalModuleId(submodule),
&mut def_collector.imports,
);
}
}

// Resolve unresolved imports collected from the crate, one by one.
Expand Down
86 changes: 51 additions & 35 deletions compiler/noirc_frontend/src/hir/def_collector/dc_mod.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use std::{collections::HashMap, path::Path, vec};
use std::{collections::HashMap, ffi::OsStr, vec};

use acvm::{AcirField, FieldElement};
use fm::{FileId, FileManager, FILE_EXTENSION};
Expand Down Expand Up @@ -573,12 +573,9 @@ impl<'a> ModCollector<'a> {
) -> Vec<(CompilationError, FileId)> {
let mut errors: Vec<(CompilationError, FileId)> = vec![];
let child_file_id =
match find_module(&context.file_manager, self.file_id, &mod_decl.ident.0.contents) {
match find_module(&context.file_manager, self.file_id, mod_decl.ident.clone()) {
Ok(child_file_id) => child_file_id,
Err(expected_path) => {
let mod_name = mod_decl.ident.clone();
let err =
DefCollectorErrorKind::UnresolvedModuleDecl { mod_name, expected_path };
Err(err) => {
errors.push((err.into(), self.file_id));
return errors;
}
Expand Down Expand Up @@ -699,8 +696,8 @@ impl<'a> ModCollector<'a> {
fn find_module(
file_manager: &FileManager,
anchor: FileId,
mod_name: &str,
) -> Result<FileId, String> {
mod_name: Ident,
) -> Result<FileId, DefCollectorErrorKind> {
let anchor_path = file_manager
.path(anchor)
.expect("File must exist in file manager in order for us to be resolving its imports.")
Expand All @@ -709,34 +706,53 @@ fn find_module(

// if `anchor` is a `main.nr`, `lib.nr`, `mod.nr` or `{mod_name}.nr`, we check siblings of
// the anchor at `base/mod_name.nr`.
let candidate = if should_check_siblings_for_module(&anchor_path, anchor_dir) {
anchor_dir.join(format!("{mod_name}.{FILE_EXTENSION}"))
} else {
// Otherwise, we check for children of the anchor at `base/anchor/mod_name.nr`
anchor_path.join(format!("{mod_name}.{FILE_EXTENSION}"))
};

file_manager
.name_to_id(candidate.clone())
.ok_or_else(|| candidate.as_os_str().to_string_lossy().to_string())
}
let mod_name_str = mod_name.0.contents.clone();
let mod_nr_candidate = anchor_dir.join(&mod_name_str).join(format!("mod.{FILE_EXTENSION}"));
let parent_candidate = anchor_dir.join(format!("{mod_name_str}.{FILE_EXTENSION}"));
let child_candidate = anchor_path.join(format!("{mod_name_str}.{FILE_EXTENSION}"));

let mod_nr_result = file_manager
.name_to_id(mod_nr_candidate.clone())
.ok_or_else(|| mod_nr_candidate.as_os_str().to_string_lossy().to_string());
let parent_result = file_manager
.name_to_id(parent_candidate.clone())
.ok_or_else(|| parent_candidate.as_os_str().to_string_lossy().to_string());

let mut path_results = vec![mod_nr_result, parent_result];
let anchor_path_suffix = anchor_path
.as_path()
.file_name()
.unwrap_or_else(|| OsStr::new(""))
.to_string_lossy()
.to_string();
if anchor_path_suffix != "main" {
let child_result = file_manager
.name_to_id(child_candidate.clone())
.ok_or_else(|| child_candidate.as_os_str().to_string_lossy().to_string());
path_results.push(child_result);
}

/// Returns true if a module's child modules are expected to be in the same directory.
/// Returns false if they are expected to be in a subdirectory matching the name of the module.
fn should_check_siblings_for_module(module_path: &Path, parent_path: &Path) -> bool {
if let Some(filename) = module_path.file_stem() {
// This check also means a `main.nr` or `lib.nr` file outside of the crate root would
// check its same directory for child modules instead of a subdirectory. Should we prohibit
// `main.nr` and `lib.nr` files outside of the crate root?
filename == "main"
|| filename == "lib"
|| filename == "mod"
|| Some(filename) == parent_path.file_stem()
} else {
// If there's no filename, we arbitrarily return true.
// Alternatively, we could panic, but this is left to a different step where we
// ideally have some source location to issue an error.
true
let found_paths: Vec<_> = path_results.into_iter().flat_map(|result| result.ok()).collect();
match found_paths.len() {
0 => {
let expected_path = parent_candidate.as_os_str().to_string_lossy().to_string();
Err(DefCollectorErrorKind::UnresolvedModuleDecl { mod_name, expected_path })
}
1 => Ok(found_paths[0]),
_ => {
let overlapping_paths: Vec<_> = found_paths
.into_iter()
.map(|found_path| {
file_manager
.path(found_path)
.expect("all modules to be in the file manager")
.as_os_str()
.to_string_lossy()
.to_string()
})
.collect();
Err(DefCollectorErrorKind::OverlappingModuleDecls { mod_name, overlapping_paths })
}
}
}

Expand Down
13 changes: 13 additions & 0 deletions compiler/noirc_frontend/src/hir/def_collector/errors.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ pub enum DefCollectorErrorKind {
Duplicate { typ: DuplicateType, first_def: Ident, second_def: Ident },
#[error("unresolved import")]
UnresolvedModuleDecl { mod_name: Ident, expected_path: String },
#[error("overlapping imports")]
OverlappingModuleDecls { mod_name: Ident, overlapping_paths: Vec<String> },
#[error("path resolution error")]
PathResolutionError(PathResolutionError),
#[error("Non-struct type used in impl")]
Expand Down Expand Up @@ -129,6 +131,17 @@ impl<'a> From<&'a DefCollectorErrorKind> for Diagnostic {
span,
)
}
DefCollectorErrorKind::OverlappingModuleDecls { mod_name, overlapping_paths } => {
let span = mod_name.0.span();
let mod_name = &mod_name.0.contents;
let overlapping_paths: String = overlapping_paths.iter().map(|overlapping_path| overlapping_path.to_owned() + "\n").collect();

Diagnostic::simple_error(
format!("Overlapping modules `{mod_name}` at paths:\n`{overlapping_paths}`"),
String::new(),
span,
)
}
DefCollectorErrorKind::PathResolutionError(error) => error.into(),
DefCollectorErrorKind::NonStructTypeInImpl { span } => Diagnostic::simple_error(
"Non-struct type used in impl".into(),
Expand Down
2 changes: 2 additions & 0 deletions compiler/noirc_frontend/src/hir/def_map/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ impl CrateDefMap {
crate_id: CrateId,
context: &mut Context,
use_elaborator: bool,
skip_prelude: bool,
macro_processors: &[&dyn MacroProcessor],
) -> Vec<(CompilationError, FileId)> {
// Check if this Crate has already been compiled
Expand Down Expand Up @@ -123,6 +124,7 @@ impl CrateDefMap {
ast,
root_file_id,
use_elaborator,
skip_prelude,
macro_processors,
));

Expand Down
10 changes: 10 additions & 0 deletions compiler/noirc_frontend/src/hir/resolution/import.rs
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,9 @@ pub fn resolve_import(
import_directive: &ImportDirective,
def_maps: &BTreeMap<CrateId, CrateDefMap>,
) -> Result<ResolvedImport, PathResolutionError> {
// // TODO cleanup
asterite marked this conversation as resolved.
Show resolved Hide resolved
// dbg!("resolve_import", crate_id, import_directive, def_maps.keys().collect::<Vec<_>>());

let allow_contracts =
allow_referencing_contracts(def_maps, crate_id, import_directive.module_id);

Expand All @@ -98,6 +101,9 @@ pub fn resolve_import(
mut error,
} = resolve_path_to_ns(import_directive, crate_id, crate_id, def_maps, allow_contracts)?;

// TODO cleanup
// dbg!("resolve_import: err", &error);

let name = resolve_path_name(import_directive);

let visibility = resolved_namespace
Expand Down Expand Up @@ -147,6 +153,9 @@ fn resolve_path_to_ns(
let import_path = &import_directive.path.segments;
let def_map = &def_maps[&crate_id];

// TODO cleanup
// dbg!("resolve_path_to_ns", &import_path);

match import_directive.path.kind {
crate::ast::PathKind::Crate => {
// Resolve from the root of the crate
Expand Down Expand Up @@ -221,6 +230,7 @@ fn resolve_name_in_module(
}

let first_segment = import_path.first().expect("ice: could not fetch first segment");

let mut current_ns = current_mod.find_name(first_segment);
if current_ns.is_none() {
return Err(PathResolutionError::Unresolved(first_segment.clone()));
Expand Down
1 change: 1 addition & 0 deletions compiler/noirc_frontend/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ pub(crate) fn get_program(src: &str) -> (ParsedModule, Context, Vec<(Compilation
program.clone().into_sorted(),
root_file_id,
false,
false,
&[], // No macro processors
));
}
Expand Down
7 changes: 7 additions & 0 deletions test_programs/compile_failure/invalid_main_sub_lib/Nargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
[package]
name = "invalid_main_sub_lib"
type = "bin"
authors = [""]
compiler_version = ">=0.30.0"

[dependencies]
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
mod lib;

use crate::lib::foo;

fn main() {}
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
pub fn foo() -> bool {
true
}
7 changes: 7 additions & 0 deletions test_programs/compile_failure/invalid_mod_mod_path/Nargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
[package]
name = "invalid_mod_mod_path"
type = "bin"
authors = [""]
compiler_version = ">=0.30.0"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
x = ""
y = ""
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
pub fn foo() -> bool {
true
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
mod crate::mod;

fn main() {
}
3 changes: 3 additions & 0 deletions test_programs/compile_failure/invalid_mod_mod_path/src/mod.nr
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
pub foo() -> bool {
true
}
TomAFrench marked this conversation as resolved.
Outdated
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
[package]
name = "overlapping_dep_and_mod"
type = "bin"
authors = [""]
compiler_version = ">=0.30.0"

[dependencies]
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
pub fn foo() -> bool {
true
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
pub fn foo() -> bool {
true
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
mod lib;

use crate::lib::foo;


fn main() { }
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
[package]
name = "overlapping_lib_and_mod"
type = "bin"
authors = [""]
compiler_version = ">=0.30.0"

[dependencies]
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
pub fn bar() -> bool {
true
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
mod foo;

use foo::bar;

fn main() {
assert(bar());
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
pub fn bar() -> bool {
true
}

Loading
Loading