# Copyright (c) 2017-2019 The University of Manchester
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
from enum import Enum
import logging
from spinn_front_end_common.utility_models import MultiCastCommand
from spinn_front_end_common.utilities.exceptions import ConfigurationException
logger = logging.getLogger(__name__)
# structure of command is KKKKKKKKKKKKKKKKKKKKK-IIIIIII-F-DDD
# K = ignored key at the top of the command
# I = instruction
# F = format
# D = device
_OFFSET_TO_IGNORED_KEY = 11
_OFFSET_TO_I = 4
_OFFSET_TO_F = 3
_OFFSET_TO_D = 0
_KEY_MASK = 0xFFFFF800
_I_MASK = 0x7F0
_F_MASK = 0x8
_D_MASK = 0x7
# UART masks and shifts
RETINA_UART_SHIFT = 3 + _OFFSET_TO_I
RETINA_WITHOUT_UART_MASK = 0x670
PUSH_BOT_LASER_LED_SPEAKER_FREQUENCY_UART_SHIFT = 1 + _OFFSET_TO_D
PUSH_BOT_MOTOR_WITHOUT_UART_MASK = 0x7C0
PUSH_BOT_MOTOR_UART_SHIFT = 0 + _OFFSET_TO_I
def _munich_key(instr, format_bit=0, device=0):
return ((instr << _OFFSET_TO_I) | (format_bit << _OFFSET_TO_F) |
(device << _OFFSET_TO_D))
def get_munich_i(key):
""" Get the instruction field from the key.
"""
return key & _I_MASK
def get_munich_f(key):
""" Get the format field from the key.
"""
return key & _F_MASK
def get_munich_d(key):
""" Get the device field from the key.
"""
return key & _D_MASK
def get_retina_i(key):
return key & RETINA_WITHOUT_UART_MASK
def get_push_bot_laser_led_speaker_frequency_i(key):
return get_munich_i(key)
def get_push_bot_motor_i(key):
return key & PUSH_BOT_MOTOR_WITHOUT_UART_MASK
# Specific fields in the key
_SENSOR_OUTGOING_OFFSET_TO_D = 2
_SENSOR_OUTGOING_OFFSET_TO_I = 7
# Payload fields
_PAYLOAD_RETINA_PAYLOAD_OFFSET = 29
_PAYLOAD_RETINA_PAYLOAD_MASK = 0xE0000000
_PAYLOAD_RETINA_KEY_OFFSET = 26
_PAYLOAD_RETINA_KEY_MASK = 0x1C000000
_PAYLOAD_SENSOR_ID_OFFSET = 27
_PAYLOAD_OFFSET_FOR_SENSOR_TIME = 0
def GET_RETINA_KEY_VALUE(payload):
return (payload & _PAYLOAD_RETINA_KEY_MASK) >> _PAYLOAD_RETINA_KEY_OFFSET
def GET_RETINA_PAYLOAD_VALUE(payload):
return (
(payload & _PAYLOAD_RETINA_PAYLOAD_MASK) >>
_PAYLOAD_RETINA_PAYLOAD_OFFSET
)
#: command key for setting up the master key of the board
CONFIGURE_MASTER_KEY = _munich_key(127, 0)
#: command key for setting up what mode of device running on the board
CHANGE_MODE = _munich_key(127, 1)
#: command for turning off retina output
DISABLE_RETINA_EVENT_STREAMING = _munich_key(0, 0)
#: command for retina where payload is events
ACTIVE_RETINA_EVENT_STREAMING_KEYS_CONFIGURATION = _munich_key(0, 1)
#: command for retina where events are the key
ACTIVE_RETINA_EVENT_STREAMING_SET_KEY = _munich_key(0, 2)
#: set timer / counter for timestamps
SET_TIMER_COUNTER_FOR_TIMESTAMPS = _munich_key(0, 3)
#: handle master / slave time sync
MASTER_SLAVE_KEY = _munich_key(0, 4)
#: command for setting bias (whatever the check that is)
BIAS_KEY = _munich_key(0, 5)
#: reset retina key.
RESET_RETINA_KEY = _munich_key(0, 7)
#: request on-board sensor data
SENSOR_REPORTING_OFF_KEY = _munich_key(1, 0)
#: poll sensors once
POLL_SENSORS_ONCE_KEY = _munich_key(1, 1)
#: poll sensors continuously
POLL_SENSORS_CONTINUOUSLY_KEY = _munich_key(1, 2)
#: disable motor
ENABLE_DISABLE_MOTOR_KEY = _munich_key(2, 0)
#: run motor for total period
MOTOR_RUN_FOR_PERIOD_KEY = _munich_key(2, 1)
#: raw output for motor 0 (permanent)
MOTOR_0_RAW_PERM_KEY = _munich_key(2, 4)
#: raw output for motor 1 (permanent)
MOTOR_1_RAW_PERM_KEY = _munich_key(2, 5)
#: raw output for motor 0 (leak towards 0)
MOTOR_0_RAW_LEAK_KEY = _munich_key(2, 6)
#: raw output for motor 1 (leak towards 0)
MOTOR_1_RAW_LEAK_KEY = _munich_key(2, 7)
#: motor output duration timer period
MOTOR_TIMER_A_TOTAL_PERIOD_KEY = _munich_key(3, 0)
#: motor output duration timer period
MOTOR_TIMER_B_TOTAL_PERIOD_KEY = _munich_key(3, 2)
#: motor output duration timer period
MOTOR_TIMER_C_TOTAL_PERIOD_KEY = _munich_key(3, 4)
#: motor output ratio active period
MOTOR_TIMER_A_CHANNEL_0_ACTIVE_PERIOD_KEY = _munich_key(4, 0)
#: motor output ratio active period
MOTOR_TIMER_A_CHANNEL_1_ACTIVE_PERIOD_KEY = _munich_key(4, 1)
#: motor output ratio active period
MOTOR_TIMER_B_CHANNEL_0_ACTIVE_PERIOD_KEY = _munich_key(4, 2)
#: motor output ratio active period
MOTOR_TIMER_B_CHANNEL_1_ACTIVE_PERIOD_KEY = _munich_key(4, 3)
#: motor output ratio active period
MOTOR_TIMER_C_CHANNEL_0_ACTIVE_PERIOD_KEY = _munich_key(4, 4)
#: motor output ratio active period
MOTOR_TIMER_C_CHANNEL_1_ACTIVE_PERIOD_KEY = _munich_key(4, 5)
#: digital IO Signals
QUERY_STATES_LINES_KEY = _munich_key(5, 0)
#: set output pattern to payload
SET_OUTPUT_PATTERN_KEY = _munich_key(5, 1)
#: add payload (logic or (PL)) to current output
ADD_PAYLOAD_TO_CURRENT_OUTPUT_KEY = _munich_key(5, 2)
#: remove payload (logic or (PL)) to current output from current output
REMOVE_PAYLOAD_TO_CURRENT_OUTPUT_KEY = _munich_key(5, 3)
#: set payload pins to high impedance
SET_PAYLOAD_TO_HIGH_IMPEDANCE_KEY = _munich_key(5, 4)
#: set laser params for PushBot
PUSH_BOT_LASER_CONFIG_TOTAL_PERIOD = _munich_key(3, 0)
#: set laser params for PushBot
PUSH_BOT_LASER_CONFIG_ACTIVE_TIME = _munich_key(4, 0)
#: set laser params for PushBot
PUSH_BOT_LASER_FREQUENCY = _munich_key(37, 1)
#: set led params for PushBot
PUSH_BOT_LED_CONFIG_TOTAL_PERIOD = _munich_key(3, 4)
#: set led params for PushBot
PUSH_BOT_LED_BACK_CONFIG_ACTIVE_TIME = _munich_key(4, 4)
#: set led params for PushBot
PUSH_BOT_LED_FRONT_CONFIG_ACTIVE_TIME = _munich_key(4, 5)
#: set led params for PushBot
PUSH_BOT_LED_FREQUENCY = _munich_key(37, 0)
#: set speaker params for PushBot
PUSH_BOT_SPEAKER_CONFIG_TOTAL_PERIOD = _munich_key(3, 2)
#: set speaker params for PushBot
PUSH_BOT_SPEAKER_CONFIG_ACTIVE_TIME = _munich_key(4, 2)
#: set speaker params for PushBot
PUSH_BOT_SPEAKER_TONE_BEEP = _munich_key(36, 0)
#: set speaker params for PushBot
PUSH_BOT_SPEAKER_TONE_MELODY = _munich_key(36, 1)
#: PushBot motor control
PUSH_BOT_MOTOR_0_PERMANENT_VELOCITY = _munich_key(32, 0)
#: PushBot motor control
PUSH_BOT_MOTOR_1_PERMANENT_VELOCITY = _munich_key(32, 1)
#: PushBot motor control
PUSH_BOT_MOTOR_0_LEAKY_VELOCITY = _munich_key(32, 2)
#: PushBot motor control
PUSH_BOT_MOTOR_1_LEAKY_VELOCITY = _munich_key(32, 3)
# payload for master slave
_PAYLOAD_MASTER_SLAVE_USE_INTERNAL_COUNTER = 0
_PAYLOAD_MASTER_SLAVE_SET_SLAVE = 1
_PAYLOAD_MASTER_SLAVE_SET_MASTER_CLOCK_NOT_STARTED = 2
_PAYLOAD_MASTER_SLAVE_SET_MASTER_CLOCK_ACTIVE = 4
class RetinaKey(Enum):
FIXED_KEY = (0, 128, 7)
NATIVE_128_X_128 = (1, 128, 7)
DOWNSAMPLE_64_X_64 = (2, 64, 6)
DOWNSAMPLE_32_X_32 = (3, 32, 5)
DOWNSAMPLE_16_X_16 = (4, 16, 4)
def __init__(self, value, pixels, bits_per_coordinate):
self._value_ = value << _PAYLOAD_RETINA_KEY_OFFSET
self._pixels = pixels
self._bits_per_coordinate = bits_per_coordinate
@property
def n_neurons(self):
return 2 * (self._pixels ** 2)
@property
def pixels(self):
return self._pixels
@property
def bits_per_coordinate(self):
return self._bits_per_coordinate
class RetinaPayload(Enum):
NO_PAYLOAD = (0, 0)
EVENTS_IN_PAYLOAD = (0, 4)
DELTA_TIMESTAMPS = (1, 4)
ABSOLUTE_2_BYTE_TIMESTAMPS = (2, 2)
ABSOLUTE_3_BYTE_TIMESTAMPS = (3, 3)
ABSOLUTE_4_BYTE_TIMESTAMPS = (4, 4)
def __init__(self, value, n_payload_bytes):
self._value_ = value << _PAYLOAD_RETINA_PAYLOAD_OFFSET
self._n_payload_bytes = n_payload_bytes
@property
def n_payload_bytes(self):
return self._n_payload_bytes
[docs]class MunichIoSpiNNakerLinkProtocol(object):
""" Provides Multicast commands for the Munich SpiNNaker-Link protocol
"""
__slots__ = ["__instance_key", "__mode", "__uart_id"]
[docs] class MODES(Enum):
"""types of modes supported by this protocol"""
RESET_TO_DEFAULT = 0
PUSH_BOT = 1
SPOMNIBOT = 2
BALL_BALANCER = 3
MY_ORO_BOTICS = 4
FREE = 5
# The instance of the protocol in use, to ensure that each vertex that is
# to send commands to the PushBot uses a different outgoing key; the top
# part of the key is ignored, so this works out!
protocol_instance = 0
# Keeps track of whether the mode has been configured already
__sent_mode_command = False
def __init__(self, mode, instance_key=None, uart_id=0):
"""
:param MODES mode:
The mode of operation of the protocol
:param instance_key: The optional instance key to use
:type instance_key: int or None
:param int uart_id: The ID of the UART when needed
"""
self.__mode = mode
# Create a key for this instance of the protocol
# - see above for reasoning
if instance_key is None:
self.__instance_key = (
MunichIoSpiNNakerLinkProtocol.protocol_instance <<
_OFFSET_TO_IGNORED_KEY
)
MunichIoSpiNNakerLinkProtocol.protocol_instance += 1
else:
self.__instance_key = instance_key
self.__uart_id = uart_id
@property
def mode(self):
"""
:rtype: MODES
"""
return self.__mode
@property
def uart_id(self):
"""
:rtype: int
"""
return self.__uart_id
[docs] @staticmethod
def sent_mode_command():
""" True if the mode command has ever been requested by any instance
"""
return MunichIoSpiNNakerLinkProtocol.__sent_mode_command
@property
def instance_key(self):
""" The key of this instance of the protocol
:rtype: int
"""
return self.__instance_key
def _get_key(self, command, offset_to_uart_id=None):
if offset_to_uart_id is None:
return command | self.__instance_key
return (
command | self.__instance_key |
(self.__uart_id << offset_to_uart_id)
)
@property
def configure_master_key_key(self):
return self._get_key(CONFIGURE_MASTER_KEY)
@property
def set_mode_key(self):
return self._get_key(CHANGE_MODE)
[docs] def set_mode(self, time=None):
MunichIoSpiNNakerLinkProtocol.__sent_mode_command = True
return MultiCastCommand(
key=self.set_mode_key, payload=self.__mode.value, time=time)
@property
def set_retina_key_key(self):
return self._get_key(
ACTIVE_RETINA_EVENT_STREAMING_SET_KEY, RETINA_UART_SHIFT)
[docs] def set_retina_key(self, new_key, time=None):
return MultiCastCommand(
key=self.set_retina_key_key,
payload=new_key, time=time)
@property
def disable_retina_key(self):
return self._get_key(DISABLE_RETINA_EVENT_STREAMING, RETINA_UART_SHIFT)
[docs] def disable_retina(self, time=None):
return MultiCastCommand(key=self.disable_retina_key, time=time)
@property
def master_slave_key(self):
return self._get_key(MASTER_SLAVE_KEY, RETINA_UART_SHIFT)
[docs] def master_slave_use_internal_counter(self, time=None):
return MultiCastCommand(
key=self.master_slave_key,
payload=_PAYLOAD_MASTER_SLAVE_USE_INTERNAL_COUNTER, time=time)
[docs] def master_slave_set_slave(self, time=None):
return MultiCastCommand(
key=self.master_slave_key,
payload=_PAYLOAD_MASTER_SLAVE_SET_SLAVE, time=time)
[docs] def master_slave_set_master_clock_not_started(self, time=None):
return MultiCastCommand(
key=self.master_slave_key,
payload=_PAYLOAD_MASTER_SLAVE_SET_MASTER_CLOCK_NOT_STARTED,
time=time)
[docs] def master_slave_set_master_clock_active(self, time=None):
return MultiCastCommand(
key=self.master_slave_key,
payload=_PAYLOAD_MASTER_SLAVE_SET_MASTER_CLOCK_ACTIVE,
time=time)
@property
def bias_values_key(self):
return self._get_key(BIAS_KEY, RETINA_UART_SHIFT)
[docs] def bias_values(self, bias_id, bias_value, time=None):
return MultiCastCommand(
key=self.bias_values_key,
payload=((bias_id << 0) | (bias_value << 8)), time=time)
@property
def reset_retina_key(self):
return self._get_key(RESET_RETINA_KEY, RETINA_UART_SHIFT)
[docs] def reset_retina(self, time=None):
return MultiCastCommand(
key=self.reset_retina_key, time=time)
@property
def turn_off_sensor_reporting_key(self):
return self._get_key(SENSOR_REPORTING_OFF_KEY)
[docs] def turn_off_sensor_reporting(self, sensor_id, time=None):
return MultiCastCommand(
key=self.turn_off_sensor_reporting_key,
payload=(sensor_id << _PAYLOAD_SENSOR_ID_OFFSET), time=time)
@property
def poll_sensors_once_key(self):
return self._get_key(POLL_SENSORS_ONCE_KEY)
[docs] def poll_sensors_once(self, sensor_id, time=None):
return MultiCastCommand(
key=self.poll_sensors_once_key,
payload=(sensor_id << _PAYLOAD_SENSOR_ID_OFFSET), time=time)
@property
def poll_individual_sensor_continuously_key(self):
return self._get_key(POLL_SENSORS_CONTINUOUSLY_KEY)
[docs] def poll_individual_sensor_continuously(
self, sensor_id, time_in_ms, time=None):
return MultiCastCommand(
key=self.poll_individual_sensor_continuously_key,
payload=((sensor_id << _PAYLOAD_SENSOR_ID_OFFSET) |
(time_in_ms << _PAYLOAD_OFFSET_FOR_SENSOR_TIME)),
time=time)
@property
def enable_disable_motor_key(self):
return self._get_key(ENABLE_DISABLE_MOTOR_KEY, RETINA_UART_SHIFT)
[docs] def generic_motor_enable(self, time=None):
return MultiCastCommand(
key=self.enable_disable_motor_key, payload=1, time=time)
[docs] def generic_motor_disable(self, time=None):
return MultiCastCommand(
key=self.enable_disable_motor_key, payload=0, time=time)
@property
def generic_motor_total_period_key(self):
return self._get_key(MOTOR_RUN_FOR_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def generic_motor_total_period(self, time_in_ms, time=None):
return MultiCastCommand(
key=self.generic_motor_total_period_key,
payload=time_in_ms, time=time)
@property
def generic_motor0_raw_output_permanent_key(self):
return self._get_key(MOTOR_0_RAW_PERM_KEY, RETINA_UART_SHIFT)
[docs] def generic_motor0_raw_output_permanent(self, pwm_signal, time=None):
return MultiCastCommand(
key=self.generic_motor0_raw_output_permanent_key,
payload=pwm_signal, time=time)
@property
def generic_motor1_raw_output_permanent_key(self):
return self._get_key(MOTOR_1_RAW_PERM_KEY, RETINA_UART_SHIFT)
[docs] def generic_motor1_raw_output_permanent(self, pwm_signal, time=None):
return MultiCastCommand(
key=self.generic_motor1_raw_output_permanent_key,
payload=pwm_signal, time=time)
@property
def generic_motor0_raw_output_leak_to_0_key(self):
return self._get_key(MOTOR_0_RAW_LEAK_KEY, RETINA_UART_SHIFT)
[docs] def generic_motor0_raw_output_leak_to_0(self, pwm_signal, time=None):
return MultiCastCommand(
key=self.generic_motor0_raw_output_leak_to_0_key,
payload=pwm_signal, time=time)
@property
def generic_motor1_raw_output_leak_to_0_key(self):
return self._get_key(MOTOR_1_RAW_LEAK_KEY, RETINA_UART_SHIFT)
[docs] def generic_motor1_raw_output_leak_to_0(self, pwm_signal, time=None):
return MultiCastCommand(
key=self.generic_motor1_raw_output_leak_to_0_key,
payload=pwm_signal, time=time)
@property
def pwm_pin_output_timer_a_duration_key(self):
return self._get_key(MOTOR_TIMER_A_TOTAL_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_a_duration(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_a_duration_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_b_duration_key(self):
return self._get_key(MOTOR_TIMER_B_TOTAL_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_b_duration(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_b_duration_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_c_duration_key(self):
return self._get_key(MOTOR_TIMER_C_TOTAL_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_c_duration(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_c_duration_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_a_channel_0_ratio_key(self):
return self._get_key(
MOTOR_TIMER_A_CHANNEL_0_ACTIVE_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_a_channel_0_ratio(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_a_channel_0_ratio_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_a_channel_1_ratio_key(self):
return self._get_key(
MOTOR_TIMER_A_CHANNEL_1_ACTIVE_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_a_channel_1_ratio(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_a_channel_1_ratio_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_b_channel_0_ratio_key(self):
return self._get_key(
MOTOR_TIMER_B_CHANNEL_0_ACTIVE_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_b_channel_0_ratio(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_b_channel_0_ratio_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_b_channel_1_ratio_key(self):
return self._get_key(
MOTOR_TIMER_B_CHANNEL_1_ACTIVE_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_b_channel_1_ratio(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_b_channel_1_ratio_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_c_channel_0_ratio_key(self):
return self._get_key(
MOTOR_TIMER_C_CHANNEL_0_ACTIVE_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_c_channel_0_ratio(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_c_channel_0_ratio_key,
payload=timer_period, time=time)
@property
def pwm_pin_output_timer_c_channel_1_ratio_key(self):
return self._get_key(
MOTOR_TIMER_C_CHANNEL_1_ACTIVE_PERIOD_KEY, RETINA_UART_SHIFT)
[docs] def pwm_pin_output_timer_c_channel_1_ratio(self, timer_period, time=None):
return MultiCastCommand(
key=self.pwm_pin_output_timer_c_channel_1_ratio_key,
payload=timer_period, time=time)
@property
def query_state_of_io_lines_key(self):
return self._get_key(QUERY_STATES_LINES_KEY)
[docs] def query_state_of_io_lines(self, time=None):
return MultiCastCommand(
key=self.query_state_of_io_lines_key, time=time)
@property
def set_output_pattern_for_payload_key(self):
return self._get_key(SET_OUTPUT_PATTERN_KEY)
[docs] def set_output_pattern_for_payload(self, payload, time=None):
return MultiCastCommand(
key=self.set_output_pattern_for_payload_key, payload=payload,
time=time)
@property
def add_payload_logic_to_current_output_key(self):
return self._get_key(ADD_PAYLOAD_TO_CURRENT_OUTPUT_KEY)
[docs] def add_payload_logic_to_current_output(self, payload, time=None):
return MultiCastCommand(
key=self.add_payload_logic_to_current_output_key,
payload=payload, time=time)
@property
def remove_payload_logic_to_current_output_key(self):
return self._get_key(REMOVE_PAYLOAD_TO_CURRENT_OUTPUT_KEY)
[docs] def remove_payload_logic_to_current_output(self, payload, time=None):
return MultiCastCommand(
key=self.remove_payload_logic_to_current_output_key,
payload=payload, time=time)
@property
def set_payload_pins_to_high_impedance_key(self):
return self._get_key(SET_PAYLOAD_TO_HIGH_IMPEDANCE_KEY)
[docs] def set_payload_pins_to_high_impedance(self, payload, time=None):
return MultiCastCommand(
key=self.set_payload_pins_to_high_impedance_key,
payload=payload, time=time)
def _check_for_pushbot_mode(self):
if self.__mode is not self.MODES.PUSH_BOT:
raise ConfigurationException(
"The mode you configured is not the PushBot, and so this "
"message is invalid for mode {}".format(self.__mode))
@property
def push_bot_laser_config_total_period_key(self):
return self._get_key(
PUSH_BOT_LASER_CONFIG_TOTAL_PERIOD, RETINA_UART_SHIFT)
[docs] def push_bot_laser_config_total_period(self, total_period, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_laser_config_total_period_key,
payload=total_period, time=time)
@property
def push_bot_laser_config_active_time_key(self):
return self._get_key(
PUSH_BOT_LASER_CONFIG_ACTIVE_TIME, RETINA_UART_SHIFT)
[docs] def push_bot_laser_config_active_time(self, active_time, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_laser_config_active_time_key,
payload=active_time, time=time)
@property
def push_bot_laser_set_frequency_key(self):
return self._get_key(
PUSH_BOT_LASER_FREQUENCY,
PUSH_BOT_LASER_LED_SPEAKER_FREQUENCY_UART_SHIFT)
[docs] def push_bot_laser_set_frequency(self, frequency, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_laser_set_frequency_key,
payload=frequency, time=time)
@property
def push_bot_speaker_config_total_period_key(self):
return self._get_key(
PUSH_BOT_SPEAKER_CONFIG_TOTAL_PERIOD, RETINA_UART_SHIFT)
[docs] def push_bot_speaker_config_total_period(
self, total_period, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_speaker_config_total_period_key,
payload=total_period, time=time)
@property
def push_bot_speaker_config_active_time_key(self):
return self._get_key(
PUSH_BOT_SPEAKER_CONFIG_ACTIVE_TIME, RETINA_UART_SHIFT)
[docs] def push_bot_speaker_config_active_time(self, active_time, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_speaker_config_active_time_key,
payload=active_time, time=time)
@property
def push_bot_speaker_set_tone_key(self):
return self._get_key(
PUSH_BOT_SPEAKER_TONE_BEEP,
PUSH_BOT_LASER_LED_SPEAKER_FREQUENCY_UART_SHIFT)
[docs] def push_bot_speaker_set_tone(self, frequency, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_speaker_set_tone_key,
payload=frequency, time=time)
@property
def push_bot_speaker_set_melody_key(self):
return self._get_key(
PUSH_BOT_SPEAKER_TONE_MELODY,
PUSH_BOT_LASER_LED_SPEAKER_FREQUENCY_UART_SHIFT)
[docs] def push_bot_speaker_set_melody(self, melody, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_speaker_set_melody_key,
payload=melody, time=time)
@property
def push_bot_led_total_period_key(self):
return self._get_key(
PUSH_BOT_LED_CONFIG_TOTAL_PERIOD, RETINA_UART_SHIFT)
[docs] def push_bot_led_total_period(self, total_period, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_led_total_period_key,
payload=total_period, time=time)
@property
def push_bot_led_back_active_time_key(self):
return self._get_key(
PUSH_BOT_LED_BACK_CONFIG_ACTIVE_TIME, RETINA_UART_SHIFT)
[docs] def push_bot_led_back_active_time(self, active_time, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_led_back_active_time_key,
payload=active_time, time=time)
@property
def push_bot_led_front_active_time_key(self):
return self._get_key(
PUSH_BOT_LED_FRONT_CONFIG_ACTIVE_TIME, RETINA_UART_SHIFT)
[docs] def push_bot_led_front_active_time(self, active_time, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_led_front_active_time_key,
payload=active_time, time=time)
@property
def push_bot_led_set_frequency_key(self):
return self._get_key(
PUSH_BOT_LED_FREQUENCY,
PUSH_BOT_LASER_LED_SPEAKER_FREQUENCY_UART_SHIFT)
[docs] def push_bot_led_set_frequency(self, frequency, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_led_set_frequency_key,
payload=frequency, time=time)
@property
def push_bot_motor_0_permanent_key(self):
return self._get_key(
PUSH_BOT_MOTOR_0_PERMANENT_VELOCITY, PUSH_BOT_MOTOR_UART_SHIFT)
[docs] def push_bot_motor_0_permanent(self, velocity, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_motor_0_permanent_key,
payload=velocity, time=time)
@property
def push_bot_motor_1_permanent_key(self):
return self._get_key(
PUSH_BOT_MOTOR_1_PERMANENT_VELOCITY, PUSH_BOT_MOTOR_UART_SHIFT)
[docs] def push_bot_motor_1_permanent(self, velocity, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_motor_1_permanent_key,
payload=velocity, time=time)
@property
def push_bot_motor_0_leaking_towards_zero_key(self):
return self._get_key(
PUSH_BOT_MOTOR_0_LEAKY_VELOCITY, PUSH_BOT_MOTOR_UART_SHIFT)
[docs] def push_bot_motor_0_leaking_towards_zero(self, velocity, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_motor_0_leaking_towards_zero_key,
payload=velocity, time=time)
@property
def push_bot_motor_1_leaking_towards_zero_key(self):
return self._get_key(
PUSH_BOT_MOTOR_1_LEAKY_VELOCITY, PUSH_BOT_MOTOR_UART_SHIFT)
[docs] def push_bot_motor_1_leaking_towards_zero(self, velocity, time=None):
self._check_for_pushbot_mode()
return MultiCastCommand(
key=self.push_bot_motor_1_leaking_towards_zero_key,
payload=velocity, time=time)
[docs] def sensor_transmission_key(self, sensor_id):
return ((sensor_id << _SENSOR_OUTGOING_OFFSET_TO_D) |
(self.__uart_id << _SENSOR_OUTGOING_OFFSET_TO_I))
@property
def set_retina_transmission_key(self):
return self._get_key(
ACTIVE_RETINA_EVENT_STREAMING_KEYS_CONFIGURATION,
RETINA_UART_SHIFT)
[docs] def set_retina_transmission(
self, retina_key=RetinaKey.NATIVE_128_X_128,
retina_payload=None, time=None):
""" Set the retina transmission key
:param RetinaKey retina_key: the new key for the retina
:param retina_payload:
the new payload for the set retina key command packet
:type retina_payload: RetinaPayload or None
:param time: when to transmit this packet
:type time: int or float or None
:return: the command to send
:rtype: ~spinn_front_end_common.utility_models.MultiCastCommand
"""
if retina_key == RetinaKey.FIXED_KEY and retina_payload is None:
retina_payload = RetinaPayload.EVENTS_IN_PAYLOAD
if retina_payload is None:
retina_payload = RetinaPayload.NO_PAYLOAD
if (retina_key == RetinaKey.FIXED_KEY and
retina_payload != RetinaPayload.EVENTS_IN_PAYLOAD):
raise ConfigurationException(
"If the Retina Key is FIXED_KEY, the payload must be"
" EVENTS_IN_PAYLOAD")
return MultiCastCommand(
key=self.set_retina_transmission_key,
payload=retina_key.value | retina_payload.value,
time=time)