Added unsigned recursive multiplier and made some bugfixes.

This commit is contained in:
honzastor 2024-03-27 23:00:13 +01:00
parent 2e1694ccd5
commit d013a40145
18 changed files with 593 additions and 68 deletions

View File

@ -5,4 +5,10 @@ cd tests
bash test_mac.sh
bash test_circuits.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

View File

@ -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.
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
```bash
chmod +x yosys_equiv_check.sh

View File

@ -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):
super().__init__(prefix, name, out_N, inner_component, inputs=[a, b], signed=signed, **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)
super().__init__(prefix, name, out_N, inner_component, inputs=[a, b], signed=signed, one_bit_circuit=one_bit_circuit, **kwargs)
""" C CODE GENERATION """
def get_prototype_c(self):

View File

@ -18,13 +18,31 @@ class GeneralCircuit():
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 == "":
self.prefix = name
else:
self.prefix = prefix + "_" + name
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:
outname = self.prefix+"_out"
self.out = Bus(outname, out_N, out_bus=True, signed=signed)
@ -67,7 +85,7 @@ class GeneralCircuit():
Args:
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."
self.components.append(component)
return component
@ -388,7 +406,7 @@ class GeneralCircuit():
"""
# Obtain proper circuit name with its bit width
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(
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
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.
"""
# 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() + \
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
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(
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
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"
# TODO del..
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.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.
"""
# 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(
N=self.N, prefix="b"), name=circuit_type)
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.
"""
# 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)}" + \
f"{self.b.get_wire_assign_blif(output=True)}" + \
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" b[{self.b.bus.index(w)}]={self.b.prefix}[{self.b.bus.index(w)}]" for w in self.b.bus]) + \
"".join(
[f" {circuit_type}_out[{self.out.bus.index(o)}]={o.name}" for o in self.out.bus]) + "\n"
"".join([f" {circuit_type}_out[{self.out.bus.index(o)}]={o.name}" for o in self.out.bus]) + "\n"
def get_circuit_blif(self):
"""Generates hierarchical Blif code subcomponent's function block.
@ -734,7 +752,7 @@ class GeneralCircuit():
"""
# Obtain proper circuit name with its bit width
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(
N=self.N, prefix="b"), name=circuit_prefix, **self._parent_kwargs)
return f"{circuit_block.get_circuit_blif()}"

View File

@ -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 (
UnsignedTruncatedCarrySaveMultiplier
)
from ariths_gen.multi_bit_circuits.approximate_multipliers.recursive_multiplier import (
UnsignedAccurateTwoBitMultiplier,
UnsignedApproximateTwoBitMultiplierM1,
UnsignedApproximateTwoBitMultiplierM2,
UnsignedApproximateTwoBitMultiplierM3,
UnsignedApproximateTwoBitMultiplierM4,
UnsignedRecursiveMultiplier
)

View File

@ -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.")

View File

@ -99,7 +99,7 @@ class Wire():
if self.is_const():
return f"({self.c_const}) << {offset};\n"
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):
"""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.

View File

@ -16,6 +16,7 @@ import itertools
if __name__ == "__main__":
N = 8
directory = f"lib_quad/lib_quad{N}"
os.makedirs(directory, exist_ok=True)
@ -30,7 +31,6 @@ if __name__ == "__main__":
import zipfile
vfile = zipfile.ZipFile(file=f"{directory}/lib_quad_{N}.zip", mode="w", compression=zipfile.ZIP_DEFLATED)
cnt = 0
N = 8
# up to 3 stages
for n in [1, 2, 3]:

View File

@ -65,6 +65,11 @@ from ariths_gen.multi_bit_circuits.multipliers import (
SignedCarrySaveMultiplier
)
from ariths_gen.multi_bit_circuits.approximate_multipliers import (
UnsignedRecursiveMultiplier,
UnsignedAccurateTwoBitMultiplier
)
from ariths_gen.multi_bit_circuits.dividers import (
ArrayDivider
)
@ -226,6 +231,11 @@ if __name__ == "__main__":
name = f"s_arrmul{N}"
circuit = SignedArrayMultiplier(a, b, name=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
name = f"u_csamul_cla{N}"

View File

@ -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 (
Wire,
ConstantWireValue0,
@ -51,7 +59,9 @@ from ariths_gen.multi_bit_circuits.approximate_multipliers import (
UnsignedTruncatedArrayMultiplier,
UnsignedTruncatedCarrySaveMultiplier,
UnsignedBrokenArrayMultiplier,
UnsignedBrokenCarrySaveMultiplier
UnsignedBrokenCarrySaveMultiplier,
UnsignedRecursiveMultiplier,
UnsignedAccurateTwoBitMultiplier
)
from ariths_gen.one_bit_circuits.logic_gates import (
@ -63,8 +73,6 @@ from ariths_gen.one_bit_circuits.logic_gates import (
XnorGate,
NotGate
)
import numpy as np
import math
def test_unsigned_approxmul(values=False):
@ -120,6 +128,19 @@ def test_unsigned_mul():
assert mul(0, 0) == 0
r = mul(av, bv)
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
for c in [UnsignedDaddaMultiplier, UnsignedCarrySaveMultiplier, UnsignedWallaceMultiplier]:
@ -342,6 +363,7 @@ def test_direct():
np.testing.assert_equal(r, expected)
print(r)
def test_wire_as_bus():
""" accept a wire as a bus """
class test_circuit(GeneralCircuit):
@ -353,8 +375,20 @@ def test_wire_as_bus():
self.out[0] = g3.out
circ = test_circuit(Wire("a"), Wire("b"), Bus("c", 2), "c1")
r = circ(np.array([0, 1]),
np.array([0, 1]).reshape(-1, 1),
np.arange(4).reshape(-1, 1, 1))
r = circ(np.array([0, 1]),
np.array([0, 1]).reshape(-1, 1),
np.arange(4).reshape(-1, 1, 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!")

View File

@ -1,21 +1,26 @@
"""
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 import GeneralCircuit
from ariths_gen.wire_components import Bus, Wire
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.multipliers import UnsignedArrayMultiplier, UnsignedDaddaMultiplier
import os, sys
import numpy as np
import itertools
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)
x = 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
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!")

View File

@ -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 (
Wire,
ConstantWireValue0,
@ -57,10 +67,6 @@ from ariths_gen.multi_bit_circuits.approximate_multipliers import (
from ariths_gen.core.cgp_circuit import UnsignedCGPCircuit, SignedCGPCircuit
import numpy as np
import math
from io import StringIO
def test_cgp_unsigned_add():
""" Test unsigned adders """
@ -199,7 +205,7 @@ def test_cgp_signed_add():
np.testing.assert_array_equal(expected, r)
def test_unsigned_mul():
def test_cgp_unsigned_mul():
""" Test unsigned multipliers """
N = 7
a = Bus(N=N, prefix="a")
@ -311,7 +317,7 @@ def test_unsigned_mul():
np.testing.assert_array_equal(expected, r)
def test_signed_mul():
def test_cgp_signed_mul():
""" Test signed multipliers """
N = 7
a = Bus(N=N, prefix="a")
@ -429,3 +435,12 @@ def test_cgp_variant1():
c = UnsignedCGPCircuit(cgp, [8, 8], name="cgp_circuit")
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!")

View File

@ -109,6 +109,7 @@ test_circuit "multiplier_signed" "s_dadda_lfa8"
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_rca8"
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"
@ -253,6 +254,7 @@ fi
# exporting u_ka8
# exporting u_lfa8
# exporting u_arrmul8
# exporting u_recmul8
# exporting u_csamul_cla8"
# exporting u_csamul_rca8"
# exporting u_csamul_pg_rca8"

View File

@ -113,6 +113,7 @@ test_circuit "multiplier_signed" "s_dadda_lfa8"
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_rca8"
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"

View File

@ -122,6 +122,7 @@ test_circuit "multiplier_signed" "s_dadda_lfa8"
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_rca8"
test_circuit "multiplier_unsigned" "u_csamul_pg_rca8"

View File

@ -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 (
Wire,
ConstantWireValue0,
@ -11,10 +21,6 @@ from ariths_gen.multi_bit_circuits.others import (
UnsignedCompareLT
)
import numpy as np
import math
from io import StringIO
def test_compare():
""" Test unsigned comparator """
@ -47,4 +53,6 @@ def test_compare():
np.testing.assert_array_equal(v, expected)
if __name__ == "__main__":
test_compare()
print("Python compare tests were successful!")

View File

@ -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 (
Wire,
ConstantWireValue0,
@ -11,10 +21,6 @@ from ariths_gen.multi_bit_circuits.others import (
UnsignedPopCount
)
import numpy as np
import math
from io import StringIO
def test_popcount():
""" Test unsigned adders """
@ -47,4 +53,6 @@ def test_popcount():
np.testing.assert_array_equal(popcnt(av), expected)
if __name__ == "__main__":
test_popcount()
print("Python popcnt tests were successful!")

View File

@ -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 (
Wire,
ConstantWireValue0,
@ -8,13 +18,11 @@ from ariths_gen.wire_components import (
from ariths_gen.core.arithmetic_circuits import GeneralCircuit
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():
""" Test unsigned adders """
@ -75,4 +83,8 @@ def test_andreduce():
np.testing.assert_array_equal(reduce(av), expected)
if __name__ == "__main__":
test_andreduce()
test_orreduce()
print("Python reduce tests were successful!")