Skip to content

Commit 382ca37

Browse files
committedMar 1, 2015
mibuild: initial Verilator support
1 parent 8f81ae6 commit 382ca37

File tree

4 files changed

+300
-0
lines changed

4 files changed

+300
-0
lines changed
 

‎mibuild/platforms/sim.py

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
from mibuild.generic_platform import *
2+
from mibuild.crg import SimpleCRG
3+
from mibuild.sim.verilator import VerilatorPlatform
4+
5+
class SimPins(Pins):
6+
def __init__(self, n):
7+
Pins.__init__(self, "s "*n)
8+
9+
_io = [
10+
("sys_clk", 0, SimPins(1)),
11+
("sys_rst", 0, SimPins(1)),
12+
("serial", 0,
13+
Subsignal("source_stb", SimPins(1)),
14+
Subsignal("source_ack", SimPins(1)),
15+
Subsignal("source_data", SimPins(8)),
16+
17+
Subsignal("sink_stb", SimPins(1)),
18+
Subsignal("sink_ack", SimPins(1)),
19+
Subsignal("sink_data", SimPins(8)),
20+
),
21+
]
22+
23+
class Platform(VerilatorPlatform):
24+
is_sim = True
25+
default_clk_name = "sys_clk"
26+
default_clk_period = 1000 # on modern computers simulate at ~ 1MHz
27+
def __init__(self):
28+
VerilatorPlatform.__init__(self, "SIM", _io)
29+
30+
def do_finalize(self, fragment):
31+
pass

‎mibuild/sim/__init__.py

Whitespace-only changes.

‎mibuild/sim/dut_tb.cpp

+158
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,158 @@
1+
// This file is Copyright (c) 2015 Florent Kermarrec <florent@enjoy-digital.fr>
2+
// License: BSD
3+
4+
#include <time.h>
5+
6+
#include "Vdut.h"
7+
#include "verilated.h"
8+
#include "verilated_vcd_c.h"
9+
10+
#include <stdio.h>
11+
#include <stdlib.h>
12+
#include <string.h>
13+
#include <unistd.h>
14+
#include <sys/select.h>
15+
#include <sys/socket.h>
16+
#include <sys/ioctl.h>
17+
#include <sys/stat.h>
18+
#include <sys/types.h>
19+
#include <termios.h>
20+
21+
int trace = 0;
22+
23+
struct termios orig_termios;
24+
25+
void reset_terminal_mode(void)
26+
{
27+
tcsetattr(0, TCSANOW, &orig_termios);
28+
}
29+
30+
void set_conio_terminal_mode(void)
31+
{
32+
struct termios new_termios;
33+
34+
/* take two copies - one for now, one for later */
35+
tcgetattr(0, &orig_termios);
36+
memcpy(&new_termios, &orig_termios, sizeof(new_termios));
37+
38+
/* register cleanup handler, and set the new terminal mode */
39+
atexit(reset_terminal_mode);
40+
cfmakeraw(&new_termios);
41+
tcsetattr(0, TCSANOW, &new_termios);
42+
}
43+
44+
int kbhit(void)
45+
{
46+
struct timeval tv = { 0L, 0L };
47+
fd_set fds;
48+
FD_ZERO(&fds);
49+
FD_SET(0, &fds);
50+
return select(1, &fds, NULL, NULL, &tv);
51+
}
52+
53+
int getch(void)
54+
{
55+
int r;
56+
unsigned char c;
57+
if ((r = read(0, &c, sizeof(c))) < 0) {
58+
return r;
59+
} else {
60+
return c;
61+
}
62+
}
63+
64+
vluint64_t main_time = 0;
65+
double sc_time_stamp () {
66+
return main_time;
67+
}
68+
69+
Vdut* dut;
70+
VerilatedVcdC* tfp;
71+
unsigned int tick;
72+
73+
/* ios */
74+
/* ios */
75+
76+
int console_service() {
77+
/* fpga --> console */
78+
SERIAL_SOURCE_ACK = 1;
79+
if(SERIAL_SOURCE_STB == 1) {
80+
if (SERIAL_SOURCE_DATA == '\n')
81+
putchar('\r');
82+
putchar(SERIAL_SOURCE_DATA);
83+
fflush(stdout);
84+
}
85+
86+
/* console --> fpga */
87+
SERIAL_SINK_STB = 0;
88+
if (tick%(1000) == 0) {
89+
if(kbhit()) {
90+
char c = getch();
91+
if (c == 27 && !kbhit()) {
92+
printf("\r\n");
93+
return -1;
94+
} else {
95+
SERIAL_SINK_STB = 1;
96+
SERIAL_SINK_DATA = c;
97+
}
98+
}
99+
}
100+
return 0;
101+
}
102+
103+
void sim_tick() {
104+
SYS_CLK = tick%2;
105+
dut->eval();
106+
if (trace)
107+
tfp->dump(tick);
108+
tick++;
109+
}
110+
111+
void sim_init() {
112+
int i;
113+
tick = 0;
114+
#ifdef SYS_RST
115+
SYS_RST = 1;
116+
SYS_CLK = 0;
117+
for (i=0; i<8; i++)
118+
sim_tick();
119+
SYS_RST = 0;
120+
#endif
121+
}
122+
123+
int main(int argc, char **argv, char **env) {
124+
125+
clock_t start;
126+
clock_t end;
127+
float speed;
128+
129+
set_conio_terminal_mode();
130+
131+
Verilated::commandArgs(argc, argv);
132+
dut = new Vdut;
133+
134+
Verilated::traceEverOn(true);
135+
tfp = new VerilatedVcdC;
136+
dut->trace(tfp, 99);
137+
tfp->open("dut.vcd");
138+
139+
start = clock();
140+
sim_init();
141+
bool run = true;
142+
while(run) {
143+
sim_tick();
144+
if (SYS_CLK) {
145+
if (console_service() != 0)
146+
run = false;
147+
}
148+
}
149+
end = clock();
150+
151+
speed = (tick/2)/((end-start)/CLOCKS_PER_SEC);
152+
153+
printf("averate speed: %3.3f MHz\n\r", speed/1000000);
154+
155+
tfp->close();
156+
157+
exit(0);
158+
}

