blob: e6f2f74a6c2813576764d03c98821a2d440372ea [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Ray Milkey34c95902015-04-15 09:47:53 -07002 * Copyright 2014-2015 Open Networking Laboratory
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 Hart26a8d952015-12-02 15:16:35 -080025import org.onosproject.net.driver.DefaultDriverData;
26import org.onosproject.net.driver.DefaultDriverHandler;
27import org.onosproject.net.driver.Driver;
Jonathan Hart3c259162015-10-21 21:31:19 -070028import org.onosproject.net.driver.DriverService;
Brian O'Connorabafb502014-12-02 22:26:20 -080029import org.onosproject.net.flow.FlowRule;
30import org.onosproject.net.flow.TrafficSelector;
BitOhenry76430852015-11-20 19:04:49 +080031import org.onosproject.net.flow.criteria.ArpHaCriterion;
BitOhenryb40129a2015-11-30 12:41:18 +080032import org.onosproject.net.flow.criteria.ArpOpCriterion;
BitOhenryc1e5fcc2015-11-23 20:47:53 +080033import org.onosproject.net.flow.criteria.ArpPaCriterion;
Saurav Dasffc5bbc2015-08-18 23:30:19 -070034import org.onosproject.net.flow.criteria.Criterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070035import org.onosproject.net.flow.criteria.EthCriterion;
36import org.onosproject.net.flow.criteria.EthTypeCriterion;
Jonathan Hart26a8d952015-12-02 15:16:35 -080037import org.onosproject.net.flow.criteria.ExtensionCriterion;
38import org.onosproject.net.flow.criteria.ExtensionSelector;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070039import org.onosproject.net.flow.criteria.IPCriterion;
40import org.onosproject.net.flow.criteria.IPDscpCriterion;
41import org.onosproject.net.flow.criteria.IPEcnCriterion;
42import org.onosproject.net.flow.criteria.IPProtocolCriterion;
43import org.onosproject.net.flow.criteria.IPv6ExthdrFlagsCriterion;
44import org.onosproject.net.flow.criteria.IPv6FlowLabelCriterion;
45import org.onosproject.net.flow.criteria.IPv6NDLinkLayerAddressCriterion;
46import org.onosproject.net.flow.criteria.IPv6NDTargetAddressCriterion;
47import org.onosproject.net.flow.criteria.IcmpCodeCriterion;
48import org.onosproject.net.flow.criteria.IcmpTypeCriterion;
49import org.onosproject.net.flow.criteria.Icmpv6CodeCriterion;
50import org.onosproject.net.flow.criteria.Icmpv6TypeCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070051import org.onosproject.net.flow.criteria.MetadataCriterion;
Saurav Dasffc5bbc2015-08-18 23:30:19 -070052import org.onosproject.net.flow.criteria.MplsBosCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070053import org.onosproject.net.flow.criteria.MplsCriterion;
Sho SHIMIZUc15ce512015-05-26 16:54:08 -070054import org.onosproject.net.flow.criteria.OchSignalCriterion;
Marc De Leenheerd24420f2015-05-27 09:40:59 -070055import org.onosproject.net.flow.criteria.OchSignalTypeCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070056import org.onosproject.net.flow.criteria.PortCriterion;
57import org.onosproject.net.flow.criteria.SctpPortCriterion;
58import org.onosproject.net.flow.criteria.TcpPortCriterion;
Hyunsun Moona08c5d02015-07-14 17:53:00 -070059import org.onosproject.net.flow.criteria.TunnelIdCriterion;
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -070060import org.onosproject.net.flow.criteria.UdpPortCriterion;
61import org.onosproject.net.flow.criteria.VlanIdCriterion;
62import org.onosproject.net.flow.criteria.VlanPcpCriterion;
Jonathan Hart26a8d952015-12-02 15:16:35 -080063import org.onosproject.openflow.controller.ExtensionSelectorInterpreter;
alshabibeec3a062014-09-17 18:01:26 -070064import org.projectfloodlight.openflow.protocol.OFFactory;
alshabib193525b2014-10-08 18:58:03 -070065import org.projectfloodlight.openflow.protocol.OFFlowAdd;
66import org.projectfloodlight.openflow.protocol.OFFlowDelete;
alshabibeec3a062014-09-17 18:01:26 -070067import org.projectfloodlight.openflow.protocol.OFFlowMod;
alshabibeec3a062014-09-17 18:01:26 -070068import org.projectfloodlight.openflow.protocol.match.Match;
69import org.projectfloodlight.openflow.protocol.match.MatchField;
Jonathan Hart26a8d952015-12-02 15:16:35 -080070import org.projectfloodlight.openflow.protocol.oxm.OFOxm;
BitOhenryb40129a2015-11-30 12:41:18 +080071import org.projectfloodlight.openflow.types.ArpOpcode;
Marc De Leenheer49087752014-10-23 13:54:09 -070072import org.projectfloodlight.openflow.types.CircuitSignalID;
alshabibeec3a062014-09-17 18:01:26 -070073import org.projectfloodlight.openflow.types.EthType;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -080074import org.projectfloodlight.openflow.types.ICMPv4Code;
75import org.projectfloodlight.openflow.types.ICMPv4Type;
alshabibeec3a062014-09-17 18:01:26 -070076import org.projectfloodlight.openflow.types.IPv4Address;
Charles M.C. Chan52fae7d2015-01-17 00:35:53 +080077import org.projectfloodlight.openflow.types.IPv6Address;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -080078import org.projectfloodlight.openflow.types.IPv6FlowLabel;
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -080079import org.projectfloodlight.openflow.types.IpDscp;
80import org.projectfloodlight.openflow.types.IpEcn;
alshabibeec3a062014-09-17 18:01:26 -070081import org.projectfloodlight.openflow.types.IpProtocol;
82import org.projectfloodlight.openflow.types.MacAddress;
83import org.projectfloodlight.openflow.types.Masked;
Saurav Dasffc5bbc2015-08-18 23:30:19 -070084import org.projectfloodlight.openflow.types.OFBooleanValue;
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -080085import org.projectfloodlight.openflow.types.OFMetadata;
alshabibeec3a062014-09-17 18:01:26 -070086import org.projectfloodlight.openflow.types.OFPort;
87import org.projectfloodlight.openflow.types.OFVlanVidMatch;
Jonathan Hart34bc6142014-10-17 11:00:43 -070088import org.projectfloodlight.openflow.types.TransportPort;
Pavlin Radoslavov5e4f7542015-02-06 18:18:21 -080089import org.projectfloodlight.openflow.types.U16;
Praseed Balakrishnan8c67d172014-11-10 10:15:41 -080090import org.projectfloodlight.openflow.types.U32;
Hyunsun Moona08c5d02015-07-14 17:53:00 -070091import org.projectfloodlight.openflow.types.U64;
Praseed Balakrishnan2dd5abd2014-11-03 14:56:28 -080092import org.projectfloodlight.openflow.types.U8;
alshabibeec3a062014-09-17 18:01:26 -070093import org.projectfloodlight.openflow.types.VlanPcp;
94import org.projectfloodlight.openflow.types.VlanVid;
95import org.slf4j.Logger;
96
Jonathan Hart3c259162015-10-21 21:31:19 -070097import java.util.Optional;
98
99import static org.slf4j.LoggerFactory.getLogger;
100
Jonathan Hart86e59352014-10-22 10:42:16 -0700101/**
102 * Builder for OpenFlow flow mods based on FlowRules.
103 */
104public abstract class FlowModBuilder {
alshabibeec3a062014-09-17 18:01:26 -0700105
106 private final Logger log = getLogger(getClass());
107
108 private final OFFactory factory;
Jonathan Hart86e59352014-10-22 10:42:16 -0700109 private final FlowRule flowRule;
alshabibeec3a062014-09-17 18:01:26 -0700110 private final TrafficSelector selector;
Brian O'Connor427a1762014-11-19 18:40:32 -0800111 protected final Long xid;
Jonathan Hart3c259162015-10-21 21:31:19 -0700112 protected final Optional<DriverService> driverService;
Jonathan Hart26a8d952015-12-02 15:16:35 -0800113 protected final DeviceId deviceId;
alshabibeec3a062014-09-17 18:01:26 -0700114
Jonathan Hart86e59352014-10-22 10:42:16 -0700115 /**
116 * Creates a new flow mod builder.
117 *
118 * @param flowRule the flow rule to transform into a flow mod
119 * @param factory the OpenFlow factory to use to build the flow mod
Pavlin Radoslavov119fd5c2014-11-25 19:08:19 -0800120 * @param xid the transaction ID
Charles Chan30ba4002015-11-05 14:45:16 -0800121 * @param driverService the device driver service
Jonathan Hart86e59352014-10-22 10:42:16 -0700122 * @return the new flow mod builder
123 */
Brian O'Connor427a1762014-11-19 18:40:32 -0800124 public static FlowModBuilder builder(FlowRule flowRule,
alshabib9af70072015-02-09 14:34:16 -0800125 OFFactory factory,
Jonathan Hart3c259162015-10-21 21:31:19 -0700126 Optional<Long> xid,
127 Optional<DriverService> driverService) {
Jonathan Hart86e59352014-10-22 10:42:16 -0700128 switch (factory.getVersion()) {
129 case OF_10:
Jonathan Hart3c259162015-10-21 21:31:19 -0700130 return new FlowModBuilderVer10(flowRule, factory, xid, driverService);
Jonathan Hart86e59352014-10-22 10:42:16 -0700131 case OF_13:
Jonathan Hart3c259162015-10-21 21:31:19 -0700132 return new FlowModBuilderVer13(flowRule, factory, xid, driverService);
Jonathan Hart86e59352014-10-22 10:42:16 -0700133 default:
134 throw new UnsupportedOperationException(
135 "No flow mod builder for protocol version " + factory.getVersion());
136 }
137 }
alshabibeec3a062014-09-17 18:01:26 -0700138
Jonathan Hart86e59352014-10-22 10:42:16 -0700139 /**
140 * Constructs a flow mod builder.
141 *
142 * @param flowRule the flow rule to transform into a flow mod
143 * @param factory the OpenFlow factory to use to build the flow mod
Charles Chan30ba4002015-11-05 14:45:16 -0800144 * @param driverService the device driver service
Pavlin Radoslavov119fd5c2014-11-25 19:08:19 -0800145 * @param xid the transaction ID
Jonathan Hart86e59352014-10-22 10:42:16 -0700146 */
Jonathan Hart3c259162015-10-21 21:31:19 -0700147 protected FlowModBuilder(FlowRule flowRule, OFFactory factory, Optional<Long> xid,
148 Optional<DriverService> driverService) {
alshabibeec3a062014-09-17 18:01:26 -0700149 this.factory = factory;
Jonathan Hart86e59352014-10-22 10:42:16 -0700150 this.flowRule = flowRule;
alshabibeec3a062014-09-17 18:01:26 -0700151 this.selector = flowRule.selector();
Sho SHIMIZUc14b2a22015-05-05 18:15:13 -0700152 this.xid = xid.orElse(0L);
Jonathan Hart3c259162015-10-21 21:31:19 -0700153 this.driverService = driverService;
Jonathan Hart26a8d952015-12-02 15:16:35 -0800154 this.deviceId = flowRule.deviceId();
alshabibeec3a062014-09-17 18:01:26 -0700155 }
156
Jonathan Hart86e59352014-10-22 10:42:16 -0700157 /**
158 * Builds an ADD flow mod.
159 *
160 * @return the flow mod
161 */
162 public abstract OFFlowAdd buildFlowAdd();
alshabibeec3a062014-09-17 18:01:26 -0700163
Jonathan Hart86e59352014-10-22 10:42:16 -0700164 /**
165 * Builds a MODIFY flow mod.
166 *
167 * @return the flow mod
168 */
169 public abstract OFFlowMod buildFlowMod();
alshabibeec3a062014-09-17 18:01:26 -0700170
Jonathan Hart86e59352014-10-22 10:42:16 -0700171 /**
172 * Builds a DELETE flow mod.
173 *
174 * @return the flow mod
175 */
176 public abstract OFFlowDelete buildFlowDel();
alshabibeec3a062014-09-17 18:01:26 -0700177
Jonathan Hart86e59352014-10-22 10:42:16 -0700178 /**
179 * Builds the match for the flow mod.
180 *
181 * @return the match
182 */
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800183 // CHECKSTYLE IGNORE MethodLength FOR NEXT 300 LINES
Jonathan Hart86e59352014-10-22 10:42:16 -0700184 protected Match buildMatch() {
alshabibeec3a062014-09-17 18:01:26 -0700185 Match.Builder mBuilder = factory.buildMatch();
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800186 Ip6Address ip6Address;
Pavlin Radoslavov23e398d2014-11-05 15:17:57 -0800187 Ip4Prefix ip4Prefix;
Charles M.C. Chan52fae7d2015-01-17 00:35:53 +0800188 Ip6Prefix ip6Prefix;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800189 EthCriterion ethCriterion;
190 IPCriterion ipCriterion;
191 TcpPortCriterion tcpPortCriterion;
192 UdpPortCriterion udpPortCriterion;
193 SctpPortCriterion sctpPortCriterion;
194 IPv6NDLinkLayerAddressCriterion llAddressCriterion;
BitOhenry76430852015-11-20 19:04:49 +0800195 ArpHaCriterion arpHaCriterion;
BitOhenryc1e5fcc2015-11-23 20:47:53 +0800196 ArpPaCriterion arpPaCriterion;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800197
alshabibeec3a062014-09-17 18:01:26 -0700198 for (Criterion c : selector.criteria()) {
199 switch (c.type()) {
200 case IN_PORT:
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -0800201 PortCriterion inPort = (PortCriterion) c;
202 mBuilder.setExact(MatchField.IN_PORT,
203 OFPort.of((int) inPort.port().toLong()));
204 break;
205 case IN_PHY_PORT:
206 PortCriterion inPhyPort = (PortCriterion) c;
207 mBuilder.setExact(MatchField.IN_PORT,
208 OFPort.of((int) inPhyPort.port().toLong()));
209 break;
210 case METADATA:
211 MetadataCriterion metadata = (MetadataCriterion) c;
212 mBuilder.setExact(MatchField.METADATA,
213 OFMetadata.ofRaw(metadata.metadata()));
alshabibeec3a062014-09-17 18:01:26 -0700214 break;
alshabibeec3a062014-09-17 18:01:26 -0700215 case ETH_DST:
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800216 ethCriterion = (EthCriterion) c;
217 mBuilder.setExact(MatchField.ETH_DST,
218 MacAddress.of(ethCriterion.mac().toLong()));
219 break;
220 case ETH_SRC:
221 ethCriterion = (EthCriterion) c;
222 mBuilder.setExact(MatchField.ETH_SRC,
223 MacAddress.of(ethCriterion.mac().toLong()));
alshabibeec3a062014-09-17 18:01:26 -0700224 break;
225 case ETH_TYPE:
226 EthTypeCriterion ethType = (EthTypeCriterion) c;
alshabibcaf1ca22015-06-25 15:18:16 -0700227 mBuilder.setExact(MatchField.ETH_TYPE, EthType.of(ethType.ethType().toShort()));
alshabibeec3a062014-09-17 18:01:26 -0700228 break;
alshabibeec3a062014-09-17 18:01:26 -0700229 case VLAN_VID:
230 VlanIdCriterion vid = (VlanIdCriterion) c;
Jonathan Hart6cd2f352015-01-13 17:44:45 -0800231
232 if (vid.vlanId().equals(VlanId.ANY)) {
233 mBuilder.setMasked(MatchField.VLAN_VID, OFVlanVidMatch.PRESENT,
234 OFVlanVidMatch.PRESENT);
Jonathan Hart1468fee2015-07-16 18:50:34 -0700235 } else if (vid.vlanId().equals(VlanId.NONE)) {
236 mBuilder.setExact(MatchField.VLAN_VID, OFVlanVidMatch.NONE);
Jonathan Hart6cd2f352015-01-13 17:44:45 -0800237 } else {
238 mBuilder.setExact(MatchField.VLAN_VID,
alshabib346b5b32015-03-06 00:42:16 -0800239 OFVlanVidMatch.ofVlanVid(VlanVid.ofVlan(vid.vlanId().toShort())));
Jonathan Hart6cd2f352015-01-13 17:44:45 -0800240 }
alshabibeec3a062014-09-17 18:01:26 -0700241 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800242 case VLAN_PCP:
243 VlanPcpCriterion vpcp = (VlanPcpCriterion) c;
244 mBuilder.setExact(MatchField.VLAN_PCP, VlanPcp.of(vpcp.priority()));
245 break;
Pavlin Radoslavovd0fd8412015-02-04 13:57:00 -0800246 case IP_DSCP:
247 IPDscpCriterion ipDscpCriterion = (IPDscpCriterion) c;
248 mBuilder.setExact(MatchField.IP_DSCP,
249 IpDscp.of(ipDscpCriterion.ipDscp()));
250 break;
251 case IP_ECN:
252 IPEcnCriterion ipEcnCriterion = (IPEcnCriterion) c;
253 mBuilder.setExact(MatchField.IP_ECN,
254 IpEcn.of(ipEcnCriterion.ipEcn()));
255 break;
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800256 case IP_PROTO:
257 IPProtocolCriterion p = (IPProtocolCriterion) c;
258 mBuilder.setExact(MatchField.IP_PROTO, IpProtocol.of(p.protocol()));
259 break;
260 case IPV4_SRC:
261 ipCriterion = (IPCriterion) c;
262 ip4Prefix = ipCriterion.ip().getIp4Prefix();
263 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
264 Ip4Address maskAddr =
265 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
266 Masked<IPv4Address> maskedIp =
267 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
268 IPv4Address.of(maskAddr.toInt()));
269 mBuilder.setMasked(MatchField.IPV4_SRC, maskedIp);
270 } else {
271 mBuilder.setExact(MatchField.IPV4_SRC,
272 IPv4Address.of(ip4Prefix.address().toInt()));
273 }
274 break;
275 case IPV4_DST:
276 ipCriterion = (IPCriterion) c;
277 ip4Prefix = ipCriterion.ip().getIp4Prefix();
278 if (ip4Prefix.prefixLength() != Ip4Prefix.MAX_MASK_LENGTH) {
279 Ip4Address maskAddr =
280 Ip4Address.makeMaskPrefix(ip4Prefix.prefixLength());
281 Masked<IPv4Address> maskedIp =
282 Masked.of(IPv4Address.of(ip4Prefix.address().toInt()),
283 IPv4Address.of(maskAddr.toInt()));
284 mBuilder.setMasked(MatchField.IPV4_DST, maskedIp);
285 } else {
286 mBuilder.setExact(MatchField.IPV4_DST,
287 IPv4Address.of(ip4Prefix.address().toInt()));
288 }
Jonathan Hart34bc6142014-10-17 11:00:43 -0700289 break;
290 case TCP_SRC:
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800291 tcpPortCriterion = (TcpPortCriterion) c;
292 mBuilder.setExact(MatchField.TCP_SRC,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700293 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800294 break;
295 case TCP_DST:
296 tcpPortCriterion = (TcpPortCriterion) c;
297 mBuilder.setExact(MatchField.TCP_DST,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700298 TransportPort.of(tcpPortCriterion.tcpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800299 break;
300 case UDP_SRC:
301 udpPortCriterion = (UdpPortCriterion) c;
302 mBuilder.setExact(MatchField.UDP_SRC,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700303 TransportPort.of(udpPortCriterion.udpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800304 break;
305 case UDP_DST:
306 udpPortCriterion = (UdpPortCriterion) c;
307 mBuilder.setExact(MatchField.UDP_DST,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700308 TransportPort.of(udpPortCriterion.udpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800309 break;
310 case SCTP_SRC:
311 sctpPortCriterion = (SctpPortCriterion) c;
312 mBuilder.setExact(MatchField.SCTP_SRC,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700313 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800314 break;
315 case SCTP_DST:
316 sctpPortCriterion = (SctpPortCriterion) c;
317 mBuilder.setExact(MatchField.SCTP_DST,
Hyunsun Mooncf732fb2015-08-22 21:04:23 -0700318 TransportPort.of(sctpPortCriterion.sctpPort().toInt()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800319 break;
320 case ICMPV4_TYPE:
321 IcmpTypeCriterion icmpType = (IcmpTypeCriterion) c;
322 mBuilder.setExact(MatchField.ICMPV4_TYPE,
323 ICMPv4Type.of(icmpType.icmpType()));
324 break;
325 case ICMPV4_CODE:
326 IcmpCodeCriterion icmpCode = (IcmpCodeCriterion) c;
327 mBuilder.setExact(MatchField.ICMPV4_CODE,
328 ICMPv4Code.of(icmpCode.icmpCode()));
329 break;
330 case IPV6_SRC:
331 ipCriterion = (IPCriterion) c;
332 ip6Prefix = ipCriterion.ip().getIp6Prefix();
333 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
334 Ip6Address maskAddr =
335 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
336 Masked<IPv6Address> maskedIp =
337 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
338 IPv6Address.of(maskAddr.toString()));
339 mBuilder.setMasked(MatchField.IPV6_SRC, maskedIp);
340 } else {
341 mBuilder.setExact(MatchField.IPV6_SRC,
342 IPv6Address.of(ip6Prefix.address().toString()));
343 }
344 break;
345 case IPV6_DST:
346 ipCriterion = (IPCriterion) c;
347 ip6Prefix = ipCriterion.ip().getIp6Prefix();
348 if (ip6Prefix.prefixLength() != Ip6Prefix.MAX_MASK_LENGTH) {
349 Ip6Address maskAddr =
350 Ip6Address.makeMaskPrefix(ip6Prefix.prefixLength());
351 Masked<IPv6Address> maskedIp =
352 Masked.of(IPv6Address.of(ip6Prefix.address().toString()),
353 IPv6Address.of(maskAddr.toString()));
354 mBuilder.setMasked(MatchField.IPV6_DST, maskedIp);
355 } else {
356 mBuilder.setExact(MatchField.IPV6_DST,
357 IPv6Address.of(ip6Prefix.address().toString()));
358 }
359 break;
360 case IPV6_FLABEL:
361 IPv6FlowLabelCriterion flowLabelCriterion =
362 (IPv6FlowLabelCriterion) c;
363 mBuilder.setExact(MatchField.IPV6_FLABEL,
364 IPv6FlowLabel.of(flowLabelCriterion.flowLabel()));
365 break;
366 case ICMPV6_TYPE:
367 Icmpv6TypeCriterion icmpv6Type = (Icmpv6TypeCriterion) c;
368 mBuilder.setExact(MatchField.ICMPV6_TYPE,
Pavlin Radoslavovf3b69332015-02-06 15:47:05 -0800369 U8.of(icmpv6Type.icmpv6Type()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800370 break;
371 case ICMPV6_CODE:
372 Icmpv6CodeCriterion icmpv6Code = (Icmpv6CodeCriterion) c;
373 mBuilder.setExact(MatchField.ICMPV6_CODE,
Pavlin Radoslavovf3b69332015-02-06 15:47:05 -0800374 U8.of(icmpv6Code.icmpv6Code()));
Pavlin Radoslavov320e6c92015-02-02 16:51:58 -0800375 break;
376 case IPV6_ND_TARGET:
377 IPv6NDTargetAddressCriterion targetAddressCriterion =
378 (IPv6NDTargetAddressCriterion) c;
379 ip6Address = targetAddressCriterion.targetAddress();
380 mBuilder.setExact(MatchField.IPV6_ND_TARGET,
381 IPv6Address.of(ip6Address.toOctets()));
382 break;
383 case IPV6_ND_SLL:
384 llAddressCriterion =
385 (IPv6NDLinkLayerAddressCriterion) c;
386 mBuilder.setExact(MatchField.IPV6_ND_SLL,
387 MacAddress.of(llAddressCriterion.mac().toLong()));
388 break;
389 case IPV6_ND_TLL:
390 llAddressCriterion =
391 (IPv6NDLinkLayerAddressCriterion) c;
392 mBuilder.setExact(MatchField.IPV6_ND_TLL,
393 MacAddress.of(llAddressCriterion.mac().toLong()));
Jonathan Hart34bc6142014-10-17 11:00:43 -0700394 break;
Praseed Balakrishnan8c67d172014-11-10 10:15:41 -0800395 case MPLS_LABEL:
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -0700396 MplsCriterion mp = (MplsCriterion) c;
Michele Santuari4b6019e2014-12-19 11:31:45 +0100397 mBuilder.setExact(MatchField.MPLS_LABEL, U32.of(mp.label().toInt()));
Praseed Balakrishnan8c67d172014-11-10 10:15:41 -0800398 break;
Pavlin Radoslavov5e4f7542015-02-06 18:18:21 -0800399 case IPV6_EXTHDR:
Sho SHIMIZUfbc80e52015-04-28 10:41:58 -0700400 IPv6ExthdrFlagsCriterion exthdrFlagsCriterion =
401 (IPv6ExthdrFlagsCriterion) c;
Pavlin Radoslavov5e4f7542015-02-06 18:18:21 -0800402 mBuilder.setExact(MatchField.IPV6_EXTHDR,
403 U16.of(exthdrFlagsCriterion.exthdrFlags()));
404 break;
Marc De Leenheer49087752014-10-23 13:54:09 -0700405 case OCH_SIGID:
Sho SHIMIZUc15ce512015-05-26 16:54:08 -0700406 try {
407 OchSignalCriterion ochSignalCriterion = (OchSignalCriterion) c;
408 OchSignal signal = ochSignalCriterion.lambda();
Sho SHIMIZUc17042d2015-05-28 12:07:23 -0700409 byte gridType = OpenFlowValueMapper.lookupGridType(signal.gridType());
410 byte channelSpacing = OpenFlowValueMapper.lookupChannelSpacing(signal.channelSpacing());
Sho SHIMIZUc15ce512015-05-26 16:54:08 -0700411 mBuilder.setExact(MatchField.OCH_SIGID,
412 new CircuitSignalID(gridType, channelSpacing,
413 (short) signal.spacingMultiplier(), (short) signal.slotGranularity()));
Sho SHIMIZUc17042d2015-05-28 12:07:23 -0700414 } catch (NoMappingFoundException e) {
Sho SHIMIZUc15ce512015-05-26 16:54:08 -0700415 log.warn(e.getMessage());
416 }
Marc De Leenheer49087752014-10-23 13:54:09 -0700417 break;
Praseed Balakrishnan2dd5abd2014-11-03 14:56:28 -0800418 case OCH_SIGTYPE:
Marc De Leenheerd24420f2015-05-27 09:40:59 -0700419 OchSignalTypeCriterion sc = (OchSignalTypeCriterion) c;
Sho SHIMIZUc17042d2015-05-28 12:07:23 -0700420 byte signalType = OpenFlowValueMapper.lookupOchSignalType(sc.signalType());
Marc De Leenheerd24420f2015-05-27 09:40:59 -0700421 mBuilder.setExact(MatchField.OCH_SIGTYPE, U8.of(signalType));
Praseed Balakrishnan2dd5abd2014-11-03 14:56:28 -0800422 break;
Hyunsun Moona08c5d02015-07-14 17:53:00 -0700423 case TUNNEL_ID:
424 TunnelIdCriterion tunnelId = (TunnelIdCriterion) c;
425 mBuilder.setExact(MatchField.TUNNEL_ID,
426 U64.of(tunnelId.tunnelId()));
427 break;
Saurav Dasffc5bbc2015-08-18 23:30:19 -0700428 case MPLS_BOS:
429 MplsBosCriterion mplsBos = (MplsBosCriterion) c;
430 mBuilder.setExact(MatchField.MPLS_BOS,
431 mplsBos.mplsBos() ? OFBooleanValue.TRUE
432 : OFBooleanValue.FALSE);
433 break;
BitOhenryb40129a2015-11-30 12:41:18 +0800434 case ARP_OP:
435 ArpOpCriterion arpOp = (ArpOpCriterion) c;
436 mBuilder.setExact(MatchField.ARP_OP,
437 ArpOpcode.of(arpOp.arpOp()));
438 break;
BitOhenrya331b182015-11-23 08:39:37 +0800439 case ARP_SHA:
440 arpHaCriterion = (ArpHaCriterion) c;
441 mBuilder.setExact(MatchField.ARP_SHA,
442 MacAddress.of(arpHaCriterion.mac().toLong()));
443 break;
BitOhenry296b4542015-11-24 08:41:58 +0800444 case ARP_SPA:
445 arpPaCriterion = (ArpPaCriterion) c;
446 mBuilder.setExact(MatchField.ARP_SPA,
447 IPv4Address.of(arpPaCriterion.ip().toInt()));
448 break;
BitOhenry76430852015-11-20 19:04:49 +0800449 case ARP_THA:
450 arpHaCriterion = (ArpHaCriterion) c;
451 mBuilder.setExact(MatchField.ARP_THA,
452 MacAddress.of(arpHaCriterion.mac().toLong()));
453 break;
BitOhenryc1e5fcc2015-11-23 20:47:53 +0800454 case ARP_TPA:
455 arpPaCriterion = (ArpPaCriterion) c;
456 mBuilder.setExact(MatchField.ARP_TPA,
457 IPv4Address.of(arpPaCriterion.ip().toInt()));
458 break;
Jonathan Hart26a8d952015-12-02 15:16:35 -0800459 case EXTENSION:
460 ExtensionCriterion extensionCriterion = (ExtensionCriterion) c;
461 OFOxm oxm = buildExtensionOxm(extensionCriterion.extensionSelector());
462 if (oxm == null) {
463 log.warn("Unable to build extension selector");
464 break;
465 }
466
467 if (oxm.isMasked()) {
468 mBuilder.setMasked(oxm.getMatchField(), oxm.getValue(), oxm.getMask());
469 } else {
470 mBuilder.setExact(oxm.getMatchField(), oxm.getValue());
471 }
472
473 break;
alshabibeec3a062014-09-17 18:01:26 -0700474 case MPLS_TC:
475 case PBB_ISID:
alshabibeec3a062014-09-17 18:01:26 -0700476 default:
477 log.warn("Match type {} not yet implemented.", c.type());
478 }
479 }
480 return mBuilder.build();
481 }
482
Jonathan Hart86e59352014-10-22 10:42:16 -0700483 /**
484 * Returns the flow rule for this builder.
485 *
486 * @return the flow rule
487 */
488 protected FlowRule flowRule() {
489 return flowRule;
490 }
alshabib219ebaa2014-09-22 15:41:24 -0700491
Jonathan Hart86e59352014-10-22 10:42:16 -0700492 /**
493 * Returns the factory used for building OpenFlow constructs.
494 *
495 * @return the factory
496 */
497 protected OFFactory factory() {
498 return factory;
499 }
alshabib219ebaa2014-09-22 15:41:24 -0700500
Jonathan Hart26a8d952015-12-02 15:16:35 -0800501 private OFOxm buildExtensionOxm(ExtensionSelector extension) {
502 if (!driverService.isPresent()) {
503 log.error("No driver service present");
504 return null;
505 }
506 Driver driver = driverService.get().getDriver(deviceId);
507 if (driver.hasBehaviour(ExtensionSelectorInterpreter.class)) {
508 DefaultDriverHandler handler =
509 new DefaultDriverHandler(new DefaultDriverData(driver, deviceId));
510 ExtensionSelectorInterpreter interpreter = handler.behaviour(ExtensionSelectorInterpreter.class);
511
512 return interpreter.mapSelector(factory(), extension);
513 }
514
515 return null;
516 }
517
alshabibeec3a062014-09-17 18:01:26 -0700518}