ngspice

One of the benefits of schematic-as-code is it avoids the tediousness of drag-and-drop GUIs. The various spice simulations often require the same drag-and-drop steps and maybe even recreating the schematic. typeCAD lets you use your code with only very minor changes.

Voltage Divider

This example will use the same voltage divider from the Simplest Walkthrough.

This is the circuit: Voltage Divider

And this is the code:

import { Schematic } from "@typecad/typecad"
import { Resistor } from '@typecad/passives/0603';

let typecad = new Schematic('voltage_divider');
let r1 = new Resistor({ value: '10kohm' });
let r2 = new Resistor({ value: '10kohm' });

typecad.named('vdiv').net(r1.pin(2), r2.pin(1));

typecad.create(r1, r2);

Install @typecad/ngspice

First, make sure you have @typecad/ngspice installed in your project. From the ./hw directory, run:

npm install @typecad/ngspice

Then you’ll need ngspice installed and ensure the binaries are in your PATH.

Power

Next, we need to add a Power component. We didn’t add one in the simple voltage divider example because it wasn’t needed, but most circuits will use Power objects. In addition to helping with simulation, they assist with ERC (electrical rule checking) by ensuring pins are connected as they should be, packages can use it to determine the power supply voltage is correct, and connections can be made.

import { Schematic } from "@typecad/typecad"
import { Schematic, Power } from "@typecad/typecad"
import { Resistor } from '@typecad/passives/0603';

let typecad = new Schematic('voltage_divider');
let r1 = new Resistor({ value: '10kohm' });
let r2 = new Resistor({ value: '10kohm' });
let vin = new Power({ power: r1.pin(1), gnd: r2.pin(2), voltage: 3.3 });

typecad.named('in').net(r1.pin(1));
typecad.named('vdiv').net(r1.pin(2), r2.pin(1));
typecad.named('gnd').net(r2.pin(2));

typecad.create(r1, r2);

In the code above, we added a Power object called vin. Power is coming in from the top of R1 and going to the bottom of R2. The voltage is 3.3 volts.

A more practical example would be a battery holder with the pins corresponding to the positive and negative terminals, but for this example, we’ll use the resistor legs as the power terminals.


Tip

Power objects represent a physical component like the pins of a battery holder or a voltage regulator. It is not the same as a schematic VCC or GND symbol that is more abstract.

Look at lines 10 and 12. You’ll see that vin.power is connected to r1.pin(1) by itself and it is ::named. @typecad/ngspice only pays attention to ::named nets that have components used in the simulation. nspice will treat each ::named net as a node and give voltage/power/current measurements for each.

Import ngspice

Now we need to add the ngspice related code.

import { Schematic, Power } from "@typecad/typecad"
import { Resistor } from '@typecad/passives/0603';
import { ngspiceSimulator } from '@typecad/ngspice';

let typecad = new Schematic('voltage_divider');
let r1 = new Resistor({ value: '10kohm', simulation: { include: true } }); 
let r2 = new Resistor({ value: '10kohm', simulation: { include: true } }); 
let vin = new Power({ power: r1.pin(1), gnd: r2.pin(2), voltage: 3.3 }); 
let ngspice = new ngspiceSimulator(typecad, vin);

typecad.named('vdiv').net(r1.pin(2), r2.pin(1));

typecad.create(r1, r2);

This code imports the package. It also adds a simulation property to the resistors. This tells @typecad/ngspice to include the component in the simulation. Every component has a simulation property, setting include to true will include the component in the simulation. There is an additional property, model, that can be used to specify an ngspice model: {model: '.model Dled D (IS=1a RS=3.3 N=1.8)'.

The last thing is to create a ngspiceSimulator object. It takes the Schematic object and all the Power objects. In this example, we only have one Power object, but if you have multiple, you can pass them all.

Simulate

Now that our circuit is created, nets are named, we can run the simulation.

Currently, there are two simulation modes available in the library: DC and transient analysis.

import { Schematic, Power } from "@typecad/typecad"
import { Resistor } from '@typecad/passives/0603';
import { ngspiceSimulator } from '@typecad/ngspice';

let typecad = new Schematic('voltage_divider');
let r1 = new Resistor({ value: '10kohm', simulation: { include: true } }); 
let r2 = new Resistor({ value: '10kohm', simulation: { include: true } }); 
let vin = new Power({ power: r1.pin(1), gnd: r2.pin(2), voltage: 3.3 }); 
let ngspice = new ngspiceSimulator(typecad, vin);

typecad.named('vdiv').net(r1.pin(2), r2.pin(1));

typecad.create(r1, r2);

ngspice.op();

When the project is built, the output will be in the console:

🌶️ Running ngspice
┌────────────────────┬───────────────┬────────────────────┐
 Variable Type Value
├────────────────────┼───────────────┼────────────────────┤
 r1:power power 2.7225 mW
├────────────────────┼───────────────┼────────────────────┤
 v(in) voltage 3.3000 V
├────────────────────┼───────────────┼────────────────────┤
 i(r1) current 1.6500 mA
├────────────────────┼───────────────┼────────────────────┤
 i(r2) current 1.6500 mA
├────────────────────┼───────────────┼────────────────────┤
 r2:power power 2.7225 mW
├────────────────────┼───────────────┼────────────────────┤
 i(v1) current -1.6500 mA
├────────────────────┼───────────────┼────────────────────┤
 v(vdiv) voltage 1.6500 V
└────────────────────┴───────────────┴────────────────────┘

Each node will be displayed with calculated voltage. ngspice doesn’t list ground nodes. Each included component will be displayed with power and current. The final bit will be information about the power supply. In this example, v1 supplies 1.65 mA of current to the circuit.

Transient

This example won’t show any variance, but you can run a transient analysis.

Add ngspice.tran('1us', '100ms'); to the end of the file and ngspice’s graphing windows will open for all the nodes and components when the project is built. tran takes all the same arguments as ngspice’s tran command, as described in the ngspice documentation, section 11.3.10.

Continuing development

This package covers the basic functionality of basic components. Future development will include the ability to use ngspice library files, additional power source options, and more analysis types.