"vscode:/vscode.git/clone" did not exist on "ed64c7bd4e7c51a6a84a82480b64317f3f951689"
simulators.py 9.07 KB
Newer Older
1
2
import math

3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
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
22
    sleep = 0
23
    cpu_freq = '3GHz'
24

25
26
27
    sync_period = 500
    pci_latency = 500

28
29
30
31
    def __init__(self):
        self.nics = []

    def full_name(self):
32
        return 'host.' + self.name
33
34

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

38
39
40
    def set_config(self, nc):
        self.node_config = nc

41
42
43
44
class NICSim(Simulator):
    network = None
    name = ''

45
46
47
48
    sync_period = 500
    pci_latency = 500
    eth_latency = 500

49
50
51
52
    def set_network(self, net):
        self.network = net
        net.nics.append(self)

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

        if extra is not None:
            cmd += ' ' + extra
        return cmd
62
63

    def full_name(self):
64
        return 'nic.' + self.name
65
66
67

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

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

    def full_name(self):
76
        return 'net.' + self.name
77
78
79


class QemuHost(HostSim):
80
    sync = False
81
    def resreq_cores(self):
82
83
84
85
        if self.sync:
            return 1
        else:
            return self.node_config.cores + 1
86
87
88
89
90
91
92

    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 '
93
            f'backing_file="{env.hd_path(self.node_config.disk_image)}" '
94
95
96
            f'{env.hdcopy_path(self)}']

    def run_cmd(self, env):
97
98
        cmd = (f'{env.qemu_path} -machine q35 -serial mon:stdio '
            '-display none -nic none '
99
100
101
102
103
104
            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" '
105
            f'-m {self.node_config.memory} -smp {self.node_config.cores} ')
106
107

        if self.sync:
108
109
110
111
112
113
114
115
116
117
118
            unit = self.cpu_freq[-3:]
            if unit.lower() == 'ghz':
                base = 0
            elif unit.lower() == 'mhz':
                base = 3
            else:
                raise Exception('cpu frequency specified in unsupported unit')
            num = float(self.cpu_freq[:-3])
            shift = base - int(math.ceil(math.log(num, 2)))

            cmd += f' -cpu Skylake-Server -icount shift={shift},sleep=off '
119
120
121
        else:
            cmd += ' -cpu host -enable-kvm '

122
123
124
125
        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 '
126
127
128
129
130
131
132
133
134
            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 += ' '

135
136
137
138
139
140
        return cmd

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

141

142
143
144
145
    def set_config(self, nc):
        nc.sim = 'gem5'
        super().set_config(nc)

146
147
148
149
150
151
    def resreq_cores(self):
        return 1

    def resreq_mem(self):
        return 4096

152
153
    def prep_cmds(self, env):
        return [f'mkdir -p {env.gem5_cpdir(self)}']
154
155
156
157
158
159
160
161
162

    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 '
163
            f'--cacheline_size=64 --cpu-clock={self.cpu_freq} '
164
165
            f'--checkpoint-dir={env.gem5_cpdir(self)} '
            f'--kernel={env.gem5_kernel_path} '
166
            f'--disk-image={env.hd_raw_path(self.node_config.disk_image)} '
167
            f'--disk-image={env.cfgtar_path(self)} '
168
            f'--cpu-type={cpu_type} --mem-size={self.node_config.memory}MB '
169
            f'--num-cpus={self.node_config.cores} '
170
171
172
173
174
175
176
177
178
179
180
181
            '--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 '
182
183
                cmd += f'--cosim-pci-lat={self.pci_latency} '
                cmd += f'--cosim-sync-int={self.sync_period} '
184
185
186
187
188
189
190
            if isinstance(nic, I40eNIC):
                cmd += '--cosim-type=i40e '
        return cmd



class CorundumVerilatorNIC(NICSim):
191
192
    clock_freq = 250 # MHz

193
194
195
196
197
    def resreq_mem(self):
        # this is a guess
        return 512

    def run_cmd(self, env):
198
199
        return self.basic_run_cmd(env, 'corundum/corundum_verilator',
            str(self.clock_freq))
200
201
202
203
204
205
206
207
208
209
210
211
212
213

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
214
        return '%s/net_wire/net_wire %s %s %d %d' % \
215
                (env.repodir, env.nic_eth_path(self.nics[0]),
216
217
                        env.nic_eth_path(self.nics[1]),
                        self.sync_period, self.eth_latency)
218
219
220
221

class SwitchNet(NetSim):
    def run_cmd(self, env):
        cmd = env.repodir + '/net_switch/net_switch'
222
        cmd += f' -S {self.sync_period} -E {self.eth_latency}'
223
224
225
        for n in self.nics:
            cmd += ' -s ' + env.nic_eth_path(n)
        return cmd
226
227


228
229
230
231
232
233
234
235
236
237
238
239
240
241
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

242
243
244
245
246
247
248
249
250
251
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
252

253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
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


269
270
271
272
273
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()
274
        #nic.name = '%s.%d' % (name_prefix, i)
275
276
277
278
        nic.set_network(net)

        host = host_class()
        host.name = '%s.%d' % (name_prefix, i)
279
280
281
282
283
284

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

285
286
287
288
289
290
291
        host.add_nic(nic)
        e.add_nic(nic)
        e.add_host(host)

        hosts.append(host)

    return hosts
292
293
294
295
296
297
298
299
300
301
302
303


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
304
        host.cpu_freq = cpu_freq
305
306
307
308
309
310
311
312
313
314
315
316
317
318

        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