blob: a281927f1f1b2253754050fc8a769a4f98fab8b5 [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2014-present Open Networking Foundation
Thomas Vachuska781d18b2014-10-27 10:31:25 -07003 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07004 * 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
Thomas Vachuska781d18b2014-10-27 10:31:25 -07007 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07008 * http://www.apache.org/licenses/LICENSE-2.0
9 *
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.
Thomas Vachuska781d18b2014-10-27 10:31:25 -070015 */
Brian O'Connorabafb502014-12-02 22:26:20 -080016package org.onosproject.provider.of.flow.impl;
alshabibeec3a062014-09-17 18:01:26 -070017
Jonathan Hart6cd2f352015-01-13 17:44:45 -080018import org.onlab.packet.Ip4Address;
19import org.onlab.packet.Ip4Prefix;
Charles M.C. Chan52fae7d2015-01-17 00:35:53 +080020import org.onlab.packet.Ip6Address;
21import org.onlab.packet.Ip6Prefix;
Jonathan Hart6cd2f352015-01-13 17:44:45 -080022import org.onlab.packet.VlanId;
Jonathan Hart26a8d952015-12-02 15:16:35 -080023import org.onosproject.net.DeviceId;
Sho SHIMIZUc15ce512015-05-26 16:54:08 -070024import org.onosproject.net.OchSignal;
Jonathan Hart42feabd2016-02-03 17:56:21 -080025import org.onosproject.net.OduSignalId;
Jonathan Hart26a8d952015-12-02 15:16:35 -080026import org.onosproject.net.driver.DefaultDriverData;
27import org.onosproject.net.driver.DefaultDriverHandler;
28import org.onosproject.net.driver.Driver;
Jonathan Hart3c259162015-10-21 21:31:19 -070029import org.onosproject.net.driver.DriverService;
Brian O'Connorabafb502014-12-02 22:26:20 -080030import org.onosproject.net.flow.FlowRule;
31import org.onosproject.net.flow.TrafficSelector;
BitOhenry76430852015-11-20 19:04:49 +080032import org.onosproject.net.flow.criteria.ArpHaCriterion;
BitOhenryb40129a2015-11-30 12:41:18 +080033import org.onosproject.net.flow.criteria.ArpOpCriterion;
BitOhenryc1e5fcc2015-11-23 20:47:53 +080034import org.onosproject.net.flow.criteria.ArpPaCriterion;
Saurav Dasffc5bbc2015-08-18 23:30:19 -070035import org.onosproject.net.flow.criteria.Criterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070036import org.onosproject.net.flow.criteria.EthCriterion;
37import org.onosproject.net.flow.criteria.EthTypeCriterion;
Jonathan Hart26a8d952015-12-02 15:16:35 -080038import org.onosproject.net.flow.criteria.ExtensionCriterion;
39import org.onosproject.net.flow.criteria.ExtensionSelector;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070040import org.onosproject.net.flow.criteria.IPCriterion;
41import org.onosproject.net.flow.criteria.IPDscpCriterion;
42import org.onosproject.net.flow.criteria.IPEcnCriterion;
43import org.onosproject.net.flow.criteria.IPProtocolCriterion;
44import org.onosproject.net.flow.criteria.IPv6ExthdrFlagsCriterion;
45import org.onosproject.net.flow.criteria.IPv6FlowLabelCriterion;
46import org.onosproject.net.flow.criteria.IPv6NDLinkLayerAddressCriterion;
47import org.onosproject.net.flow.criteria.IPv6NDTargetAddressCriterion;
48import org.onosproject.net.flow.criteria.IcmpCodeCriterion;
49import org.onosproject.net.flow.criteria.IcmpTypeCriterion;
50import org.onosproject.net.flow.criteria.Icmpv6CodeCriterion;
51import org.onosproject.net.flow.criteria.Icmpv6TypeCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070052import org.onosproject.net.flow.criteria.MetadataCriterion;
Saurav Dasffc5bbc2015-08-18 23:30:19 -070053import org.onosproject.net.flow.criteria.MplsBosCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070054import org.onosproject.net.flow.criteria.MplsCriterion;
Sho SHIMIZUc15ce512015-05-26 16:54:08 -070055import org.onosproject.net.flow.criteria.OchSignalCriterion;
Marc De Leenheerd24420f2015-05-27 09:40:59 -070056import org.onosproject.net.flow.criteria.OchSignalTypeCriterion;
Yafit Hadar73514612015-11-04 10:14:21 +020057import org.onosproject.net.flow.criteria.OduSignalIdCriterion;
58import org.onosproject.net.flow.criteria.OduSignalTypeCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070059import org.onosproject.net.flow.criteria.PortCriterion;
60import org.onosproject.net.flow.criteria.SctpPortCriterion;
61import org.onosproject.net.flow.criteria.TcpPortCriterion;
Hyunsun Moona08c5d02015-07-14 17:53:00 -070062import org.onosproject.net.flow.criteria.TunnelIdCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070063import org.onosproject.net.flow.criteria.UdpPortCriterion;
64import org.onosproject.net.flow.criteria.VlanIdCriterion;
65import org.onosproject.net.flow.criteria.VlanPcpCriterion;
Jonathan Hart26a8d952015-12-02 15:16:35 -080066import org.onosproject.openflow.controller.ExtensionSelectorInterpreter;
Thomas Vachuska95caba32016-04-04 10:42:05 -070067import org.onosproject.provider.of.flow.util.NoMappingFoundException;
68import org.onosproject.provider.of.flow.util.OpenFlowValueMapper;
alshabibeec3a062014-09-17 18:01:26 -070069import org.projectfloodlight.openflow.protocol.OFFactory;
70import org.projectfloodlight.openflow.protocol.OFFlowMod;
alshabibeec3a062014-09-17 18:01:26 -070071import org.projectfloodlight.openflow.protocol.match.Match;
72import org.projectfloodlight.openflow.protocol.match.MatchField;
Jonathan Hart26a8d952015-12-02 15:16:35 -080073import org.projectfloodlight.openflow.protocol.oxm.OFOxm;
BitOhenryb40129a2015-11-30 12:41:18 +080074import org.projectfloodlight.openflow.types.ArpOpcode;
Marc De Leenheer49087752014-10-23 13:54:09 -070075import org.projectfloodlight.openflow.types.CircuitSignalID;
alshabibeec3a062014-09-17 18:01:26 -070076import org.projectfloodlight.openflow.types.EthType;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -080077import org.projectfloodlight.openflow.types.ICMPv4Code;
78import org.projectfloodlight.openflow.types.ICMPv4Type;
alshabibeec3a062014-09-17 18:01:26 -070079import org.projectfloodlight.openflow.types.IPv4Address;
Charles M.C. Chan52fae7d2015-01-17 00:35:53 +080080import org.projectfloodlight.openflow.types.IPv6Address;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -080081import org.projectfloodlight.openflow.types.IPv6FlowLabel;
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -080082import org.projectfloodlight.openflow.types.IpDscp;
83import org.projectfloodlight.openflow.types.IpEcn;
alshabibeec3a062014-09-17 18:01:26 -070084import org.projectfloodlight.openflow.types.IpProtocol;
85import org.projectfloodlight.openflow.types.MacAddress;
86import org.projectfloodlight.openflow.types.Masked;
Saurav Dasffc5bbc2015-08-18 23:30:19 -070087import org.projectfloodlight.openflow.types.OFBooleanValue;
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -080088import org.projectfloodlight.openflow.types.OFMetadata;
alshabibeec3a062014-09-17 18:01:26 -070089import org.projectfloodlight.openflow.types.OFPort;
90import org.projectfloodlight.openflow.types.OFVlanVidMatch;
Jonathan Hart42feabd2016-02-03 17:56:21 -080091import org.projectfloodlight.openflow.types.OduSignalID;
Jonathan Hart34bc6142014-10-17 11:00:43 -070092import org.projectfloodlight.openflow.types.TransportPort;
Pavlin Radoslavov5e4f7542015-02-06 18:18:21 -080093import org.projectfloodlight.openflow.types.U16;
Praseed Balakrishnan8c67d172014-11-10 10:15:41 -080094import org.projectfloodlight.openflow.types.U32;
Hyunsun Moona08c5d02015-07-14 17:53:00 -070095import org.projectfloodlight.openflow.types.U64;
Praseed Balakrishnan2dd5abd2014-11-03 14:56:28 -080096import org.projectfloodlight.openflow.types.U8;
alshabibeec3a062014-09-17 18:01:26 -070097import org.projectfloodlight.openflow.types.VlanPcp;
98import org.projectfloodlight.openflow.types.VlanVid;
99import org.slf4j.Logger;
100
Jonathan Hart3c259162015-10-21 21:31:19 -0700101import java.util.Optional;
102
103import static org.slf4j.LoggerFactory.getLogger;
104
Jonathan Hart86e59352014-10-22 10:42:16 -0700105/**
106 * Builder for OpenFlow flow mods based on FlowRules.
107 */
108public abstract class FlowModBuilder {
alshabibeec3a062014-09-17 18:01:26 -0700109
110 private final Logger log = getLogger(getClass());
111
112 private final OFFactory factory;
Jonathan Hart86e59352014-10-22 10:42:16 -0700113 private final FlowRule flowRule;
alshabibeec3a062014-09-17 18:01:26 -0700114 private final TrafficSelector selector;
Brian O'Connor427a1762014-11-19 18:40:32 -0800115 protected final Long xid;
Jonathan Hart3c259162015-10-21 21:31:19 -0700116 protected final Optional<DriverService> driverService;
Jonathan Hart26a8d952015-12-02 15:16:35 -0800117 protected final DeviceId deviceId;
alshabibeec3a062014-09-17 18:01:26 -0700118
Jonathan Hart86e59352014-10-22 10:42:16 -0700119 /**
120 * Creates a new flow mod builder.
121 *
122 * @param flowRule the flow rule to transform into a flow mod
123 * @param factory the OpenFlow factory to use to build the flow mod
Pavlin Radoslavov119fd5c2014-11-25 19:08:19 -0800124 * @param xid the transaction ID
Charles Chan30ba4002015-11-05 14:45:16 -0800125 * @param driverService the device driver service
Jonathan Hart86e59352014-10-22 10:42:16 -0700126 * @return the new flow mod builder
127 */
Brian O'Connor427a1762014-11-19 18:40:32 -0800128 public static FlowModBuilder builder(FlowRule flowRule,
alshabib9af70072015-02-09 14:34:16 -0800129 OFFactory factory,
Jonathan Hart3c259162015-10-21 21:31:19 -0700130 Optional<Long> xid,
131 Optional<DriverService> driverService) {
Jonathan Hart86e59352014-10-22 10:42:16 -0700132 switch (factory.getVersion()) {
133 case OF_10:
Jonathan Hart3c259162015-10-21 21:31:19 -0700134 return new FlowModBuilderVer10(flowRule, factory, xid, driverService);
Jonathan Hart86e59352014-10-22 10:42:16 -0700135 case OF_13:
Yuta HIGUCHI2341e602017-03-08 20:10:08 -0800136 case OF_14:
Jonathan Hart3c259162015-10-21 21:31:19 -0700137 return new FlowModBuilderVer13(flowRule, factory, xid, driverService);
Cem Türker3baff672017-10-12 15:09:01 +0300138 case OF_15:
139 return new FlowModBuilderVer15(flowRule, factory, xid, driverService);
Jonathan Hart86e59352014-10-22 10:42:16 -0700140 default:
141 throw new UnsupportedOperationException(
142 "No flow mod builder for protocol version " + factory.getVersion());
143 }
144 }
alshabibeec3a062014-09-17 18:01:26 -0700145
Jonathan Hart86e59352014-10-22 10:42:16 -0700146 /**
147 * Constructs a flow mod builder.
148 *
149 * @param flowRule the flow rule to transform into a flow mod
150 * @param factory the OpenFlow factory to use to build the flow mod
Charles Chan30ba4002015-11-05 14:45:16 -0800151 * @param driverService the device driver service
Pavlin Radoslavov119fd5c2014-11-25 19:08:19 -0800152 * @param xid the transaction ID
Jonathan Hart86e59352014-10-22 10:42:16 -0700153 */
Jonathan Hart3c259162015-10-21 21:31:19 -0700154 protected FlowModBuilder(FlowRule flowRule, OFFactory factory, Optional<Long> xid,
155 Optional<DriverService> driverService) {
alshabibeec3a062014-09-17 18:01:26 -0700156 this.factory = factory;
Jonathan Hart86e59352014-10-22 10:42:16 -0700157 this.flowRule = flowRule;
alshabibeec3a062014-09-17 18:01:26 -0700158 this.selector = flowRule.selector();
Sho SHIMIZUc14b2a22015-05-05 18:15:13 -0700159 this.xid = xid.orElse(0L);
Jonathan Hart3c259162015-10-21 21:31:19 -0700160 this.driverService = driverService;
Jonathan Hart26a8d952015-12-02 15:16:35 -0800161 this.deviceId = flowRule.deviceId();
alshabibeec3a062014-09-17 18:01:26 -0700162 }
163
Jonathan Hart86e59352014-10-22 10:42:16 -0700164 /**
165 * Builds an ADD flow mod.
166 *
167 * @return the flow mod
168 */
Jonathan Hart42feabd2016-02-03 17:56:21 -0800169 public abstract OFFlowMod buildFlowAdd();
alshabibeec3a062014-09-17 18:01:26 -0700170
Jonathan Hart86e59352014-10-22 10:42:16 -0700171 /**
172 * Builds a MODIFY flow mod.
173 *
174 * @return the flow mod
175 */
176 public abstract OFFlowMod buildFlowMod();
alshabibeec3a062014-09-17 18:01:26 -0700177
Jonathan Hart86e59352014-10-22 10:42:16 -0700178 /**
179 * Builds a DELETE flow mod.
180 *
181 * @return the flow mod
182 */
Jonathan Hart42feabd2016-02-03 17:56:21 -0800183 public abstract OFFlowMod buildFlowDel();
alshabibeec3a062014-09-17 18:01:26 -0700184
Jonathan Hart86e59352014-10-22 10:42:16 -0700185 /**
186 * Builds the match for the flow mod.
187 *
188 * @return the match
189 */
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800190 // CHECKSTYLE IGNORE MethodLength FOR NEXT 300 LINES
Jonathan Hart86e59352014-10-22 10:42:16 -0700191 protected Match buildMatch() {
alshabibeec3a062014-09-17 18:01:26 -0700192 Match.Builder mBuilder = factory.buildMatch();
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800193 Ip6Address ip6Address;
Pavlin Radoslavov23e398d2014-11-05 15:17:57 -0800194 Ip4Prefix ip4Prefix;
Charles M.C. Chan52fae7d2015-01-17 00:35:53 +0800195 Ip6Prefix ip6Prefix;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800196 EthCriterion ethCriterion;
197 IPCriterion ipCriterion;
198 TcpPortCriterion tcpPortCriterion;
199 UdpPortCriterion udpPortCriterion;
200 SctpPortCriterion sctpPortCriterion;
201 IPv6NDLinkLayerAddressCriterion llAddressCriterion;
BitOhenry76430852015-11-20 19:04:49 +0800202 ArpHaCriterion arpHaCriterion;
BitOhenryc1e5fcc2015-11-23 20:47:53 +0800203 ArpPaCriterion arpPaCriterion;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800204
alshabibeec3a062014-09-17 18:01:26 -0700205 for (Criterion c : selector.criteria()) {
206 switch (c.type()) {
207 case IN_PORT:
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -0800208 PortCriterion inPort = (PortCriterion) c;
209 mBuilder.setExact(MatchField.IN_PORT,
210 OFPort.of((int) inPort.port().toLong()));
211 break;
212 case IN_PHY_PORT:
213 PortCriterion inPhyPort = (PortCriterion) c;
214 mBuilder.setExact(MatchField.IN_PORT,
215 OFPort.of((int) inPhyPort.port().toLong()));
216 break;
217 case METADATA:
218 MetadataCriterion metadata = (MetadataCriterion) c;
219 mBuilder.setExact(MatchField.METADATA,
220 OFMetadata.ofRaw(metadata.metadata()));
alshabibeec3a062014-09-17 18:01:26 -0700221 break;
alshabibeec3a062014-09-17 18:01:26 -0700222 case ETH_DST:
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800223 ethCriterion = (EthCriterion) c;
224 mBuilder.setExact(MatchField.ETH_DST,
225 MacAddress.of(ethCriterion.mac().toLong()));
226 break;
Saurav Das9d6c86b2016-02-19 09:01:07 -0800227 case ETH_DST_MASKED:
228 ethCriterion = (EthCriterion) c;
229 mBuilder.setMasked(MatchField.ETH_DST,
230 MacAddress.of(ethCriterion.mac().toLong()),
231 MacAddress.of(ethCriterion.mask().toLong()));
232 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800233 case ETH_SRC:
234 ethCriterion = (EthCriterion) c;
235 mBuilder.setExact(MatchField.ETH_SRC,
236 MacAddress.of(ethCriterion.mac().toLong()));
alshabibeec3a062014-09-17 18:01:26 -0700237 break;
Saurav Das9d6c86b2016-02-19 09:01:07 -0800238 case ETH_SRC_MASKED:
239 ethCriterion = (EthCriterion) c;
240 mBuilder.setMasked(MatchField.ETH_SRC,
241 MacAddress.of(ethCriterion.mac().toLong()),
242 MacAddress.of(ethCriterion.mask().toLong()));
243 break;
alshabibeec3a062014-09-17 18:01:26 -0700244 case ETH_TYPE:
245 EthTypeCriterion ethType = (EthTypeCriterion) c;
alshabibcaf1ca22015-06-25 15:18:16 -0700246 mBuilder.setExact(MatchField.ETH_TYPE, EthType.of(ethType.ethType().toShort()));
alshabibeec3a062014-09-17 18:01:26 -0700247 break;
alshabibeec3a062014-09-17 18:01:26 -0700248 case VLAN_VID:
249 VlanIdCriterion vid = (VlanIdCriterion) c;
Jonathan Hart6cd2f352015-01-13 17:44:45 -0800250
251 if (vid.vlanId().equals(VlanId.ANY)) {
252 mBuilder.setMasked(MatchField.VLAN_VID, OFVlanVidMatch.PRESENT,
253 OFVlanVidMatch.PRESENT);
Jonathan Hart1468fee2015-07-16 18:50:34 -0700254 } else if (vid.vlanId().equals(VlanId.NONE)) {
255 mBuilder.setExact(MatchField.VLAN_VID, OFVlanVidMatch.NONE);
Jonathan Hart6cd2f352015-01-13 17:44:45 -0800256 } else {
257 mBuilder.setExact(MatchField.VLAN_VID,
Charles Chan14967c22015-12-07 11:11:50 -0800258 OFVlanVidMatch.ofVlanVid(VlanVid.ofVlan(vid.vlanId().toShort())));
Jonathan Hart6cd2f352015-01-13 17:44:45 -0800259 }
alshabibeec3a062014-09-17 18:01:26 -0700260 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800261 case VLAN_PCP:
262 VlanPcpCriterion vpcp = (VlanPcpCriterion) c;
263 mBuilder.setExact(MatchField.VLAN_PCP, VlanPcp.of(vpcp.priority()));
264 break;
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -0800265 case IP_DSCP:
266 IPDscpCriterion ipDscpCriterion = (IPDscpCriterion) c;
267 mBuilder.setExact(MatchField.IP_DSCP,
268 IpDscp.of(ipDscpCriterion.ipDscp()));
269 break;
270 case IP_ECN:
271 IPEcnCriterion ipEcnCriterion = (IPEcnCriterion) c;
272 mBuilder.setExact(MatchField.IP_ECN,
273 IpEcn.of(ipEcnCriterion.ipEcn()));
274 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800275 case IP_PROTO:
276 IPProtocolCriterion p = (IPProtocolCriterion) c;
277 mBuilder.setExact(MatchField.IP_PROTO, IpProtocol.of(p.protocol()));
278 break;
279 case IPV4_SRC:
280 ipCriterion = (IPCriterion) c;
281 ip4Prefix = ipCriterion.ip().getIp4Prefix();
282 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
283 Ip4Address maskAddr =
284 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
285 Masked<IPv4Address> maskedIp =
286 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
287 IPv4Address.of(maskAddr.toInt()));
288 mBuilder.setMasked(MatchField.IPV4_SRC, maskedIp);
289 } else {
290 mBuilder.setExact(MatchField.IPV4_SRC,
291 IPv4Address.of(ip4Prefix.address().toInt()));
292 }
293 break;
294 case IPV4_DST:
295 ipCriterion = (IPCriterion) c;
296 ip4Prefix = ipCriterion.ip().getIp4Prefix();
297 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
298 Ip4Address maskAddr =
299 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
300 Masked<IPv4Address> maskedIp =
301 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
302 IPv4Address.of(maskAddr.toInt()));
303 mBuilder.setMasked(MatchField.IPV4_DST, maskedIp);
304 } else {
305 mBuilder.setExact(MatchField.IPV4_DST,
306 IPv4Address.of(ip4Prefix.address().toInt()));
307 }
Jonathan Hart34bc6142014-10-17 11:00:43 -0700308 break;
309 case TCP_SRC:
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800310 tcpPortCriterion = (TcpPortCriterion) c;
311 mBuilder.setExact(MatchField.TCP_SRC,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700312 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800313 break;
Andreas Gilbert75b882f72017-02-03 09:58:07 +0100314 case TCP_SRC_MASKED:
315 tcpPortCriterion = (TcpPortCriterion) c;
316 mBuilder.setMasked(MatchField.TCP_SRC,
317 TransportPort.of(tcpPortCriterion.tcpPort().toInt()),
318 TransportPort.of(tcpPortCriterion.mask().toInt()));
319 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800320 case TCP_DST:
321 tcpPortCriterion = (TcpPortCriterion) c;
322 mBuilder.setExact(MatchField.TCP_DST,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700323 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800324 break;
Andreas Gilbert75b882f72017-02-03 09:58:07 +0100325 case TCP_DST_MASKED:
326 tcpPortCriterion = (TcpPortCriterion) c;
327 mBuilder.setMasked(MatchField.TCP_DST,
328 TransportPort.of(tcpPortCriterion.tcpPort().toInt()),
329 TransportPort.of(tcpPortCriterion.mask().toInt()));
330 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800331 case UDP_SRC:
332 udpPortCriterion = (UdpPortCriterion) c;
333 mBuilder.setExact(MatchField.UDP_SRC,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700334 TransportPort.of(udpPortCriterion.udpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800335 break;
Andreas Gilbert75b882f72017-02-03 09:58:07 +0100336 case UDP_SRC_MASKED:
337 udpPortCriterion = (UdpPortCriterion) c;
338 mBuilder.setMasked(MatchField.UDP_SRC,
339 TransportPort.of(udpPortCriterion.udpPort().toInt()),
340 TransportPort.of(udpPortCriterion.mask().toInt()));
341 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800342 case UDP_DST:
343 udpPortCriterion = (UdpPortCriterion) c;
344 mBuilder.setExact(MatchField.UDP_DST,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700345 TransportPort.of(udpPortCriterion.udpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800346 break;
Andreas Gilbert75b882f72017-02-03 09:58:07 +0100347 case UDP_DST_MASKED:
348 udpPortCriterion = (UdpPortCriterion) c;
349 mBuilder.setMasked(MatchField.UDP_DST,
350 TransportPort.of(udpPortCriterion.udpPort().toInt()),
351 TransportPort.of(udpPortCriterion.mask().toInt()));
352 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800353 case SCTP_SRC:
354 sctpPortCriterion = (SctpPortCriterion) c;
355 mBuilder.setExact(MatchField.SCTP_SRC,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700356 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800357 break;
Andreas Gilbert75b882f72017-02-03 09:58:07 +0100358 case SCTP_SRC_MASKED:
359 sctpPortCriterion = (SctpPortCriterion) c;
360 mBuilder.setMasked(MatchField.SCTP_SRC,
361 TransportPort.of(sctpPortCriterion.sctpPort().toInt()),
362 TransportPort.of(sctpPortCriterion.mask().toInt()));
363 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800364 case SCTP_DST:
365 sctpPortCriterion = (SctpPortCriterion) c;
366 mBuilder.setExact(MatchField.SCTP_DST,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700367 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800368 break;
Andreas Gilbert75b882f72017-02-03 09:58:07 +0100369 case SCTP_DST_MASKED:
370 sctpPortCriterion = (SctpPortCriterion) c;
371 mBuilder.setMasked(MatchField.SCTP_DST,
372 TransportPort.of(sctpPortCriterion.sctpPort().toInt()),
373 TransportPort.of(sctpPortCriterion.mask().toInt()));
374 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800375 case ICMPV4_TYPE:
376 IcmpTypeCriterion icmpType = (IcmpTypeCriterion) c;
377 mBuilder.setExact(MatchField.ICMPV4_TYPE,
378 ICMPv4Type.of(icmpType.icmpType()));
379 break;
380 case ICMPV4_CODE:
381 IcmpCodeCriterion icmpCode = (IcmpCodeCriterion) c;
382 mBuilder.setExact(MatchField.ICMPV4_CODE,
383 ICMPv4Code.of(icmpCode.icmpCode()));
384 break;
385 case IPV6_SRC:
386 ipCriterion = (IPCriterion) c;
387 ip6Prefix = ipCriterion.ip().getIp6Prefix();
388 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
389 Ip6Address maskAddr =
390 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
391 Masked<IPv6Address> maskedIp =
392 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
393 IPv6Address.of(maskAddr.toString()));
394 mBuilder.setMasked(MatchField.IPV6_SRC, maskedIp);
395 } else {
396 mBuilder.setExact(MatchField.IPV6_SRC,
397 IPv6Address.of(ip6Prefix.address().toString()));
398 }
399 break;
400 case IPV6_DST:
401 ipCriterion = (IPCriterion) c;
402 ip6Prefix = ipCriterion.ip().getIp6Prefix();
403 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
404 Ip6Address maskAddr =
405 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
406 Masked<IPv6Address> maskedIp =
407 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
408 IPv6Address.of(maskAddr.toString()));
409 mBuilder.setMasked(MatchField.IPV6_DST, maskedIp);
410 } else {
411 mBuilder.setExact(MatchField.IPV6_DST,
412 IPv6Address.of(ip6Prefix.address().toString()));
413 }
414 break;
415 case IPV6_FLABEL:
416 IPv6FlowLabelCriterion flowLabelCriterion =
417 (IPv6FlowLabelCriterion) c;
418 mBuilder.setExact(MatchField.IPV6_FLABEL,
419 IPv6FlowLabel.of(flowLabelCriterion.flowLabel()));
420 break;
421 case ICMPV6_TYPE:
422 Icmpv6TypeCriterion icmpv6Type = (Icmpv6TypeCriterion) c;
423 mBuilder.setExact(MatchField.ICMPV6_TYPE,
Pavlin Radoslavovf3b69332015-02-06 15:47:05 -0800424 U8.of(icmpv6Type.icmpv6Type()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800425 break;
426 case ICMPV6_CODE:
427 Icmpv6CodeCriterion icmpv6Code = (Icmpv6CodeCriterion) c;
428 mBuilder.setExact(MatchField.ICMPV6_CODE,
Pavlin Radoslavovf3b69332015-02-06 15:47:05 -0800429 U8.of(icmpv6Code.icmpv6Code()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800430 break;
431 case IPV6_ND_TARGET:
432 IPv6NDTargetAddressCriterion targetAddressCriterion =
433 (IPv6NDTargetAddressCriterion) c;
434 ip6Address = targetAddressCriterion.targetAddress();
435 mBuilder.setExact(MatchField.IPV6_ND_TARGET,
436 IPv6Address.of(ip6Address.toOctets()));
437 break;
438 case IPV6_ND_SLL:
439 llAddressCriterion =
440 (IPv6NDLinkLayerAddressCriterion) c;
441 mBuilder.setExact(MatchField.IPV6_ND_SLL,
442 MacAddress.of(llAddressCriterion.mac().toLong()));
443 break;
444 case IPV6_ND_TLL:
445 llAddressCriterion =
446 (IPv6NDLinkLayerAddressCriterion) c;
447 mBuilder.setExact(MatchField.IPV6_ND_TLL,
448 MacAddress.of(llAddressCriterion.mac().toLong()));
Jonathan Hart34bc6142014-10-17 11:00:43 -0700449 break;
Praseed Balakrishnan8c67d172014-11-10 10:15:41 -0800450 case MPLS_LABEL:
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -0700451 MplsCriterion mp = (MplsCriterion) c;
Michele Santuari4b6019e2014-12-19 11:31:45 +0100452 mBuilder.setExact(MatchField.MPLS_LABEL, U32.of(mp.label().toInt()));
Praseed Balakrishnan8c67d172014-11-10 10:15:41 -0800453 break;
Pavlin Radoslavov5e4f7542015-02-06 18:18:21 -0800454 case IPV6_EXTHDR:
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -0700455 IPv6ExthdrFlagsCriterion exthdrFlagsCriterion =
456 (IPv6ExthdrFlagsCriterion) c;
Pavlin Radoslavov5e4f7542015-02-06 18:18:21 -0800457 mBuilder.setExact(MatchField.IPV6_EXTHDR,
458 U16.of(exthdrFlagsCriterion.exthdrFlags()));
459 break;
Marc De Leenheer49087752014-10-23 13:54:09 -0700460 case OCH_SIGID:
Sho SHIMIZUc15ce512015-05-26 16:54:08 -0700461 try {
462 OchSignalCriterion ochSignalCriterion = (OchSignalCriterion) c;
463 OchSignal signal = ochSignalCriterion.lambda();
Sho SHIMIZUc17042d2015-05-28 12:07:23 -0700464 byte gridType = OpenFlowValueMapper.lookupGridType(signal.gridType());
465 byte channelSpacing = OpenFlowValueMapper.lookupChannelSpacing(signal.channelSpacing());
Yafit Hadar73514612015-11-04 10:14:21 +0200466 mBuilder.setExact(MatchField.EXP_OCH_SIG_ID,
Sho SHIMIZUc15ce512015-05-26 16:54:08 -0700467 new CircuitSignalID(gridType, channelSpacing,
468 (short) signal.spacingMultiplier(), (short) signal.slotGranularity()));
Sho SHIMIZUc17042d2015-05-28 12:07:23 -0700469 } catch (NoMappingFoundException e) {
Sho SHIMIZUc15ce512015-05-26 16:54:08 -0700470 log.warn(e.getMessage());
471 }
Marc De Leenheer49087752014-10-23 13:54:09 -0700472 break;
Praseed Balakrishnan2dd5abd2014-11-03 14:56:28 -0800473 case OCH_SIGTYPE:
Yafit Hadar73514612015-11-04 10:14:21 +0200474 try {
475 OchSignalTypeCriterion sc = (OchSignalTypeCriterion) c;
476 byte signalType = OpenFlowValueMapper.lookupOchSignalType(sc.signalType());
477 mBuilder.setExact(MatchField.EXP_OCH_SIGTYPE, U8.of(signalType));
478 } catch (NoMappingFoundException e) {
479 log.warn(e.getMessage());
480 }
481 break;
482 case ODU_SIGID:
483 OduSignalIdCriterion oduSignalIdCriterion = (OduSignalIdCriterion) c;
484 OduSignalId oduSignalId = oduSignalIdCriterion.oduSignalId();
485 mBuilder.setExact(MatchField.EXP_ODU_SIG_ID,
486 new OduSignalID((short) oduSignalId.tributaryPortNumber(),
487 (short) oduSignalId.tributarySlotLength(),
488 oduSignalId.tributarySlotBitmap()));
489 break;
490 case ODU_SIGTYPE:
491 try {
492 OduSignalTypeCriterion oduSignalTypeCriterion = (OduSignalTypeCriterion) c;
493 byte oduSigType = OpenFlowValueMapper.lookupOduSignalType(oduSignalTypeCriterion.signalType());
494 mBuilder.setExact(MatchField.EXP_ODU_SIGTYPE, U8.of(oduSigType));
495 } catch (NoMappingFoundException e) {
496 log.warn(e.getMessage());
497 }
Praseed Balakrishnan2dd5abd2014-11-03 14:56:28 -0800498 break;
Hyunsun Moona08c5d02015-07-14 17:53:00 -0700499 case TUNNEL_ID:
500 TunnelIdCriterion tunnelId = (TunnelIdCriterion) c;
501 mBuilder.setExact(MatchField.TUNNEL_ID,
502 U64.of(tunnelId.tunnelId()));
503 break;
Saurav Dasffc5bbc2015-08-18 23:30:19 -0700504 case MPLS_BOS:
505 MplsBosCriterion mplsBos = (MplsBosCriterion) c;
506 mBuilder.setExact(MatchField.MPLS_BOS,
507 mplsBos.mplsBos() ? OFBooleanValue.TRUE
508 : OFBooleanValue.FALSE);
509 break;
BitOhenryb40129a2015-11-30 12:41:18 +0800510 case ARP_OP:
511 ArpOpCriterion arpOp = (ArpOpCriterion) c;
512 mBuilder.setExact(MatchField.ARP_OP,
513 ArpOpcode.of(arpOp.arpOp()));
514 break;
BitOhenrya331b182015-11-23 08:39:37 +0800515 case ARP_SHA:
516 arpHaCriterion = (ArpHaCriterion) c;
517 mBuilder.setExact(MatchField.ARP_SHA,
518 MacAddress.of(arpHaCriterion.mac().toLong()));
519 break;
BitOhenry296b4542015-11-24 08:41:58 +0800520 case ARP_SPA:
521 arpPaCriterion = (ArpPaCriterion) c;
522 mBuilder.setExact(MatchField.ARP_SPA,
523 IPv4Address.of(arpPaCriterion.ip().toInt()));
524 break;
BitOhenry76430852015-11-20 19:04:49 +0800525 case ARP_THA:
526 arpHaCriterion = (ArpHaCriterion) c;
527 mBuilder.setExact(MatchField.ARP_THA,
528 MacAddress.of(arpHaCriterion.mac().toLong()));
529 break;
BitOhenryc1e5fcc2015-11-23 20:47:53 +0800530 case ARP_TPA:
531 arpPaCriterion = (ArpPaCriterion) c;
532 mBuilder.setExact(MatchField.ARP_TPA,
533 IPv4Address.of(arpPaCriterion.ip().toInt()));
534 break;
Jonathan Hart26a8d952015-12-02 15:16:35 -0800535 case EXTENSION:
536 ExtensionCriterion extensionCriterion = (ExtensionCriterion) c;
537 OFOxm oxm = buildExtensionOxm(extensionCriterion.extensionSelector());
538 if (oxm == null) {
539 log.warn("Unable to build extension selector");
540 break;
541 }
542
543 if (oxm.isMasked()) {
544 mBuilder.setMasked(oxm.getMatchField(), oxm.getValue(), oxm.getMask());
545 } else {
546 mBuilder.setExact(oxm.getMatchField(), oxm.getValue());
547 }
548
549 break;
alshabibeec3a062014-09-17 18:01:26 -0700550 case MPLS_TC:
551 case PBB_ISID:
Jian Lif1ecf992015-12-02 17:48:54 -0800552 // TODO: need to implement PBB-ISID case when OpenFlowJ is ready
alshabibeec3a062014-09-17 18:01:26 -0700553 default:
554 log.warn("Match type {} not yet implemented.", c.type());
555 }
556 }
557 return mBuilder.build();
558 }
559
Jonathan Hart86e59352014-10-22 10:42:16 -0700560 /**
561 * Returns the flow rule for this builder.
562 *
563 * @return the flow rule
564 */
565 protected FlowRule flowRule() {
566 return flowRule;
567 }
alshabib219ebaa2014-09-22 15:41:24 -0700568
Jonathan Hart86e59352014-10-22 10:42:16 -0700569 /**
570 * Returns the factory used for building OpenFlow constructs.
571 *
572 * @return the factory
573 */
574 protected OFFactory factory() {
575 return factory;
576 }
alshabib219ebaa2014-09-22 15:41:24 -0700577
Jonathan Hart26a8d952015-12-02 15:16:35 -0800578 private OFOxm buildExtensionOxm(ExtensionSelector extension) {
579 if (!driverService.isPresent()) {
580 log.error("No driver service present");
581 return null;
582 }
583 Driver driver = driverService.get().getDriver(deviceId);
584 if (driver.hasBehaviour(ExtensionSelectorInterpreter.class)) {
585 DefaultDriverHandler handler =
586 new DefaultDriverHandler(new DefaultDriverData(driver, deviceId));
587 ExtensionSelectorInterpreter interpreter = handler.behaviour(ExtensionSelectorInterpreter.class);
588
589 return interpreter.mapSelector(factory(), extension);
590 }
591
592 return null;
593 }
594
alshabibeec3a062014-09-17 18:01:26 -0700595}