‎mibuild/sim/verilator.py

+111
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,111 @@
1+
# This file is Copyright (c) 2015 Florent Kermarrec <florent@enjoy-digital.fr>
2+
# License: BSD
3+
4+
import os, subprocess
5+
6+
from migen.fhdl.std import *
7+
from migen.fhdl.structure import _Fragment
8+
from mibuild.generic_platform import *
9+
10+
def _build_tb(platform, template):
11+
12+
def io_name(ressource, subsignal=None):
13+
res = platform.lookup_request(ressource)
14+
if subsignal is not None:
15+
res = getattr(res, subsignal)
16+
return platform.vns.get_name(res)
17+
18+
ios = """
19+
#define SYS_CLK dut->{sys_clk}
20+
21+
#define SERIAL_SOURCE_STB dut->{serial_source_stb}
22+
#define SERIAL_SOURCE_ACK dut->{serial_source_ack}
23+
#define SERIAL_SOURCE_DATA dut->{serial_source_data}
24+
25+
#define SERIAL_SINK_STB dut->{serial_sink_stb}
26+
#define SERIAL_SINK_ACK dut->{serial_sink_ack}
27+
#define SERIAL_SINK_DATA dut->{serial_sink_data}
28+
""".format(
29+
sys_clk=io_name("sys_clk"),
30+
31+
serial_source_stb=io_name("serial", "source_stb"),
32+
serial_source_ack=io_name("serial", "source_ack"),
33+
serial_source_data=io_name("serial", "source_data"),
34+
35+
serial_sink_stb=io_name("serial", "sink_stb"),
36+
serial_sink_ack=io_name("serial", "sink_ack"),
37+
serial_sink_data=io_name("serial", "sink_data"),
38+
)
39+
40+
content = ""
41+
f = open(template, "r")
42+
done = False
43+
for l in f:
44+
content += l
45+
if "/* ios */" in l and not done:
46+
content += ios
47+
done = True
48+
49+
f.close()
50+
tools.write_to_file("dut_tb.cpp", content)
51+
52+
def _build_sim(platform, build_name, include_paths, verilator_root_path, template_file, trace):
53+
include = ""
54+
for path in include_paths:
55+
include += "-I"+path+" "
56+
57+
build_script_contents = """# Autogenerated by mibuild
58+
rm -rf obj_dir/
59+
verilator {disable_warnings} -O3 --cc dut.v --exe dut_tb.cpp {trace} {include}
60+
make -j -C obj_dir/ -f Vdut.mk Vdut VERILATOR_ROOT={verilator_root}
Has conversations. Original line has conversations.
61+
62+
""".format(verilator_root= os.path.join("../../", verilator_root_path), # XXX
63+
disable_warnings="-Wno-lint -Wno-INITIALDLY",
Has conversations. Original line has conversations.
64+
trace="-trace" if trace else "",
65+
include=include)
66+
build_script_file = "build_" + build_name + ".sh"
67+
tools.write_to_file(build_script_file, build_script_contents, force_unix=True)
68+
69+
_build_tb(platform, os.path.join("../", template_file)) # XXX
70+
r = subprocess.call(["bash", build_script_file])
71+
if r != 0:
72+
raise OSError("Subprocess failed")
73+
74+
def _run_sim(build_name):
75+
run_script_contents = """obj_dir/Vdut
76+
"""
77+
run_script_file = "run_" + build_name + ".sh"
78+
tools.write_to_file(run_script_file, run_script_contents, force_unix=True)
79+
r = subprocess.call(["bash", run_script_file])
80+
if r != 0:
81+
raise OSError("Subprocess failed")
82+
83+
class VerilatorPlatform(GenericPlatform):
84+
# XXX fix template / verilator_path
85+
def build(self, soc, build_dir="build", build_name="top", run=True, trace=True,
86+
template_file="../migen/mibuild/sim/dut_tb.cpp",
87+
verilator_root_path="../verilator"):
88+
tools.mkdir_noerror(build_dir)
89+
os.chdir(build_dir)
90+
91+
self.soc = soc
92+
fragment = soc.get_fragment()
93+
self.finalize(fragment)
94+
v_src, vns = self.get_verilog(fragment)
95+
named_sc, named_pc = self.resolve_signals(vns)
96+
self.vns = vns
97+
v_file = "dut.v"
98+
tools.write_to_file(v_file, v_src)
99+
100+
include_paths = []
101+
for source in self.sources:
102+
path = os.path.dirname(source[0]).replace("\\", "\/")
103+
if path not in include_paths:
104+
include_paths.append(path)
105+
include_paths += self.verilog_include_paths
106+
_build_sim(self, build_name, include_paths, verilator_root_path, template_file, trace)
107+
108+
if run:
109+
_run_sim(build_name)
110+
111+
os.chdir("..")

0 commit comments

Comments
 (0)
Please sign in to comment.