use std::{
borrow::Cow,
fmt,
path::{Path as FsPath, PathBuf},
str::FromStr,
sync::Arc,
};
use miden_assembly::SourceManager;
use miden_assembly_syntax::{
Library, Path as LibraryPath,
diagnostics::{IntoDiagnostic, Report},
};
use crate::config::DebuggerConfig;
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct LinkLibrary {
pub name: Cow<'static, str>,
pub path: Option<PathBuf>,
pub kind: LibraryKind,
}
#[derive(Default, Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub enum LibraryKind {
#[default]
Masp,
Masm,
}
impl fmt::Display for LibraryKind {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Masm => f.write_str("masm"),
Self::Masp => f.write_str("masp"),
}
}
}
impl FromStr for LibraryKind {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"masm" => Ok(Self::Masm),
"masp" => Ok(Self::Masp),
_ => Err(()),
}
}
}
impl LinkLibrary {
pub fn name(&self) -> &str {
self.name.as_ref()
}
pub fn load(
&self,
config: &DebuggerConfig,
source_manager: Arc<dyn SourceManager>,
) -> Result<Arc<Library>, Report> {
if let Some(path) = self.path.as_deref() {
return self.load_from_path(path, source_manager);
}
let path = self.find(config)?;
self.load_from_path(&path, source_manager)
}
fn load_from_path(
&self,
path: &FsPath,
source_manager: Arc<dyn SourceManager>,
) -> Result<Arc<Library>, Report> {
match self.kind {
LibraryKind::Masm => {
let ns = LibraryPath::validate(self.name.as_ref()).map_err(|err| {
Report::msg(format!("invalid library namespace '{}': {err}", &self.name))
})?;
let modules = miden_assembly_syntax::parser::read_modules_from_dir(
path,
ns,
source_manager.clone(),
false,
)?;
miden_assembly::Assembler::new(source_manager).assemble_library(modules)
}
LibraryKind::Masp => {
use miden_core::serde::Deserializable;
let bytes = std::fs::read(path).into_diagnostic()?;
let package =
miden_mast_package::Package::read_from_bytes(&bytes).map_err(|e| {
Report::msg(format!(
"failed to load Miden package from {}: {e}",
path.display()
))
})?;
Ok(package.mast.clone())
}
}
}
fn find(&self, config: &DebuggerConfig) -> Result<PathBuf, Report> {
use std::fs;
let toolchain_dir = config.toolchain_dir();
let search_paths = toolchain_dir
.iter()
.chain(config.search_path.iter())
.chain(config.working_dir.iter());
for search_path in search_paths {
let reader = fs::read_dir(search_path).map_err(|err| {
Report::msg(format!(
"invalid library search path '{}': {err}",
search_path.display()
))
})?;
for entry in reader {
let Ok(entry) = entry else {
continue;
};
let path = entry.path();
let Some(stem) = path.file_stem().and_then(|stem| stem.to_str()) else {
continue;
};
if stem != self.name.as_ref() {
continue;
}
match self.kind {
LibraryKind::Masp => {
if !path.is_file() {
return Err(Report::msg(format!(
"unable to load Miden Assembly package from '{}': not a file",
path.display()
)));
}
}
LibraryKind::Masm => {
if !path.is_dir() {
return Err(Report::msg(format!(
"unable to load Miden Assembly library from '{}': not a directory",
path.display()
)));
}
}
}
return Ok(path);
}
}
Err(Report::msg(format!(
"unable to locate library '{}' using any of the provided search paths",
&self.name
)))
}
}
#[cfg(feature = "tui")]
impl clap::builder::ValueParserFactory for LinkLibrary {
type Parser = LinkLibraryParser;
fn value_parser() -> Self::Parser {
LinkLibraryParser
}
}
#[cfg(feature = "tui")]
#[doc(hidden)]
#[derive(Clone)]
pub struct LinkLibraryParser;
#[cfg(feature = "tui")]
impl clap::builder::TypedValueParser for LinkLibraryParser {
type Value = LinkLibrary;
fn possible_values(
&self,
) -> Option<Box<dyn Iterator<Item = clap::builder::PossibleValue> + '_>> {
use clap::builder::PossibleValue;
Some(Box::new(
[
PossibleValue::new("masm").help("A Miden Assembly project directory"),
PossibleValue::new("masp").help("A compiled Miden package file"),
]
.into_iter(),
))
}
fn parse_ref(
&self,
_cmd: &clap::Command,
_arg: Option<&clap::Arg>,
value: &std::ffi::OsStr,
) -> Result<Self::Value, clap::error::Error> {
use clap::error::{Error, ErrorKind};
let value = value.to_str().ok_or_else(|| Error::new(ErrorKind::InvalidUtf8))?;
let (kind, name) = value
.split_once('=')
.map(|(kind, name)| (Some(kind), name))
.unwrap_or((None, value));
if name.is_empty() {
return Err(Error::raw(
ErrorKind::ValueValidation,
"invalid link library: must specify a name or path",
));
}
let maybe_path = FsPath::new(name);
let extension = maybe_path.extension().map(|ext| ext.to_str().unwrap());
let kind = match kind {
Some(kind) if !kind.is_empty() => kind.parse::<LibraryKind>().map_err(|_| {
Error::raw(ErrorKind::InvalidValue, format!("'{kind}' is not a valid library kind"))
})?,
Some(_) | None => match extension {
Some(kind) => kind.parse::<LibraryKind>().map_err(|_| {
Error::raw(
ErrorKind::InvalidValue,
format!("'{kind}' is not a valid library kind"),
)
})?,
None => LibraryKind::default(),
},
};
if maybe_path.is_absolute() {
let meta = maybe_path.metadata().map_err(|err| {
Error::raw(
ErrorKind::ValueValidation,
format!(
"invalid link library: unable to load '{}': {err}",
maybe_path.display()
),
)
})?;
match kind {
LibraryKind::Masp if !meta.is_file() => {
return Err(Error::raw(
ErrorKind::ValueValidation,
format!("invalid link library: '{}' is not a file", maybe_path.display()),
));
}
LibraryKind::Masm if !meta.is_dir() => {
return Err(Error::raw(
ErrorKind::ValueValidation,
format!(
"invalid link library: kind 'masm' was specified, but '{}' is not a \
directory",
maybe_path.display()
),
));
}
_ => (),
}
let name = maybe_path.file_stem().unwrap().to_str().unwrap().to_string();
Ok(LinkLibrary {
name: name.into(),
path: Some(maybe_path.to_path_buf()),
kind,
})
} else if extension.is_some() {
let name = name.strip_suffix(unsafe { extension.unwrap_unchecked() }).unwrap();
let mut name = name.to_string();
name.pop();
Ok(LinkLibrary {
name: name.into(),
path: None,
kind,
})
} else {
Ok(LinkLibrary {
name: name.to_string().into(),
path: None,
kind,
})
}
}
}