-
Notifications
You must be signed in to change notification settings - Fork 235
/
Copy pathl3-load-latency.lua
134 lines (125 loc) · 4.31 KB
/
l3-load-latency.lua
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
129
130
131
132
133
local mg = require "moongen"
local memory = require "memory"
local device = require "device"
local ts = require "timestamping"
local filter = require "filter"
local hist = require "histogram"
local stats = require "stats"
local timer = require "timer"
local arp = require "proto.arp"
local log = require "log"
-- set addresses here
local DST_MAC = nil -- resolved via ARP on GW_IP or DST_IP, can be overriden with a string here
local SRC_IP_BASE = "10.0.0.10" -- actual address will be SRC_IP_BASE + random(0, flows)
local DST_IP = "10.1.0.10"
local SRC_PORT = 1234
local DST_PORT = 319
-- answer ARP requests for this IP on the rx port
-- change this if benchmarking something like a NAT device
local RX_IP = DST_IP
-- used to resolve DST_MAC
local GW_IP = DST_IP
-- used as source IP to resolve GW_IP to DST_MAC
local ARP_IP = SRC_IP_BASE
function configure(parser)
parser:description("Generates UDP traffic and measure latencies. Edit the source to modify constants like IPs.")
parser:argument("txDev", "Device to transmit from."):convert(tonumber)
parser:argument("rxDev", "Device to receive from."):convert(tonumber)
parser:option("-r --rate", "Transmit rate in Mbit/s."):default(10000):convert(tonumber)
parser:option("-f --flows", "Number of flows (randomized source IP)."):default(4):convert(tonumber)
parser:option("-s --size", "Packet size."):default(60):convert(tonumber)
end
function master(args)
txDev = device.config{port = args.txDev, rxQueues = 3, txQueues = 3}
rxDev = device.config{port = args.rxDev, rxQueues = 3, txQueues = 3}
device.waitForLinks()
-- max 1kpps timestamping traffic timestamping
-- rate will be somewhat off for high-latency links at low rates
if args.rate > 0 then
txDev:getTxQueue(0):setRate(args.rate - (args.size + 4) * 8 / 1000)
end
mg.startTask("loadSlave", txDev:getTxQueue(0), rxDev, args.size, args.flows)
mg.startTask("timerSlave", txDev:getTxQueue(1), rxDev:getRxQueue(1), args.size, args.flows)
arp.startArpTask{
-- run ARP on both ports
{ rxQueue = rxDev:getRxQueue(2), txQueue = rxDev:getTxQueue(2), ips = RX_IP },
-- we need an IP address to do ARP requests on this interface
{ rxQueue = txDev:getRxQueue(2), txQueue = txDev:getTxQueue(2), ips = ARP_IP }
}
mg.waitForTasks()
end
local function fillUdpPacket(buf, len)
buf:getUdpPacket():fill{
ethSrc = queue,
ethDst = DST_MAC,
ip4Src = SRC_IP,
ip4Dst = DST_IP,
udpSrc = SRC_PORT,
udpDst = DST_PORT,
pktLength = len
}
end
local function doArp()
if not DST_MAC then
log:info("Performing ARP lookup on %s", GW_IP)
DST_MAC = arp.blockingLookup(GW_IP, 5)
if not DST_MAC then
log:info("ARP lookup failed, using default destination mac address")
return
end
end
log:info("Destination mac: %s", DST_MAC)
end
function loadSlave(queue, rxDev, size, flows)
doArp()
local mempool = memory.createMemPool(function(buf)
fillUdpPacket(buf, size)
end)
local bufs = mempool:bufArray()
local counter = 0
local txCtr = stats:newDevTxCounter(queue, "plain")
local rxCtr = stats:newDevRxCounter(rxDev, "plain")
local baseIP = parseIPAddress(SRC_IP_BASE)
while mg.running() do
bufs:alloc(size)
for i, buf in ipairs(bufs) do
local pkt = buf:getUdpPacket()
pkt.ip4.src:set(baseIP + counter)
counter = incAndWrap(counter, flows)
end
-- UDP checksums are optional, so using just IPv4 checksums would be sufficient here
bufs:offloadUdpChecksums()
queue:send(bufs)
txCtr:update()
rxCtr:update()
end
txCtr:finalize()
rxCtr:finalize()
end
function timerSlave(txQueue, rxQueue, size, flows)
doArp()
if size < 84 then
log:warn("Packet size %d is smaller than minimum timestamp size 84. Timestamped packets will be larger than load packets.", size)
size = 84
end
local timestamper = ts:newUdpTimestamper(txQueue, rxQueue)
local hist = hist:new()
mg.sleepMillis(1000) -- ensure that the load task is running
local counter = 0
local rateLimit = timer:new(0.001)
local baseIP = parseIPAddress(SRC_IP_BASE)
while mg.running() do
hist:update(timestamper:measureLatency(size, function(buf)
fillUdpPacket(buf, size)
local pkt = buf:getUdpPacket()
pkt.ip4.src:set(baseIP + counter)
counter = incAndWrap(counter, flows)
end))
rateLimit:wait()
rateLimit:reset()
end
-- print the latency stats after all the other stuff
mg.sleepMillis(300)
hist:print()
hist:save("histogram.csv")
end