Source code for spinn_front_end_common.abstract_models.impl.machine_allocation_controller

# 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.

import logging
import sys
from threading import Thread
from typing import Dict, Optional, Tuple
from spinn_utilities.log import FormatAdapter
from spinn_utilities.abstract_base import AbstractBase, abstractmethod
from spinn_utilities.typing.coords import XY
from spinnman.constants import SCP_SCAMP_PORT
from spinnman.connections.udp_packet_connections import SCAMPConnection
from spinnman.transceiver import create_transceiver_from_hostname, Transceiver
from spinnman.connections.udp_packet_connections import EIEIOConnection
logger = FormatAdapter(logging.getLogger(__name__))


class MachineAllocationController(object, metaclass=AbstractBase):
    """
    How to manage the allocation of a machine so that it gets cleaned up
    neatly when the script dies.
    """
    __slots__ = (
        #: Boolean flag for telling this thread when the system has ended
        "_exited",
        #: the address of the root board of the allocation
        "__hostname",
        "__connection_data")

    def __init__(self, thread_name: str, hostname: Optional[str] = None,
                 connection_data: Optional[Dict[XY, str]] = None):
        """
        :param str thread_name:
        """
        thread = Thread(name=thread_name, target=self.__manage_allocation)
        thread.daemon = True
        self._exited = False
        self.__hostname = hostname
        self.__connection_data = connection_data
        thread.start()

[docs] @abstractmethod def extend_allocation(self, new_total_run_time: float): """ Extend the allocation of the machine from the original run time. :param float new_total_run_time: The total run time that is now required starting from when the machine was first allocated """ raise NotImplementedError
[docs] def close(self) -> None: """ Indicate that the use of the machine is complete. """ self._exited = True
@abstractmethod def _wait(self) -> bool: """ Wait for some bounded amount of time for a change in the status of the machine allocation. :return: Whether the machine is still (believed to be) allocated. :rtype: bool """ raise NotImplementedError
[docs] @abstractmethod def where_is_machine( self, chip_x: int, chip_y: int) -> Tuple[int, int, int]: """ Locates and returns cabinet, frame, board for a given chip in a machine allocated to this job. :param int chip_x: chip x location :param int chip_y: chip y location :return: (cabinet, frame, board) :rtype: tuple(int,int,int) """ raise NotImplementedError
def _teardown(self) -> None: """ Perform any extra tear-down that the thread requires. Does not need to be overridden if no action is desired. """ def __manage_allocation(self) -> None: machine_still_allocated = True while machine_still_allocated and not self._exited: machine_still_allocated = self._wait() self._teardown() if not self._exited: logger.error( "The allocated machine has been released before the end of" " the script; this script will now exit") sys.exit(1)
[docs] def create_transceiver(self) -> Transceiver: """ Create a transceiver for talking to the allocated machine, and make sure everything is ready for use (i.e. boot and discover connections if needed). :rtype: ~spinnman.transceiver.Transceiver """ if not self.__hostname: raise NotImplementedError("Needs a hostname") txrx = create_transceiver_from_hostname(self.__hostname) txrx.discover_scamp_connections() return txrx
[docs] def can_create_transceiver(self) -> bool: """ Detects if a call to create_transceiver could work. :rtype: bool """ return self.__hostname is not None
def __host(self, chip_x: int, chip_y: int) -> Optional[str]: if not self.__connection_data: return None return self.__connection_data.get((chip_x, chip_y))
[docs] def open_sdp_connection( self, chip_x: int, chip_y: int, udp_port: int = SCP_SCAMP_PORT) -> Optional[SCAMPConnection]: """ Open a connection to a specific Ethernet-enabled SpiNNaker chip. Caller will have to arrange for SpiNNaker to pay attention to the connection. The coordinates will be job-relative. :param int chip_x: Ethernet-enabled chip X coordinate :param int chip_y: Ethernet-enabled chip Y coordinate :param int udp_port: the UDP port on the chip to connect to; connecting to a non-SCP port will result in a connection that can't easily be configured. :rtype: ~spinnman.connections.udp_packet_connections.SDPConnection """ host = self.__host(chip_x, chip_y) if not host: return None return SCAMPConnection( chip_x=chip_x, chip_y=chip_y, remote_host=host, remote_port=udp_port)
[docs] def open_eieio_connection( self, chip_x: int, chip_y: int) -> Optional[EIEIOConnection]: """ Open an unbound EIEIO connection. This may be used to communicate with any board of the job. :rtype: ~spinnman.connections.udp_packet_connections.EIEIOConnection """ host = self.__host(chip_x, chip_y) if not host: return None return EIEIOConnection(remote_host=host, remote_port=SCP_SCAMP_PORT)
[docs] def open_eieio_listener(self) -> EIEIOConnection: """ Open an unbound EIEIO connection. This may be used to communicate with any board of the job. :rtype: ~spinnman.connections.udp_packet_connections.EIEIOConnection """ return EIEIOConnection()
@property def proxying(self) -> bool: """ Whether this is a proxying connection. False unless overridden. :rtype: bool """ return False
[docs] def make_report(self, filename: str): """ Asks the controller to make a report of details of allocations. By default, this does nothing. """