simulators.py 8.63 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
class Simulator(object):
    # number of cores required for this simulator
    def resreq_cores(self):
        return 1

    # memory required for this simulator (in MB)
    def resreq_mem(self):
        return 64

    def prep_cmds(self, env):
        return []

    def run_cmd(self, env):
        pass

class HostSim(Simulator):
    node_config = None
    name = ''
    wait = False
20
    sleep = 0
21
    cpu_freq = '3GHz'
22

23
24
25
    sync_period = 500
    pci_latency = 500

26
27
28
29
    def __init__(self):
        self.nics = []

    def full_name(self):
30
        return 'host.' + self.name
31
32

    def add_nic(self, nic):
33
        nic.name = self.name + '.' + nic.name
34
35
        self.nics.append(nic)

36
37
38
    def set_config(self, nc):
        self.node_config = nc

39
40
41
42
class NICSim(Simulator):
    network = None
    name = ''

43
44
45
46
    sync_period = 500
    pci_latency = 500
    eth_latency = 500

47
48
49
50
    def set_network(self, net):
        self.network = net
        net.nics.append(self)

51
52
    def basic_run_cmd(self, env, name, extra=None):
        cmd = '%s/%s %s %s %s 0 %d %d %d' % \
53
            (env.repodir, name, env.nic_pci_path(self), env.nic_eth_path(self),
54
55
56
57
58
59
                    env.nic_shm_path(self), self.sync_period, self.pci_latency,
                    self.eth_latency)

        if extra is not None:
            cmd += ' ' + extra
        return cmd
60
61

    def full_name(self):
62
        return 'nic.' + self.name
63
64
65

class NetSim(Simulator):
    name = ''
66
    opt = ''
67
68
    sync_period = 500
    eth_latency = 500
69
70
71
72
73

    def __init__(self):
        self.nics = []

    def full_name(self):
74
        return 'net.' + self.name
75
76
77


class QemuHost(HostSim):
78
    sync = False
79
80
81
82
83
84
85
86
87
    def resreq_cores(self):
        return self.node_config.cores + 1

    def resreq_mem(self):
        return 4096

    def prep_cmds(self, env):
        to_path = env.hdcopy_path(self)
        return [f'{env.qemu_img_path} create -f qcow2 -o '
88
            f'backing_file="{env.hd_path(self.node_config.disk_image)}" '
89
90
91
            f'{env.hdcopy_path(self)}']

    def run_cmd(self, env):
92
93
        cmd = (f'{env.qemu_path} -machine q35 -serial mon:stdio '
            '-display none -nic none '
94
95
96
97
98
99
            f'-kernel {env.qemu_kernel_path} '
            f'-drive file={env.hdcopy_path(self)},if=ide,index=0,media=disk '
            f'-drive file={env.cfgtar_path(self)},if=ide,index=1,media=disk,'
                'driver=raw '
            '-append "earlyprintk=ttyS0 console=ttyS0 root=/dev/sda1 '
                'init=/home/ubuntu/guestinit.sh rw" '
100
            f'-m {self.node_config.memory} -smp {self.node_config.cores} ')
101
102
103
104
105
106

        if self.sync:
            cmd += ' -cpu Skylake-Server -icount shift=0,sleep=off '
        else:
            cmd += ' -cpu host -enable-kvm '

107
108
109
110
        if len(self.nics) > 0:
            assert len(self.nics) == 1
            cmd += f'-chardev socket,path={env.nic_pci_path(self.nics[0])},'
            cmd += 'id=cosimcd '
111
112
113
114
115
116
117
118
119
            cmd += f'-device cosim-pci,chardev=cosimcd'
            if self.sync:
                cmd += ',sync=on'
                cmd += f',pci-latency={self.pci_latency}'
                cmd += f',sync-period={self.sync_period}'
            else:
                cmd += ',sync=off'
            cmd += ' '

120
121
122
123
124
125
        return cmd

class Gem5Host(HostSim):
    cpu_type_cp = 'X86KvmCPU'
    cpu_type = 'TimingSimpleCPU'

126

127
128
129
130
    def set_config(self, nc):
        nc.sim = 'gem5'
        super().set_config(nc)

131
132
133
134
135
136
    def resreq_cores(self):
        return 1

    def resreq_mem(self):
        return 4096

137
138
    def prep_cmds(self, env):
        return [f'mkdir -p {env.gem5_cpdir(self)}']
139
140
141
142
143
144
145
146
147

    def run_cmd(self, env):
        cpu_type = self.cpu_type
        if env.create_cp:
            cpu_type = self.cpu_type_cp

        cmd = (f'{env.gem5_path} --outdir={env.gem5_outdir(self)} '
            f'{env.gem5_py_path} --caches --l2cache --l3cache '
            '--l1d_size=32kB --l1i_size=32kB --l2_size=2MB --l3_size=32MB '
148
            f'--cacheline_size=64 --cpu-clock={self.cpu_freq} '
149
150
            f'--checkpoint-dir={env.gem5_cpdir(self)} '
            f'--kernel={env.gem5_kernel_path} '
151
            f'--disk-image={env.hd_raw_path(self.node_config.disk_image)} '
152
            f'--disk-image={env.cfgtar_path(self)} '
153
            f'--cpu-type={cpu_type} --mem-size={self.node_config.memory}MB '
154
            f'--num-cpus={self.node_config.cores} '
155
156
157
158
159
160
161
162
163
164
165
166
            '--ddio-enabled --ddio-way-part=8 --mem-type=DDR4_2400_16x4 ')

        if env.restore_cp:
            cmd += '-r 0 '

        if len(self.nics) > 0:
            assert len(self.nics) == 1
            nic = self.nics[0]
            cmd += f'--cosim-pci={env.nic_pci_path(nic)} '
            cmd += f'--cosim-shm={env.nic_shm_path(nic)} '
            if cpu_type == 'TimingSimpleCPU':
                cmd += '--cosim-sync '
