forked from p4lang/p4app-switchML
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdrop_simulator.py
128 lines (101 loc) · 4.8 KB
/
drop_simulator.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
# Copyright 2021 Intel-KAUST-Microsoft
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import logging
from control import Control
class DropSimulator(Control):
def __init__(self, target, gc, bfrt_info):
# Set up base class
super(DropSimulator, self).__init__(target, gc)
self.log = logging.getLogger(__name__)
self.tables = [bfrt_info.table_get('pipe.IngressParser.$PORT_METADATA')]
self.table = self.tables[0]
# Counter
self.simulated_drop_counter = bfrt_info.table_get(
'pipe.Ingress.egress_drop_sim.simulated_drop_packet_counter')
self.ingress_drop_probability = 0
self.egress_drop_probability = 0
# Clear table and counter (set the drop probability to zero)
self._clear()
def _clear(self):
self.table.entry_del(self.target)
self.reset_counter()
def set_drop_probabilities(self, ingress_drop_probability,
egress_drop_probability):
''' Set the probabilities of simulated drops.
Keyword arguments:
ingress_drop_probability -- probability to drop in ingress
egress_drop_probability -- probability to drop in egress
Returns:
(success flag, ingress_drop_probability or error message,
egress_drop_probability or None)
'''
if not 0 <= ingress_drop_probability <= 1:
return (False, 'Ingress drop probability must be in [0,1]', None)
if not 0 <= egress_drop_probability <= 1:
return (False, 'Egress drop probability must be in [0,1]', None)
# Compute the actual value of probabilities
ingress_drop_value = int(0xffff * ingress_drop_probability)
self.ingress_drop_probability = float(ingress_drop_value) / 0xffff
egress_drop_value = int(0xffff * egress_drop_probability)
self.egress_drop_probability = float(egress_drop_value) / 0xffff
self.log.info('Ingress drop probability: 0x{:0x} or {:.2f}'.format(
ingress_drop_value, self.ingress_drop_probability))
self.log.info('Egress drop probability: 0x{:0x} or {:.2f}'.format(
egress_drop_value, self.egress_drop_probability))
if ingress_drop_value == 0 and egress_drop_value == 0:
self.table.entry_del(self.target)
else:
# Set in all ports
num_ports = 288
self.table.entry_add(self.target, [
self.table.make_key(
[self.gc.KeyTuple('ig_intr_md.ingress_port', p)])
for p in range(num_ports)
], [
self.table.make_data([
self.gc.DataTuple('ingress_drop_probability',
ingress_drop_value),
self.gc.DataTuple('egress_drop_probability',
egress_drop_value)
])
] * num_ports)
return (True, self.ingress_drop_probability,
self.egress_drop_probability)
def get_drop_probabilities(self):
''' Get the probabilities of simulated drops '''
return self.ingress_drop_probability, self.egress_drop_probability
def reset_counter(self):
''' Reset dropped packets counter '''
self.simulated_drop_counter.entry_del(self.target)
def get_counter(self, qpn=None):
''' Get the current values of dropped packets counter per-QP.
If a queue pair number is provided, it will return only
the value for that queue pair, otherwise it will return
all of them. '''
self.simulated_drop_counter.operations_execute(self.target, 'Sync')
count = self.simulated_drop_counter.info.size
resp = self.simulated_drop_counter.entry_get(self.target, [
self.simulated_drop_counter.make_key(
[self.gc.KeyTuple('$COUNTER_INDEX', i)]) for i in range(count)
],
flags={'from_hw': False})
values = []
for v, k in resp:
v = v.to_dict()
k = k.to_dict()
idx = k['$COUNTER_INDEX']['value']
pkts = v['$COUNTER_SPEC_PKTS']
if qpn == None or qpn == idx:
values.append({'QP': idx, 'packets': pkts})
return values