From 567559de78593e6217a728f1c47875d00c61de86 Mon Sep 17 00:00:00 2001 From: sezna Date: Wed, 3 Jul 2024 18:48:26 -0700 Subject: [PATCH 1/7] stop auto-closing single quotes --- Cargo.lock | 1 - petr-playground/index.js | 4 ++-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 640f030..5071eb2 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1027,7 +1027,6 @@ dependencies = [ name = "petr-api" version = "0.1.0" dependencies = [ - "clap 4.5.8", "miette", "petr-fmt", "petr-ir", diff --git a/petr-playground/index.js b/petr-playground/index.js index 95c3d56..f8df961 100644 --- a/petr-playground/index.js +++ b/petr-playground/index.js @@ -31,13 +31,13 @@ monaco.languages.setLanguageConfiguration("petr", { autoClosingPairs: [ { open: '[', close: ']' }, { open: '(', close: ')' }, - { open: "'", close: "'" } + { open: """, close: """ } ], surroundingPairs: [ { open: '{', close: '}' }, { open: '[', close: ']' }, { open: '(', close: ')' }, - { open: "'", close: "'" } + { open: """, close: """ } ] }) From 3d739a543c062943c005bdfb8cc6bc1ce141c6a7 Mon Sep 17 00:00:00 2001 From: sezna Date: Thu, 4 Jul 2024 05:19:48 -0700 Subject: [PATCH 2/7] wip --- Cargo.lock | 5 +++++ Cargo.toml | 2 +- pete/src/main.rs | 34 +++++++++++++++++----------------- petr-ast/src/dependency.rs | 14 ++++++++++++++ petr-ast/src/lib.rs | 1 + petr-bind/src/binder.rs | 19 +++++++++---------- petr-bind/src/lib.rs | 1 + petr-playground/Cargo.toml | 1 + petr-playground/index.js | 4 ++-- petr-playground/src/lib.rs | 5 +++++ petr-resolve/src/lib.rs | 9 ++------- petr-resolve/src/resolver.rs | 8 +------- stdlib/Cargo.toml | 6 ++++++ stdlib/src/io.petr | 3 +++ stdlib/src/lib.rs | 9 +++++++++ stdlib/src/mem.petr | 6 ++++++ stdlib/src/ops.petr | 7 +++++++ 17 files changed, 90 insertions(+), 44 deletions(-) create mode 100644 petr-ast/src/dependency.rs create mode 100644 stdlib/Cargo.toml create mode 100644 stdlib/src/io.petr create mode 100644 stdlib/src/lib.rs create mode 100644 stdlib/src/mem.petr create mode 100644 stdlib/src/ops.petr diff --git a/Cargo.lock b/Cargo.lock index 5071eb2..a5c9b2d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1126,6 +1126,7 @@ name = "petr-playground" version = "0.1.0" dependencies = [ "petr-api", + "stdlib", "wasm-bindgen", ] @@ -1361,6 +1362,10 @@ version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3" +[[package]] +name = "stdlib" +version = "0.1.0" + [[package]] name = "strsim" version = "0.8.0" diff --git a/Cargo.toml b/Cargo.toml index 8a4ccc5..67443ed 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -16,4 +16,4 @@ members = [ "petr-profiling", "petr-api", "petr-playground" -] +, "stdlib"] diff --git a/pete/src/main.rs b/pete/src/main.rs index 853f072..172f403 100644 --- a/pete/src/main.rs +++ b/pete/src/main.rs @@ -257,25 +257,25 @@ pub fn load_project_and_dependencies(path: &Path) -> Result<(petr_pkg::Lockfile, Ok((lockfile, files, build_plan)) } -pub fn load_files(path: &Path) -> Vec<(PathBuf, String)> { - let mut buf = Vec::new(); - - fn read_petr_files( - dir: &PathBuf, - buf: &mut Vec<(PathBuf, String)>, - ) { - let entries = fs::read_dir(dir).expect("Failed to read directory"); - for entry in entries { - let entry = entry.expect("Failed to read directory entry"); - let path = entry.path(); - if path.is_dir() { - read_petr_files(&path, buf); - } else if path.extension().and_then(|s| s.to_str()) == Some("pt") { - let source = fs::read_to_string(&path).expect("Failed to read file"); - buf.push((path, source)); - } +fn read_petr_files( + dir: &PathBuf, + buf: &mut Vec<(PathBuf, String)>, +) { + let entries = fs::read_dir(dir).expect("Failed to read directory"); + for entry in entries { + let entry = entry.expect("Failed to read directory entry"); + let path = entry.path(); + if path.is_dir() { + read_petr_files(&path, buf); + } else if path.extension().and_then(|s| s.to_str()) == Some("pt") { + let source = fs::read_to_string(&path).expect("Failed to read file"); + buf.push((path, source)); } } +} + +pub fn load_files(path: &Path) -> Vec<(PathBuf, String)> { + let mut buf = Vec::new(); read_petr_files(&path.join("src"), &mut buf); buf diff --git a/petr-ast/src/dependency.rs b/petr-ast/src/dependency.rs new file mode 100644 index 0000000..d8fe390 --- /dev/null +++ b/petr-ast/src/dependency.rs @@ -0,0 +1,14 @@ +use crate::Ast; + +/// Describes the AST of a dependency package +pub struct Dependency { + /// The unique key that identifies this source package, agnostic of the alias given to it in + /// the manifest file + pub key: String, + /// The name of the package as given in the manifest file + pub name: petr_utils::Identifier, + /// The keys of any packages that this package depends on + pub dependencies: Vec, + /// The AST of the source code of this package + pub ast: Ast, +} diff --git a/petr-ast/src/lib.rs b/petr-ast/src/lib.rs index 635b5d0..90039a9 100644 --- a/petr-ast/src/lib.rs +++ b/petr-ast/src/lib.rs @@ -1,5 +1,6 @@ mod ast; mod comments; +pub mod dependency; mod pretty_print; pub use ast::*; pub use comments::*; diff --git a/petr-bind/src/binder.rs b/petr-bind/src/binder.rs index 0823ad2..7111ae0 100644 --- a/petr-bind/src/binder.rs +++ b/petr-bind/src/binder.rs @@ -1,6 +1,6 @@ use std::collections::BTreeMap; -use petr_ast::{Ast, Binding, ExprId, Expression, FunctionDeclaration, Ty, TypeDeclaration}; +use petr_ast::{dependency::Dependency, Ast, Binding, ExprId, Expression, FunctionDeclaration, Ty, TypeDeclaration}; use petr_utils::{idx_map_key, Identifier, IndexMap, Path, SymbolId}; // TODO: // - i don't know if type cons needs a scope. Might be good to remove that. @@ -336,18 +336,17 @@ impl Binder { pub fn from_ast_and_deps( ast: &Ast, - // TODO better type here - dependencies: Vec<( - /* Key */ String, - /*Name from manifest*/ Identifier, - /*Things this depends on*/ Vec, - Ast, - )>, + dependencies: Vec, ) -> Self { let mut binder = Self::new(); - for dependency in dependencies { - let (_key, name, _depends_on, dep_ast) = dependency; + for Dependency { + key, + name, + dependencies: depends_on, + ast: dep_ast, + } in dependencies + { let dep_scope = binder.create_scope_from_path(&Path::new(vec![name])); binder.with_specified_scope(dep_scope, |binder, _scope_id| { for module in dep_ast.modules { diff --git a/petr-bind/src/lib.rs b/petr-bind/src/lib.rs index 46557b2..96cca83 100644 --- a/petr-bind/src/lib.rs +++ b/petr-bind/src/lib.rs @@ -3,5 +3,6 @@ //! scopes define. The resolver is then able to do scope-aware name resolution in the next step. pub use binder::{Bind, Binder, BindingId, FunctionId, Item, ModuleId, Scope, ScopeId, ScopeKind, TypeId}; +pub use petr_ast::dependency::Dependency; mod binder; mod impls; diff --git a/petr-playground/Cargo.toml b/petr-playground/Cargo.toml index a7b440c..2256b9c 100644 --- a/petr-playground/Cargo.toml +++ b/petr-playground/Cargo.toml @@ -11,3 +11,4 @@ crate-type = ["cdylib"] [dependencies] wasm-bindgen = "0.2.92" petr-api = { path = "../petr-api", default-features = false, features = ["no_std"] } +stdlib = { path = "../stdlib" } diff --git a/petr-playground/index.js b/petr-playground/index.js index f8df961..363ca40 100644 --- a/petr-playground/index.js +++ b/petr-playground/index.js @@ -31,13 +31,13 @@ monaco.languages.setLanguageConfiguration("petr", { autoClosingPairs: [ { open: '[', close: ']' }, { open: '(', close: ')' }, - { open: """, close: """ } + { open: '"', close: '"' } ], surroundingPairs: [ { open: '{', close: '}' }, { open: '[', close: ']' }, { open: '(', close: ')' }, - { open: """, close: """ } + { open: '"', close: '"' } ] }) diff --git a/petr-playground/src/lib.rs b/petr-playground/src/lib.rs index ee8dd97..b451698 100644 --- a/petr-playground/src/lib.rs +++ b/petr-playground/src/lib.rs @@ -56,6 +56,11 @@ fn compile_snippet(code: String) -> Result> { let dependencies = vec![]; let (ast, parse_errs, interner, source_map) = parser.into_result(); + // add the standard library to the sources + let parser = Parser::new_with_existing_interner_and_source_map(stdlib::stdlib(), interner, source_map); + let (ast, mut new_parse_errs, interner, source_map) = parser.into_result(); + parse_errs.append(&mut new_parse_errs); + // TODO after diagnostics are implemented for these errors, append them to the errors and // return them let (_resolution_errs, resolved) = resolve_symbols(ast, interner, dependencies); diff --git a/petr-resolve/src/lib.rs b/petr-resolve/src/lib.rs index d9d9da2..395857d 100644 --- a/petr-resolve/src/lib.rs +++ b/petr-resolve/src/lib.rs @@ -14,13 +14,8 @@ mod resolver; pub fn resolve_symbols( ast: petr_ast::Ast, interner: SymbolInterner, - // TODO better type here - dependencies: Vec<( - /* Key */ String, - /*Name from manifest*/ Identifier, - /*Things this depends on*/ Vec, - Ast, - )>, + // TODO refactor tuple into struct with named fields + dependencies: Vec, ) -> (Vec, QueryableResolvedItems) { let resolver = Resolver::new(ast, interner, dependencies); resolver.into_queryable() diff --git a/petr-resolve/src/resolver.rs b/petr-resolve/src/resolver.rs index 70ce95a..d4cc5b8 100644 --- a/petr-resolve/src/resolver.rs +++ b/petr-resolve/src/resolver.rs @@ -144,13 +144,7 @@ impl Resolver { pub fn new( ast: Ast, interner: SymbolInterner, - // TODO better type here - dependencies: Vec<( - /* Key */ String, - /*Name from manifest*/ Identifier, - /*Things this depends on*/ Vec, - Ast, - )>, + dependencies: Vec, ) -> Self { let binder = Binder::from_ast_and_deps(&ast, dependencies); let mut resolver = Self { diff --git a/stdlib/Cargo.toml b/stdlib/Cargo.toml new file mode 100644 index 0000000..c4c0e7f --- /dev/null +++ b/stdlib/Cargo.toml @@ -0,0 +1,6 @@ +[package] +name = "stdlib" +version = "0.1.0" +edition = "2021" + +[dependencies] diff --git a/stdlib/src/io.petr b/stdlib/src/io.petr new file mode 100644 index 0000000..7a02e2a --- /dev/null +++ b/stdlib/src/io.petr @@ -0,0 +1,3 @@ + +Function print(content in 'string) returns 'unit + @puts content diff --git a/stdlib/src/lib.rs b/stdlib/src/lib.rs new file mode 100644 index 0000000..49d3953 --- /dev/null +++ b/stdlib/src/lib.rs @@ -0,0 +1,9 @@ +/// returns all files in the standard library +pub fn stdlib() -> Vec<(&'static str, &'static str)> { + vec![ + ("ops.petr", include_str!("ops.petr")), + ("io.petr", include_str!("io.petr")), + ("mem.petr", include_str!("mem.petr")), + ] +} + diff --git a/stdlib/src/mem.petr b/stdlib/src/mem.petr new file mode 100644 index 0000000..8556f6d --- /dev/null +++ b/stdlib/src/mem.petr @@ -0,0 +1,6 @@ + +Type Ptr = Ptr 'int + +Function malloc(size in 'int) returns Ptr + let allocated = @malloc size + Ptr allocated diff --git a/stdlib/src/ops.petr b/stdlib/src/ops.petr new file mode 100644 index 0000000..cdee3f2 --- /dev/null +++ b/stdlib/src/ops.petr @@ -0,0 +1,7 @@ +Function add(lhs in 'int, rhs in 'int) returns 'int @add lhs, rhs + +Function sub(lhs in 'int, rhs in 'int) returns 'int @subtract lhs, rhs + +Function mult(lhs in 'int, rhs in 'int) returns 'int @multiply lhs, rhs + +Function div(lhs in 'int, rhs in 'int) returns 'int @divide lhs, rhs From 12e0c86e5847700340871c5dbb521ca5a3259756 Mon Sep 17 00:00:00 2001 From: sezna Date: Sat, 6 Jul 2024 08:47:46 -0700 Subject: [PATCH 3/7] add stdlib; fix module/scoping bug --- Cargo.lock | 1 + pete/src/main.rs | 9 ++++++++- petr-api/src/lib.rs | 14 +++++++++++++- petr-bind/src/binder.rs | 13 ++++++++++++- petr-playground/index.js | 1 - petr-playground/src/lib.rs | 4 ++-- petr-resolve/src/lib.rs | 1 + petr-resolve/src/resolver.rs | 21 +++++++++++++++++---- petr-vm/Cargo.toml | 7 ++++++- petr-vm/src/lib.rs | 28 ++++++++++++++++------------ stdlib/pete.toml | 8 ++++++++ stdlib/petr.lock | 1 + stdlib/src/{io.petr => io.pt} | 0 stdlib/src/lib.rs | 7 +------ stdlib/src/mem.petr | 6 ------ stdlib/src/{ops.petr => ops.pt} | 0 16 files changed, 86 insertions(+), 35 deletions(-) create mode 100644 stdlib/pete.toml create mode 100644 stdlib/petr.lock rename stdlib/src/{io.petr => io.pt} (100%) delete mode 100644 stdlib/src/mem.petr rename stdlib/src/{ops.petr => ops.pt} (100%) diff --git a/Cargo.lock b/Cargo.lock index a5c9b2d..532f46b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1183,6 +1183,7 @@ dependencies = [ "petr-resolve", "petr-typecheck", "petr-utils", + "stdlib", "thiserror", ] diff --git a/pete/src/main.rs b/pete/src/main.rs index 172f403..63dd4cc 100644 --- a/pete/src/main.rs +++ b/pete/src/main.rs @@ -7,6 +7,7 @@ use std::{ use clap::Parser as ClapParser; use petr_api::*; use petr_pkg::BuildPlan; +use petr_resolve::Dependency; use termcolor::{ColorChoice, ColorSpec, StandardStream, WriteColor}; pub mod error { @@ -189,7 +190,13 @@ pub fn compile( let name = Identifier { id: interner.insert(Rc::from(item.manifest.name)), }; - dependencies.push((item.key, name, item.depends_on, ast)); + + dependencies.push(Dependency { + key: item.key, + name, + dependencies: item.depends_on, + ast, + }); } timings.end("parse dependencies"); diff --git a/petr-api/src/lib.rs b/petr-api/src/lib.rs index 0a01d55..1c84334 100644 --- a/petr-api/src/lib.rs +++ b/petr-api/src/lib.rs @@ -1,6 +1,13 @@ //! Top-level API for the petr programming language. //! Exposes relevant APIs from all compiler stages and tooling. +#[cfg(not(feature = "no_std"))] +use std::{ + fs, + path::{Path, PathBuf}, + rc::Rc, +}; + pub use petr_fmt::{format_sources, FormatterConfig}; pub use petr_ir::Lowerer; pub use petr_parse::Parser; @@ -149,7 +156,12 @@ pub fn compile( let name = Identifier { id: interner.insert(Rc::from(item.manifest.name)), }; - dependencies.push((item.key, name, item.depends_on, ast)); + dependencies.push(petr_resolve::Dependency { + key: item.key, + name, + dependencies: item.depends_on, + ast, + }); } timings.end("parse dependencies"); diff --git a/petr-bind/src/binder.rs b/petr-bind/src/binder.rs index 7111ae0..dae5fb5 100644 --- a/petr-bind/src/binder.rs +++ b/petr-bind/src/binder.rs @@ -102,7 +102,11 @@ impl Scope { k: SymbolId, v: T, ) { - self.items.insert(k, v); + match self.items.insert(k, v) { + // TODO: error handling and/or shadowing rules for this + Some(item) => todo!("throw error for overriding symbol name {k}"), + None => (), + }; } pub fn parent(&self) -> Option { @@ -396,6 +400,13 @@ impl Binder { ) -> ScopeId { let mut current_scope_id = self.current_scope_id(); for segment in path.iter() { + // if this scope already exists, + // just use that pre-existing ID + if let Some(Item::Module(module_id)) = self.find_symbol_in_scope(segment.id, current_scope_id) { + current_scope_id = self.modules.get(*module_id).root_scope; + continue; + } + let next_scope = self.create_scope(ScopeKind::Module(*segment)); let module = Module { root_scope: next_scope, diff --git a/petr-playground/index.js b/petr-playground/index.js index 363ca40..15bd3af 100644 --- a/petr-playground/index.js +++ b/petr-playground/index.js @@ -4,7 +4,6 @@ import { run_snippet } from './pkg'; import * as monaco from 'monaco-editor'; // or import * as monaco from 'monaco-editor/esm/vs/editor/editor.api'; // if shipping only a subset of the features & languages is desired -// // config for petr as a custom language // // Register a new language diff --git a/petr-playground/src/lib.rs b/petr-playground/src/lib.rs index b451698..d20a780 100644 --- a/petr-playground/src/lib.rs +++ b/petr-playground/src/lib.rs @@ -55,10 +55,10 @@ fn compile_snippet(code: String) -> Result> { // bring it in to this repo let dependencies = vec![]; - let (ast, parse_errs, interner, source_map) = parser.into_result(); + let (ast, mut parse_errs, interner, source_map) = parser.into_result(); // add the standard library to the sources let parser = Parser::new_with_existing_interner_and_source_map(stdlib::stdlib(), interner, source_map); - let (ast, mut new_parse_errs, interner, source_map) = parser.into_result(); + let (dep_ast, mut new_parse_errs, interner, source_map) = parser.into_result(); parse_errs.append(&mut new_parse_errs); // TODO after diagnostics are implemented for these errors, append them to the errors and diff --git a/petr-resolve/src/lib.rs b/petr-resolve/src/lib.rs index 395857d..e50d452 100644 --- a/petr-resolve/src/lib.rs +++ b/petr-resolve/src/lib.rs @@ -3,6 +3,7 @@ use petr_ast::Ast; pub use petr_ast::{Intrinsic as IntrinsicName, Literal, Ty}; +pub use petr_bind::Dependency; use petr_utils::{Identifier, SymbolInterner}; pub use resolved::QueryableResolvedItems; use resolver::Resolver; diff --git a/petr-resolve/src/resolver.rs b/petr-resolve/src/resolver.rs index d4cc5b8..08a0a8e 100644 --- a/petr-resolve/src/resolver.rs +++ b/petr-resolve/src/resolver.rs @@ -4,7 +4,7 @@ use std::rc::Rc; use petr_ast::{Ast, Commented, Expression, FunctionDeclaration, FunctionParameter, OperatorExpression}; -use petr_bind::{Binder, FunctionId, Item, ScopeId, TypeId}; +use petr_bind::{Binder, Dependency, FunctionId, Item, ScopeId, TypeId}; use petr_utils::{Identifier, Path, SpannedItem, SymbolInterner}; use thiserror::Error; @@ -59,8 +59,13 @@ impl Resolve for petr_ast::Ty { petr_ast::Ty::Unit => Type::Unit, petr_ast::Ty::Named(name) => match binder.find_symbol_in_scope(name.id, scope_id) { Some(Item::Type(id)) => Type::Named(*id), - Some(_) => { - todo!("push error -- symbol is not type"); + Some(something) => { + let name = _resolver.interner.get(name.id); + // this makes sense, the type constructor is getting resolved instead of the ty + // find_symbol_in_scope could take in what it is looking for as a parameter, + // _or_ we could have a special case when a function body is just a type + // constructorjkkj + todo!("push error -- symbol {name} is not type, it is a {something:?}"); // return None; }, None => Type::Generic(*name), @@ -184,7 +189,15 @@ impl Resolver { }, Binding(a) => { let binding = binder.get_binding(*a); - let resolved_expr = binding.val.resolve(self, binder, scope_id).expect("TODO err"); + let resolved_expr = match binding.val.resolve(self, binder, scope_id) { + Some(o) => o, + None => { + // TODO i think this is incorrect + let name = self.interner.get(binding.name.id); + self.errs.push(ResolutionError::NotFound(name.to_string())); + return; + }, + }; self.resolved.bindings.insert( *a, crate::resolver::Binding { diff --git a/petr-vm/Cargo.toml b/petr-vm/Cargo.toml index 4290061..0abba4b 100644 --- a/petr-vm/Cargo.toml +++ b/petr-vm/Cargo.toml @@ -5,7 +5,7 @@ edition = "2021" [dependencies] petr-ir = { path = "../petr-ir" } -petr-utils = { path = "../petr-utils" } +petr-utils = { path = "../petr-utils", optional = true } thiserror = "1.0.61" [dev-dependencies] @@ -13,3 +13,8 @@ petr-parse = { path = "../petr-parse" } expect-test = "1.5.0" petr-resolve = { path = "../petr-resolve" } petr-typecheck = { path = "../petr-typecheck" } +stdlib = { path = "../stdlib" } + +[features] +debug = ["petr-utils/debug"] +default = ["dep:petr-utils"] diff --git a/petr-vm/src/lib.rs b/petr-vm/src/lib.rs index fb6f333..8db7efb 100644 --- a/petr-vm/src/lib.rs +++ b/petr-vm/src/lib.rs @@ -25,18 +25,10 @@ mod tests { expect: Expect, ) { let input = input.into(); - let parser = petr_parse::Parser::new(vec![ - ( - "std/ops.pt", - " - function add(lhs in 'int, rhs in 'int) returns 'int @add lhs, rhs - function sub(lhs in 'int, rhs in 'int) returns 'int @subtract lhs, rhs - function mul(lhs in 'int, rhs in 'int) returns 'int @multiply lhs, rhs - function div(lhs in 'int, rhs in 'int) returns 'int @divide lhs, rhs - ", - ), - ("test", &input), - ]); + let mut sources = stdlib::stdlib(); + dbg!(&sources); + sources.push(("test", &input)); + let parser = petr_parse::Parser::new(sources); let (ast, errs, interner, source_map) = parser.into_result(); if !errs.is_empty() { errs.into_iter().for_each(|err| eprintln!("{:?}", render_error(&source_map, err))); @@ -77,6 +69,18 @@ function main() returns 'int ~hi(42, 3) expect!["Value(42)"], ) } + #[test] + fn import_call() { + check( + r#" +import std.io.print + +function main() returns 'unit + ~print("hello, world!") + "#, + expect!["Value(0)"], + ) + } #[test] fn addition() { diff --git a/stdlib/pete.toml b/stdlib/pete.toml new file mode 100644 index 0000000..7a68f6e --- /dev/null +++ b/stdlib/pete.toml @@ -0,0 +1,8 @@ +author = "Alex Hansen " +license = "MIT" +name = "test_project" + +[formatter] + +[dependencies.std] +git = "https://github.com/sezna/petr-std" diff --git a/stdlib/petr.lock b/stdlib/petr.lock new file mode 100644 index 0000000..f572e7d --- /dev/null +++ b/stdlib/petr.lock @@ -0,0 +1 @@ +entries = [] diff --git a/stdlib/src/io.petr b/stdlib/src/io.pt similarity index 100% rename from stdlib/src/io.petr rename to stdlib/src/io.pt diff --git a/stdlib/src/lib.rs b/stdlib/src/lib.rs index 49d3953..7446c86 100644 --- a/stdlib/src/lib.rs +++ b/stdlib/src/lib.rs @@ -1,9 +1,4 @@ /// returns all files in the standard library pub fn stdlib() -> Vec<(&'static str, &'static str)> { - vec![ - ("ops.petr", include_str!("ops.petr")), - ("io.petr", include_str!("io.petr")), - ("mem.petr", include_str!("mem.petr")), - ] + vec![("std/ops.pt", include_str!("ops.pt")), ("std/io.pt", include_str!("io.pt"))] } - diff --git a/stdlib/src/mem.petr b/stdlib/src/mem.petr deleted file mode 100644 index 8556f6d..0000000 --- a/stdlib/src/mem.petr +++ /dev/null @@ -1,6 +0,0 @@ - -Type Ptr = Ptr 'int - -Function malloc(size in 'int) returns Ptr - let allocated = @malloc size - Ptr allocated diff --git a/stdlib/src/ops.petr b/stdlib/src/ops.pt similarity index 100% rename from stdlib/src/ops.petr rename to stdlib/src/ops.pt From bb119f454187b9cda2d54684a1a65b6558c56220 Mon Sep 17 00:00:00 2001 From: sezna Date: Sat, 6 Jul 2024 09:05:31 -0700 Subject: [PATCH 4/7] wip: clean up warnings, leave breadcrumb for next time --- petr-bind/src/binder.rs | 13 ++++++------- petr-playground/src/lib.rs | 3 +-- petr-resolve/src/lib.rs | 3 +-- 3 files changed, 8 insertions(+), 11 deletions(-) diff --git a/petr-bind/src/binder.rs b/petr-bind/src/binder.rs index dae5fb5..d7dbbcb 100644 --- a/petr-bind/src/binder.rs +++ b/petr-bind/src/binder.rs @@ -102,11 +102,10 @@ impl Scope { k: SymbolId, v: T, ) { - match self.items.insert(k, v) { - // TODO: error handling and/or shadowing rules for this - Some(item) => todo!("throw error for overriding symbol name {k}"), - None => (), - }; + // TODO: error handling and/or shadowing rules for this + if self.items.insert(k, v).is_some() { + todo!("throw error for overriding symbol name {k}") + } } pub fn parent(&self) -> Option { @@ -345,9 +344,9 @@ impl Binder { let mut binder = Self::new(); for Dependency { - key, + key: _, name, - dependencies: depends_on, + dependencies: _, ast: dep_ast, } in dependencies { diff --git a/petr-playground/src/lib.rs b/petr-playground/src/lib.rs index d20a780..b1bd21e 100644 --- a/petr-playground/src/lib.rs +++ b/petr-playground/src/lib.rs @@ -51,13 +51,12 @@ fn compile_snippet(code: String) -> Result> { let buf = vec![("snippet".to_string(), code)]; let mut errs = vec![]; let parser = Parser::new(buf); - // TODO include standard library in WASM compilation target and - // bring it in to this repo let dependencies = vec![]; let (ast, mut parse_errs, interner, source_map) = parser.into_result(); // add the standard library to the sources let parser = Parser::new_with_existing_interner_and_source_map(stdlib::stdlib(), interner, source_map); + todo!("dep ast should be included here"); let (dep_ast, mut new_parse_errs, interner, source_map) = parser.into_result(); parse_errs.append(&mut new_parse_errs); diff --git a/petr-resolve/src/lib.rs b/petr-resolve/src/lib.rs index e50d452..bd4ce4e 100644 --- a/petr-resolve/src/lib.rs +++ b/petr-resolve/src/lib.rs @@ -1,10 +1,9 @@ //! given bindings, fully resolve an AST //! This crate's job is to tee up the type checker for the next stage of compilation. -use petr_ast::Ast; pub use petr_ast::{Intrinsic as IntrinsicName, Literal, Ty}; pub use petr_bind::Dependency; -use petr_utils::{Identifier, SymbolInterner}; +use petr_utils::SymbolInterner; pub use resolved::QueryableResolvedItems; use resolver::Resolver; pub use resolver::{Expr, ExprKind, Function, FunctionCall, Intrinsic, ResolutionError, Type}; From ad78ede5a120d5db27ad9a6dccd09bf190093099 Mon Sep 17 00:00:00 2001 From: sezna Date: Sat, 6 Jul 2024 15:57:08 -0700 Subject: [PATCH 5/7] playground formatter works --- petr-api/src/lib.rs | 4 ++-- petr-fmt/src/lib.rs | 2 +- petr-playground/index.html | 1 + petr-playground/index.js | 21 ++++++++++++++++++- petr-playground/src/lib.rs | 42 ++++++++++++++++++++++++++++---------- petr-vm/src/lib.rs | 27 +++++++++++++++++++----- 6 files changed, 77 insertions(+), 20 deletions(-) diff --git a/petr-api/src/lib.rs b/petr-api/src/lib.rs index 1c84334..262e935 100644 --- a/petr-api/src/lib.rs +++ b/petr-api/src/lib.rs @@ -8,12 +8,12 @@ use std::{ rc::Rc, }; -pub use petr_fmt::{format_sources, FormatterConfig}; +pub use petr_fmt::{format_sources, Formattable, FormatterConfig, FormatterContext}; pub use petr_ir::Lowerer; pub use petr_parse::Parser; #[cfg(not(feature = "no_std"))] pub use petr_pkg::{manifest::find_manifest, BuildPlan}; -pub use petr_resolve::resolve_symbols; +pub use petr_resolve::{resolve_symbols, Dependency}; pub use petr_typecheck::type_check; pub use petr_utils::{render_error, Identifier, IndexMap, SourceId, SpannedItem}; pub use petr_vm::Vm; diff --git a/petr-fmt/src/lib.rs b/petr-fmt/src/lib.rs index 6d37be3..8b2cbf4 100644 --- a/petr-fmt/src/lib.rs +++ b/petr-fmt/src/lib.rs @@ -15,7 +15,7 @@ use std::{ pub use config::{FormatterConfig, FormatterConfigBuilder}; use constants::{CLOSE_COMMENT_STR, INDENTATION_CHARACTER, OPEN_COMMENT_STR}; -use ctx::FormatterContext; +pub use ctx::FormatterContext; use petr_ast::*; use petr_parse::Parser; use petr_utils::{render_error, PrettyPrint, SpannedItem}; diff --git a/petr-playground/index.html b/petr-playground/index.html index 06a3123..dabadb2 100644 --- a/petr-playground/index.html +++ b/petr-playground/index.html @@ -25,6 +25,7 @@ #output { /* monospace font, smallish font */ font-family: monospace; + white-space: pre; font-size: 0.8em; } diff --git a/petr-playground/index.js b/petr-playground/index.js index 15bd3af..a8b1647 100644 --- a/petr-playground/index.js +++ b/petr-playground/index.js @@ -1,4 +1,4 @@ -import { run_snippet } from './pkg'; +import { run_snippet, format } from './pkg'; import * as monaco from 'monaco-editor'; @@ -120,6 +120,12 @@ export function setOutputContent(content) { } window.setOutputContent = setOutputContent; +export function setCodeEditorContent(content) { + monaco.editor.getModels()[0].setValue(content); +} + +window.setCodeEditorContent= setCodeEditorContent; + // set on-clicks for the buttons document.getElementById('run').onclick = function() { // get the text content from the monaco instance @@ -136,3 +142,16 @@ document.getElementById('run').onclick = function() { return; }; } +document.getElementById('format').onclick = function() { + // get the text content from the monaco instance + let code = monaco.editor.getModels()[0].getValue(); + // run the code + // TODO: actually render the errors on the span in the diagnostics of monaco + try { format(code); } catch (e) { + // set the output to the diagnostics + // because an Err result from wasm becomes an exception + // might be good to not use Result for that reason + document.getElementById('output').innerHTML = e; + return; + }; +} diff --git a/petr-playground/src/lib.rs b/petr-playground/src/lib.rs index b1bd21e..838698e 100644 --- a/petr-playground/src/lib.rs +++ b/petr-playground/src/lib.rs @@ -2,15 +2,20 @@ //! Nothing fancy at all, could definitely be improved over time to support better error reporting, //! etc +use std::rc::Rc; + use wasm_bindgen::prelude::*; #[wasm_bindgen] extern "C" { #[wasm_bindgen(js_name = setOutputContent)] fn set_output_content(s: &str); + + #[wasm_bindgen(js_name = setCodeEditorContent)] + fn set_code_editor_content(s: &str); } -use petr_api::{render_error, resolve_symbols, type_check, FormatterConfig, Lowerer, Parser, Vm}; +use petr_api::{render_error, resolve_symbols, type_check, Dependency, Formattable, FormatterContext, Identifier, Lowerer, Parser, Vm}; #[wasm_bindgen] pub fn run_snippet(code: &str) { @@ -26,7 +31,7 @@ pub fn run_snippet(code: &str) { let (data, instructions) = lowerer.finalize(); let vm = Vm::new(instructions, data); - let result = match vm.run() { + let (result, _stack, logs) = match vm.run() { Ok(o) => o, Err(e) => { set_output_content(&format!("Runtime error: {:#?}", e)); @@ -34,7 +39,7 @@ pub fn run_snippet(code: &str) { }, }; - set_output_content(&format!("Result: {:#?}", result)); + set_output_content(&format!("Logs:
\t{}
Result:
\t{:#?}", logs.join("\n\t"), result.inner())); } fn errors_to_html(e: &[String]) -> String { @@ -51,14 +56,20 @@ fn compile_snippet(code: String) -> Result> { let buf = vec![("snippet".to_string(), code)]; let mut errs = vec![]; let parser = Parser::new(buf); - let dependencies = vec![]; let (ast, mut parse_errs, interner, source_map) = parser.into_result(); // add the standard library to the sources let parser = Parser::new_with_existing_interner_and_source_map(stdlib::stdlib(), interner, source_map); - todo!("dep ast should be included here"); - let (dep_ast, mut new_parse_errs, interner, source_map) = parser.into_result(); + let (dep_ast, mut new_parse_errs, mut interner, source_map) = parser.into_result(); parse_errs.append(&mut new_parse_errs); + let dependencies = vec![Dependency { + key: "stdlib".to_string(), + name: Identifier { + id: interner.insert(Rc::from("std")), + }, + dependencies: vec![], + ast: dep_ast, + }]; // TODO after diagnostics are implemented for these errors, append them to the errors and // return them @@ -75,9 +86,18 @@ fn compile_snippet(code: String) -> Result> { } } -pub fn format( - _code: String, - _config: FormatterConfig, -) -> Result { - todo!() +#[wasm_bindgen] +pub fn format(code: String) { + let parser = Parser::new(vec![("snippet".to_string(), code)]); + let (ast, errs, interner, source_map) = parser.into_result(); + if !errs.is_empty() { + let errs = errs + .into_iter() + .map(|e| format!("{:?}", render_error(&source_map, e))) + .collect::>(); + set_output_content(&errs.join("
")); + } + let mut ctx = FormatterContext::from_interner(interner).with_config(Default::default()); + let formatted_content = ast.line_length_aware_format(&mut ctx).render(); + set_code_editor_content(&formatted_content); } diff --git a/petr-vm/src/lib.rs b/petr-vm/src/lib.rs index 8db7efb..6908a9d 100644 --- a/petr-vm/src/lib.rs +++ b/petr-vm/src/lib.rs @@ -42,12 +42,18 @@ mod tests { let lowerer = Lowerer::new(type_checker); let (data, ir) = lowerer.finalize(); let vm = Vm::new(ir, data); - let (res, _stack) = match vm.run() { + let (res, _stack, logs) = match vm.run() { Ok(o) => o, Err(err) => panic!("vm returned error: {err:?}"), }; - let res = format!("{res:?}"); + let mut res = format!("{res:?}"); + + if !logs.is_empty() { + res.push_str("\n___LOGS___\n"); + + res.push_str(&logs.join("\n")); + } expect.assert_eq(&res); } @@ -164,6 +170,8 @@ function main() returns 'int pub struct Vm { state: VmState, instructions: IndexMap, + /// any messages that were logged during execution + stdout: Vec, } idx_map_key!(Register); @@ -188,6 +196,12 @@ impl Default for ProgramOffset { #[derive(Clone, Copy, Debug)] pub struct Value(u64); +impl Value { + pub fn inner(&self) -> u64 { + self.0 + } +} + #[derive(Debug, Error)] pub enum VmError { #[error("Function label not found when executing opcode {0}")] @@ -209,6 +223,8 @@ enum VmControlFlow { Terminate(Value), } +pub type VmLogs = Vec; + impl Vm { pub fn new( instructions: Vec, @@ -228,10 +244,11 @@ impl Vm { call_stack: Default::default(), }, instructions: idx_map, + stdout: vec![], } } - pub fn run(mut self) -> Result<(Value, Vec)> { + pub fn run(mut self) -> Result<(Value, Vec, VmLogs)> { use VmControlFlow::*; let val = loop { match self.execute() { @@ -240,7 +257,7 @@ impl Vm { Err(e) => return Err(e), } }; - Ok((val, self.state.stack)) + Ok((val, self.state.stack, self.stdout)) } fn execute(&mut self) -> Result { @@ -322,7 +339,7 @@ impl Vm { let str = str.iter().flat_map(|num| num.to_ne_bytes()).collect::>(); // convert vec of usizes to string let string: String = str.iter().map(|&c| c as char).collect(); - println!("{}", string); + self.stdout.push(string.clone()); }, }; Ok(Continue) From 1797bbbd34e33f5513438d504371859b85da5389 Mon Sep 17 00:00:00 2001 From: sezna Date: Sat, 6 Jul 2024 17:28:03 -0700 Subject: [PATCH 6/7] update tests --- petr-playground/index.js | 5 +++-- petr-vm/src/lib.rs | Bin 13795 -> 13852 bytes 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/petr-playground/index.js b/petr-playground/index.js index a8b1647..090f076 100644 --- a/petr-playground/index.js +++ b/petr-playground/index.js @@ -14,7 +14,7 @@ monaco.languages.setMonarchTokensProvider("petr", { keywords: [ 'function', 'returns', 'in' ], tokenizer: { root: [ - [/\~[a-zA-Z][a-zA-Z0-9]*/, "function-call"], + [/\~([a-zA-Z][a-zA-Z0-9]+)(\.[a-zA-Z]([a-zA-Z0-9])+)*/, "function-call"], [/\@[a-zA-Z]+/, "intrinsic"], [/[0-9]+/, "integer-literal"], [/\".*\"/, "string-literal"], @@ -51,6 +51,7 @@ monaco.editor.defineTheme("petr-theme", { { token: "function-call", foreground: "808080", fontStyle: "bold" }, { token: "string-literal", foreground: literalColor }, { token: "integer-literal", foreground: literalColor }, + { token: "keyword", foreground: literalColor }, ], colors: { "editor.foreground": "#ffffff", @@ -110,7 +111,7 @@ monaco.languages.registerCompletionItemProvider("mySpecialLanguage", { monaco.editor.create(document.getElementById('monaco-editor'), { - value: "function main() returns 'unit \n @puts(\"Hello, World!\")", + value: "function main() returns 'unit \n ~std.io.print \"Hello, World!\"", language: 'petr', theme: "petr-theme", }); diff --git a/petr-vm/src/lib.rs b/petr-vm/src/lib.rs index 6908a9d3581a7e839c55d73352f8f10bc58258b7..88490779d3f46e873addd5b08458034cb7b85629 100644 GIT binary patch delta 119 zcmaEyJtt>FJ@e-2%(5(#?{Tqj-ow6|F($f5S&2&l4TL4;l%{GJXktjj$H)8ly9Wa) e45^INoSb|eh4TEOoD@X{1_mYN*v*ZcNjd;z3mrND delta 57 zcmbP}^Ei7$Ju`1gQo5psT5*19QF3bW<`QNNmd&%-4>NKog(c>crfL{Wp3fmVIgUqU NvK%+Z=9iqtIsjZN6HEXA From a5ff1c206a07f0b8f56564e5cb3696a29b1f5de6 Mon Sep 17 00:00:00 2001 From: sezna Date: Sat, 6 Jul 2024 19:35:55 -0700 Subject: [PATCH 7/7] remove machete --- .github/workflows/ci.yml | 8 -------- 1 file changed, 8 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 404cd3a..e2ee809 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -7,14 +7,6 @@ env: RUSTFLAGS: "-Dwarnings" jobs: - build: - runs-on: ubuntu-latest - steps: - - name: Checkout - uses: actions/checkout@v3 - - name: Machete - uses: bnjbvr/cargo-machete@main - clippy_check: runs-on: ubuntu-latest steps: