CARLsim
4.1.0
CARLsim: a GPU-accelerated SNN simulator
|
In this tutorial we will implement an 80-20 random spiking neural network with synapses of varying axonal delay that are subject to spike-timing dependent plasticity (STDP). We follow the MATLAB implementation spnet.m
, which can be obtained from Izhikevich's Website. In addition, we will use various functions from the MATLAB Offline Analysis Toolbox (OAT) to visualize network activity and weight changes.
At the end of the tutorial, you will know how to:
This tutorial assumes you know how to:
The source code of this tutorial can be found in %%CARLSIM_ROOT_DIR%%/doc/source/tutorial/2_random_spnet
.
80-20 networks get their name from the ratio of excitatory (80%) to inhibitory (20%) neurons in the network, which is thought to preserve a ratio found in the mammalian cortex (Braitenberg & Schuz, 1991), although these numbers might not be exact.
Such networks are supposed to exhibit sleeplike oscillations, gamma (40 Hz) rhythms, conversion of firing rates to spike timings, and other interesting regimes.
As always, the first step in setting up a CARLsim program is to include the libCARLsim
library and to instantiate the main simulation object:
From then on, the simulation is in CONFIG_STATE, allowing the properties of the neural network to be specified. In this tutorial, we consider a network of 800 excitatory (regular spiking; RS) and 200 inhibitory neurons (fast spiking; FS) (see ch3s1s1_izhikevich_neurons). Each neuron shall have roughly 100 incoming synapses (nSynPerNeur
) with axonal delays between 1ms and 20ms (maxDelay
). We first set a number of parameters:
and then create the groups of regular spiking and fast spiking neurons with appropriate Izhikevich parameters a, b, c, and d:
The network considered here is sparse with 10% probability of connection between any two neurons (the probability is found by dividing the number of synpases per neuron by the number of neurons):
We also set the synaptic weight magnitudes to be 6.0f
for excitatory synapses and 5.0f
for inhibitory synapses, and make sure weight magnitudes never get larger than 10.0f
:
Then the synaptic connections can be specified. Groups are connected with the "random"
primitive type of CARLsim::connect that sparsely connects groups with a fixed connection probability. Weights are initialized to wtExc
and wtInh
for excitatory and inhibitory synapses, respectively. Note that the weight of inhibitory synapses in RangeWeight is to be understood as a weight magnitude, and therefore does not have a negative sign. Weights are bounded by the interval [0, wtMax]
. Axonal delays are uniformly distributed in the range [1, 20] ms
. Connections should not be further confined to a spatial receptive field, which is denoted by RadiusRF(-1)
. SYN_PLASTIC denotes that the synapses will be subject to plasticity. All of this is accomplished with the following code snippet:
What remains to be done is to enable STDP on the connections above. STDP is enabled post-synaptically: Setting E-STDP will enable STDP on all incoming excitatory synapses (where the pre-synaptic group is excitatory), whereas setting I-STDP will enable STDP on all incoming inhibitory synapses (where the pre-synaptic group is inhibitory). We enable both types on post-synaptic group gExc
(but not gInh
on to be consistent with the original MATLAB code) and set the shape of the STDP curve to the standard exponential curve (see 5.2.2 Excitatory STDP (E-STDP) and Inhibitory STDP (I-STDP)):
Note the sign of the alpha variables. "Standard" E-STDP implies positive weight changes for pre-before-post events (alphaPlus
) and negative weight changes for post-before-pre (-alphaMinus
). With I-STDP the logic is inverted.
Finally, we disable synaptic conductances so that the network is run in CUBA mode (see 3.2.1 CUBA):
Once the spiking network has been specified, the function CARLsim::setupNetwork optimizes the network state for the chosen back-end (CPU or GPU) and moves the simulation into SETUP_STATE:
In this state, we specify a number of SpikeMonitor and ConnectionMonitor objects to record network activity and synaptic weights to a "DEFAULT"
binary file (see Chapter 7: Monitoring):
This will dump all spikes of group gExc
(in AER format) to a binary with default file name "results/spk_exc.dat"
, where "exc"
is the name that was assigned to the group gExc
in the call to CARLsim::createGroup above. The same is true for group gInh
. Analogously, two weight files "results/conn_exc_exc.dat"
and "results/conn_inh_exc.dat"
will be created that contain snapshots of the synaptic weights taken every 10,000ms.
The first call to CARLsim::runNetwork will take the simulation into RUN_STATE. The simulation will be repeatedly run (or "stepped") for 1ms, up to the point at which a total of 10,000ms are simulated.
The main simulation loop is encapsulated by a pair of SpikeMonitor::startRecording and SpikeMonitor::stopRecording calls, which instruct the SpikeMonitor objects to keep track of all spikes emitted by the neurons in their monitored group (see 7.1.2.1 Start and Stop Recording):
After (or during) the simulation these objects could be queried for more detailed spike metrics (see 7.1.2.3 Spike Metrics), but for the sake of simplicity we will only print a condensed version of spike statistics at the end:
Finally, the main simulation loop consists of repeatedly stepping the model for 1ms. At every step we randomly choose a single neuron in the network to receive an external (thalamic) input current of 20.0f
. To achieve this, we maintain a vector of external currents for every neuron in the group, initialized to zero:
Then, a neuron ID is chosen randomly, and the corresponding input current for the identified neuron is adjusted:
Finally, the resulting vectors are applied to both groups. Note that we explicitly set the unaffected input currents to zero every time step. This is because otherwise the non-zero input current from previous iterations will keep getting applied until manually reset to zero:
After that is accomplished, the model is stepped for 1ms while explicitly disabling printing the run summary at the end:
After navigating to %CARLSIM_ROOT_DIR%%/doc/source/tutorial/2_random_spnet
, the network can be compiled and run with the following commands (on Unix):
On Windows, the .vcxproj
file is already added to the CARLsim.sln
solution file. Thus the project can be built simply by opening the solution file in Visual Studio, by right-clicking the project directory and choosing "Build project".
Some of the CARLsim output is shown below:
Since the network was run in GPU_MODE, the simulation will print a summary of the GPU memory management. This indicates the amount of GPU memory that each important data structure occupies. The above was generated on a GTX 780 with 3GB of GPU memory.
Both SpikeMonitor objects produced one line of output each at the end of the simulation via SpikeMonitor::print. This line includes the number of spikes each group emitted over the time course of 10,000ms, with corresponding mean firing rate (in Hz) and standard deviation given in parentheses.
At the end of the simulation, a summary is printed that confirms the 80-20 ratio of excitatory to inhibitory neurons. A total number of 100,436 synapses were created, confirming the attempted number of (roughly) 100 synapses per neuron. The network was simulated for 10 seconds, yet its execution took only 2.92 seconds of wall-clock time, which means the network was run 3.42 times faster than real-time.
In order to plot network activity and observe weight changes in the network, we will make use of the MATLAB Offline Analysis Toolbox (OAT) (see Chapter 9: MATLAB Offline Analysis Toolbox (OAT)).
The Tutorial subdirectory "scripts/"
provides a MATLAB script "scripts/demoOAT.m"
to demonstrate the usage of the OAT. The script looks like this:
After adding the location of the OAT source code to the MATLAB path, a NetworkMonitor is opened on the simulation file "../results/sim_spnet.dat"
, which was automatically created during the CARLsim simulation (note that the string "spnet" corresponds to the name given to the network in the CARLsim constructor).
Calling the plot
method will then visualize network activity using default settings and a plot type that depends on the spatial structure of the neuron groups. Since the network does not have any particular spatial structure, the default plot type is a raster plot, shown in frames of 1000ms, as seen in Fig. 1 below.
Fig. 2 shows the temporal evolution of weights from the recurrent excitatory connection on gExc
. At the beginning of the experiment (leftmost panel), all weights are initialized to 6.0f
. Over time, these values start to diverge, but stay confined in the range [0.0f, 10.0f]
.
Similar behavior can be observed on the connection from gInh
to gExc
.
It would also be possible to open a ConnectionMonitor on the fixed connection from gExc
to gInh
, but the expected output would be similar to the leftmost panel of Fig. 2, showing all weights initialized to 5.0f
, with no weight changes to report over time.