Source code for spynnaker.pyNN.models.neuron.implementations.neuron_impl_standard

# Copyright (c) 2017 The University of Manchester
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from spinn_utilities.overrides import overrides
from spynnaker.pyNN.models.neuron.input_types import InputTypeConductance
from .abstract_neuron_impl import AbstractNeuronImpl
from spinn_front_end_common.interface.ds import DataType
from spinn_front_end_common.utilities.constants import BYTES_PER_WORD
from spynnaker.pyNN.utilities.struct import Struct, StructRepeat

# The size of the n_steps_per_timestep parameter
_N_STEPS_PER_TIMESTEP_SIZE = 1 * BYTES_PER_WORD

# The default number of steps per timestep
_DEFAULT_N_STEPS_PER_TIMESTEP = 1

_STEPS_PER_TIMESTEP = "n_steps_per_timestep"
_STEPS_PER_TIMESTEP_STRUCT = Struct(
    [(DataType.UINT32, _STEPS_PER_TIMESTEP)], repeat_type=StructRepeat.GLOBAL)


class NeuronImplStandard(AbstractNeuronImpl):
    """
    The standard componentised neuron implementation.
    """

    __slots__ = [
        "__model_name",
        "__binary",
        "__neuron_model",
        "__input_type",
        "__synapse_type",
        "__threshold_type",
        "__additional_input_type",
        "__components",
        "__n_steps_per_timestep"
    ]

    _RECORDABLES = ["v", "gsyn_exc", "gsyn_inh"]

    _RECORDABLE_DATA_TYPES = {
        "v": DataType.S1615,
        "gsyn_exc": DataType.S1615,
        "gsyn_inh": DataType.S1615
    }

    _RECORDABLE_UNITS = {
        'v': 'mV',
        'gsyn_exc': "uS",
        'gsyn_inh': "uS"}

    def __init__(
            self, model_name, binary, neuron_model, input_type,
            synapse_type, threshold_type, additional_input_type=None):
        """
        :param str model_name:
        :param str binary:
        :param AbstractNeuronModel neuron_model:
        :param AbstractInputType input_type:
        :param AbstractSynapseType synapse_type:
        :param AbstractThresholdType threshold_type:
        :param additional_input_type:
        :type additional_input_type: AbstractAdditionalInput or None
        """
        self.__model_name = model_name
        self.__binary = binary
        self.__neuron_model = neuron_model
        self.__input_type = input_type
        self.__synapse_type = synapse_type
        self.__threshold_type = threshold_type
        self.__additional_input_type = additional_input_type
        self.__n_steps_per_timestep = _DEFAULT_N_STEPS_PER_TIMESTEP

        self.__components = [
            self.__neuron_model, self.__input_type, self.__threshold_type,
            self.__synapse_type]
        if self.__additional_input_type is not None:
            self.__components.append(self.__additional_input_type)

    @property
    def n_steps_per_timestep(self):
        return self.__n_steps_per_timestep

    @n_steps_per_timestep.setter
    def n_steps_per_timestep(self, n_steps_per_timestep):
        self.__n_steps_per_timestep = n_steps_per_timestep

    @property
    @overrides(AbstractNeuronImpl.model_name)
    def model_name(self):
        return self.__model_name

    @property
    @overrides(AbstractNeuronImpl.binary_name)
    def binary_name(self):
        return self.__binary

    @property
    @overrides(AbstractNeuronImpl.structs)
    def structs(self):
        structs = [_STEPS_PER_TIMESTEP_STRUCT]
        structs.extend(s for c in self.__components for s in c.structs)
        return structs

[docs] @overrides(AbstractNeuronImpl.get_global_weight_scale) def get_global_weight_scale(self): return self.__input_type.get_global_weight_scale()
[docs] @overrides(AbstractNeuronImpl.get_n_synapse_types) def get_n_synapse_types(self): return self.__synapse_type.get_n_synapse_types()
[docs] @overrides(AbstractNeuronImpl.get_synapse_id_by_target) def get_synapse_id_by_target(self, target): return self.__synapse_type.get_synapse_id_by_target(target)
[docs] @overrides(AbstractNeuronImpl.get_synapse_targets) def get_synapse_targets(self): return self.__synapse_type.get_synapse_targets()
[docs] @overrides(AbstractNeuronImpl.get_recordable_variables) def get_recordable_variables(self): return self._RECORDABLES
[docs] @overrides(AbstractNeuronImpl.get_recordable_units) def get_recordable_units(self, variable): return self._RECORDABLE_UNITS[variable]
[docs] @overrides(AbstractNeuronImpl.get_recordable_data_types) def get_recordable_data_types(self): return self._RECORDABLE_DATA_TYPES
[docs] @overrides(AbstractNeuronImpl.is_recordable) def is_recordable(self, variable): return variable in self._RECORDABLES
[docs] @overrides(AbstractNeuronImpl.get_recordable_variable_index) def get_recordable_variable_index(self, variable): return self._RECORDABLES.index(variable)
[docs] @overrides(AbstractNeuronImpl.add_parameters) def add_parameters(self, parameters): parameters[_STEPS_PER_TIMESTEP] = self.__n_steps_per_timestep for component in self.__components: component.add_parameters(parameters)
[docs] @overrides(AbstractNeuronImpl.add_state_variables) def add_state_variables(self, state_variables): for component in self.__components: component.add_state_variables(state_variables)
[docs] @overrides(AbstractNeuronImpl.get_units) def get_units(self, variable): for component in self.__components: if component.has_variable(variable): return component.get_units(variable) raise KeyError( f"The parameter {variable} does not exist in this input " "conductance component")
@property @overrides(AbstractNeuronImpl.is_conductance_based) def is_conductance_based(self): return isinstance(self.__input_type, InputTypeConductance) def __getitem__(self, key): # Find the property in the components... for component in self.__components: if hasattr(component, key): return getattr(component, key) # ... or fail raise AttributeError( f"'{self.__class__.__name__}' object has no attribute {key}")