-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerator.py
181 lines (149 loc) · 5.63 KB
/
generator.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
HDL_file_name = str(input("Enter verilog file name: "))
file = open(HDL_file_name+'.v', 'r')
rtl_code = file.read()
test_bench=open(HDL_file_name+'_tb.v', 'w')
# remove comments from rtl_code
while("//" in rtl_code):
rtl_code = rtl_code[:rtl_code.find("//")] + rtl_code[rtl_code.find("\n",rtl_code.find("//")) :]
while("/*" in rtl_code):
rtl_code = rtl_code[:rtl_code.find("/*")] + rtl_code[rtl_code.find("*/")+2:]
rtl_code = " ".join(rtl_code.split()) #remove all white spaces
print(rtl_code)
# get module name
module_name_start_index = rtl_code.find("module") + 6
module_name_stop_index = rtl_code.find("(") if rtl_code.find("#") == -1 else rtl_code.find("#")
module_name = rtl_code[module_name_start_index:module_name_stop_index].strip()
# check if there is a clock signal
clk_signal = 1 if "clk" in rtl_code else 0
clk_period_value = 20
# check if there is a parameter
parameter = 1 if "parameter" in rtl_code else 0
# check for reset signal
reset_signal = 0
instance_name = 'uut' # default instance name
## trim the rtl code to get the inputs and outputs
# first index
if rtl_code.find("input") < rtl_code.find("output"):
first_index = rtl_code.find("input")
else:
first_index = rtl_code.find("output")
# last index
if rtl_code.rfind("output") > rtl_code.rfind("input"):
last_index = rtl_code.find(';',rtl_code.rfind("output"))+1 # 1 is added to include the semi-colon @ the end
else:
last_index = rtl_code.find(';',rtl_code.rfind("input"))+1
rtl_code = rtl_code[first_index:last_index]
rtl_code = [char for char in rtl_code if char != ' ' ] # convert the code to a list of characters
input_vector=[]
output_vector=[]
temp = ''
signal = ''
size = ''
discard_char = 1 # discard the first captured character
for i, char in enumerate(rtl_code):
temp = temp + char
if "input" in temp and "output" in temp:
temp = "input" if (temp.find("input") > temp.find("output")) else "output"
signal = ''
size = ''
discard_char = 1
if temp.count("input") > 1 or temp.count("output") > 1 :
temp = "input" if "input" in temp else "output"
signal = ''
size = ''
continue
if char == ')' :
if "input" in temp :
input_vector.append(size + " " + signal)
elif "output" in temp:
output_vector.append(size + " " + signal)
break
if "input" in temp :
if ']' in temp:
size = temp[temp.find('['): temp.find(']')+1]
signal = ''
temp = "input"
elif char == "," or char ==';':
input_vector.append(size + " " + signal)
signal = ''
elif discard_char:
discard_char = 0
continue
else:
signal += char
if "output" in temp :
if ']' in temp:
size = temp[temp.find('['): temp.find(']')+1]
signal = ''
temp = "output"
elif char == ',' or char ==';':
output_vector.append(size + " " + signal)
signal = ''
elif discard_char:
discard_char = 0
continue
else :
signal += char
# remove wire, reg from signals names
for i,signal in enumerate(input_vector):
if "wire" in signal:
input_vector[i] = signal[signal.find("wire")+4:]
elif "reg" in signal:
input_vector[i] = signal[signal.find("reg")+3:]
for i,signal in enumerate(output_vector):
if "wire" in signal:
output_vector[i] = signal[signal.find("wire")+4:]
elif "reg" in signal:
output_vector[i] = signal[signal.find("reg")+4:]
test_bench.write("/*test bench is automatically generated*/\n")
test_bench.write("`timescale 1ns / 1ps\n\n")
test_bench.write("module ")
test_bench.write(module_name+'_tb' + ';\n\n')
############ signals declaration ############
test_bench.write("// input signals\n")
for input in input_vector:
test_bench.write("reg "+input+';\n')
test_bench.write("\n// output signals\n")
for output in output_vector:
test_bench.write("wire "+ output+';\n')
############ instantiation ############
test_bench.write("\n// instantiation\n")
test_bench.write(module_name + ' ' + instance_name + "(\n")
# def instantiation():
#[0:-1]
for input in input_vector:
if ']' in input:
input = input[input.find(']')+1:]
input = input.strip()
test_bench.write("\t."+input+f'({input}),\n')
for output in output_vector:
if ']' in output:
output = output[output.find(']')+1:]
output = output.strip()
test_bench.write("\t."+output+f'({output}),\n')
test_bench.write("\t);")
############ clock generator block ############
if clk_signal:
test_bench.write("\n\n// clock signal\n")
test_bench.write(f"parameter PERIOD = {clk_period_value};\n\n")
test_bench.write("initial "+ "begin\n")
test_bench.write("clk = 0;\n")
test_bench.write("forever #(PERIOD/2) clk=~clk; \nend" +'\n')
############ active low reset block ############
for signal in input_vector:
if "reset" in signal or "rst" in signal:
test_bench.write("\n\n// reset pulse\n")
test_bench.write("initial "+ "begin\n")
test_bench.write(f"{signal} = 1'b0;\n")
test_bench.write("#(PERIOD/2);\n")
test_bench.write(f"{signal} = 1'b1;\n")
test_bench.write(" end" +'\n')
reset_signal = 1
break
############ initial block ############
test_bench.write("\n\n// test vector generator\n")
test_bench.write("initial "+ "begin\n\n")
if reset_signal:
test_bench.write(f"@(negedge {signal}); // wait for the device to reset\n\n")
test_bench.write("$finish; \nend" +'\n')
test_bench.write("\nendmodule")