mirror of
https://github.com/ehw-fit/ariths-gen.git
synced 2025-04-04 06:11:41 +01:00
Added unsigned recursive multiplier and made some bugfixes.
This commit is contained in:
parent
2e1694ccd5
commit
d013a40145
@ -5,4 +5,10 @@ cd tests
|
|||||||
bash test_mac.sh
|
bash test_mac.sh
|
||||||
bash test_circuits.sh
|
bash test_circuits.sh
|
||||||
bash test_circuits_verilog.sh
|
bash test_circuits_verilog.sh
|
||||||
bash test_circuits_cgp.sh
|
bash test_circuits_cgp.sh
|
||||||
|
python test_all.py
|
||||||
|
python test_ax.py
|
||||||
|
python test_cgp.py
|
||||||
|
python test_compare.py
|
||||||
|
python test_popcnt.py
|
||||||
|
python test_reduce.py
|
@ -141,6 +141,11 @@ print("Mean average error", np.abs(r - (va * vb)).mean())
|
|||||||
The `yosys_equiv_check.sh` script enables to formally check the equivalence of generated Verilog and BLIF representations of the same circuit.
|
The `yosys_equiv_check.sh` script enables to formally check the equivalence of generated Verilog and BLIF representations of the same circuit.
|
||||||
It uses the Yosys Open SYnthesis Suite tool by Claire Xenia Wolf. For further information, please visit: https://yosyshq.readthedocs.io/projects/yosys/en/latest/index.html.
|
It uses the Yosys Open SYnthesis Suite tool by Claire Xenia Wolf. For further information, please visit: https://yosyshq.readthedocs.io/projects/yosys/en/latest/index.html.
|
||||||
|
|
||||||
|
## Install Yosys
|
||||||
|
```bash
|
||||||
|
sudo apt-get install yosys
|
||||||
|
```
|
||||||
|
|
||||||
## Execute permission
|
## Execute permission
|
||||||
```bash
|
```bash
|
||||||
chmod +x yosys_equiv_check.sh
|
chmod +x yosys_equiv_check.sh
|
||||||
|
@ -18,25 +18,7 @@ class ArithmeticCircuit(GeneralCircuit):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, a, b, prefix: str, name: str, out_N: int, inner_component: bool = False, one_bit_circuit: bool = False, signed: bool = False, **kwargs):
|
def __init__(self, a, b, prefix: str, name: str, out_N: int, inner_component: bool = False, one_bit_circuit: bool = False, signed: bool = False, **kwargs):
|
||||||
super().__init__(prefix, name, out_N, inner_component, inputs=[a, b], signed=signed, **kwargs)
|
super().__init__(prefix, name, out_N, inner_component, inputs=[a, b], signed=signed, one_bit_circuit=one_bit_circuit, **kwargs)
|
||||||
if one_bit_circuit is False:
|
|
||||||
if prefix == "":
|
|
||||||
self.prefix = name
|
|
||||||
else:
|
|
||||||
self.prefix = prefix + "_" + name
|
|
||||||
|
|
||||||
self.inner_component = inner_component
|
|
||||||
if self.inner_component is True:
|
|
||||||
self.a = Bus(prefix=f"{self.prefix}_{a.prefix}", wires_list=a.bus)
|
|
||||||
self.b = Bus(prefix=f"{self.prefix}_{b.prefix}", wires_list=b.bus)
|
|
||||||
|
|
||||||
if a.is_output_bus():
|
|
||||||
self.a.connect_bus(connecting_bus=a)
|
|
||||||
if b.is_output_bus():
|
|
||||||
self.b.connect_bus(connecting_bus=b)
|
|
||||||
else:
|
|
||||||
self.a = Bus(prefix=f"{a.prefix}", wires_list=a.bus)
|
|
||||||
self.b = Bus(prefix=f"{b.prefix}", wires_list=b.bus)
|
|
||||||
|
|
||||||
""" C CODE GENERATION """
|
""" C CODE GENERATION """
|
||||||
def get_prototype_c(self):
|
def get_prototype_c(self):
|
||||||
|
@ -18,13 +18,31 @@ class GeneralCircuit():
|
|||||||
that are later used for generation into various representations.
|
that are later used for generation into various representations.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, prefix: str, name: str, out_N: int, inner_component: bool = False, inputs: list = [], signed: bool = False, outname=None, **kwargs):
|
def __init__(self, prefix: str, name: str, out_N: int, inner_component: bool = False, inputs: list = [], one_bit_circuit: bool = False, signed: bool = False, outname=None, **kwargs):
|
||||||
if prefix == "":
|
if prefix == "":
|
||||||
self.prefix = name
|
self.prefix = name
|
||||||
else:
|
else:
|
||||||
self.prefix = prefix + "_" + name
|
self.prefix = prefix + "_" + name
|
||||||
self.inner_component = inner_component
|
self.inner_component = inner_component
|
||||||
self.inputs = inputs
|
|
||||||
|
if one_bit_circuit is False:
|
||||||
|
# Dynamic input bus assignment
|
||||||
|
self.inputs = []
|
||||||
|
input_names = "abcdefghijklmnopqrstuvwxyz" # This should be enough..
|
||||||
|
assert len(input_names) >= len(inputs)
|
||||||
|
for i, input_bus in enumerate(inputs):
|
||||||
|
attr_name = input_names[i]
|
||||||
|
full_prefix = f"{self.prefix}_{input_bus.prefix}" if self.inner_component else f"{input_bus.prefix}"
|
||||||
|
bus = Bus(prefix=full_prefix, wires_list=input_bus.bus)
|
||||||
|
setattr(self, attr_name, bus)
|
||||||
|
self.inputs.append(bus)
|
||||||
|
|
||||||
|
# If the input bus is an output bus, connect it
|
||||||
|
if input_bus.is_output_bus():
|
||||||
|
getattr(self, attr_name).connect_bus(connecting_bus=input_bus)
|
||||||
|
else:
|
||||||
|
self.inputs = inputs
|
||||||
|
|
||||||
if not outname:
|
if not outname:
|
||||||
outname = self.prefix+"_out"
|
outname = self.prefix+"_out"
|
||||||
self.out = Bus(outname, out_N, out_bus=True, signed=signed)
|
self.out = Bus(outname, out_N, out_bus=True, signed=signed)
|
||||||
@ -67,7 +85,7 @@ class GeneralCircuit():
|
|||||||
Args:
|
Args:
|
||||||
component: Subcomponent to be added into list of components composing described circuit.
|
component: Subcomponent to be added into list of components composing described circuit.
|
||||||
"""
|
"""
|
||||||
prefixes = [c.prefix for c in self.components]
|
prefixes = [c.prefix for c in self.components] # TODO ?
|
||||||
#assert component.prefix not in prefixes, f"Component with prefix {component.prefix} already exists in the circuit."
|
#assert component.prefix not in prefixes, f"Component with prefix {component.prefix} already exists in the circuit."
|
||||||
self.components.append(component)
|
self.components.append(component)
|
||||||
return component
|
return component
|
||||||
@ -388,7 +406,7 @@ class GeneralCircuit():
|
|||||||
"""
|
"""
|
||||||
# Obtain proper circuit name with its bit width
|
# Obtain proper circuit name with its bit width
|
||||||
circuit_prefix = self.__class__(
|
circuit_prefix = self.__class__(
|
||||||
a=Bus("a"), b=Bus("b")).prefix + str(self.N)
|
a=Bus("a", self.N), b=Bus("b", self.N)).prefix + str(self.N)
|
||||||
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
||||||
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
|
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
|
||||||
return f"{circuit_block.get_circuit_c()}\n\n"
|
return f"{circuit_block.get_circuit_c()}\n\n"
|
||||||
@ -433,7 +451,7 @@ class GeneralCircuit():
|
|||||||
str: Hierarchical C code of subcomponent's C function invocation and output assignment.
|
str: Hierarchical C code of subcomponent's C function invocation and output assignment.
|
||||||
"""
|
"""
|
||||||
# Getting name of circuit type for proper C code generation without affecting actual generated composition
|
# Getting name of circuit type for proper C code generation without affecting actual generated composition
|
||||||
circuit_type = self.__class__(a=Bus("a"), b=Bus("b")).prefix + str(self.N)
|
circuit_type = self.__class__(a=Bus("a", self.N), b=Bus("b", self.N)).prefix + str(self.N)
|
||||||
return self.a.return_bus_wires_values_c_hier() + self.b.return_bus_wires_values_c_hier() + \
|
return self.a.return_bus_wires_values_c_hier() + self.b.return_bus_wires_values_c_hier() + \
|
||||||
f" {self.out.prefix} = {circuit_type}({self.a.prefix}, {self.b.prefix});\n"
|
f" {self.out.prefix} = {circuit_type}({self.a.prefix}, {self.b.prefix});\n"
|
||||||
|
|
||||||
@ -539,7 +557,7 @@ class GeneralCircuit():
|
|||||||
"""
|
"""
|
||||||
# Obtain proper circuit name with its bit width
|
# Obtain proper circuit name with its bit width
|
||||||
circuit_prefix = self.__class__(
|
circuit_prefix = self.__class__(
|
||||||
a=Bus("a"), b=Bus("b")).prefix + str(self.N)
|
a=Bus("a", self.N), b=Bus("b", self.N)).prefix + str(self.N)
|
||||||
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
||||||
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
|
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
|
||||||
return f"{circuit_block.get_circuit_v()}\n\n"
|
return f"{circuit_block.get_circuit_v()}\n\n"
|
||||||
@ -563,6 +581,7 @@ class GeneralCircuit():
|
|||||||
"""
|
"""
|
||||||
return "".join(w.get_wire_declaration_v() for w in self.inputs + [self.out]) + "\n"
|
return "".join(w.get_wire_declaration_v() for w in self.inputs + [self.out]) + "\n"
|
||||||
|
|
||||||
|
# TODO del..
|
||||||
return f" wire [{self.a.N-1}:0] {self.a.prefix};\n" + \
|
return f" wire [{self.a.N-1}:0] {self.a.prefix};\n" + \
|
||||||
f" wire [{self.b.N-1}:0] {self.b.prefix};\n" + \
|
f" wire [{self.b.N-1}:0] {self.b.prefix};\n" + \
|
||||||
f" wire [{self.out.N-1}:0] {self.out.prefix};\n"
|
f" wire [{self.out.N-1}:0] {self.out.prefix};\n"
|
||||||
@ -586,7 +605,7 @@ class GeneralCircuit():
|
|||||||
str: Hierarchical Verilog code of subcomponent's module invocation and output assignment.
|
str: Hierarchical Verilog code of subcomponent's module invocation and output assignment.
|
||||||
"""
|
"""
|
||||||
# Getting name of circuit type and insitu copying out bus for proper Verilog code generation without affecting actual generated composition
|
# Getting name of circuit type and insitu copying out bus for proper Verilog code generation without affecting actual generated composition
|
||||||
circuit_type = self.__class__(a=Bus("a"), b=Bus("b")).prefix + str(self.N)
|
circuit_type = self.__class__(a=Bus("a", self.N), b=Bus("b", self.N)).prefix + str(self.N)
|
||||||
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
||||||
N=self.N, prefix="b"), name=circuit_type)
|
N=self.N, prefix="b"), name=circuit_type)
|
||||||
return "".join([c.return_bus_wires_values_v_hier() for c in self.inputs]) + \
|
return "".join([c.return_bus_wires_values_v_hier() for c in self.inputs]) + \
|
||||||
@ -692,14 +711,13 @@ class GeneralCircuit():
|
|||||||
str: Hierarchical Blif code of subcomponent's model invocation and output assignment.
|
str: Hierarchical Blif code of subcomponent's model invocation and output assignment.
|
||||||
"""
|
"""
|
||||||
# Getting name of circuit type for proper Blif code generation without affecting actual generated composition
|
# Getting name of circuit type for proper Blif code generation without affecting actual generated composition
|
||||||
circuit_type = self.__class__(a=Bus("a"), b=Bus("b")).prefix + str(self.N)
|
circuit_type = self.__class__(a=Bus("a", self.N), b=Bus("b", self.N)).prefix + str(self.N)
|
||||||
return f"{self.a.get_wire_assign_blif(output=True)}" + \
|
return f"{self.a.get_wire_assign_blif(output=True)}" + \
|
||||||
f"{self.b.get_wire_assign_blif(output=True)}" + \
|
f"{self.b.get_wire_assign_blif(output=True)}" + \
|
||||||
f".subckt {circuit_type}" + \
|
f".subckt {circuit_type}" + \
|
||||||
"".join([f" a[{self.a.bus.index(w)}]={self.a.prefix}[{self.a.bus.index(w)}]" for w in self.a.bus]) + \
|
"".join([f" a[{self.a.bus.index(w)}]={self.a.prefix}[{self.a.bus.index(w)}]" for w in self.a.bus]) + \
|
||||||
"".join([f" b[{self.b.bus.index(w)}]={self.b.prefix}[{self.b.bus.index(w)}]" for w in self.b.bus]) + \
|
"".join([f" b[{self.b.bus.index(w)}]={self.b.prefix}[{self.b.bus.index(w)}]" for w in self.b.bus]) + \
|
||||||
"".join(
|
"".join([f" {circuit_type}_out[{self.out.bus.index(o)}]={o.name}" for o in self.out.bus]) + "\n"
|
||||||
[f" {circuit_type}_out[{self.out.bus.index(o)}]={o.name}" for o in self.out.bus]) + "\n"
|
|
||||||
|
|
||||||
def get_circuit_blif(self):
|
def get_circuit_blif(self):
|
||||||
"""Generates hierarchical Blif code subcomponent's function block.
|
"""Generates hierarchical Blif code subcomponent's function block.
|
||||||
@ -734,7 +752,7 @@ class GeneralCircuit():
|
|||||||
"""
|
"""
|
||||||
# Obtain proper circuit name with its bit width
|
# Obtain proper circuit name with its bit width
|
||||||
circuit_prefix = self.__class__(
|
circuit_prefix = self.__class__(
|
||||||
a=Bus("a"), b=Bus("b")).prefix + str(self.N)
|
a=Bus("a", self.N), b=Bus("b", self.N)).prefix + str(self.N)
|
||||||
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
circuit_block = self.__class__(a=Bus(N=self.N, prefix="a"), b=Bus(
|
||||||
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
|
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
|
||||||
return f"{circuit_block.get_circuit_blif()}"
|
return f"{circuit_block.get_circuit_blif()}"
|
||||||
|
@ -13,3 +13,12 @@ from ariths_gen.multi_bit_circuits.approximate_multipliers.broken_carry_save_mul
|
|||||||
from ariths_gen.multi_bit_circuits.approximate_multipliers.truncated_carry_save_multiplier import (
|
from ariths_gen.multi_bit_circuits.approximate_multipliers.truncated_carry_save_multiplier import (
|
||||||
UnsignedTruncatedCarrySaveMultiplier
|
UnsignedTruncatedCarrySaveMultiplier
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from ariths_gen.multi_bit_circuits.approximate_multipliers.recursive_multiplier import (
|
||||||
|
UnsignedAccurateTwoBitMultiplier,
|
||||||
|
UnsignedApproximateTwoBitMultiplierM1,
|
||||||
|
UnsignedApproximateTwoBitMultiplierM2,
|
||||||
|
UnsignedApproximateTwoBitMultiplierM3,
|
||||||
|
UnsignedApproximateTwoBitMultiplierM4,
|
||||||
|
UnsignedRecursiveMultiplier
|
||||||
|
)
|
||||||
|
@ -0,0 +1,404 @@
|
|||||||
|
from ariths_gen.wire_components import (
|
||||||
|
ConstantWireValue0,
|
||||||
|
Bus
|
||||||
|
)
|
||||||
|
from ariths_gen.core.arithmetic_circuits import (
|
||||||
|
MultiplierCircuit
|
||||||
|
)
|
||||||
|
from ariths_gen.one_bit_circuits.logic_gates import (
|
||||||
|
AndGate,
|
||||||
|
OrGate,
|
||||||
|
XorGate,
|
||||||
|
NotGate
|
||||||
|
)
|
||||||
|
|
||||||
|
from ariths_gen.multi_bit_circuits.adders import (
|
||||||
|
UnsignedCarryLookaheadAdder
|
||||||
|
)
|
||||||
|
|
||||||
|
import math
|
||||||
|
|
||||||
|
|
||||||
|
class UnsignedAccurateTwoBitMultiplier(MultiplierCircuit):
|
||||||
|
"""Class representing unsigned two-bit accurate multiplier.
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of unsigned two-bit accurate multiplier. Defaults to "".
|
||||||
|
name (str, optional): Name of unsigned two-bit accurate multiplier. Defaults to "u_2bit_accm".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "u_2bit_accm", **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
assert self.N == 2
|
||||||
|
super().__init__(a=a, b=b, prefix=prefix, name=name, out_N=self.N*2, **kwargs)
|
||||||
|
|
||||||
|
# Bus sign extension in case buses have different lengths
|
||||||
|
self.a.bus_extend(N=self.N, prefix=a.prefix)
|
||||||
|
self.b.bus_extend(N=self.N, prefix=b.prefix)
|
||||||
|
|
||||||
|
and_obj1 = AndGate(self.a.get_wire(0), self.b.get_wire(0), prefix=self.prefix+"_and0")
|
||||||
|
and_obj2 = AndGate(self.a.get_wire(0), self.b.get_wire(1), prefix=self.prefix+"_and1")
|
||||||
|
and_obj3 = AndGate(self.a.get_wire(1), self.b.get_wire(0), prefix=self.prefix+"_and2")
|
||||||
|
and_obj4 = AndGate(self.a.get_wire(1), self.b.get_wire(1), prefix=self.prefix+"_and3")
|
||||||
|
|
||||||
|
xor_obj1 = XorGate(and_obj2.out, and_obj3.out, prefix=self.prefix+"_xor0")
|
||||||
|
and_obj5 = AndGate(and_obj2.out, and_obj3.out, prefix=self.prefix+"_and4")
|
||||||
|
xor_obj2 = XorGate(and_obj5.out, and_obj4.out, prefix=self.prefix+"_xor1")
|
||||||
|
and_obj6 = AndGate(and_obj5.out, and_obj4.out, prefix=self.prefix+"_and5")
|
||||||
|
[self.add_component(obj) for obj in [and_obj1, and_obj2, and_obj3, and_obj4, xor_obj1, and_obj5, xor_obj2, and_obj6]]
|
||||||
|
|
||||||
|
self.out.connect(0, and_obj1.out)
|
||||||
|
self.out.connect(1, xor_obj1.out)
|
||||||
|
self.out.connect(2, xor_obj2.out)
|
||||||
|
self.out.connect(3, and_obj6.out)
|
||||||
|
|
||||||
|
|
||||||
|
class SignedAccurateTwoBitMultiplier(MultiplierCircuit):
|
||||||
|
"""Class representing signed two-bit accurate multiplier.
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of signed two-bit accurate multiplier. Defaults to "".
|
||||||
|
name (str, optional): Name of signed two-bit accurate multiplier. Defaults to "s_2bit_accm".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "s_2bit_accm", **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
raise NotImplementedError("SignedAccurateTwoBitMultiplier is not implemented yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class UnsignedApproximateTwoBitMultiplierM1(MultiplierCircuit):
|
||||||
|
"""Class representing unsigned two-bit approximate multiplier variant M1.
|
||||||
|
|
||||||
|
M1 ax variant defined here: https://ieeexplore.ieee.org/document/8727537
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of unsigned two-bit approximate multiplier m1. Defaults to "".
|
||||||
|
name (str, optional): Name of unsigned two-bit approximate multiplier m1. Defaults to "u_2bit_axm1".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "u_2bit_axm1", **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
assert self.N == 2
|
||||||
|
super().__init__(a=a, b=b, prefix=prefix, name=name, out_N=self.N*2, **kwargs)
|
||||||
|
|
||||||
|
# Bus sign extension in case buses have different lengths
|
||||||
|
self.a.bus_extend(N=self.N, prefix=a.prefix)
|
||||||
|
self.b.bus_extend(N=self.N, prefix=b.prefix)
|
||||||
|
|
||||||
|
and_obj1 = AndGate(self.a.get_wire(0), self.b.get_wire(0), prefix=self.prefix+"_and0")
|
||||||
|
and_obj2 = AndGate(self.a.get_wire(0), self.b.get_wire(1), prefix=self.prefix+"_and1")
|
||||||
|
and_obj3 = AndGate(self.a.get_wire(1), self.b.get_wire(0), prefix=self.prefix+"_and2")
|
||||||
|
and_obj4 = AndGate(self.a.get_wire(1), self.b.get_wire(1), prefix=self.prefix+"_and3")
|
||||||
|
|
||||||
|
or_obj1 = OrGate(and_obj2.out, and_obj3.out, prefix=self.prefix+"_or0")
|
||||||
|
[self.add_component(obj) for obj in [and_obj1, and_obj2, and_obj3, and_obj4, or_obj1]]
|
||||||
|
|
||||||
|
self.out.connect(0, and_obj1.out)
|
||||||
|
self.out.connect(1, or_obj1.out)
|
||||||
|
self.out.connect(2, and_obj4.out)
|
||||||
|
self.out.connect(3, ConstantWireValue0())
|
||||||
|
|
||||||
|
|
||||||
|
class SignedApproximateTwoBitMultiplierM1(MultiplierCircuit):
|
||||||
|
"""Class representing signed two-bit approximate multiplier variant M1.
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of signed two-bit approximate multiplier m1. Defaults to "".
|
||||||
|
name (str, optional): Name of signed two-bit approximate multiplier m1. Defaults to "s_2bit_axm1".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "s_2bit_axm1", **kwargs):
|
||||||
|
raise NotImplementedError("SignedApproximateTwoBitMultiplierM1 is not implemented yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class UnsignedApproximateTwoBitMultiplierM2(MultiplierCircuit):
|
||||||
|
"""Class representing unsigned two-bit approximate multiplier variant M2.
|
||||||
|
|
||||||
|
M2 ax variant defined here: https://ieeexplore.ieee.org/document/8727537
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of unsigned two-bit approximate multiplier m1. Defaults to "".
|
||||||
|
name (str, optional): Name of unsigned two-bit approximate multiplier m1. Defaults to "u_2bit_axm1".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "u_2bit_axm1", **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
assert self.N == 2
|
||||||
|
super().__init__(a=a, b=b, prefix=prefix, name=name, out_N=self.N*2, **kwargs)
|
||||||
|
|
||||||
|
# Bus sign extension in case buses have different lengths
|
||||||
|
self.a.bus_extend(N=self.N, prefix=a.prefix)
|
||||||
|
self.b.bus_extend(N=self.N, prefix=b.prefix)
|
||||||
|
|
||||||
|
and_obj1 = AndGate(self.a.get_wire(0), self.b.get_wire(1), prefix=self.prefix+"_and0")
|
||||||
|
and_obj2 = AndGate(self.a.get_wire(1), self.b.get_wire(0), prefix=self.prefix+"_and1")
|
||||||
|
and_obj3 = AndGate(self.a.get_wire(1), self.b.get_wire(1), prefix=self.prefix+"_and2")
|
||||||
|
|
||||||
|
and_obj4 = AndGate(and_obj1.out, and_obj2.out, prefix=self.prefix+"_and3")
|
||||||
|
xor_obj1 = XorGate(and_obj1.out, and_obj2.out, prefix=self.prefix+"_xor0")
|
||||||
|
|
||||||
|
xor_obj2 = XorGate(and_obj4.out, and_obj3.out, prefix=self.prefix+"_xor1")
|
||||||
|
[self.add_component(obj) for obj in [and_obj1, and_obj2, and_obj3, and_obj4, xor_obj1, xor_obj2]]
|
||||||
|
|
||||||
|
self.out.connect(0, and_obj4.out)
|
||||||
|
self.out.connect(1, xor_obj1.out)
|
||||||
|
self.out.connect(2, xor_obj2.out)
|
||||||
|
self.out.connect(3, and_obj4.out)
|
||||||
|
|
||||||
|
|
||||||
|
class SignedApproximateTwoBitMultiplierM2(MultiplierCircuit):
|
||||||
|
"""Class representing signed two-bit approximate multiplier variant M2.
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of signed two-bit approximate multiplier m2. Defaults to "".
|
||||||
|
name (str, optional): Name of signed two-bit approximate multiplier m2. Defaults to "s_2bit_axm2".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "s_2bit_axm2", **kwargs):
|
||||||
|
raise NotImplementedError("SignedApproximateTwoBitMultiplierM2 is not implemented yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class UnsignedApproximateTwoBitMultiplierM3(MultiplierCircuit):
|
||||||
|
"""Class representing unsigned two-bit approximate multiplier variant M3.
|
||||||
|
|
||||||
|
M3 ax variant defined here: https://ieeexplore.ieee.org/document/8727537
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of unsigned two-bit approximate multiplier m3. Defaults to "".
|
||||||
|
name (str, optional): Name of unsigned two-bit approximate multiplier m3. Defaults to "u_2bit_axm3".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "u_2bit_axm3", **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
assert self.N == 2
|
||||||
|
super().__init__(a=a, b=b, prefix=prefix, name=name, out_N=self.N*2, **kwargs)
|
||||||
|
|
||||||
|
# Bus sign extension in case buses have different lengths
|
||||||
|
self.a.bus_extend(N=self.N, prefix=a.prefix)
|
||||||
|
self.b.bus_extend(N=self.N, prefix=b.prefix)
|
||||||
|
|
||||||
|
and_obj1 = AndGate(self.a.get_wire(0), self.b.get_wire(0), prefix=self.prefix+"_and0")
|
||||||
|
and_obj2 = AndGate(self.a.get_wire(0), self.b.get_wire(1), prefix=self.prefix+"_and1")
|
||||||
|
and_obj3 = AndGate(self.a.get_wire(1), self.b.get_wire(0), prefix=self.prefix+"_and2")
|
||||||
|
and_obj4 = AndGate(self.a.get_wire(1), self.b.get_wire(1), prefix=self.prefix+"_and3")
|
||||||
|
|
||||||
|
or_obj1 = OrGate(and_obj2.out, and_obj3.out, prefix=self.prefix+"_xor0")
|
||||||
|
not_obj1 = NotGate(and_obj1.out, prefix=self.prefix+"_not0")
|
||||||
|
|
||||||
|
and_obj5 = AndGate(not_obj1.out, and_obj4.out, prefix=self.prefix+"_and4")
|
||||||
|
and_obj6 = AndGate(and_obj1.out, and_obj4.out, prefix=self.prefix+"_and5")
|
||||||
|
[self.add_component(obj) for obj in [and_obj1, and_obj2, and_obj3, and_obj4, or_obj1, not_obj1, and_obj5, and_obj6]]
|
||||||
|
|
||||||
|
self.out.connect(0, and_obj1.out)
|
||||||
|
self.out.connect(1, or_obj1.out)
|
||||||
|
self.out.connect(2, and_obj5.out)
|
||||||
|
self.out.connect(3, and_obj6.out)
|
||||||
|
|
||||||
|
|
||||||
|
class SignedApproximateTwoBitMultiplierM3(MultiplierCircuit):
|
||||||
|
"""Class representing signed two-bit approximate multiplier variant M3.
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of signed two-bit approximate multiplier m3. Defaults to "".
|
||||||
|
name (str, optional): Name of signed two-bit approximate multiplier m3. Defaults to "s_2bit_axm3".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "s_2bit_axm3", **kwargs):
|
||||||
|
raise NotImplementedError("SignedApproximateTwoBitMultiplierM3 is not implemented yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class UnsignedApproximateTwoBitMultiplierM4(MultiplierCircuit):
|
||||||
|
"""Class representing unsigned two-bit approximate multiplier variant M4.
|
||||||
|
|
||||||
|
M4 ax variant defined here: https://ieeexplore.ieee.org/document/8727537
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of unsigned two-bit approximate multiplier m4. Defaults to "".
|
||||||
|
name (str, optional): Name of unsigned two-bit approximate multiplier m4. Defaults to "u_2bit_axm4".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "u_2bit_axm4", **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
assert self.N == 2
|
||||||
|
super().__init__(a=a, b=b, prefix=prefix, name=name, out_N=self.N*2, **kwargs)
|
||||||
|
|
||||||
|
# Bus sign extension in case buses have different lengths
|
||||||
|
self.a.bus_extend(N=self.N, prefix=a.prefix)
|
||||||
|
self.b.bus_extend(N=self.N, prefix=b.prefix)
|
||||||
|
|
||||||
|
and_obj1 = AndGate(self.a.get_wire(0), self.b.get_wire(0), prefix=self.prefix+"_and0")
|
||||||
|
and_obj2 = AndGate(self.a.get_wire(0), self.b.get_wire(1), prefix=self.prefix+"_and1")
|
||||||
|
and_obj3 = AndGate(self.a.get_wire(1), self.b.get_wire(0), prefix=self.prefix+"_and2")
|
||||||
|
and_obj4 = AndGate(self.a.get_wire(1), self.b.get_wire(1), prefix=self.prefix+"_and3")
|
||||||
|
|
||||||
|
xor_obj1 = XorGate(and_obj2.out, and_obj3.out, prefix=self.prefix+"_xor0")
|
||||||
|
[self.add_component(obj) for obj in [and_obj1, and_obj2, and_obj3, and_obj4, xor_obj1]]
|
||||||
|
|
||||||
|
self.out.connect(0, and_obj1.out)
|
||||||
|
self.out.connect(1, xor_obj1.out)
|
||||||
|
self.out.connect(2, and_obj4.out)
|
||||||
|
self.out.connect(3, ConstantWireValue0())
|
||||||
|
|
||||||
|
|
||||||
|
class SignedApproximateTwoBitMultiplierM4(MultiplierCircuit):
|
||||||
|
"""Class representing signed two-bit approximate multiplier variant M4.
|
||||||
|
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of signed two-bit approximate multiplier m4. Defaults to "".
|
||||||
|
name (str, optional): Name of signed two-bit approximate multiplier m4. Defaults to "s_2bit_axm4".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "s_2bit_axm4", **kwargs):
|
||||||
|
raise NotImplementedError("SignedApproximateTwoBitMultiplierM4 is not implemented yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class UnsignedRecursiveMultiplier(MultiplierCircuit):
|
||||||
|
"""Class representing unsigned recursive multiplier.
|
||||||
|
|
||||||
|
TODO
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of unsigned recursive multiplier. Defaults to "".
|
||||||
|
name (str, optional): Name of unsigned recursive multiplier. Defaults to "u_rm".
|
||||||
|
submultipliers (list[MultiplierCircuit], optional): List of submultipliers.
|
||||||
|
Defaults (if None) to the required number of UnsignedAccurateTwoBitMultiplier instances.
|
||||||
|
unsigned_adder_class_name (str, optional): Unsigned multi bit adder used to obtain final sums of products. Defaults to UnsignedCarryLookaheadAdder.
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "u_rm", submultipliers: list[MultiplierCircuit] = None, unsigned_adder_class_name: str = UnsignedCarryLookaheadAdder, **kwargs):
|
||||||
|
self.N = max(a.N, b.N)
|
||||||
|
assert self.N > 1 and self.N & (self.N-1) == 0 # assure that N is a power of two greater than 1 (So allowed N is 2, 4, 8, ..)
|
||||||
|
super().__init__(a=a, b=b, prefix=prefix, name=name, out_N=self.N*2, **kwargs)
|
||||||
|
|
||||||
|
# Bus sign extension in case buses have different lengths
|
||||||
|
self.a.bus_extend(N=self.N, prefix=a.prefix)
|
||||||
|
self.b.bus_extend(N=self.N, prefix=b.prefix)
|
||||||
|
|
||||||
|
if submultipliers is None: # By default, we assume composition from accurate two bit submultipliers
|
||||||
|
if self.N == 2:
|
||||||
|
submultipliers = [UnsignedAccurateTwoBitMultiplier]
|
||||||
|
else:
|
||||||
|
submultipliers = [UnsignedAccurateTwoBitMultiplier for _ in range((self.N//2)**2)]
|
||||||
|
|
||||||
|
assert (self.N > 2 and len(submultipliers) == (self.N//2)**2) or (self.N == 2 and len(submultipliers) == 1)
|
||||||
|
|
||||||
|
if self.N == 2: # Base case for just one two-bit multiplier
|
||||||
|
# TODO add suffix in ariths_gen rework
|
||||||
|
mult = submultipliers[0](Bus(prefix=self.prefix + "_a", wires_list=self.a.bus), Bus(prefix=self.prefix + "_b", wires_list=self.b.bus), prefix=self.prefix + "_" + str(self.get_instance_num(cls=submultipliers[0])), **kwargs)
|
||||||
|
self.add_component(mult)
|
||||||
|
self.out.connect_bus(mult.out)
|
||||||
|
else:
|
||||||
|
# Levels of construction of the recursive multiplier
|
||||||
|
# recursive_levels = int(math.log2(self.N)-1) # Number of recursive levels based on the power ith power of 2 (e.g. for N=8, we have 2 recursive levels)
|
||||||
|
block_level = 1
|
||||||
|
partial_products = []
|
||||||
|
|
||||||
|
for m in range(len(submultipliers)): # Iterate over all 2-bit submultipliers (equals range(0, 4**recursive_levels))
|
||||||
|
a_bus_offset = 0
|
||||||
|
b_bus_offset = 0
|
||||||
|
curr_level = block_level
|
||||||
|
curr_id = m
|
||||||
|
|
||||||
|
# Determine the wires offsets from MSB (for input bus `a` and `b`) for the current 2-bit submultiplier
|
||||||
|
# There is a pattern, for example for N=8, we have 16 two-bit submultipliers with offsets:
|
||||||
|
# Mult ID: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
|
||||||
|
# MSB a_offsets: 0 0 2 2 0 0 2 2 4 4 6 6 4 4 6 6
|
||||||
|
# MSB b_offsets: 0 2 0 2 4 6 4 6 0 2 0 2 4 6 4 6
|
||||||
|
while curr_level != 0:
|
||||||
|
# A offset
|
||||||
|
if curr_id // ((4**curr_level)//2) != 0:
|
||||||
|
a_bus_offset += 2**curr_level
|
||||||
|
|
||||||
|
# B offset
|
||||||
|
if (curr_id // ((4**curr_level)//4)) % 2 != 0:
|
||||||
|
b_bus_offset += 2**curr_level
|
||||||
|
|
||||||
|
curr_level -= 1
|
||||||
|
curr_id -= (4**curr_level)*((curr_id // (4**curr_level)))
|
||||||
|
|
||||||
|
# Create 2-bit submultiplier with the corresponding input bits
|
||||||
|
# TODO add suffix in ariths_gen rework
|
||||||
|
submult_a_bus = Bus(prefix=f"mult{m}_a", wires_list=self.a.bus[::-1][0+a_bus_offset:0+a_bus_offset+2][::-1], N=2)
|
||||||
|
submult_b_bus = Bus(prefix=f"mult{m}_b", wires_list=self.b.bus[::-1][0+b_bus_offset:0+b_bus_offset+2][::-1], N=2)
|
||||||
|
submult = submultipliers[m](submult_a_bus, submult_b_bus, prefix=self.prefix + "_" + str(self.get_instance_num(cls=submultipliers[m])))
|
||||||
|
self.add_component(submult)
|
||||||
|
|
||||||
|
# Create wire vectors holding partial products for final summation
|
||||||
|
pp = Bus(prefix=f"pp_{m}", N=self.out.N, wires_list=[ConstantWireValue0() for _ in range(self.out.N)])
|
||||||
|
#[pp.connect_bus(submult.out, offset=(self.out.N-4)-(a_bus_offset+b_bus_offset))]
|
||||||
|
[pp.connect((self.out.N-1)-(a_bus_offset+b_bus_offset)-i, submult.out[3-i], inserted_wire_desired_index=3-i) for i in range(4)]
|
||||||
|
partial_products.append(pp)
|
||||||
|
|
||||||
|
# Distinction of levels of blocks to properly instantiate and connect 2-bit submultipliers
|
||||||
|
# For example, for N=8, we have 4 two-bit submultipliers in the first level, but then we have 4 times the
|
||||||
|
# same structure (4 two-bit mults) as the base component for the second level, similarly for N=16, but
|
||||||
|
# with additional third layer (consisting of 16 two-bit submultipliers)
|
||||||
|
if (m+1) % (4**block_level) == 0:
|
||||||
|
block_level += 1 # Increase the block level
|
||||||
|
|
||||||
|
# Create tree of partial product adders
|
||||||
|
while len(partial_products) != 1:
|
||||||
|
N = len(partial_products)//2
|
||||||
|
# Creation of composite unsigned multi bit adders from set of partial product vectors and addition of generated blocks outputs for next iteration
|
||||||
|
for bus_index in range(0, N):
|
||||||
|
# TODO arithsgen_rework: update check for bus declaration and assignment (if true do declare/assign again - here we would not create new bus, just assign the existing one); it should create cleaner outcode with unncecessary new bus declarations
|
||||||
|
adder_name = unsigned_adder_class_name(a=a, b=b).prefix + str(self.get_instance_num(cls=unsigned_adder_class_name))
|
||||||
|
# TODO rework the buses
|
||||||
|
bus_a = Bus(prefix=f"{adder_name}_a", wires_list=partial_products[bus_index].bus, out_bus=True) if partial_products[bus_index].out_bus else partial_products[bus_index]
|
||||||
|
bus_b = Bus(prefix=f"{adder_name}_b", wires_list=partial_products[bus_index+N].bus, out_bus=True) if partial_products[bus_index+N].out_bus else partial_products[bus_index+N]
|
||||||
|
adder = unsigned_adder_class_name(a=bus_a, b=bus_b, prefix=self.prefix, name=adder_name, inner_component=True, **kwargs)
|
||||||
|
self.add_component(adder)
|
||||||
|
partial_products.append(adder.out)
|
||||||
|
|
||||||
|
# Update partial products list for next iteration until it contains only one output vector
|
||||||
|
partial_products = partial_products[2*N:]
|
||||||
|
# Connect the final output of the recursive multiplier
|
||||||
|
self.out.connect_bus(partial_products[0])
|
||||||
|
|
||||||
|
|
||||||
|
class SignedRecursiveMultiplier(MultiplierCircuit):
|
||||||
|
"""Class representing signed recursive multiplier.
|
||||||
|
|
||||||
|
TODO
|
||||||
|
Description of the __init__ method.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
a (Bus): First input bus.
|
||||||
|
b (Bus): Second input bus.
|
||||||
|
prefix (str, optional): Prefix name of signed recursive multiplier. Defaults to "".
|
||||||
|
name (str, optional): Name of signed recursive multiplier. Defaults to "s_rm".
|
||||||
|
"""
|
||||||
|
def __init__(self, a: Bus, b: Bus, prefix: str = "", name: str = "s_rm", **kwargs):
|
||||||
|
raise NotImplementedError("SignedRecursiveMultiplier is not implemented yet.")
|
@ -99,7 +99,7 @@ class Wire():
|
|||||||
if self.is_const():
|
if self.is_const():
|
||||||
return f"({self.c_const}) << {offset};\n"
|
return f"({self.c_const}) << {offset};\n"
|
||||||
else:
|
else:
|
||||||
return f"(({self.prefix} >> {self.index}) & 0x01ull) << {offset};\n"
|
return f"(({self.name} >> 0) & 0x01ull) << {offset};\n"
|
||||||
|
|
||||||
def return_wire_value_c_hier(self, offset: int = 0):
|
def return_wire_value_c_hier(self, offset: int = 0):
|
||||||
"""Retrieves desired bit value from wire represented in C code variable and bitwise shifts it to desired position for storing it within a bus for hierarchical generation.
|
"""Retrieves desired bit value from wire represented in C code variable and bitwise shifts it to desired position for storing it within a bus for hierarchical generation.
|
||||||
|
@ -16,6 +16,7 @@ import itertools
|
|||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|
||||||
|
N = 8
|
||||||
directory = f"lib_quad/lib_quad{N}"
|
directory = f"lib_quad/lib_quad{N}"
|
||||||
os.makedirs(directory, exist_ok=True)
|
os.makedirs(directory, exist_ok=True)
|
||||||
|
|
||||||
@ -30,7 +31,6 @@ if __name__ == "__main__":
|
|||||||
import zipfile
|
import zipfile
|
||||||
vfile = zipfile.ZipFile(file=f"{directory}/lib_quad_{N}.zip", mode="w", compression=zipfile.ZIP_DEFLATED)
|
vfile = zipfile.ZipFile(file=f"{directory}/lib_quad_{N}.zip", mode="w", compression=zipfile.ZIP_DEFLATED)
|
||||||
cnt = 0
|
cnt = 0
|
||||||
N = 8
|
|
||||||
|
|
||||||
# up to 3 stages
|
# up to 3 stages
|
||||||
for n in [1, 2, 3]:
|
for n in [1, 2, 3]:
|
||||||
|
@ -65,6 +65,11 @@ from ariths_gen.multi_bit_circuits.multipliers import (
|
|||||||
SignedCarrySaveMultiplier
|
SignedCarrySaveMultiplier
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from ariths_gen.multi_bit_circuits.approximate_multipliers import (
|
||||||
|
UnsignedRecursiveMultiplier,
|
||||||
|
UnsignedAccurateTwoBitMultiplier
|
||||||
|
)
|
||||||
|
|
||||||
from ariths_gen.multi_bit_circuits.dividers import (
|
from ariths_gen.multi_bit_circuits.dividers import (
|
||||||
ArrayDivider
|
ArrayDivider
|
||||||
)
|
)
|
||||||
@ -226,6 +231,11 @@ if __name__ == "__main__":
|
|||||||
name = f"s_arrmul{N}"
|
name = f"s_arrmul{N}"
|
||||||
circuit = SignedArrayMultiplier(a, b, name=name)
|
circuit = SignedArrayMultiplier(a, b, name=name)
|
||||||
export_circuit(circuit, name)
|
export_circuit(circuit, name)
|
||||||
|
|
||||||
|
# Accurate recursive multiplier
|
||||||
|
name = f"u_recmul{N}"
|
||||||
|
circuit = UnsignedRecursiveMultiplier(a, b, name=name)
|
||||||
|
export_circuit(circuit, name)
|
||||||
|
|
||||||
# Csamul (Braun multiplier) – the ppa adders are also configurable as above if desirable
|
# Csamul (Braun multiplier) – the ppa adders are also configurable as above if desirable
|
||||||
name = f"u_csamul_cla{N}"
|
name = f"u_csamul_cla{N}"
|
||||||
|
@ -1,3 +1,11 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
# Add the parent directory to the system path
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
import numpy as np
|
||||||
|
import math
|
||||||
|
|
||||||
from ariths_gen.wire_components import (
|
from ariths_gen.wire_components import (
|
||||||
Wire,
|
Wire,
|
||||||
ConstantWireValue0,
|
ConstantWireValue0,
|
||||||
@ -51,7 +59,9 @@ from ariths_gen.multi_bit_circuits.approximate_multipliers import (
|
|||||||
UnsignedTruncatedArrayMultiplier,
|
UnsignedTruncatedArrayMultiplier,
|
||||||
UnsignedTruncatedCarrySaveMultiplier,
|
UnsignedTruncatedCarrySaveMultiplier,
|
||||||
UnsignedBrokenArrayMultiplier,
|
UnsignedBrokenArrayMultiplier,
|
||||||
UnsignedBrokenCarrySaveMultiplier
|
UnsignedBrokenCarrySaveMultiplier,
|
||||||
|
UnsignedRecursiveMultiplier,
|
||||||
|
UnsignedAccurateTwoBitMultiplier
|
||||||
)
|
)
|
||||||
|
|
||||||
from ariths_gen.one_bit_circuits.logic_gates import (
|
from ariths_gen.one_bit_circuits.logic_gates import (
|
||||||
@ -63,8 +73,6 @@ from ariths_gen.one_bit_circuits.logic_gates import (
|
|||||||
XnorGate,
|
XnorGate,
|
||||||
NotGate
|
NotGate
|
||||||
)
|
)
|
||||||
import numpy as np
|
|
||||||
import math
|
|
||||||
|
|
||||||
|
|
||||||
def test_unsigned_approxmul(values=False):
|
def test_unsigned_approxmul(values=False):
|
||||||
@ -120,6 +128,19 @@ def test_unsigned_mul():
|
|||||||
assert mul(0, 0) == 0
|
assert mul(0, 0) == 0
|
||||||
r = mul(av, bv)
|
r = mul(av, bv)
|
||||||
np.testing.assert_array_equal(expected, r)
|
np.testing.assert_array_equal(expected, r)
|
||||||
|
|
||||||
|
# Accurate variant of recursive multiplier
|
||||||
|
for c in [UnsignedRecursiveMultiplier]:
|
||||||
|
N_rec = 8
|
||||||
|
a_rec = Bus(N=N_rec, prefix="a")
|
||||||
|
b_rec = Bus(N=N_rec, prefix="b")
|
||||||
|
av_rec = np.arange(2**N_rec)
|
||||||
|
bv_rec = av_rec.reshape(-1, 1)
|
||||||
|
expected_rec = av_rec * bv_rec
|
||||||
|
mul = c(a_rec, b_rec, submultipliers=[UnsignedAccurateTwoBitMultiplier for _ in range((N_rec//2)**2)])
|
||||||
|
assert mul(0, 0) == 0
|
||||||
|
r = mul(av_rec, bv_rec)
|
||||||
|
np.testing.assert_array_equal(expected_rec, r)
|
||||||
|
|
||||||
# Configurable PPA
|
# Configurable PPA
|
||||||
for c in [UnsignedDaddaMultiplier, UnsignedCarrySaveMultiplier, UnsignedWallaceMultiplier]:
|
for c in [UnsignedDaddaMultiplier, UnsignedCarrySaveMultiplier, UnsignedWallaceMultiplier]:
|
||||||
@ -342,6 +363,7 @@ def test_direct():
|
|||||||
np.testing.assert_equal(r, expected)
|
np.testing.assert_equal(r, expected)
|
||||||
print(r)
|
print(r)
|
||||||
|
|
||||||
|
|
||||||
def test_wire_as_bus():
|
def test_wire_as_bus():
|
||||||
""" accept a wire as a bus """
|
""" accept a wire as a bus """
|
||||||
class test_circuit(GeneralCircuit):
|
class test_circuit(GeneralCircuit):
|
||||||
@ -353,8 +375,20 @@ def test_wire_as_bus():
|
|||||||
self.out[0] = g3.out
|
self.out[0] = g3.out
|
||||||
|
|
||||||
circ = test_circuit(Wire("a"), Wire("b"), Bus("c", 2), "c1")
|
circ = test_circuit(Wire("a"), Wire("b"), Bus("c", 2), "c1")
|
||||||
r = circ(np.array([0, 1]),
|
r = circ(np.array([0, 1]),
|
||||||
np.array([0, 1]).reshape(-1, 1),
|
np.array([0, 1]).reshape(-1, 1),
|
||||||
np.arange(4).reshape(-1, 1, 1))
|
np.arange(4).reshape(-1, 1, 1))
|
||||||
assert r.sum() == 1
|
assert r.sum() == 1
|
||||||
assert r[-1, -1, -1] == 1
|
assert r[-1, -1, -1] == 1
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
test_unsigned_approxmul()
|
||||||
|
test_unsigned_mul()
|
||||||
|
test_signed_mul()
|
||||||
|
test_unsigned_add()
|
||||||
|
test_signed_add()
|
||||||
|
test_mac()
|
||||||
|
test_direct()
|
||||||
|
test_wire_as_bus()
|
||||||
|
print("Python tests were successful!")
|
||||||
|
@ -1,21 +1,26 @@
|
|||||||
"""
|
"""
|
||||||
Testing the QuAdder
|
Testing the QuAdder
|
||||||
"""
|
"""
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
# Add the parent directory to the system path
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
import numpy as np
|
||||||
|
import itertools
|
||||||
|
|
||||||
from ariths_gen.core.arithmetic_circuits.arithmetic_circuit import ArithmeticCircuit
|
from ariths_gen.core.arithmetic_circuits.arithmetic_circuit import ArithmeticCircuit
|
||||||
from ariths_gen.core.arithmetic_circuits import GeneralCircuit
|
from ariths_gen.core.arithmetic_circuits import GeneralCircuit
|
||||||
from ariths_gen.wire_components import Bus, Wire
|
from ariths_gen.wire_components import Bus, Wire
|
||||||
from ariths_gen.multi_bit_circuits.adders import UnsignedRippleCarryAdder
|
from ariths_gen.multi_bit_circuits.adders import UnsignedRippleCarryAdder
|
||||||
from ariths_gen.multi_bit_circuits.approximate_adders import QuAdder
|
from ariths_gen.multi_bit_circuits.approximate_adders import QuAdder
|
||||||
from ariths_gen.multi_bit_circuits.multipliers import UnsignedArrayMultiplier, UnsignedDaddaMultiplier
|
from ariths_gen.multi_bit_circuits.multipliers import UnsignedArrayMultiplier, UnsignedDaddaMultiplier
|
||||||
import os, sys
|
|
||||||
|
|
||||||
import numpy as np
|
|
||||||
import itertools
|
|
||||||
|
|
||||||
def test_quadder():
|
def test_quadder():
|
||||||
c = QuAdder(Bus("a", 8), Bus("b", 8), R = [4, 2, 2], P=[0, 2, 2], prefix="quad")
|
c = QuAdder(Bus("a", 8), Bus("b", 8), R=[4, 2, 2], P=[0, 2, 2], prefix="quad")
|
||||||
c.get_v_code_hier(file_object=sys.stdout)
|
c.get_v_code_hier(file_object=sys.stdout)
|
||||||
|
|
||||||
x = np.arange(0, 256).reshape(-1, 1)
|
x = np.arange(0, 256).reshape(-1, 1)
|
||||||
y = np.arange(0, 256).reshape(1, -1)
|
y = np.arange(0, 256).reshape(1, -1)
|
||||||
|
|
||||||
@ -23,4 +28,9 @@ def test_quadder():
|
|||||||
r2 = x + y
|
r2 = x + y
|
||||||
|
|
||||||
assert np.abs(r - r2).max() == 64
|
assert np.abs(r - r2).max() == 64
|
||||||
np.testing.assert_equal(np.abs(r - r2).mean(), 7.5)
|
np.testing.assert_equal(np.abs(r - r2).mean(), 7.5)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
test_quadder()
|
||||||
|
print("Quadder Python tests were successful!")
|
||||||
|
@ -1,3 +1,13 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
# Add the parent directory to the system path
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
import math
|
||||||
|
from io import StringIO
|
||||||
|
|
||||||
from ariths_gen.wire_components import (
|
from ariths_gen.wire_components import (
|
||||||
Wire,
|
Wire,
|
||||||
ConstantWireValue0,
|
ConstantWireValue0,
|
||||||
@ -57,10 +67,6 @@ from ariths_gen.multi_bit_circuits.approximate_multipliers import (
|
|||||||
|
|
||||||
from ariths_gen.core.cgp_circuit import UnsignedCGPCircuit, SignedCGPCircuit
|
from ariths_gen.core.cgp_circuit import UnsignedCGPCircuit, SignedCGPCircuit
|
||||||
|
|
||||||
import numpy as np
|
|
||||||
import math
|
|
||||||
from io import StringIO
|
|
||||||
|
|
||||||
|
|
||||||
def test_cgp_unsigned_add():
|
def test_cgp_unsigned_add():
|
||||||
""" Test unsigned adders """
|
""" Test unsigned adders """
|
||||||
@ -199,7 +205,7 @@ def test_cgp_signed_add():
|
|||||||
np.testing.assert_array_equal(expected, r)
|
np.testing.assert_array_equal(expected, r)
|
||||||
|
|
||||||
|
|
||||||
def test_unsigned_mul():
|
def test_cgp_unsigned_mul():
|
||||||
""" Test unsigned multipliers """
|
""" Test unsigned multipliers """
|
||||||
N = 7
|
N = 7
|
||||||
a = Bus(N=N, prefix="a")
|
a = Bus(N=N, prefix="a")
|
||||||
@ -311,7 +317,7 @@ def test_unsigned_mul():
|
|||||||
np.testing.assert_array_equal(expected, r)
|
np.testing.assert_array_equal(expected, r)
|
||||||
|
|
||||||
|
|
||||||
def test_signed_mul():
|
def test_cgp_signed_mul():
|
||||||
""" Test signed multipliers """
|
""" Test signed multipliers """
|
||||||
N = 7
|
N = 7
|
||||||
a = Bus(N=N, prefix="a")
|
a = Bus(N=N, prefix="a")
|
||||||
@ -429,3 +435,12 @@ def test_cgp_variant1():
|
|||||||
|
|
||||||
c = UnsignedCGPCircuit(cgp, [8, 8], name="cgp_circuit")
|
c = UnsignedCGPCircuit(cgp, [8, 8], name="cgp_circuit")
|
||||||
assert c(0, 0) == 8 # TypeError: 'int' object is not subscriptable
|
assert c(0, 0) == 8 # TypeError: 'int' object is not subscriptable
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
test_cgp_unsigned_add()
|
||||||
|
test_cgp_signed_add()
|
||||||
|
test_cgp_unsigned_mul()
|
||||||
|
test_cgp_signed_mul()
|
||||||
|
test_cgp_variant1()
|
||||||
|
print("CGP Python tests were successful!")
|
||||||
|
@ -109,6 +109,7 @@ test_circuit "multiplier_signed" "s_dadda_lfa8"
|
|||||||
|
|
||||||
|
|
||||||
test_circuit "multiplier_unsigned" "u_arrmul8"
|
test_circuit "multiplier_unsigned" "u_arrmul8"
|
||||||
|
test_circuit "multiplier_unsigned" "u_recmul8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_cla8"
|
test_circuit "multiplier_unsigned" "u_csamul_cla8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_rca8"
|
test_circuit "multiplier_unsigned" "u_csamul_rca8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
|
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
|
||||||
@ -253,6 +254,7 @@ fi
|
|||||||
# exporting u_ka8
|
# exporting u_ka8
|
||||||
# exporting u_lfa8
|
# exporting u_lfa8
|
||||||
# exporting u_arrmul8
|
# exporting u_arrmul8
|
||||||
|
# exporting u_recmul8
|
||||||
# exporting u_csamul_cla8"
|
# exporting u_csamul_cla8"
|
||||||
# exporting u_csamul_rca8"
|
# exporting u_csamul_rca8"
|
||||||
# exporting u_csamul_pg_rca8"
|
# exporting u_csamul_pg_rca8"
|
||||||
|
@ -113,6 +113,7 @@ test_circuit "multiplier_signed" "s_dadda_lfa8"
|
|||||||
|
|
||||||
|
|
||||||
test_circuit "multiplier_unsigned" "u_arrmul8"
|
test_circuit "multiplier_unsigned" "u_arrmul8"
|
||||||
|
test_circuit "multiplier_unsigned" "u_recmul8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_cla8"
|
test_circuit "multiplier_unsigned" "u_csamul_cla8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_rca8"
|
test_circuit "multiplier_unsigned" "u_csamul_rca8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
|
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
|
||||||
|
@ -122,6 +122,7 @@ test_circuit "multiplier_signed" "s_dadda_lfa8"
|
|||||||
|
|
||||||
|
|
||||||
test_circuit "multiplier_unsigned" "u_arrmul8"
|
test_circuit "multiplier_unsigned" "u_arrmul8"
|
||||||
|
test_circuit "multiplier_unsigned" "u_recmul8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_cla8"
|
test_circuit "multiplier_unsigned" "u_csamul_cla8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_rca8"
|
test_circuit "multiplier_unsigned" "u_csamul_rca8"
|
||||||
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
|
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
|
||||||
|
@ -1,3 +1,13 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
# Add the parent directory to the system path
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
import math
|
||||||
|
from io import StringIO
|
||||||
|
|
||||||
from ariths_gen.wire_components import (
|
from ariths_gen.wire_components import (
|
||||||
Wire,
|
Wire,
|
||||||
ConstantWireValue0,
|
ConstantWireValue0,
|
||||||
@ -11,10 +21,6 @@ from ariths_gen.multi_bit_circuits.others import (
|
|||||||
UnsignedCompareLT
|
UnsignedCompareLT
|
||||||
)
|
)
|
||||||
|
|
||||||
import numpy as np
|
|
||||||
import math
|
|
||||||
from io import StringIO
|
|
||||||
|
|
||||||
|
|
||||||
def test_compare():
|
def test_compare():
|
||||||
""" Test unsigned comparator """
|
""" Test unsigned comparator """
|
||||||
@ -47,4 +53,6 @@ def test_compare():
|
|||||||
np.testing.assert_array_equal(v, expected)
|
np.testing.assert_array_equal(v, expected)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
test_compare()
|
||||||
|
print("Python compare tests were successful!")
|
@ -1,3 +1,13 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
# Add the parent directory to the system path
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
import math
|
||||||
|
from io import StringIO
|
||||||
|
|
||||||
from ariths_gen.wire_components import (
|
from ariths_gen.wire_components import (
|
||||||
Wire,
|
Wire,
|
||||||
ConstantWireValue0,
|
ConstantWireValue0,
|
||||||
@ -11,10 +21,6 @@ from ariths_gen.multi_bit_circuits.others import (
|
|||||||
UnsignedPopCount
|
UnsignedPopCount
|
||||||
)
|
)
|
||||||
|
|
||||||
import numpy as np
|
|
||||||
import math
|
|
||||||
from io import StringIO
|
|
||||||
|
|
||||||
|
|
||||||
def test_popcount():
|
def test_popcount():
|
||||||
""" Test unsigned adders """
|
""" Test unsigned adders """
|
||||||
@ -47,4 +53,6 @@ def test_popcount():
|
|||||||
np.testing.assert_array_equal(popcnt(av), expected)
|
np.testing.assert_array_equal(popcnt(av), expected)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
test_popcount()
|
||||||
|
print("Python popcnt tests were successful!")
|
||||||
|
@ -1,3 +1,13 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
# Add the parent directory to the system path
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
import math
|
||||||
|
from io import StringIO
|
||||||
|
|
||||||
from ariths_gen.wire_components import (
|
from ariths_gen.wire_components import (
|
||||||
Wire,
|
Wire,
|
||||||
ConstantWireValue0,
|
ConstantWireValue0,
|
||||||
@ -8,13 +18,11 @@ from ariths_gen.wire_components import (
|
|||||||
from ariths_gen.core.arithmetic_circuits import GeneralCircuit
|
from ariths_gen.core.arithmetic_circuits import GeneralCircuit
|
||||||
|
|
||||||
from ariths_gen.multi_bit_circuits.others import (
|
from ariths_gen.multi_bit_circuits.others import (
|
||||||
BitReduce, AndReduce, OrReduce
|
BitReduce,
|
||||||
|
AndReduce,
|
||||||
|
OrReduce
|
||||||
)
|
)
|
||||||
|
|
||||||
import numpy as np
|
|
||||||
import math
|
|
||||||
from io import StringIO
|
|
||||||
|
|
||||||
|
|
||||||
def test_orreduce():
|
def test_orreduce():
|
||||||
""" Test unsigned adders """
|
""" Test unsigned adders """
|
||||||
@ -75,4 +83,8 @@ def test_andreduce():
|
|||||||
|
|
||||||
np.testing.assert_array_equal(reduce(av), expected)
|
np.testing.assert_array_equal(reduce(av), expected)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
test_andreduce()
|
||||||
|
test_orreduce()
|
||||||
|
print("Python reduce tests were successful!")
|
||||||
|
Loading…
x
Reference in New Issue
Block a user