blob: c42488c2b56fd4b87a735257cb02e5584263b2a2 [file] [log] [blame]
Jonathan Hart1caaa932013-11-04 15:28:28 -08001package net.onrc.onos.ofcontroller.forwarding;
2
Jonathan Harte93aed42013-12-05 18:39:50 -08003import java.util.ArrayList;
4import java.util.Collection;
Jonathan Hartd33a6cf2013-12-10 14:29:08 -08005import java.util.HashMap;
Jonathan Hart1caaa932013-11-04 15:28:28 -08006import java.util.Iterator;
Jonathan Harte93aed42013-12-05 18:39:50 -08007import java.util.List;
8import java.util.Map;
Jonathan Hart1caaa932013-11-04 15:28:28 -08009
10import net.floodlightcontroller.core.FloodlightContext;
11import net.floodlightcontroller.core.IFloodlightProviderService;
12import net.floodlightcontroller.core.IOFMessageListener;
13import net.floodlightcontroller.core.IOFSwitch;
Jonathan Harte93aed42013-12-05 18:39:50 -080014import net.floodlightcontroller.core.module.FloodlightModuleContext;
15import net.floodlightcontroller.core.module.IFloodlightModule;
16import net.floodlightcontroller.core.module.IFloodlightService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080017import net.floodlightcontroller.packet.Ethernet;
18import net.floodlightcontroller.util.MACAddress;
Jonathan Hart17672992013-12-12 16:15:16 -080019import net.onrc.onos.datagrid.IDatagridService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080020import net.onrc.onos.ofcontroller.core.IDeviceStorage;
21import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.IDeviceObject;
22import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.IPortObject;
23import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.ISwitchObject;
24import net.onrc.onos.ofcontroller.core.internal.DeviceStorageImpl;
Jonathan Hartd857ad62013-12-14 18:08:17 -080025import net.onrc.onos.ofcontroller.devicemanager.IOnosDeviceService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080026import net.onrc.onos.ofcontroller.flowmanager.IFlowService;
Jonathan Hart7e6df362013-12-10 23:33:59 -080027import net.onrc.onos.ofcontroller.flowprogrammer.IFlowPusherService;
Jonathan Hart7804bea2014-01-07 10:50:52 -080028import net.onrc.onos.ofcontroller.proxyarp.IProxyArpService;
29import net.onrc.onos.ofcontroller.proxyarp.BroadcastPacketOutNotification;
Jonathan Hart1caaa932013-11-04 15:28:28 -080030import net.onrc.onos.ofcontroller.topology.TopologyManager;
31import net.onrc.onos.ofcontroller.util.CallerId;
32import net.onrc.onos.ofcontroller.util.DataPath;
33import net.onrc.onos.ofcontroller.util.Dpid;
Jonathan Hart4fb16d82013-11-07 22:41:32 -080034import net.onrc.onos.ofcontroller.util.FlowEntryMatch;
Jonathan Hart1caaa932013-11-04 15:28:28 -080035import net.onrc.onos.ofcontroller.util.FlowId;
36import net.onrc.onos.ofcontroller.util.FlowPath;
37import net.onrc.onos.ofcontroller.util.FlowPathType;
38import net.onrc.onos.ofcontroller.util.FlowPathUserState;
39import net.onrc.onos.ofcontroller.util.Port;
40import net.onrc.onos.ofcontroller.util.SwitchPort;
41
42import org.openflow.protocol.OFMessage;
43import org.openflow.protocol.OFPacketIn;
Jonathan Hart41d1e912013-11-24 16:50:25 -080044import org.openflow.protocol.OFPacketOut;
45import org.openflow.protocol.OFPort;
Jonathan Hart1caaa932013-11-04 15:28:28 -080046import org.openflow.protocol.OFType;
Jonathan Hart41d1e912013-11-24 16:50:25 -080047import org.openflow.protocol.action.OFAction;
48import org.openflow.protocol.action.OFActionOutput;
Jonathan Hart1caaa932013-11-04 15:28:28 -080049import org.openflow.util.HexString;
50import org.slf4j.Logger;
51import org.slf4j.LoggerFactory;
52
Jonathan Hartd857ad62013-12-14 18:08:17 -080053import com.google.common.collect.LinkedListMultimap;
54import com.google.common.collect.ListMultimap;
Jonathan Hart5e448782013-12-10 12:36:35 -080055
56public class Forwarding implements IOFMessageListener, IFloodlightModule,
57 IForwardingService {
Jonathan Hart1caaa932013-11-04 15:28:28 -080058 private final static Logger log = LoggerFactory.getLogger(Forwarding.class);
59
Jonathan Hart7e6df362013-12-10 23:33:59 -080060 private final int IDLE_TIMEOUT = 5; // seconds
61 private final int HARD_TIMEOUT = 0; // seconds
Jonathan Harte789d6e2013-12-17 17:50:11 -080062
63 private final int PATH_PUSHED_TIMEOUT = 3000; // milliseconds
Jonathan Hart7e6df362013-12-10 23:33:59 -080064
Jonathan Hart1caaa932013-11-04 15:28:28 -080065 private IFloodlightProviderService floodlightProvider;
66 private IFlowService flowService;
Jonathan Hart7e6df362013-12-10 23:33:59 -080067 private IFlowPusherService flowPusher;
Jonathan Hart17672992013-12-12 16:15:16 -080068 private IDatagridService datagrid;
Jonathan Hart1caaa932013-11-04 15:28:28 -080069
70 private IDeviceStorage deviceStorage;
71 private TopologyManager topologyService;
72
Jonathan Harte789d6e2013-12-17 17:50:11 -080073 //private Map<Path, Long> pendingFlows;
74 // TODO it seems there is a Guava collection that will time out entries.
75 // We should see if this will work here.
76 private Map<Path, PushedFlow> pendingFlows;
Jonathan Hartd857ad62013-12-14 18:08:17 -080077 private ListMultimap<Long, PacketToPush> waitingPackets;
Jonathan Hart5e448782013-12-10 12:36:35 -080078
Jonathan Hart7e6df362013-12-10 23:33:59 -080079 private final Object lock = new Object();
80
Jonathan Harte789d6e2013-12-17 17:50:11 -080081 private class PacketToPush {
Jonathan Hart5e448782013-12-10 12:36:35 -080082 public final OFPacketOut packet;
83 public final long dpid;
Jonathan Hart1caaa932013-11-04 15:28:28 -080084
Jonathan Hart5e448782013-12-10 12:36:35 -080085 public PacketToPush(OFPacketOut packet, long dpid) {
86 this.packet = packet;
87 this.dpid = dpid;
88 }
89 }
90
Jonathan Harte789d6e2013-12-17 17:50:11 -080091 private class PushedFlow {
92 public final long flowId;
93 private final long pushedTime;
94 public short firstHopOutPort = OFPort.OFPP_NONE.getValue();
95
96 public PushedFlow(long flowId) {
97 this.flowId = flowId;
98 pushedTime = System.currentTimeMillis();
99 }
100
101 public boolean isExpired() {
102 return (System.currentTimeMillis() - pushedTime) > PATH_PUSHED_TIMEOUT;
103 }
104 }
105
106 private final class Path {
Jonathan Hart5e448782013-12-10 12:36:35 -0800107 public final SwitchPort srcPort;
108 public final SwitchPort dstPort;
Jonathan Harte789d6e2013-12-17 17:50:11 -0800109 public final MACAddress srcMac;
110 public final MACAddress dstMac;
Jonathan Hart5e448782013-12-10 12:36:35 -0800111
Jonathan Harte789d6e2013-12-17 17:50:11 -0800112 public Path(SwitchPort src, SwitchPort dst,
113 MACAddress srcMac, MACAddress dstMac) {
Jonathan Hart5e448782013-12-10 12:36:35 -0800114 srcPort = new SwitchPort(new Dpid(src.dpid().value()),
115 new Port(src.port().value()));
116 dstPort = new SwitchPort(new Dpid(dst.dpid().value()),
117 new Port(dst.port().value()));
Jonathan Harte789d6e2013-12-17 17:50:11 -0800118 this.srcMac = srcMac;
119 this.dstMac = dstMac;
Jonathan Hart5e448782013-12-10 12:36:35 -0800120 }
121
122 @Override
123 public boolean equals(Object other) {
124 if (!(other instanceof Path)) {
125 return false;
126 }
127
128 Path otherPath = (Path) other;
129 return srcPort.equals(otherPath.srcPort) &&
Jonathan Harte789d6e2013-12-17 17:50:11 -0800130 dstPort.equals(otherPath.dstPort) &&
131 srcMac.equals(otherPath.srcMac) &&
132 dstMac.equals(otherPath.dstMac);
Jonathan Hart5e448782013-12-10 12:36:35 -0800133 }
134
135 @Override
136 public int hashCode() {
137 int hash = 17;
138 hash = 31 * hash + srcPort.hashCode();
139 hash = 31 * hash + dstPort.hashCode();
Jonathan Harte789d6e2013-12-17 17:50:11 -0800140 hash = 31 * hash + srcMac.hashCode();
141 hash = 31 * hash + dstMac.hashCode();
Jonathan Hart5e448782013-12-10 12:36:35 -0800142 return hash;
143 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800144
145 @Override
146 public String toString() {
147 return "(" + srcMac + " at " + srcPort + ") => ("
148 + dstPort + " at " + dstMac + ")";
149 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800150 }
151
Jonathan Harte93aed42013-12-05 18:39:50 -0800152 @Override
153 public Collection<Class<? extends IFloodlightService>> getModuleServices() {
Jonathan Hartd33a6cf2013-12-10 14:29:08 -0800154 List<Class<? extends IFloodlightService>> services =
155 new ArrayList<Class<? extends IFloodlightService>>(1);
156 services.add(IForwardingService.class);
157 return services;
Jonathan Harte93aed42013-12-05 18:39:50 -0800158 }
159
160 @Override
161 public Map<Class<? extends IFloodlightService>, IFloodlightService> getServiceImpls() {
Jonathan Hartd33a6cf2013-12-10 14:29:08 -0800162 Map<Class<? extends IFloodlightService>, IFloodlightService> impls =
163 new HashMap<Class<? extends IFloodlightService>, IFloodlightService>(1);
164 impls.put(IForwardingService.class, this);
165 return impls;
Jonathan Harte93aed42013-12-05 18:39:50 -0800166 }
167
168 @Override
169 public Collection<Class<? extends IFloodlightService>> getModuleDependencies() {
170 List<Class<? extends IFloodlightService>> dependencies =
171 new ArrayList<Class<? extends IFloodlightService>>();
172 dependencies.add(IFloodlightProviderService.class);
173 dependencies.add(IFlowService.class);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800174 dependencies.add(IFlowPusherService.class);
Jonathan Hartd857ad62013-12-14 18:08:17 -0800175 dependencies.add(IOnosDeviceService.class);
Jonathan Hart7804bea2014-01-07 10:50:52 -0800176 // We don't use the IProxyArpService directly, but reactive forwarding
177 // requires it to be loaded and answering ARP requests
178 dependencies.add(IProxyArpService.class);
Jonathan Harte93aed42013-12-05 18:39:50 -0800179 return dependencies;
180 }
181
182 @Override
183 public void init(FloodlightModuleContext context) {
Jonathan Hart7e6df362013-12-10 23:33:59 -0800184 floodlightProvider =
Jonathan Harte93aed42013-12-05 18:39:50 -0800185 context.getServiceImpl(IFloodlightProviderService.class);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800186 flowService = context.getServiceImpl(IFlowService.class);
187 flowPusher = context.getServiceImpl(IFlowPusherService.class);
Jonathan Hart17672992013-12-12 16:15:16 -0800188 datagrid = context.getServiceImpl(IDatagridService.class);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800189
190 floodlightProvider.addOFMessageListener(OFType.PACKET_IN, this);
191
Jonathan Harte789d6e2013-12-17 17:50:11 -0800192 //pendingFlows = new ConcurrentHashMap<Path, Long>();
193 pendingFlows = new HashMap<Path, PushedFlow>();
Jonathan Hart7e6df362013-12-10 23:33:59 -0800194 //waitingPackets = Multimaps.synchronizedSetMultimap(
195 //HashMultimap.<Long, PacketToPush>create());
Jonathan Hartd857ad62013-12-14 18:08:17 -0800196 //waitingPackets = HashMultimap.create();
197 waitingPackets = LinkedListMultimap.create();
Jonathan Hart5e448782013-12-10 12:36:35 -0800198
Jonathan Hart1caaa932013-11-04 15:28:28 -0800199 deviceStorage = new DeviceStorageImpl();
200 deviceStorage.init("");
201 topologyService = new TopologyManager();
202 topologyService.init("");
203 }
204
Jonathan Harte93aed42013-12-05 18:39:50 -0800205 @Override
206 public void startUp(FloodlightModuleContext context) {
Jonathan Hart1caaa932013-11-04 15:28:28 -0800207 // no-op
208 }
209
210 @Override
211 public String getName() {
212 return "onosforwarding";
213 }
214
215 @Override
216 public boolean isCallbackOrderingPrereq(OFType type, String name) {
217 return (type == OFType.PACKET_IN) &&
Jonathan Hartd857ad62013-12-14 18:08:17 -0800218 (name.equals("devicemanager") || name.equals("proxyarpmanager")
219 || name.equals("onosdevicemanager"));
Jonathan Hart1caaa932013-11-04 15:28:28 -0800220 }
221
222 @Override
223 public boolean isCallbackOrderingPostreq(OFType type, String name) {
224 return false;
225 }
226
227 @Override
228 public Command receive(
229 IOFSwitch sw, OFMessage msg, FloodlightContext cntx) {
230
231 if (msg.getType() != OFType.PACKET_IN) {
232 return Command.CONTINUE;
233 }
234
235 OFPacketIn pi = (OFPacketIn) msg;
236
237 Ethernet eth = IFloodlightProviderService.bcStore.
238 get(cntx, IFloodlightProviderService.CONTEXT_PI_PAYLOAD);
239
Jonathan Hart17672992013-12-12 16:15:16 -0800240 if (eth.getEtherType() != Ethernet.TYPE_IPv4) {
Jonathan Hart1caaa932013-11-04 15:28:28 -0800241 return Command.CONTINUE;
242 }
243
Jonathan Hart17672992013-12-12 16:15:16 -0800244 if (eth.isBroadcast() || eth.isMulticast()) {
245 handleBroadcast(sw, pi, eth);
246 //return Command.CONTINUE;
247 }
248 else {
249 // Unicast
250 handlePacketIn(sw, pi, eth);
251 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800252
253 return Command.STOP;
254 }
255
Jonathan Hart17672992013-12-12 16:15:16 -0800256 private void handleBroadcast(IOFSwitch sw, OFPacketIn pi, Ethernet eth) {
257 if (log.isTraceEnabled()) {
258 log.trace("Sending broadcast packet to other ONOS instances");
259 }
Jonathan Hart7804bea2014-01-07 10:50:52 -0800260
261 datagrid.sendPacketOutNotification(new BroadcastPacketOutNotification(
262 eth.serialize(), sw.getId(), pi.getInPort()));
Jonathan Hart17672992013-12-12 16:15:16 -0800263 }
264
Jonathan Hart1caaa932013-11-04 15:28:28 -0800265 private void handlePacketIn(IOFSwitch sw, OFPacketIn pi, Ethernet eth) {
Jonathan Hart5e448782013-12-10 12:36:35 -0800266 String destinationMac =
267 HexString.toHexString(eth.getDestinationMACAddress());
Jonathan Hart1caaa932013-11-04 15:28:28 -0800268
269 IDeviceObject deviceObject = deviceStorage.getDeviceByMac(
270 destinationMac);
271
272 if (deviceObject == null) {
Jonathan Hart18ad9502013-12-15 18:28:00 -0800273 log.debug("No device entry found for {} - broadcasting packet",
274 destinationMac);
275 handleBroadcast(sw, pi, eth);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800276 return;
277 }
278
Jonathan Hart5e448782013-12-10 12:36:35 -0800279 Iterator<IPortObject> ports = deviceObject.getAttachedPorts().iterator();
Jonathan Hart1caaa932013-11-04 15:28:28 -0800280 if (!ports.hasNext()) {
Jonathan Hart18ad9502013-12-15 18:28:00 -0800281 log.debug("No attachment point found for device {} - broadcasting packet",
282 destinationMac);
283 handleBroadcast(sw, pi, eth);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800284 return;
285 }
286 IPortObject portObject = ports.next();
287 short destinationPort = portObject.getNumber();
288 ISwitchObject switchObject = portObject.getSwitch();
289 long destinationDpid = HexString.toLong(switchObject.getDPID());
290
Jonathan Hart41d1e912013-11-24 16:50:25 -0800291 // TODO SwitchPort, Dpid and Port should probably be immutable
Jonathan Hart1caaa932013-11-04 15:28:28 -0800292 SwitchPort srcSwitchPort = new SwitchPort(
293 new Dpid(sw.getId()), new Port(pi.getInPort()));
294 SwitchPort dstSwitchPort = new SwitchPort(
295 new Dpid(destinationDpid), new Port(destinationPort));
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800296
297 MACAddress srcMacAddress = MACAddress.valueOf(eth.getSourceMACAddress());
298 MACAddress dstMacAddress = MACAddress.valueOf(eth.getDestinationMACAddress());
Jonathan Hart1caaa932013-11-04 15:28:28 -0800299
Jonathan Hart5e448782013-12-10 12:36:35 -0800300
Jonathan Hart7e6df362013-12-10 23:33:59 -0800301 FlowPath flowPath, reverseFlowPath;
Jonathan Hart5e448782013-12-10 12:36:35 -0800302
Jonathan Harte789d6e2013-12-17 17:50:11 -0800303 Path pathspec = new Path(srcSwitchPort, dstSwitchPort,
304 srcMacAddress, dstMacAddress);
Jonathan Hart5e448782013-12-10 12:36:35 -0800305 // TODO check concurrency
Jonathan Hart7e6df362013-12-10 23:33:59 -0800306 synchronized (lock) {
Jonathan Harte789d6e2013-12-17 17:50:11 -0800307 PushedFlow existingFlow = pendingFlows.get(pathspec);
308 //Long existingFlowId = pendingFlows.get(pathspec);
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800309
Jonathan Harte789d6e2013-12-17 17:50:11 -0800310 if (existingFlow != null && !existingFlow.isExpired()) {
Jonathan Hart7e6df362013-12-10 23:33:59 -0800311 log.debug("Found existing flow {}",
Jonathan Harte789d6e2013-12-17 17:50:11 -0800312 HexString.toHexString(existingFlow.flowId));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800313
314 OFPacketOut po = constructPacketOut(pi, sw);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800315
316 if (existingFlow.firstHopOutPort != OFPort.OFPP_NONE.getValue()) {
317 // Flow has been sent to the switches so it is safe to
318 // send a packet out now
319 sendPacketOut(sw, po, existingFlow.firstHopOutPort);
320 }
321 else {
322 // Flow has not yet been sent to switches so save the
323 // packet out for later
324 waitingPackets.put(existingFlow.flowId,
325 new PacketToPush(po, sw.getId()));
326 }
Jonathan Hart7e6df362013-12-10 23:33:59 -0800327 return;
328 }
329
Jonathan Harte789d6e2013-12-17 17:50:11 -0800330 //log.debug("Couldn't match {} in {}", pathspec, pendingFlows);
331
Jonathan Hart7e6df362013-12-10 23:33:59 -0800332 log.debug("Adding new flow between {} at {} and {} at {}",
333 new Object[]{srcMacAddress, srcSwitchPort, dstMacAddress, dstSwitchPort});
334
335
336 CallerId callerId = new CallerId("Forwarding");
337
338 DataPath datapath = new DataPath();
339 datapath.setSrcPort(srcSwitchPort);
340 datapath.setDstPort(dstSwitchPort);
341
342 flowPath = new FlowPath();
343 flowPath.setInstallerId(callerId);
344
345 flowPath.setFlowPathType(FlowPathType.FP_TYPE_SHORTEST_PATH);
346 flowPath.setFlowPathUserState(FlowPathUserState.FP_USER_ADD);
347 flowPath.setFlowEntryMatch(new FlowEntryMatch());
348 flowPath.setIdleTimeout(IDLE_TIMEOUT);
349 flowPath.setHardTimeout(HARD_TIMEOUT);
350 flowPath.flowEntryMatch().enableSrcMac(srcMacAddress);
351 flowPath.flowEntryMatch().enableDstMac(dstMacAddress);
352 flowPath.flowEntryMatch().enableEthernetFrameType(Ethernet.TYPE_IPv4);
353 flowPath.setDataPath(datapath);
354
355
356 DataPath reverseDataPath = new DataPath();
357 // Reverse the ports for the reverse path
358 reverseDataPath.setSrcPort(dstSwitchPort);
359 reverseDataPath.setDstPort(srcSwitchPort);
360
361 // TODO implement copy constructor for FlowPath
362 reverseFlowPath = new FlowPath();
363 reverseFlowPath.setInstallerId(callerId);
364 reverseFlowPath.setFlowPathType(FlowPathType.FP_TYPE_SHORTEST_PATH);
365 reverseFlowPath.setFlowPathUserState(FlowPathUserState.FP_USER_ADD);
366 reverseFlowPath.setIdleTimeout(IDLE_TIMEOUT);
367 reverseFlowPath.setHardTimeout(HARD_TIMEOUT);
368 reverseFlowPath.setFlowEntryMatch(new FlowEntryMatch());
369 // Reverse the MAC addresses for the reverse path
370 reverseFlowPath.flowEntryMatch().enableSrcMac(dstMacAddress);
371 reverseFlowPath.flowEntryMatch().enableDstMac(srcMacAddress);
372 reverseFlowPath.flowEntryMatch().enableEthernetFrameType(Ethernet.TYPE_IPv4);
373 reverseFlowPath.setDataPath(reverseDataPath);
374 reverseFlowPath.dataPath().srcPort().dpid().toString();
375
376 // TODO what happens if no path exists? cleanup
377
378 FlowId flowId = new FlowId(flowService.getNextFlowEntryId());
379 FlowId reverseFlowId = new FlowId(flowService.getNextFlowEntryId());
380
381 flowPath.setFlowId(flowId);
382 reverseFlowPath.setFlowId(reverseFlowId);
383
384 OFPacketOut po = constructPacketOut(pi, sw);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800385 Path reversePathSpec = new Path(dstSwitchPort, srcSwitchPort,
386 dstMacAddress, srcMacAddress);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800387
388 // Add to waiting lists
Jonathan Harte789d6e2013-12-17 17:50:11 -0800389 //pendingFlows.put(pathspec, flowId.value());
390 //pendingFlows.put(reversePathSpec, reverseFlowId.value());
391 pendingFlows.put(pathspec, new PushedFlow(flowId.value()));
392 pendingFlows.put(reversePathSpec, new PushedFlow(reverseFlowId.value()));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800393 waitingPackets.put(flowId.value(), new PacketToPush(po, sw.getId()));
394
Jonathan Hart41d1e912013-11-24 16:50:25 -0800395 }
396
Jonathan Hart5e448782013-12-10 12:36:35 -0800397 flowService.addFlow(reverseFlowPath);
398 flowService.addFlow(flowPath);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800399
Jonathan Hart1caaa932013-11-04 15:28:28 -0800400 }
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800401
Jonathan Hart5e448782013-12-10 12:36:35 -0800402 /*
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800403 private boolean flowExists(SwitchPort srcPort, MACAddress srcMac,
404 SwitchPort dstPort, MACAddress dstMac) {
405 for (FlowPath flow : datagridService.getAllFlows()) {
406 FlowEntryMatch match = flow.flowEntryMatch();
407 // TODO implement FlowEntryMatch.equals();
408 // This is painful to do properly without support in the FlowEntryMatch
409 boolean same = true;
410 if (!match.srcMac().equals(srcMac) ||
411 !match.dstMac().equals(dstMac)) {
412 same = false;
413 }
414 if (!flow.dataPath().srcPort().equals(srcPort) ||
415 !flow.dataPath().dstPort().equals(dstPort)) {
416 same = false;
417 }
418
419 if (same) {
420 log.debug("found flow entry that's the same {}-{}:::{}-{}",
421 new Object[] {srcPort, srcMac, dstPort, dstMac});
422 return true;
423 }
424 }
425
426 return false;
427 }
Jonathan Hart5e448782013-12-10 12:36:35 -0800428 */
Jonathan Hart1caaa932013-11-04 15:28:28 -0800429
Jonathan Hart7e6df362013-12-10 23:33:59 -0800430 private OFPacketOut constructPacketOut(OFPacketIn pi, IOFSwitch sw) {
Jonathan Hart41d1e912013-11-24 16:50:25 -0800431 OFPacketOut po = new OFPacketOut();
432 po.setInPort(OFPort.OFPP_NONE)
433 .setInPort(pi.getInPort())
Jonathan Hart5e448782013-12-10 12:36:35 -0800434 .setActions(new ArrayList<OFAction>())
435 .setLengthU(OFPacketOut.MINIMUM_LENGTH);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800436
437 if (sw.getBuffers() == 0) {
438 po.setBufferId(OFPacketOut.BUFFER_ID_NONE)
439 .setPacketData(pi.getPacketData())
440 .setLengthU(po.getLengthU() + po.getPacketData().length);
441 }
442 else {
443 po.setBufferId(pi.getBufferId());
444 }
445
Jonathan Hart5e448782013-12-10 12:36:35 -0800446 return po;
447 }
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800448
Jonathan Hart5e448782013-12-10 12:36:35 -0800449 @Override
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800450 public void flowsInstalled(Collection<FlowPath> installedFlowPaths) {
451 for (FlowPath flowPath : installedFlowPaths) {
452 flowInstalled(flowPath);
453 }
454 }
455
456 private void flowInstalled(FlowPath installedFlowPath) {
Jonathan Hart5e448782013-12-10 12:36:35 -0800457 long flowId = installedFlowPath.flowId().value();
Jonathan Hart5e448782013-12-10 12:36:35 -0800458
Jonathan Harte789d6e2013-12-17 17:50:11 -0800459 short outPort =
460 installedFlowPath.flowEntries().get(0).outPort().value();
461
462 MACAddress srcMacAddress = installedFlowPath.flowEntryMatch().srcMac();
463 MACAddress dstMacAddress = installedFlowPath.flowEntryMatch().dstMac();
464
Jonathan Hart7e6df362013-12-10 23:33:59 -0800465 Collection<PacketToPush> packets;
466 synchronized (lock) {
Jonathan Harte789d6e2013-12-17 17:50:11 -0800467 log.debug("Flow {} has been installed, sending queued packets",
468 installedFlowPath.flowId());
469
Jonathan Hart7e6df362013-12-10 23:33:59 -0800470 packets = waitingPackets.removeAll(flowId);
471
Jonathan Harte789d6e2013-12-17 17:50:11 -0800472 // remove pending flows entry
473 Path installedPath = new Path(installedFlowPath.dataPath().srcPort(),
474 installedFlowPath.dataPath().dstPort(),
475 srcMacAddress, dstMacAddress);
476 //pendingFlows.remove(pathToRemove);
Pavlin Radoslavov3e6a0bc2013-12-17 22:20:36 -0800477 PushedFlow existingFlow = pendingFlows.get(installedPath);
478 if (existingFlow != null)
479 existingFlow.firstHopOutPort = outPort;
Jonathan Hart7e6df362013-12-10 23:33:59 -0800480 }
Jonathan Hart5e448782013-12-10 12:36:35 -0800481
482 for (PacketToPush packet : packets) {
483 IOFSwitch sw = floodlightProvider.getSwitches().get(packet.dpid);
484
Jonathan Harte789d6e2013-12-17 17:50:11 -0800485 sendPacketOut(sw, packet.packet, outPort);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800486 }
487 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800488
489 private void sendPacketOut(IOFSwitch sw, OFPacketOut po, short outPort) {
490 po.getActions().add(new OFActionOutput(outPort));
491 po.setActionsLength((short)
492 (po.getActionsLength() + OFActionOutput.MINIMUM_LENGTH));
493 po.setLengthU(po.getLengthU() + OFActionOutput.MINIMUM_LENGTH);
494
495 flowPusher.add(sw, po);
496 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800497}