From c4d060570d3d62c59f4565092bed01019a87d681 Mon Sep 17 00:00:00 2001 From: Robier Al Kaadi <robal695@student.liu.se> Date: Mon, 17 Jun 2024 15:42:01 +0200 Subject: [PATCH] Add is_commutative and is_distributive --- b_asic/core_operations.py | 1 + b_asic/gui_utils/color_button.py | 10 ++- b_asic/operation.py | 39 +++++++++++ b_asic/signal_flow_graph.py | 112 +++++++++++++++++++++++++++++++ 4 files changed, 160 insertions(+), 2 deletions(-) diff --git a/b_asic/core_operations.py b/b_asic/core_operations.py index df0868ee..c670b15a 100644 --- a/b_asic/core_operations.py +++ b/b_asic/core_operations.py @@ -1053,6 +1053,7 @@ class MAD(AbstractOperation): _execution_time: Optional[int] is_swappable = True + is_distributive = True def __init__( self, diff --git a/b_asic/gui_utils/color_button.py b/b_asic/gui_utils/color_button.py index 834d0727..ea54c6b5 100644 --- a/b_asic/gui_utils/color_button.py +++ b/b_asic/gui_utils/color_button.py @@ -31,7 +31,10 @@ class ColorButton(QPushButton): self.set_color(self._default) def set_color(self, color: QColor): - """Set new color.""" + """Set new color. + color : QColor + The new color of the button. + """ if color != self._color: self._color = color self._color_changed.emit(color) @@ -42,7 +45,10 @@ class ColorButton(QPushButton): self.setStyleSheet("") def set_text_color(self, color: QColor): - """Set text color.""" + """Set text color. + color : QColor + The new color of the text in the button. + """ self.setStyleSheet(f"color: {color.name()};") @property diff --git a/b_asic/operation.py b/b_asic/operation.py index 02995d3f..8e550874 100644 --- a/b_asic/operation.py +++ b/b_asic/operation.py @@ -425,6 +425,22 @@ class Operation(GraphComponent, SignalSourceProvider): """ raise NotImplementedError + @property + @abstractmethod + def is_commutative(self) -> bool: + """ + Return True if the operation is commutative. + """ + raise NotImplementedError + + @property + @abstractmethod + def is_distributive(self) -> bool: + """ + Return True if the operation is distributive. + """ + raise NotImplementedError + @property @abstractmethod def is_swappable(self) -> bool: @@ -1064,6 +1080,29 @@ class AbstractOperation(Operation, AbstractGraphComponent): input_.connected_source.operation.is_constant for input_ in self.inputs ) + @property + def is_commutative(self) -> bool: + """ + Checks if the operation is commutative. + + An operation is commutative if the order of the inputs does not change the result. + For example, addition is commutative because `a + b == b + a`, but subtraction is not + because `a - b != b - a`. + + Returns: + bool: True if the operation is commutative, False otherwise. + + """ + # doc-string inherited + if self.input_count == 2: + return self.is_swappable + return False + + @property + def is_distributive(self) -> bool: + # doc-string inherited + return False + @property def is_swappable(self) -> bool: # doc-string inherited diff --git a/b_asic/signal_flow_graph.py b/b_asic/signal_flow_graph.py index cbd491d7..651751f2 100644 --- a/b_asic/signal_flow_graph.py +++ b/b_asic/signal_flow_graph.py @@ -2111,6 +2111,118 @@ class SFG(AbstractOperation): def is_constant(self) -> bool: return all(output.is_constant for output in self._output_operations) + @property + def is_commutative(self) -> bool: + """ + Checks if all operations in the sfg are commutative. + + An operation is considered commutative if it is not an 'in' or 'out' operation, + and its `is_commutative` property is `True`. + + Returns: + bool: `True` if all operations are commutative, `False` otherwise. + """ + return all( + (op.is_commutative if op.type_name() not in ['in', 'out'] else True) + for op in self.split() + ) + + @property + def is_distributive(self) -> bool: + """ + Checks if the sfg is distributive. + + An operation is considered distributive if it can be applied to each element of a set separately. + For example, multiplication is distributive over addition, meaning that `a * (b + c)` is equivalent to `a * b + a * c`. + + Returns: + bool: True if the sfg is distributive, False otherwise. + + Example: + >>> Mad_op = MAD(Input(), Input(), Input()) # Creates an instance of the Mad operation, MAD is defined in b_asic.core_operations + >>> Mad_sfg = Mad_op.to_sfg() # The operation is turned into a sfg + >>> Mad_sfg.is_distributive # True # if the distributive property holds, False otherwise + + """ + structures = [] + operations = self.get_operations_topological_order() + for op in operations: + if not ( + op.type_name() == "in" + or op.type_name() == "out" + or op.type_name() == "c" + or op.type_name() == "t" + ): + structures.append(op) + return ( + all(self.has_distributive_structure(op) for op in structures) + if len(structures) > 1 + else False + ) + + def has_distributive_structure(self, op: Operation) -> bool: + """ + Checks if the sfg contains distributive structures. + Parameters + ========== + op : Operation + The operation that is the start of the structure to check for distributivity. + Returns: + bool: True if a distributive structures is found, False otherwise. + """ + if op.type_name() == 'mac': + return True + elif op.type_name() in ['cmul', 'mul', 'div']: + for subsequent_op in op.subsequent_operations: + if subsequent_op.type_name() in [ + 'add', + 'sub', + 'addsub', + 'min', + 'max', + 'sqrt', + 'rec', + 'out', + 't', + ]: + return True + else: + return False + elif op.type_name() in ['add', 'sub', 'addsub']: + for subsequent_op in op.subsequent_operations: + if subsequent_op.type_name() in [ + 'mul', + 'div', + 'min', + 'max', + 'out', + 'cmul', + 't', + ]: + return True + else: + return False + elif op.type_name() in ['min', 'max']: + for subsequent_op in op.subsequent_operations: + if subsequent_op.type_name() in [ + 'add', + 'sub', + 'addsub', + 'cmul', + 'out', + 't', + ]: + return True + else: + return False + + # elif op.type_name() == 'cmul': + # for subsequent_op in op.subsequent_operations: + # if subsequent_op.type_name() in ['max', 'min', 'out', 't']: + # return True + # else: return False + return False + 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}) -- GitLab