]> git.proxmox.com Git - mirror_ovs.git/blob - tests/flowgen.pl
ofp-util: Wildcard VLAN PCP in OF1.0 matches when 802.1Q not present.
[mirror_ovs.git] / tests / flowgen.pl
1 #! /usr/bin/perl
2
3 # Copyright (c) 2009, 2010, 2011, 2012 Nicira, Inc.
4 #
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at:
8 #
9 # http://www.apache.org/licenses/LICENSE-2.0
10 #
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
16
17 use strict;
18 use warnings;
19
20 open(FLOWS, ">&=3");# or die "failed to open fd 3 for writing: $!\n";
21 open(PACKETS, ">&=4");# or die "failed to open fd 4 for writing: $!\n";
22
23 # Print pcap file header.
24 print PACKETS pack('NnnNNNN',
25 0xa1b2c3d4, # magic number
26 2, # major version
27 4, # minor version
28 0, # time zone offset
29 0, # time stamp accuracy
30 1518, # snaplen
31 1); # Ethernet
32
33 output(DL_HEADER => '802.2');
34
35 for my $dl_header (qw(802.2+SNAP Ethernet)) {
36 my %a = (DL_HEADER => $dl_header);
37 for my $dl_vlan (qw(none zero nonzero)) {
38 my %b = (%a, DL_VLAN => $dl_vlan);
39
40 # Non-IP case.
41 output(%b, DL_TYPE => 'non-ip');
42
43 for my $ip_options (qw(no yes)) {
44 my %c = (%b, DL_TYPE => 'ip', IP_OPTIONS => $ip_options);
45 for my $ip_fragment (qw(no first middle last)) {
46 my %d = (%c, IP_FRAGMENT => $ip_fragment);
47 for my $tp_proto (qw(TCP TCP+options UDP ICMP other)) {
48 output(%d, TP_PROTO => $tp_proto);
49 }
50 }
51 }
52 }
53 }
54
55 sub output {
56 my (%attrs) = @_;
57
58 # Compose flow.
59 my (%flow);
60 $flow{DL_SRC} = "00:02:e3:0f:80:a4";
61 $flow{DL_DST} = "00:1a:92:40:ac:05";
62 $flow{NW_PROTO} = 0;
63 $flow{NW_TOS} = 0;
64 $flow{NW_SRC} = '0.0.0.0';
65 $flow{NW_DST} = '0.0.0.0';
66 $flow{TP_SRC} = 0;
67 $flow{TP_DST} = 0;
68 if (defined($attrs{DL_VLAN})) {
69 my (%vlan_map) = ('none' => 0xffff,
70 'zero' => 0,
71 'nonzero' => 0x0123);
72 $flow{DL_VLAN} = $vlan_map{$attrs{DL_VLAN}};
73 } else {
74 $flow{DL_VLAN} = 0xffff; # OFP_VLAN_NONE
75 }
76 if ($attrs{DL_HEADER} eq '802.2') {
77 $flow{DL_TYPE} = 0x5ff; # OFP_DL_TYPE_NOT_ETH_TYPE
78 } elsif ($attrs{DL_TYPE} eq 'ip') {
79 $flow{DL_TYPE} = 0x0800; # ETH_TYPE_IP
80 $flow{NW_SRC} = '10.0.2.15';
81 $flow{NW_DST} = '192.168.1.20';
82 $flow{NW_TOS} = 44;
83 if ($attrs{TP_PROTO} eq 'other') {
84 $flow{NW_PROTO} = 42;
85 } elsif ($attrs{TP_PROTO} eq 'TCP' ||
86 $attrs{TP_PROTO} eq 'TCP+options') {
87 $flow{NW_PROTO} = 6; # IPPROTO_TCP
88 $flow{TP_SRC} = 6667;
89 $flow{TP_DST} = 9998;
90 } elsif ($attrs{TP_PROTO} eq 'UDP') {
91 $flow{NW_PROTO} = 17; # IPPROTO_UDP
92 $flow{TP_SRC} = 1112;
93 $flow{TP_DST} = 2223;
94 } elsif ($attrs{TP_PROTO} eq 'ICMP') {
95 $flow{NW_PROTO} = 1; # IPPROTO_ICMP
96 $flow{TP_SRC} = 8; # echo request
97 $flow{TP_DST} = 0; # code
98 } else {
99 die;
100 }
101 if ($attrs{IP_FRAGMENT} ne 'no' && $attrs{IP_FRAGMENT} ne 'first') {
102 $flow{TP_SRC} = $flow{TP_DST} = 0;
103 }
104 } elsif ($attrs{DL_TYPE} eq 'non-ip') {
105 $flow{DL_TYPE} = 0x5678;
106 } else {
107 die;
108 }
109
110 # Compose packet.
111 my $packet = '';
112 my $wildcards = 0;
113 $packet .= pack_ethaddr($flow{DL_DST});
114 $packet .= pack_ethaddr($flow{DL_SRC});
115 if ($flow{DL_VLAN} != 0xffff) {
116 $packet .= pack('nn', 0x8100, $flow{DL_VLAN});
117 } else {
118 $wildcards |= 1 << 20; # OFPFW10_DL_VLAN_PCP
119 }
120 my $len_ofs = length($packet);
121 $packet .= pack('n', 0) if $attrs{DL_HEADER} =~ /^802.2/;
122 if ($attrs{DL_HEADER} eq '802.2') {
123 $packet .= pack('CCC', 0x42, 0x42, 0x03); # LLC for 802.1D STP.
124 } else {
125 if ($attrs{DL_HEADER} eq '802.2+SNAP') {
126 $packet .= pack('CCC', 0xaa, 0xaa, 0x03); # LLC for SNAP.
127 $packet .= pack('CCC', 0, 0, 0); # SNAP OUI.
128 }
129 $packet .= pack('n', $flow{DL_TYPE});
130 if ($attrs{DL_TYPE} eq 'ip') {
131 my $ip = pack('CCnnnCCnNN',
132 (4 << 4) | 5, # version, hdrlen
133 $flow{NW_TOS}, # type of service
134 0, # total length (filled in later)
135 65432, # id
136 0, # frag offset
137 64, # ttl
138 $flow{NW_PROTO}, # protocol
139 0, # checksum
140 0x0a00020f, # source
141 0xc0a80114); # dest
142 if ($attrs{IP_OPTIONS} eq 'yes') {
143 substr($ip, 0, 1) = pack('C', (4 << 4) | 8);
144 $ip .= pack('CCnnnCCCx',
145 130, # type
146 11, # length
147 0x6bc5, # top secret
148 0xabcd,
149 0x1234,
150 1,
151 2,
152 3);
153 }
154 if ($attrs{IP_FRAGMENT} ne 'no') {
155 my (%frag_map) = ('first' => 0x2000, # more frags, ofs 0
156 'middle' => 0x2111, # more frags, ofs 0x888
157 'last' => 0x0222); # last frag, ofs 0x1110
158 substr($ip, 6, 2)
159 = pack('n', $frag_map{$attrs{IP_FRAGMENT}});
160 }
161
162 if ($attrs{TP_PROTO} =~ '^TCP') {
163 my $tcp = pack('nnNNnnnn',
164 $flow{TP_SRC}, # source port
165 $flow{TP_DST}, # dest port
166 87123455, # seqno
167 712378912, # ackno
168 (5 << 12) | 0x02 | 0x10, # hdrlen, SYN, ACK
169 5823, # window size
170 18923, # checksum
171 12893); # urgent pointer
172 if ($attrs{TP_PROTO} eq 'TCP+options') {
173 substr($tcp, 12, 2) = pack('n', (6 << 12) | 0x02 | 0x10);
174 $tcp .= pack('CCn', 2, 4, 1975); # MSS option
175 }
176 $tcp .= 'payload';
177 $ip .= $tcp;
178 } elsif ($attrs{TP_PROTO} eq 'UDP') {
179 my $len = 15;
180 my $udp = pack('nnnn', $flow{TP_SRC}, $flow{TP_DST}, $len, 0);
181 $udp .= chr($len) while length($udp) < $len;
182 $ip .= $udp;
183 } elsif ($attrs{TP_PROTO} eq 'ICMP') {
184 $ip .= pack('CCnnn',
185 8, # echo request
186 0, # code
187 0, # checksum
188 736, # identifier
189 931); # sequence number
190 } elsif ($attrs{TP_PROTO} eq 'other') {
191 $ip .= 'other header';
192 } else {
193 die;
194 }
195 substr($ip, 2, 2) = pack('n', length($ip));
196 $packet .= $ip;
197 }
198 }
199 if ($attrs{DL_HEADER} =~ /^802.2/) {
200 my $len = length ($packet);
201 $len -= 4 if $flow{DL_VLAN} != 0xffff;
202 substr($packet, $len_ofs, 2) = pack('n', $len);
203 }
204
205 print join(' ', map("$_=$attrs{$_}", keys(%attrs))), "\n";
206 print join(' ', map("$_=$flow{$_}", keys(%flow))), "\n";
207 print "\n";
208
209 print FLOWS pack('Nn',
210 $wildcards, # wildcards
211 1); # in_port
212 print FLOWS pack_ethaddr($flow{DL_SRC});
213 print FLOWS pack_ethaddr($flow{DL_DST});
214 print FLOWS pack('nCxnCCxxNNnn',
215 $flow{DL_VLAN},
216 0, # DL_VLAN_PCP
217 $flow{DL_TYPE},
218 $flow{NW_TOS},
219 $flow{NW_PROTO},
220 inet_aton($flow{NW_SRC}),
221 inet_aton($flow{NW_DST}),
222 $flow{TP_SRC},
223 $flow{TP_DST});
224
225 print PACKETS pack('NNNN',
226 0, # timestamp seconds
227 0, # timestamp microseconds
228 length($packet), # bytes saved
229 length($packet)), # total length
230 $packet;
231 }
232
233 sub pack_ethaddr {
234 local ($_) = @_;
235 my $xx = '([0-9a-fA-F][0-9a-fA-F])';
236 my (@octets) = /$xx:$xx:$xx:$xx:$xx:$xx/;
237 @octets == 6 or die $_;
238 my ($out) = '';
239 $out .= pack('C', hex($_)) foreach @octets;
240 return $out;
241 }
242
243 sub inet_aton {
244 local ($_) = @_;
245 my ($a, $b, $c, $d) = /^(\d+)\.(\d+)\.(\d+)\.(\d+)$/;
246 defined $d or die $_;
247 return ($a << 24) | ($b << 16) | ($c << 8) | $d;
248 }