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.Criterion;
28 import org.onosproject.net.flow.criteria.EthCriterion;
29 import org.onosproject.net.flow.criteria.EthTypeCriterion;
30 import org.onosproject.net.flow.criteria.IPCriterion;
31 import org.onosproject.net.flow.criteria.IPDscpCriterion;
32 import org.onosproject.net.flow.criteria.IPEcnCriterion;
33 import org.onosproject.net.flow.criteria.IPProtocolCriterion;
34 import org.onosproject.net.flow.criteria.IPv6ExthdrFlagsCriterion;
35 import org.onosproject.net.flow.criteria.IPv6FlowLabelCriterion;
36 import org.onosproject.net.flow.criteria.IPv6NDLinkLayerAddressCriterion;
37 import org.onosproject.net.flow.criteria.IPv6NDTargetAddressCriterion;
38 import org.onosproject.net.flow.criteria.IcmpCodeCriterion;
39 import org.onosproject.net.flow.criteria.IcmpTypeCriterion;
40 import org.onosproject.net.flow.criteria.Icmpv6CodeCriterion;
41 import org.onosproject.net.flow.criteria.Icmpv6TypeCriterion;
42 import org.onosproject.net.flow.criteria.MetadataCriterion;
43 import org.onosproject.net.flow.criteria.MplsBosCriterion;
44 import org.onosproject.net.flow.criteria.MplsCriterion;
45 import org.onosproject.net.flow.criteria.OchSignalCriterion;
46 import org.onosproject.net.flow.criteria.OchSignalTypeCriterion;
47 import org.onosproject.net.flow.criteria.PortCriterion;
48 import org.onosproject.net.flow.criteria.SctpPortCriterion;
49 import org.onosproject.net.flow.criteria.TcpPortCriterion;
50 import org.onosproject.net.flow.criteria.TunnelIdCriterion;
51 import org.onosproject.net.flow.criteria.UdpPortCriterion;
52 import org.onosproject.net.flow.criteria.VlanIdCriterion;
53 import org.onosproject.net.flow.criteria.VlanPcpCriterion;
54 import org.projectfloodlight.openflow.protocol.OFFactory;
55 import org.projectfloodlight.openflow.protocol.OFFlowAdd;
56 import org.projectfloodlight.openflow.protocol.OFFlowDelete;
57 import org.projectfloodlight.openflow.protocol.OFFlowMod;
58 import org.projectfloodlight.openflow.protocol.match.Match;
59 import org.projectfloodlight.openflow.protocol.match.MatchField;
60 import org.projectfloodlight.openflow.types.CircuitSignalID;
61 import org.projectfloodlight.openflow.types.EthType;
62 import org.projectfloodlight.openflow.types.ICMPv4Code;
63 import org.projectfloodlight.openflow.types.ICMPv4Type;
64 import org.projectfloodlight.openflow.types.IPv4Address;
65 import org.projectfloodlight.openflow.types.IPv6Address;
66 import org.projectfloodlight.openflow.types.IPv6FlowLabel;
67 import org.projectfloodlight.openflow.types.IpDscp;
68 import org.projectfloodlight.openflow.types.IpEcn;
69 import org.projectfloodlight.openflow.types.IpProtocol;
70 import org.projectfloodlight.openflow.types.MacAddress;
71 import org.projectfloodlight.openflow.types.Masked;
72 import org.projectfloodlight.openflow.types.OFBooleanValue;
73 import org.projectfloodlight.openflow.types.OFMetadata;
74 import org.projectfloodlight.openflow.types.OFPort;
75 import org.projectfloodlight.openflow.types.OFVlanVidMatch;
76 import org.projectfloodlight.openflow.types.TransportPort;
77 import org.projectfloodlight.openflow.types.U16;
78 import org.projectfloodlight.openflow.types.U32;
79 import org.projectfloodlight.openflow.types.U64;
80 import org.projectfloodlight.openflow.types.U8;
81 import org.projectfloodlight.openflow.types.VlanPcp;
82 import org.projectfloodlight.openflow.types.VlanVid;
83 import org.slf4j.Logger;
85 import java.util.Optional;
87 import static org.slf4j.LoggerFactory.getLogger;
90 * Builder for OpenFlow flow mods based on FlowRules.
92 public abstract class FlowModBuilder {
94 private final Logger log = getLogger(getClass());
96 private final OFFactory factory;
97 private final FlowRule flowRule;
98 private final TrafficSelector selector;
99 protected final Long xid;
100 protected final Optional<DriverService> driverService;
103 * Creates a new flow mod builder.
105 * @param flowRule the flow rule to transform into a flow mod
106 * @param factory the OpenFlow factory to use to build the flow mod
107 * @param xid the transaction ID
108 * @return the new flow mod builder
110 public static FlowModBuilder builder(FlowRule flowRule,
113 Optional<DriverService> driverService) {
114 switch (factory.getVersion()) {
116 return new FlowModBuilderVer10(flowRule, factory, xid, driverService);
118 return new FlowModBuilderVer13(flowRule, factory, xid, driverService);
120 throw new UnsupportedOperationException(
121 "No flow mod builder for protocol version " + factory.getVersion());
126 * Constructs a flow mod builder.
128 * @param flowRule the flow rule to transform into a flow mod
129 * @param factory the OpenFlow factory to use to build the flow mod
130 * @param xid the transaction ID
132 protected FlowModBuilder(FlowRule flowRule, OFFactory factory, Optional<Long> xid,
133 Optional<DriverService> driverService) {
134 this.factory = factory;
135 this.flowRule = flowRule;
136 this.selector = flowRule.selector();
137 this.xid = xid.orElse(0L);
138 this.driverService = driverService;
142 * Builds an ADD flow mod.
144 * @return the flow mod
146 public abstract OFFlowAdd buildFlowAdd();
149 * Builds a MODIFY flow mod.
151 * @return the flow mod
153 public abstract OFFlowMod buildFlowMod();
156 * Builds a DELETE flow mod.
158 * @return the flow mod
160 public abstract OFFlowDelete buildFlowDel();
163 * Builds the match for the flow mod.
167 // CHECKSTYLE IGNORE MethodLength FOR NEXT 300 LINES
168 protected Match buildMatch() {
169 Match.Builder mBuilder = factory.buildMatch();
170 Ip6Address ip6Address;
173 EthCriterion ethCriterion;
174 IPCriterion ipCriterion;
175 TcpPortCriterion tcpPortCriterion;
176 UdpPortCriterion udpPortCriterion;
177 SctpPortCriterion sctpPortCriterion;
178 IPv6NDLinkLayerAddressCriterion llAddressCriterion;
180 for (Criterion c : selector.criteria()) {
183 PortCriterion inPort = (PortCriterion) c;
184 mBuilder.setExact(MatchField.IN_PORT,
185 OFPort.of((int) inPort.port().toLong()));
188 PortCriterion inPhyPort = (PortCriterion) c;
189 mBuilder.setExact(MatchField.IN_PORT,
190 OFPort.of((int) inPhyPort.port().toLong()));
193 MetadataCriterion metadata = (MetadataCriterion) c;
194 mBuilder.setExact(MatchField.METADATA,
195 OFMetadata.ofRaw(metadata.metadata()));
198 ethCriterion = (EthCriterion) c;
199 mBuilder.setExact(MatchField.ETH_DST,
200 MacAddress.of(ethCriterion.mac().toLong()));
203 ethCriterion = (EthCriterion) c;
204 mBuilder.setExact(MatchField.ETH_SRC,
205 MacAddress.of(ethCriterion.mac().toLong()));
208 EthTypeCriterion ethType = (EthTypeCriterion) c;
209 mBuilder.setExact(MatchField.ETH_TYPE, EthType.of(ethType.ethType().toShort()));
212 VlanIdCriterion vid = (VlanIdCriterion) c;
214 if (vid.vlanId().equals(VlanId.ANY)) {
215 mBuilder.setMasked(MatchField.VLAN_VID, OFVlanVidMatch.PRESENT,
216 OFVlanVidMatch.PRESENT);
217 } else if (vid.vlanId().equals(VlanId.NONE)) {
218 mBuilder.setExact(MatchField.VLAN_VID, OFVlanVidMatch.NONE);
220 mBuilder.setExact(MatchField.VLAN_VID,
221 OFVlanVidMatch.ofVlanVid(VlanVid.ofVlan(vid.vlanId().toShort())));
225 VlanPcpCriterion vpcp = (VlanPcpCriterion) c;
226 mBuilder.setExact(MatchField.VLAN_PCP, VlanPcp.of(vpcp.priority()));
229 IPDscpCriterion ipDscpCriterion = (IPDscpCriterion) c;
230 mBuilder.setExact(MatchField.IP_DSCP,
231 IpDscp.of(ipDscpCriterion.ipDscp()));
234 IPEcnCriterion ipEcnCriterion = (IPEcnCriterion) c;
235 mBuilder.setExact(MatchField.IP_ECN,
236 IpEcn.of(ipEcnCriterion.ipEcn()));
239 IPProtocolCriterion p = (IPProtocolCriterion) c;
240 mBuilder.setExact(MatchField.IP_PROTO, IpProtocol.of(p.protocol()));
243 ipCriterion = (IPCriterion) c;
244 ip4Prefix = ipCriterion.ip().getIp4Prefix();
245 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
246 Ip4Address maskAddr =
247 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
248 Masked<IPv4Address> maskedIp =
249 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
250 IPv4Address.of(maskAddr.toInt()));
251 mBuilder.setMasked(MatchField.IPV4_SRC, maskedIp);
253 mBuilder.setExact(MatchField.IPV4_SRC,
254 IPv4Address.of(ip4Prefix.address().toInt()));
258 ipCriterion = (IPCriterion) c;
259 ip4Prefix = ipCriterion.ip().getIp4Prefix();
260 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
261 Ip4Address maskAddr =
262 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
263 Masked<IPv4Address> maskedIp =
264 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
265 IPv4Address.of(maskAddr.toInt()));
266 mBuilder.setMasked(MatchField.IPV4_DST, maskedIp);
268 mBuilder.setExact(MatchField.IPV4_DST,
269 IPv4Address.of(ip4Prefix.address().toInt()));
273 tcpPortCriterion = (TcpPortCriterion) c;
274 mBuilder.setExact(MatchField.TCP_SRC,
275 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
278 tcpPortCriterion = (TcpPortCriterion) c;
279 mBuilder.setExact(MatchField.TCP_DST,
280 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
283 udpPortCriterion = (UdpPortCriterion) c;
284 mBuilder.setExact(MatchField.UDP_SRC,
285 TransportPort.of(udpPortCriterion.udpPort().toInt()));
288 udpPortCriterion = (UdpPortCriterion) c;
289 mBuilder.setExact(MatchField.UDP_DST,
290 TransportPort.of(udpPortCriterion.udpPort().toInt()));
293 sctpPortCriterion = (SctpPortCriterion) c;
294 mBuilder.setExact(MatchField.SCTP_SRC,
295 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
298 sctpPortCriterion = (SctpPortCriterion) c;
299 mBuilder.setExact(MatchField.SCTP_DST,
300 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
303 IcmpTypeCriterion icmpType = (IcmpTypeCriterion) c;
304 mBuilder.setExact(MatchField.ICMPV4_TYPE,
305 ICMPv4Type.of(icmpType.icmpType()));
308 IcmpCodeCriterion icmpCode = (IcmpCodeCriterion) c;
309 mBuilder.setExact(MatchField.ICMPV4_CODE,
310 ICMPv4Code.of(icmpCode.icmpCode()));
313 ipCriterion = (IPCriterion) c;
314 ip6Prefix = ipCriterion.ip().getIp6Prefix();
315 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
316 Ip6Address maskAddr =
317 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
318 Masked<IPv6Address> maskedIp =
319 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
320 IPv6Address.of(maskAddr.toString()));
321 mBuilder.setMasked(MatchField.IPV6_SRC, maskedIp);
323 mBuilder.setExact(MatchField.IPV6_SRC,
324 IPv6Address.of(ip6Prefix.address().toString()));
328 ipCriterion = (IPCriterion) c;
329 ip6Prefix = ipCriterion.ip().getIp6Prefix();
330 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
331 Ip6Address maskAddr =
332 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
333 Masked<IPv6Address> maskedIp =
334 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
335 IPv6Address.of(maskAddr.toString()));
336 mBuilder.setMasked(MatchField.IPV6_DST, maskedIp);
338 mBuilder.setExact(MatchField.IPV6_DST,
339 IPv6Address.of(ip6Prefix.address().toString()));
343 IPv6FlowLabelCriterion flowLabelCriterion =
344 (IPv6FlowLabelCriterion) c;
345 mBuilder.setExact(MatchField.IPV6_FLABEL,
346 IPv6FlowLabel.of(flowLabelCriterion.flowLabel()));
349 Icmpv6TypeCriterion icmpv6Type = (Icmpv6TypeCriterion) c;
350 mBuilder.setExact(MatchField.ICMPV6_TYPE,
351 U8.of(icmpv6Type.icmpv6Type()));
354 Icmpv6CodeCriterion icmpv6Code = (Icmpv6CodeCriterion) c;
355 mBuilder.setExact(MatchField.ICMPV6_CODE,
356 U8.of(icmpv6Code.icmpv6Code()));
359 IPv6NDTargetAddressCriterion targetAddressCriterion =
360 (IPv6NDTargetAddressCriterion) c;
361 ip6Address = targetAddressCriterion.targetAddress();
362 mBuilder.setExact(MatchField.IPV6_ND_TARGET,
363 IPv6Address.of(ip6Address.toOctets()));
367 (IPv6NDLinkLayerAddressCriterion) c;
368 mBuilder.setExact(MatchField.IPV6_ND_SLL,
369 MacAddress.of(llAddressCriterion.mac().toLong()));
373 (IPv6NDLinkLayerAddressCriterion) c;
374 mBuilder.setExact(MatchField.IPV6_ND_TLL,
375 MacAddress.of(llAddressCriterion.mac().toLong()));
378 MplsCriterion mp = (MplsCriterion) c;
379 mBuilder.setExact(MatchField.MPLS_LABEL, U32.of(mp.label().toInt()));
382 IPv6ExthdrFlagsCriterion exthdrFlagsCriterion =
383 (IPv6ExthdrFlagsCriterion) c;
384 mBuilder.setExact(MatchField.IPV6_EXTHDR,
385 U16.of(exthdrFlagsCriterion.exthdrFlags()));
389 OchSignalCriterion ochSignalCriterion = (OchSignalCriterion) c;
390 OchSignal signal = ochSignalCriterion.lambda();
391 byte gridType = OpenFlowValueMapper.lookupGridType(signal.gridType());
392 byte channelSpacing = OpenFlowValueMapper.lookupChannelSpacing(signal.channelSpacing());
393 mBuilder.setExact(MatchField.OCH_SIGID,
394 new CircuitSignalID(gridType, channelSpacing,
395 (short) signal.spacingMultiplier(), (short) signal.slotGranularity()));
396 } catch (NoMappingFoundException e) {
397 log.warn(e.getMessage());
401 OchSignalTypeCriterion sc = (OchSignalTypeCriterion) c;
402 byte signalType = OpenFlowValueMapper.lookupOchSignalType(sc.signalType());
403 mBuilder.setExact(MatchField.OCH_SIGTYPE, U8.of(signalType));
406 TunnelIdCriterion tunnelId = (TunnelIdCriterion) c;
407 mBuilder.setExact(MatchField.TUNNEL_ID,
408 U64.of(tunnelId.tunnelId()));
411 MplsBosCriterion mplsBos = (MplsBosCriterion) c;
412 mBuilder.setExact(MatchField.MPLS_BOS,
413 mplsBos.mplsBos() ? OFBooleanValue.TRUE
414 : OFBooleanValue.FALSE);
424 log.warn("Match type {} not yet implemented.", c.type());
427 return mBuilder.build();
431 * Returns the flow rule for this builder.
433 * @return the flow rule
435 protected FlowRule flowRule() {
440 * Returns the factory used for building OpenFlow constructs.
442 * @return the factory
444 protected OFFactory factory() {