167
168
                cmd += f'--cosim-pci-lat={self.pci_latency} '
                cmd += f'--cosim-sync-int={self.sync_period} '
169
170
171
172
173
174
175
            if isinstance(nic, I40eNIC):
                cmd += '--cosim-type=i40e '
        return cmd



class CorundumVerilatorNIC(NICSim):
176
177
    clock_freq = 250 # MHz

178
179
180
181
182
    def resreq_mem(self):
        # this is a guess
        return 512

    def run_cmd(self, env):
183
184
        return self.basic_run_cmd(env, 'corundum/corundum_verilator',
            str(self.clock_freq))
185
186
187
188
189
190
191
192
193
194
195
196
197
198

class CorundumBMNIC(NICSim):
    def run_cmd(self, env):
        return self.basic_run_cmd(env, 'corundum_bm/corundum_bm')

class I40eNIC(NICSim):
    def run_cmd(self, env):
        return self.basic_run_cmd(env, 'i40e_bm/i40e_bm')



class WireNet(NetSim):
    def run_cmd(self, env):
        assert len(self.nics) == 2
199
        return '%s/net_wire/net_wire %s %s %d %d' % \
200
                (env.repodir, env.nic_eth_path(self.nics[0]),
201
202
                        env.nic_eth_path(self.nics[1]),
                        self.sync_period, self.eth_latency)
203
204
205
206

class SwitchNet(NetSim):
    def run_cmd(self, env):
        cmd = env.repodir + '/net_switch/net_switch'
207
        cmd += f' -S {self.sync_period} -E {self.eth_latency}'
208
209
210
        for n in self.nics:
            cmd += ' -s ' + env.nic_eth_path(n)
        return cmd
211
212


213
214
215
216
217
218
219
220
221
222
223
224
225
226
class NS3DumbbellNet(NetSim):
    def run_cmd(self, env):
        ports = ''
        for n in self.nics:
            if 'server' in n.name:
                ports += '--CosimPortLeft=' + env.nic_eth_path(n) + ' '
            else:
                ports += '--CosimPortRight=' + env.nic_eth_path(n) + ' '

        cmd = env.repodir + '/ns-3' + '/cosim-run.sh cosim cosim-dumbbell-example ' + ports + ' ' + self.opt
        print(cmd)

        return cmd

227
228
229
230
231
232
233
234
235
236
class NS3BridgeNet(NetSim):
    def run_cmd(self, env):
        ports = ''
        for n in self.nics:
            ports += '--CosimPort=' + env.nic_eth_path(n) + ' '

        cmd = env.repodir + '/ns-3' + '/cosim-run.sh cosim cosim-bridge-example ' + ports + ' ' + self.opt
        print(cmd)

        return cmd
237

238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
class NS3SequencerNet(NetSim):
    def run_cmd(self, env):
        ports = ''
        for n in self.nics:
            if 'client' in n.name:
                ports += '--ClientPort=' + env.nic_eth_path(n) + ' '
            elif 'replica' in n.name:
                ports += '--ServerPort=' + env.nic_eth_path(n) + ' '
            elif 'sequencer' in n.name:
                ports += '--EndhostSequencerPort=' + env.nic_eth_path(n) + ' '
            else:
                raise Exception('Wrong NIC type')
        cmd = env.repodir + '/ns-3' + '/cosim-run.sh sequencer sequencer-single-switch-example ' + ports + ' ' + self.opt
        return cmd


254
255
256
257
258
def create_basic_hosts(e, num, name_prefix, net, nic_class, host_class,
        nc_class, app_class, ip_start=1):
    hosts = []
    for i in range(0, num):
        nic = nic_class()
259
        #nic.name = '%s.%d' % (name_prefix, i)
260
261
262
263
        nic.set_network(net)

        host = host_class()
        host.name = '%s.%d' % (name_prefix, i)
264
265
266
267
268
269

        node_config = nc_class()
        node_config.ip = '10.0.0.%d' % (ip_start + i)
        node_config.app = app_class()
        host.set_config(node_config)

270
271
272
273
274
275
276
        host.add_nic(nic)
        e.add_nic(nic)
        e.add_host(host)

        hosts.append(host)

    return hosts
277
278
279
280
281
282
283
284
285
286
287
288


def create_dctcp_hosts(e, num, name_prefix, net, nic_class, host_class,
        nc_class, app_class, cpu_freq, mtu, ip_start=1):
    hosts = []
    for i in range(0, num):
        nic = nic_class()
        #nic.name = '%s.%d' % (name_prefix, i)
        nic.set_network(net)

        host = host_class()
        host.name = '%s.%d' % (name_prefix, i)
Hejing Li's avatar
Hejing Li committed
289
        host.cpu_freq = cpu_freq
290
291
292
293
294
295
296
297
298
299
300
301
302
303

        node_config = nc_class()
        node_config.mtu = mtu
        node_config.ip = '192.168.64.%d' % (ip_start + i)
        node_config.app = app_class()
        host.set_config(node_config)

        host.add_nic(nic)
        e.add_nic(nic)
        e.add_host(host)

        hosts.append(host)

    return hosts