-
Notifications
You must be signed in to change notification settings - Fork 23
/
plot_layout_new_model.py
724 lines (662 loc) · 25.9 KB
/
plot_layout_new_model.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
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
"""
Program to plot vias in the whole layout using DEF and LEF data.
Author: Tri Minh Cao
Email: [email protected]
Date: September 2016
"""
from def_parser import *
from lef_parser import *
from util import *
import plot_cell
import matplotlib.pyplot as plt
import numpy as np
import time
import img_util
import pickle
import random
import os
import time
import shutil
def sort_vias_by_row(layout_area, row_height, vias):
"""
Sort the vias by row
:param layout_area: a list [x, y] that stores the area of the layout
:param vias: a list of vias that need to be sorted
:return: a list of rows, each containing a list of vias in that row.
"""
num_rows = layout_area[1] // row_height + 1
rows = []
for i in range(num_rows):
rows.append([])
for via in vias:
via_y = via[0][1]
row_dest = via_y // row_height
rows[row_dest].append(via)
# sort vias in each row based on x-coordinate
for each_row in rows:
each_row.sort(key = lambda x: x[0][0])
return rows
def plot_window(left_pt, width, height, vias, lef_data, macro=None, comp=None):
"""
Method to plot a window from the layout with all vias inside it.
:param left_pt: bottom left point (origin) of the window
:param width: width of the window
:param height: height of the window
:param vias: a list containing all vias on a row
:return: void
"""
# get the corners for the window
corners = [left_pt]
corners.append((left_pt[0] + width, left_pt[1] + height))
# compose the output file name
out_folder = './images/'
# current_time = time.strftime('%H%M%d%m%Y')
pos = (str(corners[0][0]) + '_' + str(corners[0][1]) + '_' +
str(corners[1][0]) + '_' + str(corners[1][1]))
# out_file = out_folder + pos
out_file = out_folder
# out_file += str(corners[0][0])
out_file += pos
if macro:
out_file += '_' + macro
if comp:
out_file += '_' + comp
# current_time = time.strftime('%H%M%S%d%m%Y')
# out_file += '_' + current_time
if os.path.exists(out_file + '.png'):
return out_file + '.png'
plt.figure(figsize=(1, 1.6), dpi=80, frameon=False)
# scale the axis of the subplot
# draw the window boundary
# scaled_pts = rect_to_polygon(corners)
# draw_shape = plt.Polygon(scaled_pts, closed=True, fill=None,
# color="blue")
# plt.gca().add_patch(draw_shape)
# plot the vias inside the windows
# look for the vias
for via in vias:
if (via[0][0] - left_pt[0] > width):
break
via_name = via[1]
via_info = lef_data.via_dict[via_name]
via_loc = via[0]
plot_cell.draw_via(via_loc, via_info)
# scale the axis of the subplot
axis = [corners[0][0], corners[1][0], corners[0][1], corners[1][1]]
# print (test_axis)
plt.axis(axis)
plt.axis('off')
plt.gca().set_aspect('equal', adjustable='box')
plt.savefig(out_file)
# plt.show()
plt.close('all')
return out_file + '.png'
def group_via(via_list, max_number, max_distance):
"""
Method to group the vias together to check if they belong to a cell.
:param via_list: a list of all vias.
:return: a list of groups of vias.
"""
groups = []
length = len(via_list)
for i in range(length):
# one_group = [via_list[i]]
curr_via = via_list[i]
curr_list = []
for j in range(2, max_number + 1):
if i + j - 1 < length:
right_via = via_list[i + j - 1]
dist = right_via[0][0] - curr_via[0][0]
if dist < max_distance:
curr_list.append(via_list[i:i+j])
# only add via group list that is not empty
if len(curr_list) > 0:
groups.append(curr_list)
return groups
def predict_cell(candidates, row, model, lef_data, std_cells):
"""
Use the trained model to choose the most probable cell from via groups.
:param candidates: 2-via and 3-via groups that could make a cell
:return: a tuple (chosen via group, predicted cell name)
"""
margin = 350
img_width = 126
img_height = 66
img_shape = img_width * img_height
possible_candidates = []
for i in range(len(candidates)):
# dataset = np.ndarray(shape=(len(candidates), img_height, img_width),
# dtype=np.float32)
if candidates[i] != -1:
possible_candidates.append(i)
# dataset = np.ndarray(shape=(1, img_height, img_width),
# dtype=np.float32)
dataset = np.ndarray(shape=(1, img_width, img_height),
dtype=np.float32)
each_group = candidates[i]
left_pt = [each_group[0][0][0] - margin, CELL_HEIGHT * row]
width = each_group[-1][0][0] - left_pt[0] + margin
# print (width)
img_file = plot_window(left_pt, width, CELL_HEIGHT, each_group, lef_data)
# print (img_file)
image_data = img_util.load_image(img_file)
# print (image_data.shape)
dataset[0, :, :] = image_data
X_test = dataset.reshape(dataset.shape[0], img_shape)
result = model.decision_function(X_test)
result = result[0]
# check for result
if result[i] == max(result):
return candidates[i], i
# if we cannot find a solution, randomly select a choice
choice = random.choice(possible_candidates)
return candidates[choice], choice
def sorted_components(layout_area, row_height, comps):
"""
Sort the components by row
:param layout_area: a list [x, y] that stores the area of the layout
:param comps: a list of components that need to be sorted
:return: a list of rows, each containing a list of components in that row.
"""
num_rows = layout_area[1] // row_height + 1
rows = []
for i in range(num_rows):
rows.append([])
for comp in comps:
comp_y = comp.placed[1]
row_dest = comp_y // row_height
rows[row_dest].append(comp)
# sort vias in each row based on x-coordinate
for each_row in rows:
each_row.sort(key = lambda x: x.placed[0])
return rows
def predict_score(predicts, actuals):
"""
Find the number of correct cell predictions.
:param predicts: a list of predictions.
:param actuals: a list of actual cells.
:return: # correct predictions, # cells
"""
len_preds = len(predicts)
len_actuals = len(actuals)
shorter_len = min(len_preds, len_actuals)
gap_predict = 0
gap_actual = 0
num_correct = 0
# print (shorter_len)
for i in range(shorter_len):
# print (i)
# print (gap_predict)
# print (gap_actual)
# print ()
if predicts[i + gap_predict] == actuals[i + gap_actual]:
num_correct += 1
else:
if len_preds < len_actuals:
gap_actual += 1
len_preds += 1
elif len_preds > len_actuals:
gap_predict += 1
len_actuals += 1
return num_correct, len(actuals)
def plot_cell_w_vias():
# process each row, plot all cells
# for i in range(num_rows):
margin = 350
for i in range(1):
via_idx = 0
print (len(components[i]))
print (len(via1_sorted[i]))
for each_comp in components[i]:
comp_name = each_comp.name
macro_name = each_comp.macro
macro_data = lef_parser.macro_dict[macro_name]
num_vias = len(macro_data.pin_dict) - 2 # because of VDD and GND pins
# get the vias
cell_vias = via1_sorted[i][via_idx:via_idx + num_vias]
# update via_idx
via_idx += num_vias
# plot the cell
left_pt = [cell_vias[0][0][0] - margin, CELL_HEIGHT * i]
width = cell_vias[-1][0][0] - left_pt[0] + margin
# print (width)
img_file = plot_window(left_pt, width, CELL_HEIGHT, cell_vias,
lef_parser, macro=macro_name, comp = comp_name)
print (comp_name)
print (macro_name)
print (cell_vias)
print (via_idx)
print('Finished!')
def check_via_group(via_group, source_sink):
"""
Check the validity of each via set in the via group.
:param via_group: the via_group in question.
:return: via_group with all valid candidate(s)
"""
# valid for 2-via cell: 1 source, 1 sink
# valid for 3-via cell: 2 sink, 1 source
valid_group = []
for each_group in via_group:
num_vias = len(each_group)
num_source = 0
num_sink = 0
for each_via in each_group:
# 0 = sink, 1 = source
if source_sink[each_via[2]] == 1:
num_source += 1
elif source_sink[each_via[2]] == 0:
num_sink += 1
if num_source <= 1 and num_sink <=2:
valid_group.append(each_group)
return valid_group
def get_candidates(first_via_idx, via_list, std_cells):
"""
Generate a list of candidates from the first via.
Each standard cell will be considered for candidates.
If the standard cell cannot be placed there, the value is -1,
otherwise, it will be a list of vias.
:param first_via_idx: first via index in the via_list
:param via_list: the list of all vias (in a row)
:param std_cells: a list that stores information of std cells
:return: a list of groups of vias, or -1
"""
# candidates = [-1 for i in range(len(std_cells))]
candidates = []
first_via = via_list[first_via_idx]
# print (first_via)
first_via_x = first_via[0][0]
for i in range(len(std_cells)):
cell_width = std_cells[i][2]
min_vias = std_cell_info[i][0]
max_vias = std_cells[i][1]
pin_left_dist = std_cells[i][3]
boundary = first_via_x + cell_width - pin_left_dist
# boundary = first_via_x + cell_width
# possible vias contain the vias inside the boundary
possible_vias = [first_via]
for j in range(first_via_idx + 1, len(via_list)):
if via_list[j][0][0] <= boundary:
possible_vias.append(via_list[j])
else:
break
# check the candidate against cell info
if len(possible_vias) > max_vias or len(possible_vias) < min_vias:
candidates.append(-1)
else:
candidates.append(possible_vias)
for each_cand in candidates:
if each_cand != -1:
return candidates
return -1
def get_inputs_outputs(def_info):
"""
Method to get all inputs and outputs nets from a DEF file.
:param def_info: def info (already parsed).
:return: inputs and outputs
"""
pins = def_parser.pins.pins
inputs = []
outputs = []
for each_pin in pins:
pin_name = each_pin.name
direction = each_pin.direction.lower()
if direction == 'input':
inputs.append(pin_name)
elif direction == 'output':
outputs.append(pin_name)
return inputs, outputs
def recover_netlist(def_info, inputs, outputs, recovered_cells):
"""
Method to create a netlist from predicted cells
:param def_info: information from the DEF file
:param inputs: input pins of the design
:param outputs: output pins of the design
:param recovered_cells: recovered cells with input nets and output nets
:return: recovered netlist file name
"""
# NOTE: the order of nets is not like that in original netlist
design = def_info.design_name
nets = set(def_info.nets.net_dict.keys())
inputs_set = set(inputs)
outputs_set = set(outputs)
io = inputs_set | outputs_set
wires = nets - io
# print(wires)
# print(len(wires))
# save the cells_reco for later inspection
# filename = './recovered/' + design + '.pickle'
# try:
# with open(filename, 'wb') as f:
# pickle.dump(cells_reco, f, pickle.HIGHEST_PROTOCOL)
# except Exception as e:
# print('Unable to save data to', filename, ':', e)
## dd/mm/yyyy format
date = time.strftime("%m/%d/%Y %H:%M:%S")
s = '/////////////////////////////\n'
s += '// Generated by TMC\n'
s += '// Design: ' + design + '\n'
s += '// Date: ' + date + '\n'
s += '/////////////////////////////\n\n'
# add module definition
s += 'module ' + design + ' ( '
num_ios = len(io)
idx = 0
for each_pin in io:
s += each_pin
idx += 1
if idx < num_ios:
s += ', '
s += ' );\n'
indent = ' '
# add input
num_in = len(inputs)
idx = 0
s += indent + 'input '
for each_in in inputs:
s += each_in
idx += 1
if idx < num_in:
s += ', '
s += ';\n'
# add output
num_out = len(outputs)
idx = 0
s += indent + 'output '
for each_out in outputs:
s += each_out
idx += 1
if idx < num_out:
s += ', '
s += ';\n'
# add wire
num_wire = len(wires)
idx = 0
s += indent + 'wire '
for each_wire in wires:
s += each_wire
idx += 1
if idx < num_wire:
s += ', '
s += ';\n'
# add cells
s += '\n'
cell_idx = 2
for each_cell in cells_reco:
cell_idx += 1
s += indent + each_cell[0] + ' U' + str(cell_idx) + ' ( '
in_nets = each_cell[1]
s += '.A(' + in_nets[0] + ')' + ', '
if len(in_nets) == 2:
s += '.B(' + in_nets[1] + ')' + ', '
out_net = each_cell[2]
s += '.Y(' + out_net + ')'
s += ' );\n'
s += 'endmodule'
# write to an output file
folder = './recovered/'
filename = design + '_recovered' + '.v'
print('Writing recovered netlist file...')
f = open(folder + filename, mode="w+")
f.write(s)
f.close()
print('Writing done.')
return filename
def closest_via_pair_y(via_group):
"""
Method to find the closest pair of via based on y-coordinate.
:param via_group: a list of vias
:return: the pair of vias that are closest based on y-coordinate.
"""
closest_pair = 0 # initialize the answer
min_dist = float('inf')
for i in range(len(via_group)):
for j in range(i + 1, len(via_group)):
y_i = via_group[i][0][1]
y_j = via_group[j][0][1]
if abs(y_i - y_j) < min_dist:
min_dist = abs(y_i - y_j)
closest_pair = (via_group[i], via_group[j])
return closest_pair
# Main Class
if __name__ == '__main__':
path = './libraries/layout_yujie/'
files = os.listdir(path)
files = ['b18_C_gscl45nm_tri_routing_layer9.def']
for f in files:
start_time = time.time()
# def_path = './libraries/layout_yujie/c432.def'
def_path = path + f
def_parser = DefParser(def_path)
def_parser.parse()
scale = def_parser.scale
lef_file = "./libraries/FreePDK45/gscl45nm.lef"
lef_parser = LefParser(lef_file)
lef_parser.parse()
macro_dict = lef_parser.macro_dict
CELL_HEIGHT = int(float(scale) * lef_parser.cell_height)
# print (CELL_HEIGHT)
print ("Process file:", def_path)
all_via1 = get_all_vias(def_parser, via_type="M2_M1_via")
# create the netlist on-the-fly, for each net, 0 = input only, 1 = input or output
netlist_fly = dict()
# build the net_via dictionary
nets = def_parser.nets.nets
# initialize the nets_via_dict
nets_vias_dict = {}
for net in nets:
net_name = net.name
nets_vias_dict[net_name] = []
# initialize all the nets = 1
netlist_fly[net_name] = 1
# add vias to nets_dict
for each_via in all_via1:
net = each_via[2]
nets_vias_dict[net].append(each_via)
# sort the vias by row
via1_sorted = sort_vias_by_row(def_parser.diearea[1], CELL_HEIGHT, all_via1)
# add inputs and outputs from the design to via info
inputs, outputs = get_inputs_outputs(def_parser)
for each_in in inputs:
# print(each_in)
netlist_fly[each_in] = 0
for each_via in nets_vias_dict[each_in]:
each_via[3] = 0
for each_out in outputs:
netlist_fly[each_out] = 2
for each_via in nets_vias_dict[each_out]:
each_via[3] = 1
MAX_DISTANCE = 2280 # OR2 cell width, can be changed later
components = sorted_components(def_parser.diearea[1], CELL_HEIGHT,
def_parser.components.comps)
num_rows = len(components)
###############
# DO PREDICTION
# predict_row()
# We can load the trained model
pickle_filename = "./trained_models/logit_model_111816.pickle"
try:
with open(pickle_filename, 'rb') as f:
logit_model = pickle.load(f)
except Exception as e:
print('Unable to read data from', pickle_filename, ':', e)
labels = {0: 'and2', 1: 'invx1', 2: 'invx8', 3: 'nand2', 4: 'nor2',
5: 'or2'}
macro_from_labels = {0: 'AND2X1', 1: 'INVX1', 2: 'INVX8', 3: 'NAND2X1',
4: 'NOR2X1', 5: 'OR2X1'}
cell_labels = {'AND2X1': 'and2', 'INVX1': 'invx1', 'NAND2X1': 'nand2',
'NOR2X1': 'nor2', 'OR2X1': 'or2', 'INVX8': 'invx8'}
##############
# List of standard cells
std_cell_info = {}
# info includes (min num vias, max num vias, width,
# distance from left boundary to first pin)
# also need distance from right most pin to right boundary
# I wonder if max num vias should be used, actually I don't know what is the
# maximum number of vias, but I guess +1 is fine.
# 0 is and2, 1 is invx1, etc.
std_cell_info[0] = (3, 4, 2280, 295 + 200)
std_cell_info[1] = (2, 3, 1140, 315 + 245)
std_cell_info[2] = (2, 3, 2660, 695 + 500)
std_cell_info[3] = (3, 4, 1520, 90 + 200)
std_cell_info[4] = (3, 4, 1520, 315 + 200)
std_cell_info[5] = (3, 4, 2280, 695 + 150)
# process
# print the sorted components
components = sorted_components(def_parser.diearea[1], CELL_HEIGHT,
def_parser.components.comps)
correct = 0
total_cells = 0
predicts = []
actuals = []
cells_reco = [] # a list of recovered cells
# vias_reco = [] # a list of vias in the predicted cell, for debug purpose
# via_groups is only one row
# for i in range(len(via1_sorted)):
for i in range(0, 1):
print ('Process row', (i + 1))
# each via group in via_groups consist of two candidates
# via_groups = group_via(via1_sorted[i], 3, MAX_DISTANCE)
visited_vias = [] # later, make visited_vias a set to run faster
cells_pred = []
via_idx = 0
while via_idx < len(via1_sorted[i]):
# choosing candidates
candidates = get_candidates(via_idx, via1_sorted[i], std_cell_info)
print(via_idx)
print(via1_sorted[i][via_idx])
print(candidates)
if candidates == -1:
print('Something wrong!')
via_idx += 1
else:
# corner case: no possible candidates
best_group, prediction = predict_cell(candidates, i, logit_model,
lef_parser, std_cell_info)
# recover the cell information
macro_name = macro_from_labels[prediction]
macro_info = macro_dict[macro_from_labels[prediction]]
num_pins = len(macro_info.info["PIN"]) - 2
# NOTE: we assume inputs are A, B and output is Y
# for each_pin in pins:
# print(each_pin.name)
recover = []
input_nets = []
if macro_name == 'INVX1':
output_net = False
for each_via in best_group:
net_name = each_via[2]
if netlist_fly[net_name] == 2:
output_net = net_name
break
for each_via in best_group:
net_name = each_via[2]
if net_name != output_net:
input_nets.append(net_name)
break
if not output_net:
output_net = best_group[-1][2]
input_nets.append(best_group[0][2])
elif macro_name == 'INVX8':
output_net = False
for each_via in best_group:
net_name = each_via[2]
if netlist_fly[net_name] == 2:
output_net = net_name
break
for each_via in best_group:
net_name = each_via[2]
if net_name != output_net:
input_nets.append(net_name)
break
# second approach
if output_net == False:
input_nets = []
middle_y = CELL_HEIGHT * i + CELL_HEIGHT/2
min_dist = float('inf')
min_via = None
for each_via in best_group:
y_dist = abs(each_via[0][1] - middle_y)
if y_dist < min_dist:
min_via = each_via
input_nets.append(min_via[2])
for each_via in best_group:
net_name = each_via[2]
if net_name != input_nets[0]:
output_net = net_name
break
else:
input_vias = closest_via_pair_y(best_group)
for each_via in input_vias:
input_nets.append(each_via[2])
for each_via in best_group:
if each_via not in input_vias:
output_net = each_via[2]
netlist_fly[each_via[2]] = 0
break
# for j in range(len(best_group) - 1, -1, -1):
# net_name = best_group[j][2]
# if netlist_fly[net_name] == 1:
# output_net = net_name
# netlist_fly[net_name] = 0
# break
# for j in range(len(best_group)):
# net_name = best_group[j][2]
# if net_name != output_net and len(input_nets) + 1 < num_pins:
# input_nets.append(net_name)
# corner case: same nets for some or all vias in the group
# in this case, actually we can ignore the result
# num_inputs = len(input_nets)
# for i in range(num_inputs, num_pins - 1):
# input_nets.append(best_group[i][2])
# NOTE: the following lines only work for 2-pin and 3-pin cell
if len(input_nets) + 1 == num_pins:
recover.append(macro_name)
recover.append(input_nets)
recover.append(output_net)
cells_reco.append(recover)
# vias_reco.append((best_group, macro_from_labels[prediction]))
via_idx += len(best_group)
# print (best_group)
# print (labels[prediction])
cells_pred.append(labels[prediction])
for each_via in best_group:
visited_vias.append(each_via)
print (cells_pred)
print (len(cells_pred))
actual_comp = []
actual_macro = []
for each_comp in components[i]:
actual_comp.append(cell_labels[each_comp.macro])
actual_macro.append(each_comp.macro)
print (actual_comp)
print (len(actual_comp))
num_correct, num_cells = predict_score(cells_pred, actual_comp)
correct += num_correct
total_cells += num_cells
predicts.append(cells_pred)
actuals.append(actual_comp)
print ()
print ("\nTotal number of cells: ", total_cells)
print ("Number of correct cells predicted: ", correct)
print ("Accuracy rate (%): ", correct / total_cells * 100)
# print the execution time
print("\n--- Execution time:")
print("--- %s seconds ---" % (time.time() - start_time))
print("\n")
# remove images used
shutil.rmtree("./images")
if not os.path.exists("./images"):
os.makedirs("./images")
# count the time to generate the netlist separately
start_time = time.time()
# write the recovered verilog netlist
recover_netlist(def_parser, inputs, outputs, cells_reco)
print("\n--- Generate netlist time:")
print("--- %s seconds ---" % (time.time() - start_time))
# debug = (cells_reco, vias_reco)
# filename = './recovered/debug/c5315_debug' + '.pickle'
# try:
# with open(filename, 'wb') as f:
# pickle.dump(debug, f, pickle.HIGHEST_PROTOCOL)
# except Exception as e:
# print('Unable to save data to', filename, ':', e)