Source code for pacman.operations.routing_table_generators.zoned_routing_table_generator

# Copyright (c) 2019 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 collections import defaultdict
from spinn_utilities.progress_bar import ProgressBar
from spinn_machine import MulticastRoutingEntry
from pacman.model.routing_tables import (
    UnCompressedMulticastRoutingTable, MulticastRoutingTables)


class SharedEntry(object):
    slots = [

        #: the edges this path entry goes down
        "link_ids",

        #: the processors this path entry goes to
        "processor_ids",

        #: whether this entry is defaultable
        "defaultable"
        ]

    def __init__(self, entry):
        """
        :param MulticastRoutingTableByPartitionEntry entry:
        """
        self.link_ids = entry.link_ids
        self.processor_ids = entry.processor_ids
        self.defaultable = entry.defaultable

    def still_defaultable(self, entry):
        """
        :param MulticastRoutingTableByPartitionEntry entry:
        """
        self.defaultable = self.defaultable & entry.defaultable


[docs]class ZonedRoutingTableGenerator(object): """ An algorithm that can produce routing tables in zones. """ __slots__ = []
[docs] def __call__( self, routing_infos, routing_table_by_partitions, machine, info_by_app_vertex): """ :param RoutingInfo routing_infos: :param MulticastRoutingTableByPartition routing_table_by_partitions: :param ~spinn_machine.Machine machine: :param dict(ApplicationVertex,BaseKeyAndMask) info_by_app_vertex: :rtype: MulticastRoutingTables """ progress = ProgressBar(machine.n_chips, "Generating routing tables") routing_tables = MulticastRoutingTables() for chip in progress.over(machine.chips): partitions_in_table = routing_table_by_partitions.\ get_entries_for_router(chip.x, chip.y) if partitions_in_table: routing_tables.add_routing_table(self._create_routing_table( chip, partitions_in_table, routing_infos, info_by_app_vertex)) return routing_tables
def _create_routing_table(self, chip, partitions_in_table, routing_infos, info_by_app_vertex): """ :param ~spinn_machine.Chip chip: :param partitions_in_table: :type partitions_in_table: dict(AbstractSingleSourcePartition, MulticastRoutingTableByPartitionEntry) :param RoutingInfo routing_infos: :param dict(ApplicationVertex,BaseKeyAndMask) info_by_app_vertex: :rtype: MulticastRoutingTable """ table = UnCompressedMulticastRoutingTable(chip.x, chip.y) partitions_by_app_vertex = defaultdict(set) for partition in partitions_in_table: partitions_by_app_vertex[partition.pre_vertex.app_vertex].add( partition) for app_vertex in partitions_by_app_vertex: if app_vertex in info_by_app_vertex: shared_entry = self._find_shared_entry( partitions_by_app_vertex[app_vertex], partitions_in_table) else: shared_entry = None if shared_entry is None: self._add_partition_based( partitions_by_app_vertex[app_vertex], routing_infos, partitions_in_table, table) else: self.__add_key_and_mask( info_by_app_vertex[app_vertex], shared_entry, table) return table def _find_shared_entry(self, partitions, partitions_in_table): """ :param set(AbstractSingleSourcePartition) partitions: :param partitions_in_table: :type partitions_in_table: dict(AbstractSingleSourcePartition, MulticastRoutingTableByPartitionEntry) :rtype: SharedEntry or None """ shared_entry = None for partition in partitions: entry = partitions_in_table[partition] if shared_entry is None: shared_entry = SharedEntry(entry) else: if shared_entry.link_ids != entry.link_ids: return None if shared_entry.processor_ids != entry.processor_ids: return None shared_entry.still_defaultable(entry) return shared_entry def _add_partition_based( self, partitions, routing_infos, partitions_in_table, table): """ :param set(AbstractSingleSourcePartition) partitions: :param RoutingInfo routing_infos: :param partitions_in_table: :type partitions_in_table: dict(AbstractSingleSourcePartition, MulticastRoutingTableByPartitionEntry) :param MulticastRoutingTable table: """ for partition in partitions: r_info = routing_infos.get_routing_info_from_partition(partition) entry = partitions_in_table[partition] self.__add_key_and_mask(r_info.key_and_mask, entry, table) @staticmethod def __add_key_and_mask(key_and_mask, entry, table): """ :param BaseKeyAndMask key_and_mask: :param MulticastRoutingTableByPartitionEntry entry: :param MulticastRoutingTable table: """ table.add_multicast_routing_entry(MulticastRoutingEntry( routing_entry_key=key_and_mask.key_combo, defaultable=entry.defaultable, mask=key_and_mask.mask, link_ids=entry.link_ids, processor_ids=entry.processor_ids))