Skip to content

Commit

Permalink
feat: support libraries in REPL
Browse files Browse the repository at this point in the history
  • Loading branch information
Fumuran committed Nov 29, 2023
1 parent 350904d commit ecb1699
Show file tree
Hide file tree
Showing 4 changed files with 121 additions and 18 deletions.
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,9 @@
- Updated Winterfell dependency to v0.7 (#1121).
- Added methods `StackOutputs::get_stack_item()` and `StackOutputs::get_stack_word()` (#1155).

#### CLI
- Introduced the `!use` command for the Miden REPL (#1162).

## 0.7.0 (2023-10-11)

#### Assembly
Expand Down
34 changes: 34 additions & 0 deletions docs/src/tools/repl.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,11 @@ Miden REPL can be started via the CLI [repl](../intro/usage.md#cli-interface) co
./target/optimized/miden repl
```

It is also possible to initialize REPL with libraries. To create it with Miden standard library you need to specify `-s` or `--stdlib` subcommand, it is also possible to add a third-party library by specifying `-l` or `--libraries` subcommand with paths to `.masl` library files. For example:
```Shell
./target/optimized/miden repl -s -l example/library.masl
```

### Miden assembly instruction

All Miden instructions mentioned in the [Miden Assembly sections](../user_docs/assembly/main.md) are valid. One can either input instructions one by one or multiple instructions in one input.
Expand Down Expand Up @@ -115,6 +120,35 @@ If the `addr` has not been initialized:
Memory at address 87 is empty
```

### !use

The `!use` command prints out the list of all modules available for import.

If the stdlib was added to the available libraries list `!use` command will print all its modules:
```
>> !use
Modules available for importing:
std::collections::mmr
std::collections::smt
std::collections::smt64
...
std::mem
std::sys
std::utils
```

Using the `!use` command with a module name will add the specified module to the program imports:
```
>> !use std::math::u64
>> !program
use.std::math::u64
begin
end
```

### !undo

The `!undo` command reverts to the previous state of the stack and memory by dropping off the last executed assembly instruction from the program. One could use `!undo` as often as they want to restore the state of a stack and memory $n$ instructions ago (provided there are $n$ instructions in the program). The `!undo` command will result in an error if no remaining instructions are left in the Miden program.
Expand Down
13 changes: 11 additions & 2 deletions miden/src/cli/repl.rs
Original file line number Diff line number Diff line change
@@ -1,15 +1,24 @@
use clap::Parser;
use std::path::PathBuf;

use crate::repl::start_repl;

#[derive(Debug, Clone, Parser)]
#[clap(about = "Initiates the Miden REPL tool")]
pub struct ReplCmd {}
pub struct ReplCmd {
/// Paths to .masl library files
#[clap(short = 'l', long = "libraries", value_parser)]
library_paths: Vec<PathBuf>,

/// Usage of standard library
#[clap(short = 's', long = "stdlib")]
use_stdlib: bool,
}

impl ReplCmd {
pub fn execute(&self) -> Result<(), String> {
// initiates repl tool.
start_repl();
start_repl(&self.library_paths, self.use_stdlib);
Ok(())
}
}
89 changes: 73 additions & 16 deletions miden/src/repl/mod.rs
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
use super::ProgramError;
use assembly::{Assembler, Library, MaslLibrary};
use miden::{
math::{Felt, StarkField},
DefaultHost, StackInputs, Word,
};
use processor::ContextId;
use rustyline::{error::ReadlineError, DefaultEditor};
use std::{collections::BTreeSet, path::PathBuf};
use stdlib::StdLibrary;

/// This work is in continuation to the amazing work done by team `Scribe`
/// [here](https://github.com/ControlCplusControlV/Scribe/blob/main/transpiler/src/repl.rs#L8)
Expand Down Expand Up @@ -125,9 +127,24 @@ use rustyline::{error::ReadlineError, DefaultEditor};
/// Memory at address 87 is empty
/// Initiates the Miden Repl tool.
pub fn start_repl() {
pub fn start_repl(library_paths: &Vec<PathBuf>, use_stdlib: bool) {
let mut program_lines: Vec<String> = Vec::new();

// set of user imported modules
let mut imported_modules: BTreeSet<String> = BTreeSet::new();

// load libraries from files
let mut provided_libraries = Vec::new();
for path in library_paths {
let library = MaslLibrary::read_from_file(path)
.map_err(|e| format!("Failed to read library: {e}"))
.unwrap();
provided_libraries.push(library);
}
if use_stdlib {
provided_libraries.push(MaslLibrary::from(StdLibrary::default()));
}

println!("========================== Miden REPL ============================");
println!();
// prints out all the available commands in the Miden Repl tool.
Expand All @@ -143,16 +160,21 @@ pub fn start_repl() {
// initializing readline.
let mut rl = DefaultEditor::new().expect("Readline couldn't be initialized");
loop {
let program = format!(
"begin\n{}\nend",
let mut program = String::new();
for module in imported_modules.iter() {
program.push_str(module);
program.push('\n');
}
program.push_str(&format!(
"\nbegin\n{}\nend",
program_lines
.iter()
.map(|l| format!(" {}", l))
.collect::<Vec<_>>()
.join("\n")
);
));

let result = execute(program.clone());
let result = execute(program.clone(), &provided_libraries);

if !program_lines.is_empty() {
match result {
Expand Down Expand Up @@ -210,7 +232,7 @@ pub fn start_repl() {
break;
}
}
// incase the flag has not been initialized.
// in case the flag has not been initialized.
if !mem_at_addr_present {
println!("Memory at address {} is empty", addr);
}
Expand All @@ -232,6 +254,8 @@ pub fn start_repl() {
};
} else if line == "!stack" {
should_print_stack = true;
} else if line.starts_with("!use") {
handle_use_command(line, &provided_libraries, &mut imported_modules);
} else {
rl.add_history_entry(line.clone()).expect("Failed to add a history entry");
program_lines.push(line.clone());
Expand Down Expand Up @@ -262,18 +286,26 @@ pub fn start_repl() {
/// Compiles and executes a compiled Miden program, returning the stack, memory and any Miden errors.
/// The program is passed in as a String, passed to the Miden Assembler, and then passed into the Miden
/// Processor to be executed.
fn execute(program: String) -> Result<(Vec<(u64, Word)>, Vec<Felt>), ProgramError> {
let program = assembly::Assembler::default()
.compile(&program)
.map_err(ProgramError::AssemblyError)?;
fn execute(
program: String,
provided_libraries: &Vec<MaslLibrary>,
) -> Result<(Vec<(u64, Word)>, Vec<Felt>), String> {
// compile program
let mut assembler = Assembler::default();

assembler = assembler
.with_libraries(provided_libraries.clone().into_iter())
.map_err(|err| format!("{err}"))?;

let program = assembler.compile(&program).map_err(|err| format!("{err}"))?;

let stack_inputs = StackInputs::default();
let host = DefaultHost::default();

let state_iter = processor::execute_iter(&program, stack_inputs, host);
let (system, _, stack, chiplets, err) = state_iter.into_parts();
if let Some(err) = err {
return Err(ProgramError::ExecutionError(err));
return Err(format!("{err}"));
}

// loads the memory at the latest clock cycle.
Expand Down Expand Up @@ -306,16 +338,41 @@ fn read_mem_address(mem_str: &str) -> Result<u64, String> {
Ok(*addr)
}

/// Parses `!use` command. Adds the provided module to the program imports, or prints the list of
/// all available modules if no module name was provided.
fn handle_use_command(
line: String,
provided_libraries: &Vec<MaslLibrary>,
imported_modules: &mut BTreeSet<String>,
) {
let tokens: Vec<&str> = line.split_whitespace().collect();

match tokens.len() {
1 => {
println!("Modules available for importing:");
for lib in provided_libraries {
lib.modules().for_each(|module| println!("{}", module.path));
}
}
2 => {
imported_modules.insert(format!("use.{}", tokens[1]).to_string());
}
_ => println!("malformed instruction '!use': too many parameters provided"),
}
}

/// Prints out all the available command present in the Miden Repl tool.
fn print_instructions() {
println!("Available commands:");
println!();
println!("!stack: displays the complete state of the stack");
println!("!mem: displays the state of the entire memory");
println!("!mem[i]: displays the state of the memory at address i");
println!("!stack: display the complete state of the stack");
println!("!mem: display the state of the entire memory");
println!("!mem[i]: display the state of the memory at address i");
println!("!undo: remove the last instruction");
println!("!use: display a list of modules available for import");
println!("!use <full_module_name>: import the specified module");
println!("!program: display the program");
println!("!help: prints out all the available commands");
println!("!help: print out all the available commands");
println!();
}

Expand Down

0 comments on commit ecb1699

Please sign in to comment.