A Quick Comparison of LTSpice and Julia DiffEq

11.0 μs

LTSpice is probably the most popular version of SPICE (Simulation Program with Integrated Circuit Emphasis), a simulator for electronic circuits. It is proprietary software distributed as freeware by Analog Devices but originally developed at Linear Technology which was acquired by Analog.

The Julia programming language has perhaps the best suite of tools for solving differential equations in DifferentialEquations.jl, so it's natural to compare the performance of them.

31.4 μs

Here we're comparing simulations of the following schematic, which I copied from this article by Michael Engelhardt, the author of LTSpice.

13.6 μs
64.8 ms

This circuit features a lossless resonant tank circuit, so after a starting kick from the piece-wise linear (PWL) current source it should repeat the same sinewaves forever.

7.1 μs

Here's the circuit expressed as a pair of ODEs, written with a function signature compatible with the DifferentialEquations.jl solvers:

7.2 μs
LCIode (generic function with 1 method)
65.5 μs

Here's the piecewise-linear current source:

7.1 μs
I1 (generic function with 1 method)
22.7 μs

Here we set up the solution with initial conditions, a time span, and tolerance options.

It was necessary to tighten the tolerances from the default values so that the resonance would not lose or gain energy significantly over the 1.0 second span.

9.6 μs
run (generic function with 1 method)
41.6 μs

And here we run it.

7.1 μs
9.1 ms

Meanwhile, let's run the LTSpice version and import the voltage waveform to Julia.

7.1 μs
22.6 s

Let's plot the results against each other.

6.8 μs
plotboth (generic function with 1 method)
39.5 μs

These start off the same:

7.1 μs
522 ms

But somehwere along the way they get out of phase. This plot shows the final 2 ms of the simulation. These should be the same frequency but they're not!

7.8 μs
513 ms

Let's multiply the results together; the envelope shows us how different the frequencies are. We seem to be off by about 19.5 cycles over 1 second.

8.8 μs
385 ms

The resonant frequency should be 1(2π)LC or 2250.8 Hz. Which simulation got it right? Did either of them? Let's write a simple frequency counter.

16.3 ms
freqcount (generic function with 1 method)
79.7 μs

Measure the frequency of the Julia DiffEq solution:

9.6 μs
2250.866948589712
113 μs

Measure the frequency of the LTSpice solution:

7.6 μs
2231.357208116857
133 ms

Okay, it looks like we found out which solution is right!

7.1 μs

Did we pay for that correct answer with more compute time? How long did Julia take?

6.6 μs
  8.582 ms (84241 allocations: 11.33 MiB)
11.9 s

How long did LTSpice take?

8.3 μs

From LTSPice menus: View, SPICE Error Log:

Circuit: * D:\SPICE\SDFig2.asc

.OP point found by inspection.

Date: Wed Dec 16 12:25:34 2020
Total elapsed time: 0.983 seconds.

tnom = 27
temp = 27
method = modified trap
totiter = 130620
traniter = 130620
tranpoints = 65311
accept = 46151
rejected = 19160
matrix size = 1
fillins = 0
solver = Normal
Matrix Compiler1: off  0.0/0.0/[0.0]
Matrix Compiler2: 43 bytes object code size  0.0/0.0/[0.0]
20.9 μs

So for this particular problem Julia is approximately 100x faster!

Is this an unfair comparison?

One possible reason is that LTSpice uses "self-authoring assembly language source written at run time", which sounds a bit like JIT compilation so it may be unfair to compare the @btime result of Julia (which removes compilation time) to the LTSpice time.

So let's extend the simulation time by 10x. If the LTSpice assembly authoring took a big chunk of the 1.003 s, it should become less significant.

17.6 μs

Temporarily making that change gives 97 ms for Julia and 8 s for LTSpice. So the 100x remains (as does the frequency error).

6.1 μs
Some final comments
  • This is just a quick experiment and not necessarily representative of other problems.

  • I'm sure I did not explore all options for making these faster. Maybe there is some low hanging fruit on the LTSpice side. If history is any guide, I'm sure the Julia code can be sped up, maybe by picking a different solver algorithm.

  • Perhaps LTSpice is optimized for larger circuits and performs relatively slowly here. This is in some ways a toy example. A more complicated example should be tried.

  • When using LTSpice you don't have to write your own differential equations! While not a big deal for this circuit, it's not something one wants to do for a circuit of any complexity. Anyone got a SPICE netlist to DiffEq translator?

  • This notebook was prepared using the Pluto reactive notebook system for Julia.

5.5 ms

Revised 2022-12-17 to resolve inconsistent timings due to laptop power settings.

5.4 μs
Julia Version 1.5.3
Commit 788b2c77c1 (2020-11-09 13:37 UTC)
Platform Info:
  OS: Windows (x86_64-w64-mingw32)
  CPU: Intel(R) Core(TM) i7-9750H CPU @ 2.60GHz
  WORD_SIZE: 64
  LIBM: libopenlibm
  LLVM: libLLVM-9.0.1 (ORCJIT, skylake)
Environment:
  JULIA_REVISE_WORKER_ONLY = 1
Status `C:\Users\klaff\.julia\environments\v1.5\Project.toml`
  [0c46a032] DifferentialEquations v6.15.0
  [90137ffa] StaticArrays v0.12.5
5.7 s
76.0 s