From 1d37163dc4e8252af1bf98ee242be2ce75855cb3 Mon Sep 17 00:00:00 2001
From: Oscar Gustafsson <oscar.gustafsson@gmail.com>
Date: Wed, 10 May 2023 19:20:47 +0200
Subject: [PATCH] Add get_used_type_names methods

---
 b_asic/schedule.py                      |  6 +++-
 b_asic/scheduler_gui/scheduler_event.py |  4 +++
 b_asic/scheduler_gui/scheduler_item.py  |  1 +
 b_asic/signal_flow_graph.py             |  6 ++++
 docs_sphinx/conf.py                     |  6 ++--
 test/test_schedule.py                   | 18 +++++++---
 test/test_sfg.py                        | 47 ++++++++++++++++++-------
 7 files changed, 68 insertions(+), 20 deletions(-)

diff --git a/b_asic/schedule.py b/b_asic/schedule.py
index 61b5bd84..ce5f96e9 100644
--- a/b_asic/schedule.py
+++ b/b_asic/schedule.py
@@ -35,6 +35,7 @@ from b_asic.process import MemoryVariable, OperatorProcess
 from b_asic.resources import ProcessCollection
 from b_asic.signal_flow_graph import SFG
 from b_asic.special_operations import Delay, Input, Output
+from b_asic.types import TypeName
 
 # Need RGB from 0 to 1
 _EXECUTION_TIME_COLOR = tuple(c / 255 for c in EXECUTION_TIME_COLOR)
@@ -557,7 +558,6 @@ class Schedule:
         time : int
             The time to move. If positive move forward, if negative move backward.
         """
-        print(f"schedule.move_operation({graph_id!r}, {time})")
         if graph_id not in self._start_times:
             raise ValueError(f"No operation with graph_id {graph_id} in schedule")
 
@@ -782,6 +782,10 @@ class Schedule:
             self.cyclic,
         )
 
+    def get_used_type_names(self) -> List[TypeName]:
+        """Get a list of all TypeNames used in the Schedule."""
+        return self._sfg.get_used_type_names()
+
     def _get_y_position(
         self, graph_id, operation_height=1.0, operation_gap=None
     ) -> float:
diff --git a/b_asic/scheduler_gui/scheduler_event.py b/b_asic/scheduler_gui/scheduler_event.py
index b7fce257..7610d510 100644
--- a/b_asic/scheduler_gui/scheduler_event.py
+++ b/b_asic/scheduler_gui/scheduler_event.py
@@ -223,6 +223,10 @@ class SchedulerEvent:  # PyQt5
                 math.ceil(pos_y),
                 (pos_y % 1) != 0,
             )
+            print(
+                f"schedule.move_y_location({item.operation.graph_id!r},"
+                f" {math.ceil(pos_y)}, {(pos_y % 1) != 0})"
+            )
             self._signals.redraw_all.emit()
         # Operation has been moved in x-direction
         if redraw:
diff --git a/b_asic/scheduler_gui/scheduler_item.py b/b_asic/scheduler_gui/scheduler_item.py
index 74051251..2f5f6eca 100644
--- a/b_asic/scheduler_gui/scheduler_item.py
+++ b/b_asic/scheduler_gui/scheduler_item.py
@@ -227,6 +227,7 @@ class SchedulerItem(SchedulerEvent, QGraphicsItemGroup):  # PySide2 / PyQt5
         move_time = new_start_time - op_start_time
         if move_time:
             self.schedule.move_operation(item.graph_id, move_time)
+            print(f"schedule.move_operation({item.graph_id!r}, {move_time})")
 
     def is_valid_delta_time(self, delta_time: int) -> bool:
         """
diff --git a/b_asic/signal_flow_graph.py b/b_asic/signal_flow_graph.py
index 5869e38a..43c6d085 100644
--- a/b_asic/signal_flow_graph.py
+++ b/b_asic/signal_flow_graph.py
@@ -1716,3 +1716,9 @@ class SFG(AbstractOperation):
     @property
     def is_constant(self) -> bool:
         return all(output.is_constant for output in self._output_operations)
