Init
This commit is contained in:
commit
b7f83bfed2
11
MTCodes/CMakeLists.txt
Normal file
11
MTCodes/CMakeLists.txt
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
file(GLOB FILENAMES *.cpp)
|
||||||
|
|
||||||
|
foreach(filename ${FILENAMES})
|
||||||
|
get_filename_component(basename ${filename} NAME_WE)
|
||||||
|
add_executable(${basename} ${filename})
|
||||||
|
target_link_libraries(${basename} PUBLIC mockturtle)
|
||||||
|
|
||||||
|
if(EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/${basename}.cmake")
|
||||||
|
include(${basename}.cmake)
|
||||||
|
endif()
|
||||||
|
endforeach()
|
26
MTCodes/generate.cpp
Normal file
26
MTCodes/generate.cpp
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
#include <lorina/aiger.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
std::vector<mockturtle::signal<mockturtle::mig_network>> a;
|
||||||
|
std::vector<mockturtle::signal<mockturtle::mig_network>> b;
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < 8; i++) {
|
||||||
|
a.push_back(mig.create_pi());
|
||||||
|
b.push_back(mig.create_pi());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto out = mockturtle::carry_ripple_multiplier(mig, a, b);
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < out.size(); i++) {
|
||||||
|
mig.create_po(out[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
17
MTCodes/passAig.cpp
Normal file
17
MTCodes/passAig.cpp
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#include <lorina/aiger.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
auto const result = lorina::read_aiger("aig.aig", mockturtle::aiger_reader(mig) );
|
||||||
|
if ( result != lorina::return_code::success ) {
|
||||||
|
std::cout << "Read AIG failed\n";
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
34
MTCodes/passMig.cpp
Normal file
34
MTCodes/passMig.cpp
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
#include <lorina/aiger.hpp>
|
||||||
|
#include <lorina/bench.hpp>
|
||||||
|
#include <lorina/blif.hpp>
|
||||||
|
#include <lorina/pla.hpp>
|
||||||
|
#include <lorina/verilog.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
#include <lorina/diagnostics.hpp>
|
||||||
|
|
||||||
|
class diagnostics : public lorina::diagnostic_consumer
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void handle_diagnostic( lorina::diagnostic_level level, std::string const& message ) const override {
|
||||||
|
printf("[%d] %s\n", int(level), message.c_str());
|
||||||
|
}
|
||||||
|
}; /* diagnostics */
|
||||||
|
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
diagnostics consumer;
|
||||||
|
lorina::diagnostic_engine diag( &consumer );
|
||||||
|
|
||||||
|
auto const result = lorina::read_verilog("adder.v", mockturtle::verilog_reader(mig), &diag);
|
||||||
|
if ( result != lorina::return_code::success ) {
|
||||||
|
std::cout << "Read Verilog failed " << int(result) << "\n";
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
33
MTCodes/resubstitution.cpp
Normal file
33
MTCodes/resubstitution.cpp
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#include <lorina/aiger.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
auto const result = lorina::read_aiger("aig.aig", mockturtle::aiger_reader(mig) );
|
||||||
|
if ( result != lorina::return_code::success ) {
|
||||||
|
std::cout << "Read AIG failed\n";
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//apply rewriring
|
||||||
|
//mockturtle::mig_algebraic_depth_rewriting(mig);
|
||||||
|
|
||||||
|
mockturtle::resubstitution_params ps;
|
||||||
|
mockturtle::resubstitution_stats st;
|
||||||
|
|
||||||
|
ps.max_pis = 8u;
|
||||||
|
ps.max_inserts = 1u;
|
||||||
|
ps.progress = false;
|
||||||
|
|
||||||
|
mockturtle::depth_view depth_mig{ mig };
|
||||||
|
mockturtle::fanout_view fanout_mig{ depth_mig };
|
||||||
|
|
||||||
|
mockturtle::mig_resubstitution( fanout_mig, ps, &st );
|
||||||
|
mig = mockturtle::cleanup_dangling( mig );
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
31
MTCodes/rewriting.cpp
Normal file
31
MTCodes/rewriting.cpp
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#include <lorina/aiger.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
auto const result = lorina::read_aiger("aig.aig", mockturtle::aiger_reader(mig) );
|
||||||
|
if ( result != lorina::return_code::success ) {
|
||||||
|
std::cout << "Read AIG failed\n";
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
//apply rewriring
|
||||||
|
//mockturtle::mig_algebraic_depth_rewriting(mig);
|
||||||
|
|
||||||
|
mockturtle::mig_algebraic_depth_rewriting_params rp;
|
||||||
|
|
||||||
|
//rp.strategy = mockturtle::mig_algebraic_depth_rewriting_params::aggressive;
|
||||||
|
//rp.overhead = 1.5;
|
||||||
|
|
||||||
|
mockturtle::depth_view depth_mig{ mig };
|
||||||
|
mockturtle::fanout_view fanout_mig{ depth_mig };
|
||||||
|
|
||||||
|
mockturtle::mig_algebraic_depth_rewriting(fanout_mig, rp);
|
||||||
|
mig = mockturtle::cleanup_dangling( mig );
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
4
formal.ys
Normal file
4
formal.ys
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
read_verilog mig.v
|
||||||
|
hierarchy
|
||||||
|
proc; opt; memory; opt; techmap; opt
|
||||||
|
write_blif formal.blif
|
18
setup.sh
Normal file
18
setup.sh
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
git clone https://github.com/ehw-fit/ariths-gen
|
||||||
|
git clone https://github.com/Lukas0025/ariths-gen-mig
|
||||||
|
git clone https://github.com/lsils/mockturtle
|
||||||
|
|
||||||
|
rm -rf mockturtle/examples/*
|
||||||
|
cp MTCodes/* mockturtle/examples/
|
||||||
|
cd mockturtle
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake ..
|
||||||
|
make
|
||||||
|
cd ../..
|
||||||
|
|
||||||
|
python3 test.py
|
||||||
|
|
||||||
|
rm -rf ariths-gen
|
||||||
|
rm -rf ariths-gen-mig
|
||||||
|
rm -rf mockturtle
|
510
test.py
Normal file
510
test.py
Normal file
@ -0,0 +1,510 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
import pyverilog
|
||||||
|
from pyverilog.dataflow.dataflow_analyzer import VerilogDataflowAnalyzer
|
||||||
|
|
||||||
|
import graphviz
|
||||||
|
import networkx as nx
|
||||||
|
import os
|
||||||
|
import numpy as np
|
||||||
|
|
||||||
|
def toOrientedGraph(nodes):
|
||||||
|
tmp = []
|
||||||
|
for node in nodes.values():
|
||||||
|
for ins in node.getIns():
|
||||||
|
opts = {'style': 'solid'}
|
||||||
|
|
||||||
|
if ins["inv"]:
|
||||||
|
opts['style'] = 'dashed'
|
||||||
|
|
||||||
|
tmp.append(
|
||||||
|
(
|
||||||
|
ins["node"],
|
||||||
|
node.getName(),
|
||||||
|
opts
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return tmp
|
||||||
|
|
||||||
|
depthBuffer = {}
|
||||||
|
|
||||||
|
def getSubDelay(node, nodes, depth):
|
||||||
|
global depthBuffer
|
||||||
|
|
||||||
|
if str(node.getName()) in depthBuffer:
|
||||||
|
return depthBuffer[str(node.getName())]
|
||||||
|
|
||||||
|
maxLen = 0
|
||||||
|
if depth >= 100:
|
||||||
|
maxLen = np.inf
|
||||||
|
else:
|
||||||
|
for ins in node.getIns():
|
||||||
|
subLen = getSubDelay(nodes[str(ins["node"])], nodes, depth + 1)
|
||||||
|
if subLen > maxLen:
|
||||||
|
maxLen = subLen
|
||||||
|
|
||||||
|
depthBuffer[str(node.getName())] = maxLen + 1
|
||||||
|
|
||||||
|
return maxLen + 1
|
||||||
|
|
||||||
|
def getDelay(nodes):
|
||||||
|
global depthBuffer
|
||||||
|
|
||||||
|
depthBuffer.clear() # clear deph buffer
|
||||||
|
|
||||||
|
maxLen = 0
|
||||||
|
for node in nodes.values():
|
||||||
|
if node.isOutput():
|
||||||
|
subLen = getSubDelay(node, nodes, 0)
|
||||||
|
if subLen > maxLen:
|
||||||
|
maxLen = subLen
|
||||||
|
return maxLen - 2 # exclude input and output node
|
||||||
|
|
||||||
|
def countGates(nodes):
|
||||||
|
count = 0
|
||||||
|
for node in nodes.values():
|
||||||
|
if node.tree is not None and node.tree.__class__.__name__ == "DFOperator":
|
||||||
|
count += 1
|
||||||
|
return count
|
||||||
|
|
||||||
|
class node:
|
||||||
|
def __init__(self, name, types):
|
||||||
|
self.name = name
|
||||||
|
self.types = types
|
||||||
|
self.tree = None
|
||||||
|
self.ins = []
|
||||||
|
|
||||||
|
# ((a >> 0) & 0x01) & ((b >> 0) & 0x01)) | (((b >> 0) & 0x01) & (0x01)) | (((a >> 0) & 0x01) & (0x01))
|
||||||
|
def addMajiIn(self, tree, pos1, pos2, pos3):
|
||||||
|
a = None
|
||||||
|
aInv = False
|
||||||
|
|
||||||
|
if not(str(tree) == "Or" and str(tree.nextnodes[0]) == "Or" and str(tree.nextnodes[1]) == "And" and str(tree.nextnodes[0].nextnodes[0]) == "And" and str(tree.nextnodes[0].nextnodes[1]) == "And"):
|
||||||
|
return a, aInv
|
||||||
|
|
||||||
|
if tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].__class__.__name__ == "DFOperator": # is classic input
|
||||||
|
if tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].operator == "Unot":
|
||||||
|
if tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].nextnodes[0].__class__.__name__ == "DFIntConst": # is constant
|
||||||
|
a = str(tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].nextnodes[0])
|
||||||
|
else:
|
||||||
|
a = tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].nextnodes[0].nextnodes[0].nextnodes[0].name
|
||||||
|
aInv = True
|
||||||
|
else:
|
||||||
|
a = tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].nextnodes[0].nextnodes[0].name
|
||||||
|
elif tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].__class__.__name__ == "DFIntConst": # is constant
|
||||||
|
a = str(tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3])
|
||||||
|
else:
|
||||||
|
return None, False
|
||||||
|
|
||||||
|
return a, aInv
|
||||||
|
|
||||||
|
# ( ~x0 & n18 ) | ( ~x0 & n19 ) | ( n18 & n19 )
|
||||||
|
def addMTMaji(self, tree, pos1, pos2, pos3):
|
||||||
|
a = None
|
||||||
|
aInv = False
|
||||||
|
|
||||||
|
if not(str(tree) == "Or" and str(tree.nextnodes[0]) == "Or" and str(tree.nextnodes[1]) == "And" and str(tree.nextnodes[0].nextnodes[0]) == "And" and str(tree.nextnodes[0].nextnodes[1]) == "And"):
|
||||||
|
return a, aInv
|
||||||
|
|
||||||
|
if tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].__class__.__name__ == "DFOperator": # is classic input
|
||||||
|
if tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].operator == "Unot":
|
||||||
|
a = str(tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3].nextnodes[0])
|
||||||
|
aInv = True
|
||||||
|
else:
|
||||||
|
return None, False
|
||||||
|
else:
|
||||||
|
a = str(tree.nextnodes[pos1].nextnodes[pos2].nextnodes[pos3])
|
||||||
|
|
||||||
|
return a, aInv
|
||||||
|
|
||||||
|
def parseMaji(self, tree):
|
||||||
|
if tree.__class__.__name__ == "DFOperator":
|
||||||
|
a, aInv, b, bInv, c, cInv = None, None, None, None, None, None
|
||||||
|
|
||||||
|
try:
|
||||||
|
a, aInv = self.addMajiIn(tree, 0, 0, 0)
|
||||||
|
b, bInv = self.addMajiIn(tree, 0, 0, 1)
|
||||||
|
c, cInv = self.addMajiIn(tree, 0, 1, 1)
|
||||||
|
except:
|
||||||
|
a, aInv, b, bInv, c, cInv = None, None, None, None, None, None
|
||||||
|
|
||||||
|
try:
|
||||||
|
if a is None or b is None or c is None:
|
||||||
|
a, aInv = self.addMTMaji(tree, 0, 0, 0)
|
||||||
|
b, bInv = self.addMTMaji(tree, 0, 0, 1)
|
||||||
|
c, cInv = self.addMTMaji(tree, 0, 1, 1)
|
||||||
|
except:
|
||||||
|
a, aInv, b, bInv, c, cInv = None, None, None, None, None, None
|
||||||
|
|
||||||
|
if a is None or b is None or c is None:
|
||||||
|
self.addInFromTree(tree, False)
|
||||||
|
else:
|
||||||
|
self.addIn(a, aInv)
|
||||||
|
self.addIn(b, bInv)
|
||||||
|
self.addIn(c, cInv)
|
||||||
|
else:
|
||||||
|
self.addInFromTree(tree, False)
|
||||||
|
|
||||||
|
def addInFromTree(self, tree, invert = False):
|
||||||
|
if tree.__class__.__name__ == "DFOperator":
|
||||||
|
for node in tree.nextnodes:
|
||||||
|
self.addInFromTree(node, invert)
|
||||||
|
elif tree.__class__.__name__ == "DFIntConst":
|
||||||
|
self.addIn(str(tree), invert)
|
||||||
|
else:
|
||||||
|
self.addIn(tree.name, invert)
|
||||||
|
|
||||||
|
def isInput(self):
|
||||||
|
return "Input" in self.types
|
||||||
|
|
||||||
|
def isOutput(self):
|
||||||
|
return "Output" in self.types
|
||||||
|
|
||||||
|
def getIns(self):
|
||||||
|
return self.ins
|
||||||
|
|
||||||
|
def getName(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
def addIn(self, name, invert = False):
|
||||||
|
self.ins.append({
|
||||||
|
"node": name,
|
||||||
|
"inv": invert
|
||||||
|
})
|
||||||
|
|
||||||
|
def getNodes(file, top = "top"):
|
||||||
|
nodes = {}
|
||||||
|
|
||||||
|
nodes['0'] = node('0', {"Input"})
|
||||||
|
nodes['1'] = node('1', {"Input"})
|
||||||
|
|
||||||
|
|
||||||
|
analyzer = VerilogDataflowAnalyzer([file], top)
|
||||||
|
analyzer.generate()
|
||||||
|
|
||||||
|
# First create nodes
|
||||||
|
terms = analyzer.getTerms()
|
||||||
|
for tk, tv in sorted(terms.items(), key=lambda x: str(x[0])):
|
||||||
|
nodes[str(tv.name)] = node(tv.name, tv.termtype)
|
||||||
|
|
||||||
|
# Create connections between nodes
|
||||||
|
binddict = analyzer.getBinddict()
|
||||||
|
for bk, bv in sorted(binddict.items(), key=lambda x: str(x[0])):
|
||||||
|
for bvi in bv:
|
||||||
|
nodes[str(bvi.dest)].parseMaji(bvi.tree)
|
||||||
|
nodes[str(bvi.dest)].tree = bvi.tree
|
||||||
|
|
||||||
|
return nodes
|
||||||
|
|
||||||
|
def plotNodes(nodes, name):
|
||||||
|
# create a directed multi-graph
|
||||||
|
G = nx.MultiDiGraph()
|
||||||
|
G.add_edges_from(toOrientedGraph(nodes))
|
||||||
|
# plot the graph
|
||||||
|
a = nx.nx_agraph.to_agraph(G)
|
||||||
|
src = graphviz.Source(a.to_string())
|
||||||
|
src.render(name)
|
||||||
|
|
||||||
|
def generateAg(name):
|
||||||
|
return """from io import StringIO
|
||||||
|
import os, sys
|
||||||
|
|
||||||
|
DIR_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||||
|
sys.path.insert(0, os.path.join(DIR_PATH, '..'))
|
||||||
|
|
||||||
|
from ariths_gen.core.arithmetic_circuits import GeneralCircuit
|
||||||
|
from ariths_gen.multi_bit_circuits.adders import UnsignedRippleCarryAdder, UnsignedCarryLookaheadAdder, UnsignedBrentKungAdder, UnsignedCarryIncrementAdder, UnsignedCarrySkipAdder, UnsignedCarrySelectAdder, UnsignedConditionalSumAdder, UnsignedHanCarlsonAdder, UnsignedKnowlesAdder, UnsignedKoggeStoneAdder, UnsignedLadnerFischerAdder, UnsignedPGRippleCarryAdder, UnsignedSklanskyAdder
|
||||||
|
from ariths_gen.multi_bit_circuits.multipliers import UnsignedArrayMultiplier, UnsignedCarrySaveMultiplier, UnsignedDaddaMultiplier, UnsignedWallaceMultiplier
|
||||||
|
from ariths_gen.wire_components import Bus
|
||||||
|
|
||||||
|
from ariths_gen.pdk import *
|
||||||
|
import os
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
uut = """ + name + """
|
||||||
|
maji = uut(Bus("a", 8), Bus("b", 8), prefix="", name="top")
|
||||||
|
|
||||||
|
maji.get_v_code_flat(open("mig.v", "w"))
|
||||||
|
"""
|
||||||
|
|
||||||
|
def generateMT(name, outputA = True):
|
||||||
|
if outputA:
|
||||||
|
return """#include <lorina/aiger.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
std::vector<mockturtle::signal<mockturtle::mig_network>> a;
|
||||||
|
std::vector<mockturtle::signal<mockturtle::mig_network>> b;
|
||||||
|
auto cin = mig.get_constant(false);
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < 8; i++) {
|
||||||
|
a.push_back(mig.create_pi());
|
||||||
|
b.push_back(mig.create_pi());
|
||||||
|
}
|
||||||
|
|
||||||
|
mockturtle::""" + name + """(mig, a, b, cin);
|
||||||
|
|
||||||
|
auto out = a;
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < out.size(); i++) {
|
||||||
|
mig.create_po(out[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
else:
|
||||||
|
return """#include <lorina/aiger.hpp>
|
||||||
|
#include <mockturtle/mockturtle.hpp>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
mockturtle::mig_network mig;
|
||||||
|
|
||||||
|
std::vector<mockturtle::signal<mockturtle::mig_network>> a;
|
||||||
|
std::vector<mockturtle::signal<mockturtle::mig_network>> b;
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < 8; i++) {
|
||||||
|
a.push_back(mig.create_pi());
|
||||||
|
b.push_back(mig.create_pi());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto out = mockturtle::""" + name + """(mig, a, b);
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < out.size(); i++) {
|
||||||
|
mig.create_po(out[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// output
|
||||||
|
mockturtle::write_verilog(mig, "mig.v");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
|
||||||
|
# https://mockturtle.readthedocs.io/en/latest/generators/arithmetic.html#addition-and-subtraction
|
||||||
|
circs = [
|
||||||
|
# ADDRS
|
||||||
|
{
|
||||||
|
"name": "UnsignedRippleCarryAdder",
|
||||||
|
"ag": "UnsignedRippleCarryAdder",
|
||||||
|
"mt": "carry_ripple_adder_inplace",
|
||||||
|
"mtAOut": True
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedCarryLookaheadAdder",
|
||||||
|
"ag": "UnsignedCarryLookaheadAdder",
|
||||||
|
"mt": "carry_lookahead_adder_inplace",
|
||||||
|
"mtAOut": True
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedBrentKungAdder",
|
||||||
|
"ag": "UnsignedBrentKungAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedCarryIncrementAdder",
|
||||||
|
"ag": "UnsignedCarryIncrementAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedCarrySkipAdder",
|
||||||
|
"ag": "UnsignedCarrySkipAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedCarrySelectAdder",
|
||||||
|
"ag": "UnsignedCarrySelectAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedConditionalSumAdder",
|
||||||
|
"ag": "UnsignedConditionalSumAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedHanCarlsonAdder",
|
||||||
|
"ag": "UnsignedHanCarlsonAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedKnowlesAdder",
|
||||||
|
"ag": "UnsignedKnowlesAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedKoggeStoneAdder",
|
||||||
|
"ag": "UnsignedKoggeStoneAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedLadnerFischerAdder",
|
||||||
|
"ag": "UnsignedLadnerFischerAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedPGRippleCarryAdder",
|
||||||
|
"ag": "UnsignedPGRippleCarryAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedSklanskyAdder",
|
||||||
|
"ag": "UnsignedSklanskyAdder",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
# MULS
|
||||||
|
{
|
||||||
|
"name": "UnsignedArrayMultiplier",
|
||||||
|
"ag": "UnsignedArrayMultiplier",
|
||||||
|
"mt": "carry_ripple_multiplier",
|
||||||
|
"mtAOut": False
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedCarrySaveMultiplier",
|
||||||
|
"ag": "UnsignedCarrySaveMultiplier",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedDaddaMultiplier",
|
||||||
|
"ag": "UnsignedDaddaMultiplier",
|
||||||
|
"mt": None
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "UnsignedWallaceMultiplier",
|
||||||
|
"ag": "UnsignedWallaceMultiplier",
|
||||||
|
"mt": None
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
#
|
||||||
|
# Begin main
|
||||||
|
#
|
||||||
|
|
||||||
|
MT_PATH = "./mockturtle/"
|
||||||
|
AGM_PATH = "./ariths-gen-mig/"
|
||||||
|
AG_PATH = "./ariths-gen/"
|
||||||
|
PWD = os.getcwd()
|
||||||
|
|
||||||
|
table = open(f"{PWD}/compare.csv", "w")
|
||||||
|
|
||||||
|
print("type;MT (GATES);MT (DELAY);ariths-gen->AIG->MT (GATES);ariths-gen->AIG->MT (DELAY);ariths-gen->AIG->MT+RESUBSTITUTION (GATES);ariths-gen->AIG->MT+RESUBSTITUTION (DELAY);ariths-gen-mig (GATES);ariths-gen-mig (DELAY);ariths-gen-mi->AIG->MT+RESUBSTITUTION (GATES);ariths-gen-mi->AIG->MT+RESUBSTITUTION (DELAY)", file=table)
|
||||||
|
|
||||||
|
for circ in circs:
|
||||||
|
|
||||||
|
MTDirectGates = "-"
|
||||||
|
MTDirectDelay = "-"
|
||||||
|
|
||||||
|
## MockTurtle direct generator
|
||||||
|
|
||||||
|
if circ["mt"] is not None:
|
||||||
|
|
||||||
|
with open(f"{MT_PATH}/examples/generate.cpp", "w") as Vfile:
|
||||||
|
print(generateMT(circ["mt"], circ["mtAOut"]), file=Vfile)
|
||||||
|
|
||||||
|
os.system(f"cd {MT_PATH}/build/examples && make && ./generate && cp mig.v {PWD}/mig.v")
|
||||||
|
|
||||||
|
# generate blif for verification
|
||||||
|
os.system(f"yosys formal.ys")
|
||||||
|
|
||||||
|
os.system(f"mv formal.blif formalMT.blif && mv mig.v {circ['name']}_MT.v")
|
||||||
|
|
||||||
|
nodes = getNodes(f"{circ['name']}_MT.v")
|
||||||
|
|
||||||
|
MTDirectGates = countGates(nodes)
|
||||||
|
MTDirectDelay = getDelay(nodes)
|
||||||
|
|
||||||
|
plotNodes(nodes, f"{circ['name']}_MT")
|
||||||
|
|
||||||
|
## ArithsGen
|
||||||
|
|
||||||
|
with open(f"{AG_PATH}/generate.py", "w") as Vfile:
|
||||||
|
print(generateAg(circ["ag"]), file=Vfile)
|
||||||
|
|
||||||
|
os.system(f"cd {AG_PATH} && python3 ./generate.py && cp mig.v {PWD}/mig.v")
|
||||||
|
|
||||||
|
os.system(f"sed -i -e 's/0x00/0/g' mig.v && sed -i -e 's/0x01/1/g' mig.v") # 0x00 and 0x01 to 0 and 1
|
||||||
|
|
||||||
|
# generate blif for verification
|
||||||
|
os.system(f"yosys formal.ys")
|
||||||
|
|
||||||
|
os.system(f"yosys toAig.ys") # yosys VERILOG to AIG
|
||||||
|
|
||||||
|
os.system(f"./{MT_PATH}/build/examples/passAig") # aig.aig to mig.v
|
||||||
|
|
||||||
|
os.system(f"mv mig.v {circ['name']}_AG.v")
|
||||||
|
|
||||||
|
os.system(f"./{MT_PATH}/build/examples/resubstitution") # aig.aig to mig.v (resubstitution)
|
||||||
|
|
||||||
|
os.system(f"mv mig.v {circ['name']}_AG_RE.v")
|
||||||
|
|
||||||
|
nodes = getNodes(f"{circ['name']}_AG.v")
|
||||||
|
|
||||||
|
print(f"Working on {circ['name']}_AG")
|
||||||
|
|
||||||
|
AGDirectGates = countGates(nodes)
|
||||||
|
AGDirectDelay = getDelay(nodes)
|
||||||
|
|
||||||
|
plotNodes(nodes, f"{circ['name']}_AG")
|
||||||
|
|
||||||
|
nodes = getNodes(f"{circ['name']}_AG_RE.v")
|
||||||
|
|
||||||
|
print(f"Working on {circ['name']}_AG_RE")
|
||||||
|
|
||||||
|
AGREDirectGates = countGates(nodes)
|
||||||
|
AGREDirectDelay = getDelay(nodes)
|
||||||
|
|
||||||
|
plotNodes(nodes, f"{circ['name']}_AG_RE")
|
||||||
|
|
||||||
|
## ArithsGenMig
|
||||||
|
|
||||||
|
with open(f"{AGM_PATH}/generate.py", "w") as Vfile:
|
||||||
|
print(generateAg(circ["ag"]), file=Vfile)
|
||||||
|
|
||||||
|
os.system(f"cd {AGM_PATH} && python3 ./generate.py && cp mig.v {PWD}/{circ['name']}_AGM.v")
|
||||||
|
|
||||||
|
os.system(f"sed -i -e 's/0x00/0/g' {circ['name']}_AGM.v && sed -i -e 's/0x01/1/g' {circ['name']}_AGM.v") # 0x00 and 0x01 to 0 and 1
|
||||||
|
|
||||||
|
nodes = getNodes(f"{circ['name']}_AGM.v")
|
||||||
|
|
||||||
|
print(f"Working on {circ['name']}_AGM")
|
||||||
|
|
||||||
|
AGMDirectGates = countGates(nodes)
|
||||||
|
AGMDirectDelay = getDelay(nodes)
|
||||||
|
|
||||||
|
plotNodes(nodes, f"{circ['name']}_AGM")
|
||||||
|
|
||||||
|
os.system(f"cp {circ['name']}_AGM.v mig.v")
|
||||||
|
|
||||||
|
os.system(f"yosys toAig.ys") # yosys VERILOG to AIG
|
||||||
|
|
||||||
|
os.system(f"./{MT_PATH}/build/examples/resubstitution") # aig.aig to mig.v (resubstitution)
|
||||||
|
|
||||||
|
os.system(f"mv mig.v {circ['name']}_AGM_RE.v")
|
||||||
|
|
||||||
|
nodes = getNodes(f"{circ['name']}_AGM_RE.v")
|
||||||
|
|
||||||
|
print(f"Working on {circ['name']}_AGM_RE")
|
||||||
|
|
||||||
|
AGMREDirectGates = countGates(nodes)
|
||||||
|
AGMREDirectDelay = getDelay(nodes)
|
||||||
|
|
||||||
|
plotNodes(nodes, f"{circ['name']}_AGM_RE")
|
||||||
|
|
||||||
|
ec = os.system(f"bash yosys_equiv_check.sh -v {circ['name']}_AGM.v -b formal.blif -m sat")
|
||||||
|
if ec != 0:
|
||||||
|
print("Formal verification fail")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
print(f"{circ['name']};{MTDirectGates};{MTDirectDelay};{AGDirectGates};{AGDirectDelay};{AGREDirectGates};{AGREDirectDelay};{AGMDirectGates};{AGMDirectDelay};{AGMREDirectGates};{AGMREDirectDelay}", file=table)
|
||||||
|
|
||||||
|
os.system("zip data.zip *.pdf *.v")
|
||||||
|
os.system("mkdir stash")
|
||||||
|
os.system("mv toAig.ys stash && mv test.py stash && mv compare.csv stash && mv setup.sh stash && mv data.zip stash && mv yosys_equiv_check.sh stash && mv formal.ys stash")
|
||||||
|
os.system("rm -f *")
|
||||||
|
os.system("mv stash/* . && rm -rf stash")
|
||||||
|
table.close()
|
8
toAig.ys
Normal file
8
toAig.ys
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
read_verilog mig.v
|
||||||
|
proc
|
||||||
|
opt
|
||||||
|
techmap
|
||||||
|
opt
|
||||||
|
abc
|
||||||
|
aigmap
|
||||||
|
write_aiger aig.aig
|
200
yosys_equiv_check.sh
Normal file
200
yosys_equiv_check.sh
Normal file
@ -0,0 +1,200 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Script enables formal verification and equivalence of Verilog/BLIF circuit designs
|
||||||
|
# using Yosys Open SYnthesis Suite by Claire Xenia Wolf.
|
||||||
|
# For more information, please visit: http://bygone.clairexen.net/yosys/documentation.html
|
||||||
|
|
||||||
|
# Echo script help
|
||||||
|
help () {
|
||||||
|
echo "–––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––"
|
||||||
|
echo "––– HELP –––"
|
||||||
|
echo "–––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––"
|
||||||
|
echo "– SCRIPT FOR CHECKING FORMAL EQUIVALENCE BETWEEN VERILOG AND BLIF DESIGNS USING YOSYS –"
|
||||||
|
echo
|
||||||
|
echo "Input files should have the same name used for corresponding file types filenames as well"
|
||||||
|
echo "as for their inner design's top module names."
|
||||||
|
echo "Formal verification and equivalence of Verilog/BLIF circuit designs achieved using Yosys"
|
||||||
|
echo "Open SYnthesis Suite."
|
||||||
|
echo "–––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––––"
|
||||||
|
echo "Input parameters:"
|
||||||
|
echo "-h, --help"
|
||||||
|
echo " help information"
|
||||||
|
echo "-v 'verilog_file', --verilog_file 'verilog_file'"
|
||||||
|
echo " specifies input Verilog design file"
|
||||||
|
echo "-b 'blif_file', --blif_file 'blif_file'"
|
||||||
|
echo " specifies input BLIF design file"
|
||||||
|
echo "-m 'method', --mode 'method'"
|
||||||
|
echo " specifies chosen formal equivalence method (sat|equiv)"
|
||||||
|
echo "-H, --hier"
|
||||||
|
echo " specifies whether designs are in hierarchical representation (default is flat)"
|
||||||
|
}
|
||||||
|
|
||||||
|
# No input parameters present, echo help and exit.
|
||||||
|
if [[ "$#" -eq 0 ]]; then
|
||||||
|
help
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Defaultly flat designs are considered
|
||||||
|
HIERARCHICAL=false
|
||||||
|
|
||||||
|
# Parsing and extraction of input parameters and their arguments into variables.
|
||||||
|
while [[ "$#" -gt 0 ]] ; do
|
||||||
|
case "$1" in
|
||||||
|
"-v" | "--verilog_file")
|
||||||
|
if [[ -f "$2" && "$2" == *.v ]]; then
|
||||||
|
VERILOG_FILE=$2
|
||||||
|
else
|
||||||
|
echo "$2 is not a Verilog file!";
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
shift 2;;
|
||||||
|
|
||||||
|
"-b" | "--blif_file")
|
||||||
|
if [[ -f "$2" && "$2" == *.blif ]]; then
|
||||||
|
BLIF_FILE="$2"
|
||||||
|
else
|
||||||
|
echo "$2 is not a BLIF file!";
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
shift 2;;
|
||||||
|
|
||||||
|
"-m" | "--mode")
|
||||||
|
if [[ "$2" == "sat" || "$2" == "equiv" ]]; then
|
||||||
|
METHOD="$2"
|
||||||
|
else
|
||||||
|
echo "$2 is not a supported formal equivalence method!";
|
||||||
|
echo
|
||||||
|
echo "Type -h | --help for more information."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
shift 2;;
|
||||||
|
|
||||||
|
"-H" | "--hier")
|
||||||
|
HIERARCHICAL=true
|
||||||
|
shift 1;;
|
||||||
|
"-h" | "--help")
|
||||||
|
help
|
||||||
|
exit 0;;
|
||||||
|
*)
|
||||||
|
echo "Unknown input parameter $1!"
|
||||||
|
echo
|
||||||
|
echo "Type -h | --help for more information."
|
||||||
|
exit 1;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
# Check if both files compulsory parameters are set and if they have the same name.
|
||||||
|
# To proper equiv check both designs top modules must be of same names (assumption that filename == top module name)
|
||||||
|
if [[ -z "$VERILOG_FILE" || -z "$BLIF_FILE" || -z "$METHOD" ]]; then
|
||||||
|
[ -z "$VERILOG_FILE" ] && echo "Missing compulsory Verilog file for comparison!"
|
||||||
|
[ -z "$BLIF_FILE" ] && echo "Missing compulsory BLIF file for comparison!"
|
||||||
|
[ -z "$METHOD" ] && echo "Missing choice of formal equivalence method!"
|
||||||
|
echo
|
||||||
|
echo "Type -h | --help for more information."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
TOP_MODULE="top"
|
||||||
|
|
||||||
|
|
||||||
|
# Formal verification with equiv_* for flat designs
|
||||||
|
if [ "$METHOD" = "equiv" ]; then
|
||||||
|
if [ "$HIERARCHICAL" = false ]; then
|
||||||
|
yosys -p "
|
||||||
|
# Gold design
|
||||||
|
read_verilog $VERILOG_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
splitnets -ports;;
|
||||||
|
design -stash gold
|
||||||
|
|
||||||
|
# Gate design
|
||||||
|
read_blif $BLIF_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
design -stash gate
|
||||||
|
|
||||||
|
# Prove equivalence
|
||||||
|
design -copy-from gold -as gold $TOP_MODULE
|
||||||
|
design -copy-from gate -as gate $TOP_MODULE
|
||||||
|
|
||||||
|
equiv_make gold gate equiv
|
||||||
|
hierarchy -top equiv
|
||||||
|
equiv_simple
|
||||||
|
equiv_status -assert
|
||||||
|
"
|
||||||
|
# Formal verification with equiv_* for hierarchical designs
|
||||||
|
else
|
||||||
|
yosys -p "
|
||||||
|
# Gold design
|
||||||
|
read_verilog $VERILOG_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
flatten
|
||||||
|
splitnets -ports;;
|
||||||
|
design -stash gold
|
||||||
|
|
||||||
|
# Gate design
|
||||||
|
read_blif $BLIF_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
flatten
|
||||||
|
splitnets -ports;;
|
||||||
|
design -stash gate
|
||||||
|
|
||||||
|
# Prove equivalence
|
||||||
|
design -copy-from gold -as gold $TOP_MODULE
|
||||||
|
design -copy-from gate -as gate $TOP_MODULE
|
||||||
|
|
||||||
|
equiv_make gold gate equiv
|
||||||
|
hierarchy -top equiv
|
||||||
|
equiv_simple
|
||||||
|
equiv_status -assert
|
||||||
|
"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
if [ "$HIERARCHICAL" = false ]; then
|
||||||
|
yosys -p "
|
||||||
|
# Gold design
|
||||||
|
read_verilog $VERILOG_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
splitnets -ports;;
|
||||||
|
design -stash gold
|
||||||
|
|
||||||
|
# Gate design
|
||||||
|
read_blif $BLIF_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
design -stash gate
|
||||||
|
|
||||||
|
# Prove equivalence
|
||||||
|
design -copy-from gold -as gold $TOP_MODULE
|
||||||
|
design -copy-from gate -as gate $TOP_MODULE
|
||||||
|
|
||||||
|
miter -equiv -flatten gold gate miter
|
||||||
|
hierarchy -top miter
|
||||||
|
sat -verify -tempinduct -prove trigger 0
|
||||||
|
"
|
||||||
|
# Formal verification with equiv_* for hierarchical designs
|
||||||
|
else
|
||||||
|
yosys -p "
|
||||||
|
# Gold design
|
||||||
|
read_verilog $VERILOG_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
flatten
|
||||||
|
splitnets -ports;;
|
||||||
|
design -stash gold
|
||||||
|
|
||||||
|
# Gate design
|
||||||
|
read_blif $BLIF_FILE
|
||||||
|
prep -top $TOP_MODULE
|
||||||
|
flatten
|
||||||
|
splitnets -ports;;
|
||||||
|
design -stash gate
|
||||||
|
|
||||||
|
# Prove equivalence
|
||||||
|
design -copy-from gold -as gold $TOP_MODULE
|
||||||
|
design -copy-from gate -as gate $TOP_MODULE
|
||||||
|
|
||||||
|
miter -equiv -flatten gold gate miter
|
||||||
|
hierarchy -top miter
|
||||||
|
sat -verify -tempinduct -prove trigger 0
|
||||||
|
"
|
||||||
|
fi
|
||||||
|
fi
|
Loading…
x
Reference in New Issue
Block a user