Copyright (c) 2025 - The University of Texas at Austin This work was produced under contract #2317831 to National Technology and Engineering Solutions of Sandia, LLC which is under contract No. DE-NA0003525 with the U.S. Department of Energy.
Simulating Advanced Neuromorphic Architectures for Fast Exploration (SANA-FE)
A framework for modeling the energy usage and performance of different neuromorphic hardware.
We hope that you find this project useful. If you use SANA-FE in your work, please cite our paper:
James A. Boyle, Mark Plagge, Suma George Cardwell, Frances S. Chance, and Andreas Gerstlauer, "SANA-FE: Simulating Advanced Neuromorphic Architectures for Fast Exploration," in IEEE Transactions on Computer-Aided Design of Integrated Circuits and Systems, 2025, doi:10.1109/TCAD.2025.3537971.
@article{boyle2025sanafe,
title={SANA-FE: Simulating Advanced Neuromorphic Architectures for Fast Exploration},
author={James A. Boyle and Mark Plagge and Suma George Cardwell and Frances S. Chance and Andreas Gerstlauer},
journal={IEEE Transactions on Computer-Aided Design of Integrated Circuits and Systems},
volume={44},
number={8},
pages={3165--3178},
year={2025},
doi={10.1109/TCAD.2025.3537971}
}
This project uses CMake as its build system and dependency manager. To setup
compilation, first create a temporary build directory:
mkdir build && cd build
Run the following command in this build directory:
cmake ..
Then compile SANA-FE by using the command:
make -j 12
The option -j
indicates the number of parallel threads to use. This should
be less than the threads supported by your system.
Building this project requires cmake
, make
, and a compiler that supports the
C++17 standard (e.g., GCC >= 8, Clang >= 5). This project uses RapidYAML for all
YAML file parsing, and Booksim 2 for optional cycle-accurate NoC modeling. To
build the Python interfaces, you must also have Python >= 3.8 installed with
PyBind11. You can install PyBind11 using:
pip install pybind11
Booksim 2 requires the bison
and flex
packages for config parsing.
For example, in Ubuntu these can be installed using apt
.
apt install bison flex
./sim arch/example.yaml snn/example.yaml 100
This simulates 100 time-steps of a tiny connected spiking neural network (SNN).
General usage:
./sim [optional flags] <architecture description> <SNN description> <N timesteps>
In addition to the standlone simulator, SANA-FE can also be scripted using a
Python API. For an example of how this can be done, see the Jupyter
notebook-based tutorials in the tutorial/
directory.
Additional examples and experiments may be found in the scripts/
directory.
SANA-FE takes command line arguments, an architecture description file (YAML)
and an SNN description file (YAML). The description files both use custom
file formats. Examples for architectures may be found in arch/
. Examples for
SNNs may be found in snn/
.
Optional command line flags can be used to enable simulation traces. Note that after enabling traces globally, you will still have to create probes at the neuron level to get trace output.
Flags:
-m
: Enable message traces tomessages.csv
-n
: Use the (legacy) netlist format for SNNs, instead of YAML.-o
: Output directory-p
: Record the simulated performance of each timestep toperf.csv
-s
: Enable spike traces tospikes.csv
-t [simple/detailed/cycle]
: Specify the timing model (default=detailed
)-v
: Enable potential (voltage) traces topotential.csv
-N
: Number of neuron/message processing threads (default=1)-S
: Number of scheduling threads (default=0, use main thread)
The SNN description format is based on the YAML file format.
Different mapped SNNs can be defined flexibly and generally using sections for
neuron groups
, edges
, and hardware mappings
. Each section allows for
custom attributes
to be defined, which are converted to model parameters
within the simulator. While the keywords for sections are fixed, attributes
allow for custom user-defined parameters to be associated with neurons and
connections.
The SNN must be defined under the main network
section. All other top-level
sections are ignored. Then, we have groups
, edges
, and mappings
sub-sections.
Groups of neurons are one or more neurons that may share some common attributes.
This is similar to how other frameworks may define populations, or layers of
similar neurons. How neurons are grouped is up to the user, but they can be
useful for sharing common attributes or connections. Under the groups
subsection, you must create a list of named neuron groups. Within each group is
an attributes
section and a neurons
section.
In each neurons
subsection, list all sets of neurons belonging to the group.
For conciseness we support specifying multiple neurons using the range (..)
notation. Following each neuron, give an ordered or unordered list of
attributes e.g.,
- 0..2: [attribute1: value1]
- 3: {attribute1: value1}
In the edges
section, define neuron to neuron connections or group to group
hyper-edges, including any edge attributes. The edge format uses a notation
similar to the graph DOT format e.g.,
- layer1.0 -> layer2.1: [weight: 1]
- layer1 -> layer2: [weight: 1]
Finally, in the mappings
section we map neurons to hardware cores. Under the
section heading is a list of mappings, with an example of one mapping as
follows:
- layer1.0..1: [core: 0.0]
Similar to before, neurons may give as a range for brevity. This maps two neurons to tile 0, core 0 (the first core in the chip).
As long as valid YAML syntax is used, SANA-FE does not distinguish between
different styles (block style, flow style, or a mix of the two). For one example
of a simple SNN, see snn/example.yaml
.
The architecture description format is also based on the YAML file format.
Different architectures are defined using a hierarchical description. This tool models neuromorphic designs with several assumptions, in order to simplify the tool.
- The chip is time-step based. A time-step is a small discrete amount of time. This is as opposed to a purely event driven simulation e.g. ROSS.
- The neural cores adhere to some common design patterns
At the top level, the description begins with the "architecture" keyword. Any other top-level sections will be ignored. This defines anything at the chip level, including the NoC interconnect.
A chip contains one or more network tiles, representing some shared network
resources e.g., a router. Each tile
contains one or more cores, where a core
performs computation. Each neuromorphic core
contains a fixed spike processing
hardware pipeline. It is assumed that tiles and cores are all parallel processing
elements.
Each core is assumed to have a neuromorphic pipeline which processes the updates for one or more neurons. The pipeline is a fixed sequence of niche hardware units. Those hardware units could contain digital logic, analog circuits or even novel devices.
The pipeline contains the following units:
-
The input axon unit receive spike packets from the network and generate synaptic addresses for memory lookups.
-
The synaptic unit looks up connectivity for incoming spikes and updates any relevant synaptic currents.
-
The dendritic unit combines currents based some internal structure and a set of operations.
-
The soma unit updates membrane potentials based on the dendritic current and neuron model. If the firing criteria is met, it generates a spike for that neuron.
-
The output axon unit send spikes from the soma out to the network to go to other cores' pipelines.
For an example, see arch/loihi.yaml
. There are a nested series of
keywords, where keywords define required hardware blocks. Each block must be
contain a name
keyword, which may optionally specify the number of instances.
Blocks are duplicated the number specified in the range, for example:
# Define 8 cores, 0 through 7
-name: neuromorphic_core[0..7]
Blocks much also have both an attributes
section and the next hardware blocks
in the hierarchy. The attributes section will generate one or more parameters
that are passed to be parsed by the simulator and the relevant hardware models
implemented either internally (models.cpp
) or externally (plugins).
If corresponding traces are enabled, output is saved to trace files with hard-coded names using either csv or yaml extensions.
spikes.csv
: The spikes for each time-step on probed neurons
potential.csv
: The potentials for each time-step on probed neurons
perf.csv
: Detailed statistics for each timestep and each hardware unit
messages.csv
: Information on spike messages for each time-step
run_summary.yaml
: High-level statistics for the simulation e.g. runtime
SANA-FE uses a user-provided spiking architecture, a mapped SNN, and run-time
configuration to simulate a spiking chip as it executes a spiking application.
SANA-FE uses the Architecture
to compile a SpikingChip
, which it then loads
the mapped SNN. SANA-FE then rapidly simulates the design at a time-step
granularity.
During each time-step SANA-FE models custom spike-processing pipelines executing
within each core, modeling the processing of neurons and spike messages. Using
our spiking hardware template, we enable custom hardware blocks to be
incorporated for axonal, synaptic, dendritic and somatic hardware. Each hardware
unit is implemented using a model
- you can take the built-in hardware unit
models provided in models.cpp
, or implement models externally as hardware unit
plugins using the fixed base class interfaces. The SANA-FE kernel coordinates
all on-chip activity, makes calls to the models and tracks the total energy and
latency across the chip.
SANA-FE includes efficient but detailed semi-analytical timing models. This
takes aggregated information about all spike messages generated in a time-step
and calls a custom scheduler in schedule.cpp
. The on chip schedule ultimately
gives you a reasonably accurate prediction of the chip timings, accounting for
effects such as blocking in the NoC and custom latency simulations within
hardware units.
As part of SANA-FE, the user can implement different hardware models using
custom plugins. Models for synapses, dendrites and somas are all supported.
SANA-FE supports a base hardware model base class, with which it implements
all of its synaptic, dendritic and somatic hardware models. Using SANA-FE's
PipelineUnit
base class, you can implement your own models as hardware
plugins.
There is one example already provided in the /plugins
folder implementing a
Hodgkin-Huxley neuron model (hodgkin_huxley.cpp
). There are a few steps
required to use plugins in SANA-FE:
- Specify the plugin path in the architecture yaml file, in the corresponding
synapse
,dendrite
orsoma
hardware section. Specify the plugin path using the attributeplugin: <pathname>
. - Specify the model name using the attribute
model: <name>
. - Map neurons to the hardware unit as usual with the attribute:
soma_hw_name
.
For example, for the Hodgkin-Huxley example provided with SANA-FE, you could use it as follows:
# Rest of arch description
...
soma:
- name: plugin_example_soma
attributes:
plugin: plugins/hodgkin_huxley.cpp
model: HodgkinHuxley
...
SANA-FE can run any models provided as user plugins. The plugin must be compiled
as a shared library containing one or more hardware models.
Models can execute arbitrary code, but interfaces must be derived either from
the general PipelineUnit
class, or one of the specialized SynapseUnit
,
DendriteUnit
or SomaUnit
base classes.
SANA-FE's plugin mechanism makes it easy to integrate plugins with your architectural simulations. However, a few steps are needed to get plugins running:
- You must make sure your plugin has been built as a shared library (
.so
), either by updating the plugin CMake file or providing your own build scripts. - Your new plugin must implement a hardware model class with the hardware
functionality you want. The model class you implement must be derived from
PipelineUnit
inchip.hpp
, which defines the required interfaces. These are enforced by pure virtual methods, including attribute parsing methods update methods. For examples of different hardware models, see eithermodels.cpp
or theplugins
folder. - Finally, provide a class factory function that returns a new instance of
your model class. This has to be in the format
create_<modelname>
. For example, for aHodgkinHuxley
model, we would specify the following code in the plugin C++ file:
extern "C" sanafe::PipelineUnit *create_HodgkinHuxley()
{
return (sanafe::PipelineUnit *) new HodgkinHuxley();
}
It is recommended new users look through the rest of the hodgkin_huxley.cpp
file to see what an example plugin looks like.
Version 1 of SANA-FE (written in C) defined a simpler, less capable SNN description format (compared to the current YAML-based format). For back-compatability, the netlist-style format is still supported. To use this format, use the command-line flag (-n).
In the netlist format each line defines a new entry, which may either be a neuron group (g), neuron (n), edge (e), or mapping (&). Each line starts with the type of entry followed by one required field and then any number of named attributes. Fields are separated by one or more spaces.
Attributes are defined using the syntax: <attribute>=<value>
. Note, there
is no space before or after the equals. The attribute soma_hw_name
is required
to be set for every neuron or neuron group.
A neuron group helps reduce the number of repeated, shared parameters for a population of neurons e.g., for a layer of neurons in a deep SNN.
g <number of neurons> <common attributes>
Neurons are addressed (using the group number followed by neuron number), and then all attributes are specified. Note the group must be defined first.
n group_id.neuron_id <unique attributes>
An edge connects one source neuron (presynaptic) to one destination neuron (postsynaptic). The edge may also have attributes such as synaptic weight.
e src_group_id.src_neuron_id->dest_group_id.dest_neuron_id <edge attributes>
Finally, mappings place predefined neurons on a hardware core. Here we specify the neuron and the core.
& group_id.neuron_id@tile_id.core_id
An example of how to use the netlist format is given in snn/example.net
This project has been written in C++ and Python. All code is in the /src
directory. See header files for more detail on supported classes and functions.
C++ code has been written using the C++17 standard.
James A. Boyle, Jason Ho, Mark Plagge, Suma George Cardwell, Frances S. Chance, and Andreas Gerstlauer, "Exploring Dendrites in Large-Scale Neuromorphic Architectures," in International Conference on Neuromorphic Systems (ICONS), Seattle, WA, USA, doi:.
James A. Boyle, Mark Plagge, Suma George Cardwell, Frances S. Chance, and Andreas Gerstlauer, "SANA-FE: Simulating Advanced Neuromorphic Architectures for Fast Exploration," in IEEE Transactions on Computer-Aided Design of Integrated Circuits and Systems, 2025, doi:10.1109/TCAD.2025.3537971.
James A. Boyle, Mark Plagge, Suma George Cardwell, Frances S. Chance, and Andreas Gerstlauer, "Tutorial: Large-Scale Spiking Neuromorphic Architecture Exploration using SANA-FE," in International Conference on Hardware/Software Codesign and System Synthesis (CODES+ISSS), Raleigh, NC, USA, doi:10.1109/CODES-ISSS60120.2024.00007.
James A. Boyle, Mark Plagge, Suma George Cardwell, Frances S. Chance, and Andreas Gerstlauer, "Performance and Energy Simulation of Spiking Neuromorphic Architectures for Fast Exploration," in International Conference on Neuromorphic Systems (ICONS), Santa Fe, NM, USA, doi:10.1145/3589737.3605970.
James Boyle: [email protected]