#![warn(missing_docs)]
use std::collections::{HashMap, HashSet};
use std::io::Write;
#[cfg(any(unix, target_os = "redox"))]
use std::os::unix::prelude::PermissionsExt;
use std::path::PathBuf;
use std::sync::Arc;
use crate::blocks::Vsource;
use crate::tran::Tran;
use arcstr::ArcStr;
use blocks::Isource;
use cache::error::TryInnerError;
use cache::CacheableWithState;
use error::*;
use nutlex::parser::Data;
use scir::schema::{FromSchema, NoSchema, NoSchemaError};
use scir::{ChildId, Library, NetlistLibConversion, SignalInfo, SignalPathTail, SliceOnePath};
use serde::{Deserialize, Serialize};
use spice::netlist::{
HasSpiceLikeNetlist, Include, NetlistKind, NetlistOptions, NetlisterInstance, RenameGround,
};
use spice::{ComponentValue, Spice};
use substrate::block::Block;
use substrate::context::Installation;
use substrate::execute::Executor;
use substrate::io::schematic::HardwareType;
use substrate::schematic::primitives::{RawInstance, Resistor};
use substrate::schematic::schema::Schema;
use substrate::schematic::{CellBuilder, PrimitiveBinding, Schematic};
use substrate::simulation::{SimulationContext, Simulator};
use templates::{write_run_script, RunScriptContext};
use unicase::UniCase;
pub mod blocks;
pub mod error;
pub(crate) mod templates;
pub mod tran;
#[derive(Debug, Clone)]
pub enum Primitive {
Spice(spice::Primitive),
Vsource(Vsource),
Isource(Isource),
}
impl Primitive {
fn ports(&self) -> Vec<ArcStr> {
match self {
Primitive::Spice(prim) => prim.ports(),
Primitive::Vsource(_) => vec!["1".into(), "2".into()],
Primitive::Isource(_) => vec!["1".into(), "2".into()],
}
}
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Ord, PartialOrd)]
pub enum SaveStmt {
Raw(ArcStr),
ScirVoltage(SliceOnePath),
ResistorCurrent(scir::InstancePath),
}
impl<T: Into<ArcStr>> From<T> for SaveStmt {
fn from(value: T) -> Self {
Self::Raw(value.into())
}
}
impl SaveStmt {
pub fn new(path: impl Into<ArcStr>) -> Self {
Self::from(path)
}
pub(crate) fn to_save_string(
&self,
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
) -> ArcStr {
match self {
SaveStmt::Raw(raw) => raw.clone(),
SaveStmt::ScirVoltage(scir) => arcstr::format!(
"v({})",
node_voltage_path(lib, conv, &lib.simplify_path(scir.clone()),)
),
SaveStmt::ResistorCurrent(scir) => {
arcstr::format!(
"@{}{}[i]",
if scir.len() == 1 { "" } else { "R." },
instance_path(lib, conv, scir)
)
}
}
}
pub(crate) fn to_data_string(
&self,
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
) -> ArcStr {
match self {
SaveStmt::Raw(raw) => raw.clone(),
SaveStmt::ScirVoltage(_) => self.to_save_string(lib, conv),
SaveStmt::ResistorCurrent(_) => {
arcstr::format!("i({})", self.to_save_string(lib, conv).to_lowercase())
}
}
}
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Ord, PartialOrd)]
pub enum ProbeStmt {
Raw(ArcStr),
ScirCurrent(SliceOnePath),
}
impl<T: Into<ArcStr>> From<T> for ProbeStmt {
fn from(value: T) -> Self {
Self::Raw(value.into())
}
}
impl ProbeStmt {
pub fn new(path: impl Into<ArcStr>) -> Self {
Self::from(path)
}
pub(crate) fn to_probe_string(
&self,
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
) -> ArcStr {
match self {
ProbeStmt::Raw(raw) => raw.clone(),
ProbeStmt::ScirCurrent(scir) => {
arcstr::format!("i({})", node_current_path(lib, conv, scir, true))
}
}
}
pub(crate) fn to_data_string(
&self,
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
) -> ArcStr {
match self {
ProbeStmt::Raw(raw) => raw.clone(),
ProbeStmt::ScirCurrent(scir) => {
arcstr::format!(
"i({})",
node_current_path(lib, conv, scir, false).to_lowercase()
)
}
}
}
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Ord, PartialOrd)]
pub(crate) enum SavedData {
Save(SaveStmt),
Probe(ProbeStmt),
}
impl SavedData {
pub(crate) fn netlist<W: Write>(
&self,
out: &mut W,
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
) -> std::io::Result<()> {
match self {
Self::Save(save) => write!(out, ".save {}", save.to_save_string(lib, conv)),
Self::Probe(probe) => write!(out, ".probe {}", probe.to_probe_string(lib, conv)),
}
}
pub(crate) fn to_data_string(
&self,
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
) -> ArcStr {
match self {
Self::Save(save) => save.to_data_string(lib, conv),
Self::Probe(probe) => probe.to_data_string(lib, conv),
}
}
}
impl From<SaveStmt> for SavedData {
fn from(value: SaveStmt) -> Self {
Self::Save(value)
}
}
impl From<ProbeStmt> for SavedData {
fn from(value: ProbeStmt) -> Self {
Self::Probe(value)
}
}
#[derive(Debug, Clone, Default)]
pub struct Ngspice {}
#[derive(Debug, Clone, Default)]
pub struct Options {
includes: HashSet<Include>,
saves: HashMap<SavedData, u64>,
next_save_key: u64,
}
impl Options {
pub fn include(&mut self, path: impl Into<PathBuf>) {
self.includes.insert(Include::new(path));
}
pub fn include_section(&mut self, path: impl Into<PathBuf>, section: impl Into<ArcStr>) {
self.includes.insert(Include::new(path).section(section));
}
fn save_inner(&mut self, save: impl Into<SavedData>) -> u64 {
let save = save.into();
if let Some(key) = self.saves.get(&save) {
*key
} else {
let save_key = self.next_save_key;
self.next_save_key += 1;
self.saves.insert(save, save_key);
save_key
}
}
pub fn save_tran_voltage(&mut self, save: impl Into<SaveStmt>) -> tran::VoltageSavedKey {
tran::VoltageSavedKey(self.save_inner(save.into()))
}
pub fn save_tran_current(&mut self, save: impl Into<SaveStmt>) -> tran::CurrentSavedKey {
tran::CurrentSavedKey(vec![self.save_inner(save.into())])
}
pub fn probe_tran_current(&mut self, save: impl Into<ProbeStmt>) -> tran::CurrentSavedKey {
tran::CurrentSavedKey(vec![self.save_inner(save.into())])
}
}
#[derive(Debug, Clone, Serialize, Deserialize, Hash, PartialEq, Eq)]
struct CachedSim {
simulation_netlist: Vec<u8>,
}
struct CachedSimState {
input: Vec<Input>,
netlist: PathBuf,
output_file: PathBuf,
log: PathBuf,
err_log: PathBuf,
run_script: PathBuf,
work_dir: PathBuf,
executor: Arc<dyn Executor>,
}
impl CacheableWithState<CachedSimState> for CachedSim {
type Output = Vec<HashMap<String, Vec<f64>>>;
type Error = Arc<Error>;
fn generate_with_state(
&self,
state: CachedSimState,
) -> std::result::Result<Self::Output, Self::Error> {
let inner = || -> Result<Self::Output> {
let CachedSimState {
input,
netlist,
output_file,
log,
err_log,
run_script,
work_dir,
executor,
} = state;
write_run_script(
RunScriptContext {
netlist: &netlist,
raw_output_file: &output_file,
log_path: &log,
err_path: &err_log,
bashrc: None,
flags: "",
},
&run_script,
)?;
let mut perms = std::fs::metadata(&run_script)?.permissions();
#[cfg(any(unix, target_os = "redox"))]
perms.set_mode(0o744);
std::fs::set_permissions(&run_script, perms)?;
let mut command = std::process::Command::new("/bin/bash");
command.arg(&run_script).current_dir(&work_dir);
executor
.execute(command, Default::default())
.map_err(|_| Error::NgspiceError)?;
let contents = std::fs::read(&output_file)?;
let rawfile = nutlex::parse(
&contents,
nutlex::Options {
endianness: nutlex::ByteOrder::LittleEndian,
},
)?;
let mut raw_outputs = Vec::with_capacity(input.len());
for (an, results) in input.iter().zip(rawfile.analyses.into_iter()) {
match an {
Input::Tran(_) => match results.data {
Data::Real(real) => raw_outputs.push(HashMap::from_iter(
results
.variables
.into_iter()
.map(|var| (var.name.to_string(), real[var.idx].clone())),
)),
_ => {
return Err(Error::NgspiceError);
}
},
}
}
Ok(raw_outputs)
};
inner().map_err(Arc::new)
}
}
impl Ngspice {
fn simulate(
&self,
ctx: &SimulationContext<Ngspice>,
options: Options,
input: Vec<Input>,
) -> Result<Vec<Output>> {
std::fs::create_dir_all(&ctx.work_dir)?;
let netlist = ctx.work_dir.join("netlist.spice");
let mut f = std::fs::File::create(&netlist)?;
let mut w = Vec::new();
let mut includes = options.includes.into_iter().collect::<Vec<_>>();
let mut saves = options.saves.keys().cloned().collect::<Vec<_>>();
includes.sort();
saves.sort();
let netlister = NetlisterInstance::new(
self,
&ctx.lib.scir,
&mut w,
NetlistOptions::new(
NetlistKind::Testbench(RenameGround::Yes(arcstr::literal!("0"))),
&includes,
),
);
let conv = netlister.export()?;
writeln!(w)?;
for save in saves {
save.netlist(&mut w, &ctx.lib.scir, &conv)?;
writeln!(w)?;
}
writeln!(w)?;
for an in input.iter() {
an.netlist(&mut w)?;
writeln!(w)?;
}
f.write_all(&w)?;
let output_file = ctx.work_dir.join("data.raw");
let log = ctx.work_dir.join("ngspice.log");
let err_log = ctx.work_dir.join("ngspice.err");
let run_script = ctx.work_dir.join("simulate.sh");
let work_dir = ctx.work_dir.clone();
let executor = ctx.ctx.executor.clone();
let raw_outputs = ctx
.ctx
.cache
.get_with_state(
"ngspice.simulation.outputs",
CachedSim {
simulation_netlist: w,
},
CachedSimState {
input,
netlist,
output_file,
log,
err_log,
run_script,
work_dir,
executor,
},
)
.try_inner()
.map_err(|e| match e {
TryInnerError::CacheError(e) => Error::Caching(e),
TryInnerError::GeneratorError(e) => Error::Generator(e.clone()),
})?
.clone();
let conv = Arc::new(conv);
let outputs = raw_outputs
.into_iter()
.map(|mut raw_values| {
tran::Output {
time: Arc::new(raw_values.remove("time").unwrap()),
raw_values: raw_values
.into_iter()
.map(|(k, v)| (ArcStr::from(k), Arc::new(v)))
.collect(),
saved_values: options
.saves
.iter()
.map(|(k, v)| (*v, k.to_data_string(&ctx.lib.scir, &conv)))
.collect(),
}
.into()
})
.collect();
Ok(outputs)
}
}
impl scir::schema::Schema for Ngspice {
type Primitive = Primitive;
}
impl FromSchema<NoSchema> for Ngspice {
type Error = NoSchemaError;
fn convert_primitive(
_primitive: <NoSchema as Schema>::Primitive,
) -> std::result::Result<<Self as Schema>::Primitive, Self::Error> {
Err(NoSchemaError)
}
fn convert_instance(
_instance: &mut scir::Instance,
_primitive: &<NoSchema as Schema>::Primitive,
) -> std::result::Result<(), Self::Error> {
Err(NoSchemaError)
}
}
impl Schematic<Ngspice> for RawInstance {
fn schematic(
&self,
io: &<<Self as Block>::Io as HardwareType>::Bundle,
cell: &mut CellBuilder<Ngspice>,
) -> substrate::error::Result<Self::NestedData> {
let mut prim = PrimitiveBinding::new(Primitive::Spice(spice::Primitive::RawInstance {
cell: self.cell.clone(),
ports: self.ports.clone(),
params: self
.params
.clone()
.into_iter()
.map(|(k, v)| (UniCase::new(k), v))
.collect(),
}));
for (i, port) in self.ports.iter().enumerate() {
prim.connect(port, io[i]);
}
cell.set_primitive(prim);
Ok(())
}
}
impl Schematic<Ngspice> for Resistor {
fn schematic(
&self,
io: &<<Self as Block>::Io as HardwareType>::Bundle,
cell: &mut CellBuilder<Ngspice>,
) -> substrate::error::Result<Self::NestedData> {
let mut prim = PrimitiveBinding::new(Primitive::Spice(spice::Primitive::Res2 {
value: ComponentValue::Fixed(self.value()),
params: Default::default(),
}));
prim.connect("1", io.p);
prim.connect("2", io.n);
cell.set_primitive(prim);
Ok(())
}
}
impl Installation for Ngspice {}
impl Simulator for Ngspice {
type Schema = Ngspice;
type Input = Input;
type Options = Options;
type Output = Output;
type Error = Error;
fn simulate_inputs(
&self,
config: &substrate::simulation::SimulationContext<Self>,
options: Self::Options,
input: Vec<Self::Input>,
) -> Result<Vec<Self::Output>> {
self.simulate(config, options, input)
}
}
pub(crate) fn instance_path(
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
path: &scir::InstancePath,
) -> String {
lib.convert_instance_path_with_conv(conv, path.clone())
.join(".")
}
pub(crate) fn node_voltage_path(
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
path: &SliceOnePath,
) -> String {
lib.convert_slice_one_path_with_conv(conv, path.clone(), |name, index| {
if let Some(index) = index {
arcstr::format!("{}[{}]", name, index)
} else {
name.clone()
}
})
.join(".")
}
pub(crate) fn node_current_path(
lib: &Library<Ngspice>,
conv: &NetlistLibConversion,
path: &SliceOnePath,
save: bool,
) -> String {
assert_eq!(
path.instances().len(),
1,
"ngspice only supports saving currents of top level instance terminals"
);
let annotated_path = lib.annotate_instance_path(path.instances().clone());
let named_path = lib.convert_instance_path_with_conv(conv, path.instances().clone());
let mut str_path = named_path.join(".");
str_path.push(':');
match annotated_path.instances.last().unwrap().child {
Some(ChildId::Cell(id)) => {
let cell = lib.cell(id);
if save {
let signal = match path.tail() {
SignalPathTail::Id(slice) => cell.signal(slice.signal()),
SignalPathTail::Name(slice) => cell.signal_named(slice.signal()),
};
let idx = signal.port.expect("signal is not a valid terminal");
str_path.push_str(&format!(
"{}",
idx + path.tail().index().unwrap_or_default() + 1
));
} else {
let name = match path.tail() {
SignalPathTail::Id(slice) => cell.signal(slice.signal()).name.clone(),
SignalPathTail::Name(slice) => slice.signal().clone(),
};
str_path.push_str(&name);
if let Some(index) = path.tail().index() {
str_path.push_str(&format!("[{}]", index));
}
}
}
Some(ChildId::Primitive(id)) => {
let prim = lib.primitive(id);
let tail = path.tail().as_ref().unwrap_name();
if save {
str_path.push_str(&format!(
"{}",
prim.ports()
.iter()
.position(|x| x == tail.signal())
.unwrap()
+ 1
));
} else {
str_path.push_str(&format!("n{}", tail.signal().to_lowercase()));
}
}
None => {
panic!("cannot save or recover paths that do not exist")
}
}
str_path
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub enum Input {
Tran(Tran),
}
impl From<Tran> for Input {
fn from(value: Tran) -> Self {
Self::Tran(value)
}
}
#[derive(Debug, Clone)]
pub enum Output {
Tran(tran::Output),
}
impl From<tran::Output> for Output {
fn from(value: tran::Output) -> Self {
Self::Tran(value)
}
}
impl TryFrom<Output> for tran::Output {
type Error = Error;
fn try_from(value: Output) -> Result<Self> {
match value {
Output::Tran(t) => Ok(t),
}
}
}
impl Input {
fn netlist<W: Write>(&self, out: &mut W) -> Result<()> {
match self {
Self::Tran(t) => t.netlist(out),
}
}
}
impl Tran {
fn netlist<W: Write>(&self, out: &mut W) -> Result<()> {
write!(out, ".tran {} {}", self.step, self.stop)?;
if let Some(ref start) = self.start {
write!(out, "{start}")?;
}
Ok(())
}
}
impl HasSpiceLikeNetlist for Ngspice {
fn write_prelude<W: Write>(&self, out: &mut W, _lib: &Library<Self>) -> std::io::Result<()> {
writeln!(out, "* Substrate SPICE library")?;
writeln!(out, "* This is a generated file. Be careful when editing manually: this file may be overwritten.\n")?;
Ok(())
}
fn write_include<W: Write>(
&self,
out: &mut W,
include: &spice::netlist::Include,
) -> std::io::Result<()> {
Spice.write_include(out, include)
}
fn write_start_subckt<W: Write>(
&self,
out: &mut W,
name: &ArcStr,
ports: &[&SignalInfo],
) -> std::io::Result<()> {
Spice.write_start_subckt(out, name, ports)
}
fn write_end_subckt<W: Write>(&self, out: &mut W, name: &ArcStr) -> std::io::Result<()> {
Spice.write_end_subckt(out, name)
}
fn write_instance<W: Write>(
&self,
out: &mut W,
name: &ArcStr,
connections: Vec<ArcStr>,
child: &ArcStr,
) -> std::io::Result<ArcStr> {
Spice.write_instance(out, name, connections, child)
}
fn write_primitive_inst<W: Write>(
&self,
out: &mut W,
name: &ArcStr,
mut connections: HashMap<ArcStr, Vec<ArcStr>>,
primitive: &<Self as Schema>::Primitive,
) -> std::io::Result<ArcStr> {
match primitive {
Primitive::Spice(spice_primitive) => {
Spice.write_primitive_inst(out, name, connections, spice_primitive)
}
Primitive::Vsource(vsource) => {
let name = arcstr::format!("V{}", name);
write!(out, "{}", name)?;
for port in ["P", "N"] {
for part in connections.remove(port).unwrap() {
write!(out, " {}", part)?;
}
}
match vsource {
Vsource::Dc(dc) => {
write!(out, " DC {}", dc)?;
}
Vsource::Pulse(pulse) => {
write!(
out,
" PULSE({} {} {} {} {} {} {} {})",
pulse.val0,
pulse.val1,
pulse.delay.unwrap_or_default(),
pulse.rise.unwrap_or_default(),
pulse.fall.unwrap_or_default(),
pulse.width.unwrap_or_default(),
pulse.period.unwrap_or_default(),
pulse.num_pulses.unwrap_or_default(),
)?;
}
}
Ok(name)
}
Primitive::Isource(isource) => {
let name = arcstr::format!("I{}", name);
write!(out, "{}", name)?;
for port in ["P", "N"] {
for part in connections.remove(port).unwrap() {
write!(out, " {}", part)?;
}
}
match isource {
Isource::Dc(dc) => {
write!(out, " DC {}", dc)?;
}
Isource::Pulse(pulse) => {
write!(
out,
" PULSE({} {} {} {} {} {} {} {})",
pulse.val0,
pulse.val1,
pulse.delay.unwrap_or_default(),
pulse.rise.unwrap_or_default(),
pulse.fall.unwrap_or_default(),
pulse.width.unwrap_or_default(),
pulse.period.unwrap_or_default(),
pulse.num_pulses.unwrap_or_default(),
)?;
}
}
Ok(name)
}
}
}
}