2 * Copyright 2014-2015 Open Networking Laboratory
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
16 package org.onosproject.provider.of.flow.impl;
18 import org.onlab.packet.Ip4Address;
19 import org.onlab.packet.Ip4Prefix;
20 import org.onlab.packet.Ip6Address;
21 import org.onlab.packet.Ip6Prefix;
22 import org.onlab.packet.VlanId;
23 import org.onosproject.net.OchSignal;
24 import org.onosproject.net.driver.DriverService;
25 import org.onosproject.net.flow.FlowRule;
26 import org.onosproject.net.flow.TrafficSelector;
27 import org.onosproject.net.flow.criteria.ArpHaCriterion;
28 import org.onosproject.net.flow.criteria.ArpOpCriterion;
29 import org.onosproject.net.flow.criteria.ArpPaCriterion;
30 import org.onosproject.net.flow.criteria.Criterion;
31 import org.onosproject.net.flow.criteria.EthCriterion;
32 import org.onosproject.net.flow.criteria.EthTypeCriterion;
33 import org.onosproject.net.flow.criteria.IPCriterion;
34 import org.onosproject.net.flow.criteria.IPDscpCriterion;
35 import org.onosproject.net.flow.criteria.IPEcnCriterion;
36 import org.onosproject.net.flow.criteria.IPProtocolCriterion;
37 import org.onosproject.net.flow.criteria.IPv6ExthdrFlagsCriterion;
38 import org.onosproject.net.flow.criteria.IPv6FlowLabelCriterion;
39 import org.onosproject.net.flow.criteria.IPv6NDLinkLayerAddressCriterion;
40 import org.onosproject.net.flow.criteria.IPv6NDTargetAddressCriterion;
41 import org.onosproject.net.flow.criteria.IcmpCodeCriterion;
42 import org.onosproject.net.flow.criteria.IcmpTypeCriterion;
43 import org.onosproject.net.flow.criteria.Icmpv6CodeCriterion;
44 import org.onosproject.net.flow.criteria.Icmpv6TypeCriterion;
45 import org.onosproject.net.flow.criteria.MetadataCriterion;
46 import org.onosproject.net.flow.criteria.MplsBosCriterion;
47 import org.onosproject.net.flow.criteria.MplsCriterion;
48 import org.onosproject.net.flow.criteria.OchSignalCriterion;
49 import org.onosproject.net.flow.criteria.OchSignalTypeCriterion;
50 import org.onosproject.net.flow.criteria.PortCriterion;
51 import org.onosproject.net.flow.criteria.SctpPortCriterion;
52 import org.onosproject.net.flow.criteria.TcpPortCriterion;
53 import org.onosproject.net.flow.criteria.TunnelIdCriterion;
54 import org.onosproject.net.flow.criteria.UdpPortCriterion;
55 import org.onosproject.net.flow.criteria.VlanIdCriterion;
56 import org.onosproject.net.flow.criteria.VlanPcpCriterion;
57 import org.projectfloodlight.openflow.protocol.OFFactory;
58 import org.projectfloodlight.openflow.protocol.OFFlowAdd;
59 import org.projectfloodlight.openflow.protocol.OFFlowDelete;
60 import org.projectfloodlight.openflow.protocol.OFFlowMod;
61 import org.projectfloodlight.openflow.protocol.match.Match;
62 import org.projectfloodlight.openflow.protocol.match.MatchField;
63 import org.projectfloodlight.openflow.types.ArpOpcode;
64 import org.projectfloodlight.openflow.types.CircuitSignalID;
65 import org.projectfloodlight.openflow.types.EthType;
66 import org.projectfloodlight.openflow.types.ICMPv4Code;
67 import org.projectfloodlight.openflow.types.ICMPv4Type;
68 import org.projectfloodlight.openflow.types.IPv4Address;
69 import org.projectfloodlight.openflow.types.IPv6Address;
70 import org.projectfloodlight.openflow.types.IPv6FlowLabel;
71 import org.projectfloodlight.openflow.types.IpDscp;
72 import org.projectfloodlight.openflow.types.IpEcn;
73 import org.projectfloodlight.openflow.types.IpProtocol;
74 import org.projectfloodlight.openflow.types.MacAddress;
75 import org.projectfloodlight.openflow.types.Masked;
76 import org.projectfloodlight.openflow.types.OFBooleanValue;
77 import org.projectfloodlight.openflow.types.OFMetadata;
78 import org.projectfloodlight.openflow.types.OFPort;
79 import org.projectfloodlight.openflow.types.OFVlanVidMatch;
80 import org.projectfloodlight.openflow.types.TransportPort;
81 import org.projectfloodlight.openflow.types.U16;
82 import org.projectfloodlight.openflow.types.U32;
83 import org.projectfloodlight.openflow.types.U64;
84 import org.projectfloodlight.openflow.types.U8;
85 import org.projectfloodlight.openflow.types.VlanPcp;
86 import org.projectfloodlight.openflow.types.VlanVid;
87 import org.slf4j.Logger;
89 import java.util.Optional;
91 import static org.slf4j.LoggerFactory.getLogger;
94 * Builder for OpenFlow flow mods based on FlowRules.
96 public abstract class FlowModBuilder {
98 private final Logger log = getLogger(getClass());
100 private final OFFactory factory;
101 private final FlowRule flowRule;
102 private final TrafficSelector selector;
103 protected final Long xid;
104 protected final Optional<DriverService> driverService;
107 * Creates a new flow mod builder.
109 * @param flowRule the flow rule to transform into a flow mod
110 * @param factory the OpenFlow factory to use to build the flow mod
111 * @param xid the transaction ID
112 * @param driverService the device driver service
113 * @return the new flow mod builder
115 public static FlowModBuilder builder(FlowRule flowRule,
118 Optional<DriverService> driverService) {
119 switch (factory.getVersion()) {
121 return new FlowModBuilderVer10(flowRule, factory, xid, driverService);
123 return new FlowModBuilderVer13(flowRule, factory, xid, driverService);
125 throw new UnsupportedOperationException(
126 "No flow mod builder for protocol version " + factory.getVersion());
131 * Constructs a flow mod builder.
133 * @param flowRule the flow rule to transform into a flow mod
134 * @param factory the OpenFlow factory to use to build the flow mod
135 * @param driverService the device driver service
136 * @param xid the transaction ID
138 protected FlowModBuilder(FlowRule flowRule, OFFactory factory, Optional<Long> xid,
139 Optional<DriverService> driverService) {
140 this.factory = factory;
141 this.flowRule = flowRule;
142 this.selector = flowRule.selector();
143 this.xid = xid.orElse(0L);
144 this.driverService = driverService;
148 * Builds an ADD flow mod.
150 * @return the flow mod
152 public abstract OFFlowAdd buildFlowAdd();
155 * Builds a MODIFY flow mod.
157 * @return the flow mod
159 public abstract OFFlowMod buildFlowMod();
162 * Builds a DELETE flow mod.
164 * @return the flow mod
166 public abstract OFFlowDelete buildFlowDel();
169 * Builds the match for the flow mod.
173 // CHECKSTYLE IGNORE MethodLength FOR NEXT 300 LINES
174 protected Match buildMatch() {
175 Match.Builder mBuilder = factory.buildMatch();
176 Ip6Address ip6Address;
179 EthCriterion ethCriterion;
180 IPCriterion ipCriterion;
181 TcpPortCriterion tcpPortCriterion;
182 UdpPortCriterion udpPortCriterion;
183 SctpPortCriterion sctpPortCriterion;
184 IPv6NDLinkLayerAddressCriterion llAddressCriterion;
185 ArpHaCriterion arpHaCriterion;
186 ArpPaCriterion arpPaCriterion;
188 for (Criterion c : selector.criteria()) {
191 PortCriterion inPort = (PortCriterion) c;
192 mBuilder.setExact(MatchField.IN_PORT,
193 OFPort.of((int) inPort.port().toLong()));
196 PortCriterion inPhyPort = (PortCriterion) c;
197 mBuilder.setExact(MatchField.IN_PORT,
198 OFPort.of((int) inPhyPort.port().toLong()));
201 MetadataCriterion metadata = (MetadataCriterion) c;
202 mBuilder.setExact(MatchField.METADATA,
203 OFMetadata.ofRaw(metadata.metadata()));
206 ethCriterion = (EthCriterion) c;
207 mBuilder.setExact(MatchField.ETH_DST,
208 MacAddress.of(ethCriterion.mac().toLong()));
211 ethCriterion = (EthCriterion) c;
212 mBuilder.setExact(MatchField.ETH_SRC,
213 MacAddress.of(ethCriterion.mac().toLong()));
216 EthTypeCriterion ethType = (EthTypeCriterion) c;
217 mBuilder.setExact(MatchField.ETH_TYPE, EthType.of(ethType.ethType().toShort()));
220 VlanIdCriterion vid = (VlanIdCriterion) c;
222 if (vid.vlanId().equals(VlanId.ANY)) {
223 mBuilder.setMasked(MatchField.VLAN_VID, OFVlanVidMatch.PRESENT,
224 OFVlanVidMatch.PRESENT);
225 } else if (vid.vlanId().equals(VlanId.NONE)) {
226 mBuilder.setExact(MatchField.VLAN_VID, OFVlanVidMatch.NONE);
228 mBuilder.setExact(MatchField.VLAN_VID,
229 OFVlanVidMatch.ofVlanVid(VlanVid.ofVlan(vid.vlanId().toShort())));
233 VlanPcpCriterion vpcp = (VlanPcpCriterion) c;
234 mBuilder.setExact(MatchField.VLAN_PCP, VlanPcp.of(vpcp.priority()));
237 IPDscpCriterion ipDscpCriterion = (IPDscpCriterion) c;
238 mBuilder.setExact(MatchField.IP_DSCP,
239 IpDscp.of(ipDscpCriterion.ipDscp()));
242 IPEcnCriterion ipEcnCriterion = (IPEcnCriterion) c;
243 mBuilder.setExact(MatchField.IP_ECN,
244 IpEcn.of(ipEcnCriterion.ipEcn()));
247 IPProtocolCriterion p = (IPProtocolCriterion) c;
248 mBuilder.setExact(MatchField.IP_PROTO, IpProtocol.of(p.protocol()));
251 ipCriterion = (IPCriterion) c;
252 ip4Prefix = ipCriterion.ip().getIp4Prefix();
253 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
254 Ip4Address maskAddr =
255 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
256 Masked<IPv4Address> maskedIp =
257 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
258 IPv4Address.of(maskAddr.toInt()));
259 mBuilder.setMasked(MatchField.IPV4_SRC, maskedIp);
261 mBuilder.setExact(MatchField.IPV4_SRC,
262 IPv4Address.of(ip4Prefix.address().toInt()));
266 ipCriterion = (IPCriterion) c;
267 ip4Prefix = ipCriterion.ip().getIp4Prefix();
268 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
269 Ip4Address maskAddr =
270 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
271 Masked<IPv4Address> maskedIp =
272 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
273 IPv4Address.of(maskAddr.toInt()));
274 mBuilder.setMasked(MatchField.IPV4_DST, maskedIp);
276 mBuilder.setExact(MatchField.IPV4_DST,
277 IPv4Address.of(ip4Prefix.address().toInt()));
281 tcpPortCriterion = (TcpPortCriterion) c;
282 mBuilder.setExact(MatchField.TCP_SRC,
283 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
286 tcpPortCriterion = (TcpPortCriterion) c;
287 mBuilder.setExact(MatchField.TCP_DST,
288 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
291 udpPortCriterion = (UdpPortCriterion) c;
292 mBuilder.setExact(MatchField.UDP_SRC,
293 TransportPort.of(udpPortCriterion.udpPort().toInt()));
296 udpPortCriterion = (UdpPortCriterion) c;
297 mBuilder.setExact(MatchField.UDP_DST,
298 TransportPort.of(udpPortCriterion.udpPort().toInt()));
301 sctpPortCriterion = (SctpPortCriterion) c;
302 mBuilder.setExact(MatchField.SCTP_SRC,
303 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
306 sctpPortCriterion = (SctpPortCriterion) c;
307 mBuilder.setExact(MatchField.SCTP_DST,
308 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
311 IcmpTypeCriterion icmpType = (IcmpTypeCriterion) c;
312 mBuilder.setExact(MatchField.ICMPV4_TYPE,
313 ICMPv4Type.of(icmpType.icmpType()));
316 IcmpCodeCriterion icmpCode = (IcmpCodeCriterion) c;
317 mBuilder.setExact(MatchField.ICMPV4_CODE,
318 ICMPv4Code.of(icmpCode.icmpCode()));
321 ipCriterion = (IPCriterion) c;
322 ip6Prefix = ipCriterion.ip().getIp6Prefix();
323 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
324 Ip6Address maskAddr =
325 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
326 Masked<IPv6Address> maskedIp =
327 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
328 IPv6Address.of(maskAddr.toString()));
329 mBuilder.setMasked(MatchField.IPV6_SRC, maskedIp);
331 mBuilder.setExact(MatchField.IPV6_SRC,
332 IPv6Address.of(ip6Prefix.address().toString()));
336 ipCriterion = (IPCriterion) c;
337 ip6Prefix = ipCriterion.ip().getIp6Prefix();
338 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
339 Ip6Address maskAddr =
340 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
341 Masked<IPv6Address> maskedIp =
342 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
343 IPv6Address.of(maskAddr.toString()));
344 mBuilder.setMasked(MatchField.IPV6_DST, maskedIp);
346 mBuilder.setExact(MatchField.IPV6_DST,
347 IPv6Address.of(ip6Prefix.address().toString()));
351 IPv6FlowLabelCriterion flowLabelCriterion =
352 (IPv6FlowLabelCriterion) c;
353 mBuilder.setExact(MatchField.IPV6_FLABEL,
354 IPv6FlowLabel.of(flowLabelCriterion.flowLabel()));
357 Icmpv6TypeCriterion icmpv6Type = (Icmpv6TypeCriterion) c;
358 mBuilder.setExact(MatchField.ICMPV6_TYPE,
359 U8.of(icmpv6Type.icmpv6Type()));
362 Icmpv6CodeCriterion icmpv6Code = (Icmpv6CodeCriterion) c;
363 mBuilder.setExact(MatchField.ICMPV6_CODE,
364 U8.of(icmpv6Code.icmpv6Code()));
367 IPv6NDTargetAddressCriterion targetAddressCriterion =
368 (IPv6NDTargetAddressCriterion) c;
369 ip6Address = targetAddressCriterion.targetAddress();
370 mBuilder.setExact(MatchField.IPV6_ND_TARGET,
371 IPv6Address.of(ip6Address.toOctets()));
375 (IPv6NDLinkLayerAddressCriterion) c;
376 mBuilder.setExact(MatchField.IPV6_ND_SLL,
377 MacAddress.of(llAddressCriterion.mac().toLong()));
381 (IPv6NDLinkLayerAddressCriterion) c;
382 mBuilder.setExact(MatchField.IPV6_ND_TLL,
383 MacAddress.of(llAddressCriterion.mac().toLong()));
386 MplsCriterion mp = (MplsCriterion) c;
387 mBuilder.setExact(MatchField.MPLS_LABEL, U32.of(mp.label().toInt()));
390 IPv6ExthdrFlagsCriterion exthdrFlagsCriterion =
391 (IPv6ExthdrFlagsCriterion) c;
392 mBuilder.setExact(MatchField.IPV6_EXTHDR,
393 U16.of(exthdrFlagsCriterion.exthdrFlags()));
397 OchSignalCriterion ochSignalCriterion = (OchSignalCriterion) c;
398 OchSignal signal = ochSignalCriterion.lambda();
399 byte gridType = OpenFlowValueMapper.lookupGridType(signal.gridType());
400 byte channelSpacing = OpenFlowValueMapper.lookupChannelSpacing(signal.channelSpacing());
401 mBuilder.setExact(MatchField.OCH_SIGID,
402 new CircuitSignalID(gridType, channelSpacing,
403 (short) signal.spacingMultiplier(), (short) signal.slotGranularity()));
404 } catch (NoMappingFoundException e) {
405 log.warn(e.getMessage());
409 OchSignalTypeCriterion sc = (OchSignalTypeCriterion) c;
410 byte signalType = OpenFlowValueMapper.lookupOchSignalType(sc.signalType());
411 mBuilder.setExact(MatchField.OCH_SIGTYPE, U8.of(signalType));
414 TunnelIdCriterion tunnelId = (TunnelIdCriterion) c;
415 mBuilder.setExact(MatchField.TUNNEL_ID,
416 U64.of(tunnelId.tunnelId()));
419 MplsBosCriterion mplsBos = (MplsBosCriterion) c;
420 mBuilder.setExact(MatchField.MPLS_BOS,
421 mplsBos.mplsBos() ? OFBooleanValue.TRUE
422 : OFBooleanValue.FALSE);
425 ArpOpCriterion arpOp = (ArpOpCriterion) c;
426 mBuilder.setExact(MatchField.ARP_OP,
427 ArpOpcode.of(arpOp.arpOp()));
430 arpHaCriterion = (ArpHaCriterion) c;
431 mBuilder.setExact(MatchField.ARP_SHA,
432 MacAddress.of(arpHaCriterion.mac().toLong()));
435 arpPaCriterion = (ArpPaCriterion) c;
436 mBuilder.setExact(MatchField.ARP_SPA,
437 IPv4Address.of(arpPaCriterion.ip().toInt()));
440 arpHaCriterion = (ArpHaCriterion) c;
441 mBuilder.setExact(MatchField.ARP_THA,
442 MacAddress.of(arpHaCriterion.mac().toLong()));
445 arpPaCriterion = (ArpPaCriterion) c;
446 mBuilder.setExact(MatchField.ARP_TPA,
447 IPv4Address.of(arpPaCriterion.ip().toInt()));
452 log.warn("Match type {} not yet implemented.", c.type());
455 return mBuilder.build();
459 * Returns the flow rule for this builder.
461 * @return the flow rule
463 protected FlowRule flowRule() {
468 * Returns the factory used for building OpenFlow constructs.
470 * @return the factory
472 protected OFFactory factory() {