+
+    def get_used_type_names(self) -> List[TypeName]:
+        """Get a list of all TypeNames used in the SFG."""
+        ret = list({op.type_name() for op in self.operations})
+        ret.sort()
+        return ret
diff --git a/docs_sphinx/conf.py b/docs_sphinx/conf.py
index dc42428a..f8148bc4 100644
--- a/docs_sphinx/conf.py
+++ b/docs_sphinx/conf.py
@@ -71,17 +71,17 @@ sphinx_gallery_conf = {
     'doc_module': ('b_asic',),
     'reference_url': {'b_asic': None},
     'image_scrapers': (
-        'matplotlib',
         qtgallery.qtscraper,
+        'matplotlib',
     ),
     'reset_modules': (
-        'matplotlib',
         qtgallery.reset_qapp,
+        'matplotlib',
     ),
 }
 
 qtgallery_conf = {
-    "xvfb_size": (640, 480),
+    "xvfb_size": (800, 600),
     "xvfb_color_depth": 24,
     "xfvb_use_xauth": False,
     "xfvb_extra_args": [],
diff --git a/test/test_schedule.py b/test/test_schedule.py
index 3a7059b3..e5fc64b3 100644
--- a/test/test_schedule.py
+++ b/test/test_schedule.py
@@ -499,6 +499,11 @@ class TestProcesses:
         pc = secondorder_iir_schedule.get_memory_variables()
         assert len(pc) == 12
 
+    def test_get_operations(self, secondorder_iir_schedule_with_execution_times):
+        pc = secondorder_iir_schedule_with_execution_times.get_operations()
+        assert len(pc) == 13
+        assert all(isinstance(operand, OperatorProcess) for operand in pc.collection)
+
 
 class TestFigureGeneration:
     @pytest.mark.mpl_image_compare(remove_text=True, style='mpl20')
@@ -564,7 +569,12 @@ class TestErrors:
         ):
             Schedule(sfg_simple_filter, scheduling_algorithm="foo")
 
-    def test_get_operations(self, secondorder_iir_schedule_with_execution_times):
-        pc = secondorder_iir_schedule_with_execution_times.get_operations()
-        assert len(pc) == 13
-        assert all(isinstance(operand, OperatorProcess) for operand in pc.collection)
+
+class TestGetUsedTypeNames:
+    def test_secondorder_iir_schedule(self, secondorder_iir_schedule):
+        assert secondorder_iir_schedule.get_used_type_names() == [
+            'add',
+            'cmul',
+            'in',
+            'out',
+        ]
diff --git a/test/test_sfg.py b/test/test_sfg.py
index d88c6289..5cc0d416 100644
--- a/test/test_sfg.py
+++ b/test/test_sfg.py
@@ -83,7 +83,8 @@ class TestPrintSfg:
             sfg.__str__()
             == "id: no_id, \tname: SFG1, \tinputs: {0: '-'}, \toutputs: {0: '-'}\n"
             + "Internal Operations:\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
             + str(sfg.find_by_name("INP1")[0])
             + "\n"
             + str(sfg.find_by_name("INP2")[0])
@@ -92,7 +93,8 @@ class TestPrintSfg:
             + "\n"
             + str(sfg.find_by_name("OUT1")[0])
             + "\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
         )
 
     def test_add_mul(self):
@@ -108,7 +110,8 @@ class TestPrintSfg:
             sfg.__str__()
             == "id: no_id, \tname: mac_sfg, \tinputs: {0: '-'}, \toutputs: {0: '-'}\n"
             + "Internal Operations:\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
             + str(sfg.find_by_name("INP1")[0])
             + "\n"
             + str(sfg.find_by_name("INP2")[0])
