-
Notifications
You must be signed in to change notification settings - Fork 82
/
Copy pathexample5-introspection.py
77 lines (60 loc) · 2.63 KB
/
example5-introspection.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
""" Example 5: Making use of PyRTL and Introspection. """
import pyrtl
# The following example shows how PyRTL can be used to make some interesting
# hardware structures using Python introspection. In particular, this example
# makes a N-stage pipeline structure. Any specific pipeline is then a derived
# class of SimplePipeline where methods with names starting with "stage" are
# stages, and new members with names not starting with "_" are to be registered
# for the next stage.
class SimplePipeline(object):
""" Pipeline builder with auto generation of pipeline registers. """
def __init__(self):
self._pipeline_register_map = {}
self._current_stage_num = 0
stage_list = [method for method in dir(self) if method.startswith('stage')]
for stage in sorted(stage_list):
stage_method = getattr(self, stage)
stage_method()
self._current_stage_num += 1
def __getattr__(self, name):
try:
return self._pipeline_register_map[self._current_stage_num][name]
except KeyError:
raise pyrtl.PyrtlError(
'error, no pipeline register "%s" defined for stage %d'
% (name, self._current_stage_num))
def __setattr__(self, name, value):
if name.startswith('_'):
# do not do anything tricky with variables starting with '_'
object.__setattr__(self, name, value)
else:
next_stage = self._current_stage_num + 1
pipereg_id = str(self._current_stage_num) + 'to' + str(next_stage)
rname = 'pipereg_' + pipereg_id + '_' + name
new_pipereg = pyrtl.Register(bitwidth=len(value), name=rname)
if next_stage not in self._pipeline_register_map:
self._pipeline_register_map[next_stage] = {}
self._pipeline_register_map[next_stage][name] = new_pipereg
new_pipereg.next <<= value
class SimplePipelineExample(SimplePipeline):
""" A very simple pipeline to show how registers are inferred. """
def __init__(self):
self._loopback = pyrtl.WireVector(1, 'loopback')
super(SimplePipelineExample, self).__init__()
def stage0(self):
self.n = ~ self._loopback
def stage1(self):
self.n = self.n
def stage2(self):
self.n = self.n
def stage3(self):
self.n = self.n
def stage4(self):
self._loopback <<= self.n
simplepipeline = SimplePipelineExample()
print(pyrtl.working_block())
# Simulation of the core
sim_trace = pyrtl.SimulationTrace()
sim = pyrtl.Simulation(tracer=sim_trace)
sim.step_multiple({}, nsteps=15)
sim_trace.render_trace()