@@ -121,7 +124,8 @@ class TestPrintSfg:
             + "\n"
             + str(sfg.find_by_name("OUT1")[0])
             + "\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
         )
 
     def test_constant(self):
@@ -136,7 +140,8 @@ class TestPrintSfg:
             sfg.__str__()
             == "id: no_id, \tname: sfg, \tinputs: {0: '-'}, \toutputs: {0: '-'}\n"
             + "Internal Operations:\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
             + str(sfg.find_by_name("CONST")[0])
             + "\n"
             + str(sfg.find_by_name("INP1")[0])
@@ -145,7 +150,8 @@ class TestPrintSfg:
             + "\n"
             + str(sfg.find_by_name("OUT1")[0])
             + "\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
         )
 
     def test_simple_filter(self, sfg_simple_filter):
@@ -154,7 +160,8 @@ class TestPrintSfg:
             == "id: no_id, \tname: simple_filter, \tinputs: {0: '-'},"
             " \toutputs: {0: '-'}\n"
             + "Internal Operations:\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
             + str(sfg_simple_filter.find_by_name("IN1")[0])
             + "\n"
             + str(sfg_simple_filter.find_by_name("ADD1")[0])
@@ -165,7 +172,8 @@ class TestPrintSfg:
             + "\n"
             + str(sfg_simple_filter.find_by_name("OUT1")[0])
             + "\n"
-            + "----------------------------------------------------------------------------------------------------\n"
+            + "--------------------------------------------------------------------"
+            + "--------------------------------\n"
         )
 
 
@@ -818,7 +826,8 @@ class TestConnectExternalSignalsToComponentsSoloComp:
     def test_connect_external_signals_to_components_operation_tree(
         self, operation_tree
     ):
-        """Replaces an SFG with only a operation_tree component with its inner components
+        """
+        Replaces an SFG with only a operation_tree component with its inner components
         """
         sfg1 = SFG(outputs=[Output(operation_tree)])
         out1 = Output(None, "OUT1")
@@ -832,7 +841,9 @@ class TestConnectExternalSignalsToComponentsSoloComp:
     def test_connect_external_signals_to_components_large_operation_tree(
         self, large_operation_tree
     ):
-        """Replaces an SFG with only a large_operation_tree component with its inner components
+        """
+        Replaces an SFG with only a large_operation_tree component with its inner
+        components
         """
         sfg1 = SFG(outputs=[Output(large_operation_tree)])
         out1 = Output(None, "OUT1")
@@ -1474,8 +1485,8 @@ class TestUnfold:
         for k in count1.keys():
             assert count1[k] * multiple == count2[k]
 
-    # This is horrifying, but I can't figure out a way to run the test on multiple fixtures,
-    # so this is an ugly hack until someone that knows pytest comes along
+    # This is horrifying, but I can't figure out a way to run the test on multiple
+    # fixtures, so this is an ugly hack until someone that knows pytest comes along
     def test_two_inputs_two_outputs(self, sfg_two_inputs_two_outputs: SFG):
         self.do_tests(sfg_two_inputs_two_outputs)
 
@@ -1639,3 +1650,15 @@ class TestInsertComponentAfter:
         sfg = SFG(outputs=[Output(large_operation_tree_names)])
         with pytest.raises(ValueError, match="Unknown component:"):
             sfg.insert_operation_after('foo', SquareRoot())
+
+
+class TestGetUsedTypeNames:
+    def test_single_accumulator(self, sfg_simple_accumulator: SFG):
+        assert sfg_simple_accumulator.get_used_type_names() == ['add', 'in', 'out', 't']
+
+    def test_sfg_nested(self, sfg_nested: SFG):
+        assert sfg_nested.get_used_type_names() == ['in', 'out', 'sfg']
+
+    def test_large_operation_tree(self, large_operation_tree):
+        sfg = SFG(outputs=[Output(large_operation_tree)])
+        assert sfg.get_used_type_names() == ['add', 'c', 'out']
-- 
GitLab