blob: 133f29ce97a2240f59d37cc123b7386ea248b7a6 [file] [log] [blame]
Jonathan Hart1caaa932013-11-04 15:28:28 -08001package net.onrc.onos.ofcontroller.forwarding;
2
Jonathan Hart17672992013-12-12 16:15:16 -08003import java.net.InetAddress;
Jonathan Harte93aed42013-12-05 18:39:50 -08004import java.util.ArrayList;
5import java.util.Collection;
Jonathan Hartd33a6cf2013-12-10 14:29:08 -08006import java.util.HashMap;
Jonathan Hart1caaa932013-11-04 15:28:28 -08007import java.util.Iterator;
Jonathan Harte93aed42013-12-05 18:39:50 -08008import java.util.List;
9import java.util.Map;
Jonathan Hart1caaa932013-11-04 15:28:28 -080010
11import net.floodlightcontroller.core.FloodlightContext;
12import net.floodlightcontroller.core.IFloodlightProviderService;
13import net.floodlightcontroller.core.IOFMessageListener;
14import net.floodlightcontroller.core.IOFSwitch;
Jonathan Harte93aed42013-12-05 18:39:50 -080015import net.floodlightcontroller.core.module.FloodlightModuleContext;
16import net.floodlightcontroller.core.module.IFloodlightModule;
17import net.floodlightcontroller.core.module.IFloodlightService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080018import net.floodlightcontroller.packet.Ethernet;
Jonathan Hart17672992013-12-12 16:15:16 -080019import net.floodlightcontroller.packet.IPv4;
Jonathan Hart1caaa932013-11-04 15:28:28 -080020import net.floodlightcontroller.util.MACAddress;
Jonathan Hart17672992013-12-12 16:15:16 -080021import net.onrc.onos.datagrid.IDatagridService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080022import net.onrc.onos.ofcontroller.core.IDeviceStorage;
23import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.IDeviceObject;
24import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.IPortObject;
25import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.ISwitchObject;
26import net.onrc.onos.ofcontroller.core.internal.DeviceStorageImpl;
Jonathan Hartd857ad62013-12-14 18:08:17 -080027import net.onrc.onos.ofcontroller.devicemanager.IOnosDeviceService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080028import net.onrc.onos.ofcontroller.flowmanager.IFlowService;
Jonathan Hart7e6df362013-12-10 23:33:59 -080029import net.onrc.onos.ofcontroller.flowprogrammer.IFlowPusherService;
Jonathan Hart17672992013-12-12 16:15:16 -080030import net.onrc.onos.ofcontroller.proxyarp.ArpMessage;
Jonathan Hart1caaa932013-11-04 15:28:28 -080031import net.onrc.onos.ofcontroller.topology.TopologyManager;
32import net.onrc.onos.ofcontroller.util.CallerId;
33import net.onrc.onos.ofcontroller.util.DataPath;
34import net.onrc.onos.ofcontroller.util.Dpid;
Jonathan Hart4fb16d82013-11-07 22:41:32 -080035import net.onrc.onos.ofcontroller.util.FlowEntryMatch;
Jonathan Hart1caaa932013-11-04 15:28:28 -080036import net.onrc.onos.ofcontroller.util.FlowId;
37import net.onrc.onos.ofcontroller.util.FlowPath;
38import net.onrc.onos.ofcontroller.util.FlowPathType;
39import net.onrc.onos.ofcontroller.util.FlowPathUserState;
40import net.onrc.onos.ofcontroller.util.Port;
41import net.onrc.onos.ofcontroller.util.SwitchPort;
42
43import org.openflow.protocol.OFMessage;
44import org.openflow.protocol.OFPacketIn;
Jonathan Hart41d1e912013-11-24 16:50:25 -080045import org.openflow.protocol.OFPacketOut;
46import org.openflow.protocol.OFPort;
Jonathan Hart1caaa932013-11-04 15:28:28 -080047import org.openflow.protocol.OFType;
Jonathan Hart41d1e912013-11-24 16:50:25 -080048import org.openflow.protocol.action.OFAction;
49import org.openflow.protocol.action.OFActionOutput;
Jonathan Hart1caaa932013-11-04 15:28:28 -080050import org.openflow.util.HexString;
51import org.slf4j.Logger;
52import org.slf4j.LoggerFactory;
53
Jonathan Hartd857ad62013-12-14 18:08:17 -080054import com.google.common.collect.LinkedListMultimap;
55import com.google.common.collect.ListMultimap;
Jonathan Hart17672992013-12-12 16:15:16 -080056import com.google.common.net.InetAddresses;
Jonathan Hart5e448782013-12-10 12:36:35 -080057
58public class Forwarding implements IOFMessageListener, IFloodlightModule,
59 IForwardingService {
Jonathan Hart1caaa932013-11-04 15:28:28 -080060 private final static Logger log = LoggerFactory.getLogger(Forwarding.class);
61
Jonathan Hart7e6df362013-12-10 23:33:59 -080062 private final int IDLE_TIMEOUT = 5; // seconds
63 private final int HARD_TIMEOUT = 0; // seconds
Jonathan Harte789d6e2013-12-17 17:50:11 -080064
65 private final int PATH_PUSHED_TIMEOUT = 3000; // milliseconds
Jonathan Hart7e6df362013-12-10 23:33:59 -080066
Jonathan Hart1caaa932013-11-04 15:28:28 -080067 private IFloodlightProviderService floodlightProvider;
68 private IFlowService flowService;
Jonathan Hart7e6df362013-12-10 23:33:59 -080069 private IFlowPusherService flowPusher;
Jonathan Hart17672992013-12-12 16:15:16 -080070 private IDatagridService datagrid;
Jonathan Hart1caaa932013-11-04 15:28:28 -080071
72 private IDeviceStorage deviceStorage;
73 private TopologyManager topologyService;
74
Jonathan Harte789d6e2013-12-17 17:50:11 -080075 //private Map<Path, Long> pendingFlows;
76 // TODO it seems there is a Guava collection that will time out entries.
77 // We should see if this will work here.
78 private Map<Path, PushedFlow> pendingFlows;
Jonathan Hartd857ad62013-12-14 18:08:17 -080079 private ListMultimap<Long, PacketToPush> waitingPackets;
Jonathan Hart5e448782013-12-10 12:36:35 -080080
Jonathan Hart7e6df362013-12-10 23:33:59 -080081 private final Object lock = new Object();
82
Jonathan Harte789d6e2013-12-17 17:50:11 -080083 private class PacketToPush {
Jonathan Hart5e448782013-12-10 12:36:35 -080084 public final OFPacketOut packet;
85 public final long dpid;
Jonathan Hart1caaa932013-11-04 15:28:28 -080086
Jonathan Hart5e448782013-12-10 12:36:35 -080087 public PacketToPush(OFPacketOut packet, long dpid) {
88 this.packet = packet;
89 this.dpid = dpid;
90 }
91 }
92
Jonathan Harte789d6e2013-12-17 17:50:11 -080093 private class PushedFlow {
94 public final long flowId;
95 private final long pushedTime;
96 public short firstHopOutPort = OFPort.OFPP_NONE.getValue();
97
98 public PushedFlow(long flowId) {
99 this.flowId = flowId;
100 pushedTime = System.currentTimeMillis();
101 }
102
103 public boolean isExpired() {
104 return (System.currentTimeMillis() - pushedTime) > PATH_PUSHED_TIMEOUT;
105 }
106 }
107
108 private final class Path {
Jonathan Hart5e448782013-12-10 12:36:35 -0800109 public final SwitchPort srcPort;
110 public final SwitchPort dstPort;
Jonathan Harte789d6e2013-12-17 17:50:11 -0800111 public final MACAddress srcMac;
112 public final MACAddress dstMac;
Jonathan Hart5e448782013-12-10 12:36:35 -0800113
Jonathan Harte789d6e2013-12-17 17:50:11 -0800114 public Path(SwitchPort src, SwitchPort dst,
115 MACAddress srcMac, MACAddress dstMac) {
Jonathan Hart5e448782013-12-10 12:36:35 -0800116 srcPort = new SwitchPort(new Dpid(src.dpid().value()),
117 new Port(src.port().value()));
118 dstPort = new SwitchPort(new Dpid(dst.dpid().value()),
119 new Port(dst.port().value()));
Jonathan Harte789d6e2013-12-17 17:50:11 -0800120 this.srcMac = srcMac;
121 this.dstMac = dstMac;
Jonathan Hart5e448782013-12-10 12:36:35 -0800122 }
123
124 @Override
125 public boolean equals(Object other) {
126 if (!(other instanceof Path)) {
127 return false;
128 }
129
130 Path otherPath = (Path) other;
131 return srcPort.equals(otherPath.srcPort) &&
Jonathan Harte789d6e2013-12-17 17:50:11 -0800132 dstPort.equals(otherPath.dstPort) &&
133 srcMac.equals(otherPath.srcMac) &&
134 dstMac.equals(otherPath.dstMac);
Jonathan Hart5e448782013-12-10 12:36:35 -0800135 }
136
137 @Override
138 public int hashCode() {
139 int hash = 17;
140 hash = 31 * hash + srcPort.hashCode();
141 hash = 31 * hash + dstPort.hashCode();
Jonathan Harte789d6e2013-12-17 17:50:11 -0800142 hash = 31 * hash + srcMac.hashCode();
143 hash = 31 * hash + dstMac.hashCode();
Jonathan Hart5e448782013-12-10 12:36:35 -0800144 return hash;
145 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800146
147 @Override
148 public String toString() {
149 return "(" + srcMac + " at " + srcPort + ") => ("
150 + dstPort + " at " + dstMac + ")";
151 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800152 }
153
Jonathan Harte93aed42013-12-05 18:39:50 -0800154 @Override
155 public Collection<Class<? extends IFloodlightService>> getModuleServices() {
Jonathan Hartd33a6cf2013-12-10 14:29:08 -0800156 List<Class<? extends IFloodlightService>> services =
157 new ArrayList<Class<? extends IFloodlightService>>(1);
158 services.add(IForwardingService.class);
159 return services;
Jonathan Harte93aed42013-12-05 18:39:50 -0800160 }
161
162 @Override
163 public Map<Class<? extends IFloodlightService>, IFloodlightService> getServiceImpls() {
Jonathan Hartd33a6cf2013-12-10 14:29:08 -0800164 Map<Class<? extends IFloodlightService>, IFloodlightService> impls =
165 new HashMap<Class<? extends IFloodlightService>, IFloodlightService>(1);
166 impls.put(IForwardingService.class, this);
167 return impls;
Jonathan Harte93aed42013-12-05 18:39:50 -0800168 }
169
170 @Override
171 public Collection<Class<? extends IFloodlightService>> getModuleDependencies() {
172 List<Class<? extends IFloodlightService>> dependencies =
173 new ArrayList<Class<? extends IFloodlightService>>();
174 dependencies.add(IFloodlightProviderService.class);
175 dependencies.add(IFlowService.class);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800176 dependencies.add(IFlowPusherService.class);
Jonathan Hartd857ad62013-12-14 18:08:17 -0800177 dependencies.add(IOnosDeviceService.class);
Jonathan Harte93aed42013-12-05 18:39:50 -0800178 return dependencies;
179 }
180
181 @Override
182 public void init(FloodlightModuleContext context) {
Jonathan Hart7e6df362013-12-10 23:33:59 -0800183 floodlightProvider =
Jonathan Harte93aed42013-12-05 18:39:50 -0800184 context.getServiceImpl(IFloodlightProviderService.class);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800185 flowService = context.getServiceImpl(IFlowService.class);
186 flowPusher = context.getServiceImpl(IFlowPusherService.class);
Jonathan Hart17672992013-12-12 16:15:16 -0800187 datagrid = context.getServiceImpl(IDatagridService.class);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800188
189 floodlightProvider.addOFMessageListener(OFType.PACKET_IN, this);
190
Jonathan Harte789d6e2013-12-17 17:50:11 -0800191 //pendingFlows = new ConcurrentHashMap<Path, Long>();
192 pendingFlows = new HashMap<Path, PushedFlow>();
Jonathan Hart7e6df362013-12-10 23:33:59 -0800193 //waitingPackets = Multimaps.synchronizedSetMultimap(
194 //HashMultimap.<Long, PacketToPush>create());
Jonathan Hartd857ad62013-12-14 18:08:17 -0800195 //waitingPackets = HashMultimap.create();
196 waitingPackets = LinkedListMultimap.create();
Jonathan Hart5e448782013-12-10 12:36:35 -0800197
Jonathan Hart1caaa932013-11-04 15:28:28 -0800198 deviceStorage = new DeviceStorageImpl();
199 deviceStorage.init("");
200 topologyService = new TopologyManager();
201 topologyService.init("");
202 }
203
Jonathan Harte93aed42013-12-05 18:39:50 -0800204 @Override
205 public void startUp(FloodlightModuleContext context) {
Jonathan Hart1caaa932013-11-04 15:28:28 -0800206 // no-op
207 }
208
209 @Override
210 public String getName() {
211 return "onosforwarding";
212 }
213
214 @Override
215 public boolean isCallbackOrderingPrereq(OFType type, String name) {
216 return (type == OFType.PACKET_IN) &&
Jonathan Hartd857ad62013-12-14 18:08:17 -0800217 (name.equals("devicemanager") || name.equals("proxyarpmanager")
218 || name.equals("onosdevicemanager"));
Jonathan Hart1caaa932013-11-04 15:28:28 -0800219 }
220
221 @Override
222 public boolean isCallbackOrderingPostreq(OFType type, String name) {
223 return false;
224 }
225
226 @Override
227 public Command receive(
228 IOFSwitch sw, OFMessage msg, FloodlightContext cntx) {
229
230 if (msg.getType() != OFType.PACKET_IN) {
231 return Command.CONTINUE;
232 }
233
234 OFPacketIn pi = (OFPacketIn) msg;
235
236 Ethernet eth = IFloodlightProviderService.bcStore.
237 get(cntx, IFloodlightProviderService.CONTEXT_PI_PAYLOAD);
238
Jonathan Hart17672992013-12-12 16:15:16 -0800239 if (eth.getEtherType() != Ethernet.TYPE_IPv4) {
Jonathan Hart1caaa932013-11-04 15:28:28 -0800240 return Command.CONTINUE;
241 }
242
Jonathan Hart17672992013-12-12 16:15:16 -0800243 if (eth.isBroadcast() || eth.isMulticast()) {
244 handleBroadcast(sw, pi, eth);
245 //return Command.CONTINUE;
246 }
247 else {
248 // Unicast
249 handlePacketIn(sw, pi, eth);
250 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800251
252 return Command.STOP;
253 }
254
Jonathan Hart17672992013-12-12 16:15:16 -0800255 private void handleBroadcast(IOFSwitch sw, OFPacketIn pi, Ethernet eth) {
256 if (log.isTraceEnabled()) {
257 log.trace("Sending broadcast packet to other ONOS instances");
258 }
259
260 IPv4 ipv4Packet = (IPv4) eth.getPayload();
261
262 // TODO We'll put the destination address here, because the current
263 // architecture needs an address. Addresses are only used for replies
264 // however, which don't apply to non-ARP packets. The ArpMessage class
265 // has become a bit too overloaded and should be refactored to
266 // handle all use cases nicely.
267 InetAddress targetAddress =
268 InetAddresses.fromInteger(ipv4Packet.getDestinationAddress());
269
270 // Piggy-back on the ARP mechanism to broadcast this packet out the
271 // edge. Luckily the ARP module doesn't check that the packet is
272 // actually ARP before broadcasting, so we can trick it into sending
273 // our non-ARP packets.
274 // TODO This should be refactored later to account for the new use case.
Jonathan Hartf9bd98d2013-12-13 11:40:55 -0800275 datagrid.sendArpRequest(ArpMessage.newRequest(targetAddress, eth.serialize(),
Jonathan Hartd857ad62013-12-14 18:08:17 -0800276 -1L, (short)-1, sw.getId(), pi.getInPort()));
Jonathan Hart17672992013-12-12 16:15:16 -0800277 }
278
Jonathan Hart1caaa932013-11-04 15:28:28 -0800279 private void handlePacketIn(IOFSwitch sw, OFPacketIn pi, Ethernet eth) {
Jonathan Hart5e448782013-12-10 12:36:35 -0800280 String destinationMac =
281 HexString.toHexString(eth.getDestinationMACAddress());
Jonathan Hart1caaa932013-11-04 15:28:28 -0800282
283 IDeviceObject deviceObject = deviceStorage.getDeviceByMac(
284 destinationMac);
285
286 if (deviceObject == null) {
Jonathan Hart18ad9502013-12-15 18:28:00 -0800287 log.debug("No device entry found for {} - broadcasting packet",
288 destinationMac);
289 handleBroadcast(sw, pi, eth);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800290 return;
291 }
292
Jonathan Hart5e448782013-12-10 12:36:35 -0800293 Iterator<IPortObject> ports = deviceObject.getAttachedPorts().iterator();
Jonathan Hart1caaa932013-11-04 15:28:28 -0800294 if (!ports.hasNext()) {
Jonathan Hart18ad9502013-12-15 18:28:00 -0800295 log.debug("No attachment point found for device {} - broadcasting packet",
296 destinationMac);
297 handleBroadcast(sw, pi, eth);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800298 return;
299 }
300 IPortObject portObject = ports.next();
301 short destinationPort = portObject.getNumber();
302 ISwitchObject switchObject = portObject.getSwitch();
303 long destinationDpid = HexString.toLong(switchObject.getDPID());
304
Jonathan Hart41d1e912013-11-24 16:50:25 -0800305 // TODO SwitchPort, Dpid and Port should probably be immutable
306 // (also, are Dpid and Port are even necessary?)
Jonathan Hart1caaa932013-11-04 15:28:28 -0800307 SwitchPort srcSwitchPort = new SwitchPort(
308 new Dpid(sw.getId()), new Port(pi.getInPort()));
309 SwitchPort dstSwitchPort = new SwitchPort(
310 new Dpid(destinationDpid), new Port(destinationPort));
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800311
312 MACAddress srcMacAddress = MACAddress.valueOf(eth.getSourceMACAddress());
313 MACAddress dstMacAddress = MACAddress.valueOf(eth.getDestinationMACAddress());
Jonathan Hart1caaa932013-11-04 15:28:28 -0800314
Jonathan Hart5e448782013-12-10 12:36:35 -0800315
Jonathan Hart7e6df362013-12-10 23:33:59 -0800316 FlowPath flowPath, reverseFlowPath;
Jonathan Hart5e448782013-12-10 12:36:35 -0800317
Jonathan Harte789d6e2013-12-17 17:50:11 -0800318 Path pathspec = new Path(srcSwitchPort, dstSwitchPort,
319 srcMacAddress, dstMacAddress);
Jonathan Hart5e448782013-12-10 12:36:35 -0800320 // TODO check concurrency
Jonathan Hart7e6df362013-12-10 23:33:59 -0800321 synchronized (lock) {
Jonathan Harte789d6e2013-12-17 17:50:11 -0800322 PushedFlow existingFlow = pendingFlows.get(pathspec);
323 //Long existingFlowId = pendingFlows.get(pathspec);
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800324
Jonathan Harte789d6e2013-12-17 17:50:11 -0800325 if (existingFlow != null && !existingFlow.isExpired()) {
Jonathan Hart7e6df362013-12-10 23:33:59 -0800326 log.debug("Found existing flow {}",
Jonathan Harte789d6e2013-12-17 17:50:11 -0800327 HexString.toHexString(existingFlow.flowId));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800328
329 OFPacketOut po = constructPacketOut(pi, sw);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800330
331 if (existingFlow.firstHopOutPort != OFPort.OFPP_NONE.getValue()) {
332 // Flow has been sent to the switches so it is safe to
333 // send a packet out now
334 sendPacketOut(sw, po, existingFlow.firstHopOutPort);
335 }
336 else {
337 // Flow has not yet been sent to switches so save the
338 // packet out for later
339 waitingPackets.put(existingFlow.flowId,
340 new PacketToPush(po, sw.getId()));
341 }
Jonathan Hart7e6df362013-12-10 23:33:59 -0800342 return;
343 }
344
Jonathan Harte789d6e2013-12-17 17:50:11 -0800345 //log.debug("Couldn't match {} in {}", pathspec, pendingFlows);
346
Jonathan Hart7e6df362013-12-10 23:33:59 -0800347 log.debug("Adding new flow between {} at {} and {} at {}",
348 new Object[]{srcMacAddress, srcSwitchPort, dstMacAddress, dstSwitchPort});
349
350
351 CallerId callerId = new CallerId("Forwarding");
352
353 DataPath datapath = new DataPath();
354 datapath.setSrcPort(srcSwitchPort);
355 datapath.setDstPort(dstSwitchPort);
356
357 flowPath = new FlowPath();
358 flowPath.setInstallerId(callerId);
359
360 flowPath.setFlowPathType(FlowPathType.FP_TYPE_SHORTEST_PATH);
361 flowPath.setFlowPathUserState(FlowPathUserState.FP_USER_ADD);
362 flowPath.setFlowEntryMatch(new FlowEntryMatch());
363 flowPath.setIdleTimeout(IDLE_TIMEOUT);
364 flowPath.setHardTimeout(HARD_TIMEOUT);
365 flowPath.flowEntryMatch().enableSrcMac(srcMacAddress);
366 flowPath.flowEntryMatch().enableDstMac(dstMacAddress);
367 flowPath.flowEntryMatch().enableEthernetFrameType(Ethernet.TYPE_IPv4);
368 flowPath.setDataPath(datapath);
369
370
371 DataPath reverseDataPath = new DataPath();
372 // Reverse the ports for the reverse path
373 reverseDataPath.setSrcPort(dstSwitchPort);
374 reverseDataPath.setDstPort(srcSwitchPort);
375
376 // TODO implement copy constructor for FlowPath
377 reverseFlowPath = new FlowPath();
378 reverseFlowPath.setInstallerId(callerId);
379 reverseFlowPath.setFlowPathType(FlowPathType.FP_TYPE_SHORTEST_PATH);
380 reverseFlowPath.setFlowPathUserState(FlowPathUserState.FP_USER_ADD);
381 reverseFlowPath.setIdleTimeout(IDLE_TIMEOUT);
382 reverseFlowPath.setHardTimeout(HARD_TIMEOUT);
383 reverseFlowPath.setFlowEntryMatch(new FlowEntryMatch());
384 // Reverse the MAC addresses for the reverse path
385 reverseFlowPath.flowEntryMatch().enableSrcMac(dstMacAddress);
386 reverseFlowPath.flowEntryMatch().enableDstMac(srcMacAddress);
387 reverseFlowPath.flowEntryMatch().enableEthernetFrameType(Ethernet.TYPE_IPv4);
388 reverseFlowPath.setDataPath(reverseDataPath);
389 reverseFlowPath.dataPath().srcPort().dpid().toString();
390
391 // TODO what happens if no path exists? cleanup
392
393 FlowId flowId = new FlowId(flowService.getNextFlowEntryId());
394 FlowId reverseFlowId = new FlowId(flowService.getNextFlowEntryId());
395
396 flowPath.setFlowId(flowId);
397 reverseFlowPath.setFlowId(reverseFlowId);
398
399 OFPacketOut po = constructPacketOut(pi, sw);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800400 Path reversePathSpec = new Path(dstSwitchPort, srcSwitchPort,
401 dstMacAddress, srcMacAddress);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800402
403 // Add to waiting lists
Jonathan Harte789d6e2013-12-17 17:50:11 -0800404 //pendingFlows.put(pathspec, flowId.value());
405 //pendingFlows.put(reversePathSpec, reverseFlowId.value());
406 pendingFlows.put(pathspec, new PushedFlow(flowId.value()));
407 pendingFlows.put(reversePathSpec, new PushedFlow(reverseFlowId.value()));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800408 waitingPackets.put(flowId.value(), new PacketToPush(po, sw.getId()));
409
Jonathan Hart41d1e912013-11-24 16:50:25 -0800410 }
411
Jonathan Hart5e448782013-12-10 12:36:35 -0800412 flowService.addFlow(reverseFlowPath);
413 flowService.addFlow(flowPath);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800414
Jonathan Hart1caaa932013-11-04 15:28:28 -0800415 }
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800416
Jonathan Hart5e448782013-12-10 12:36:35 -0800417 /*
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800418 private boolean flowExists(SwitchPort srcPort, MACAddress srcMac,
419 SwitchPort dstPort, MACAddress dstMac) {
420 for (FlowPath flow : datagridService.getAllFlows()) {
421 FlowEntryMatch match = flow.flowEntryMatch();
422 // TODO implement FlowEntryMatch.equals();
423 // This is painful to do properly without support in the FlowEntryMatch
424 boolean same = true;
425 if (!match.srcMac().equals(srcMac) ||
426 !match.dstMac().equals(dstMac)) {
427 same = false;
428 }
429 if (!flow.dataPath().srcPort().equals(srcPort) ||
430 !flow.dataPath().dstPort().equals(dstPort)) {
431 same = false;
432 }
433
434 if (same) {
435 log.debug("found flow entry that's the same {}-{}:::{}-{}",
436 new Object[] {srcPort, srcMac, dstPort, dstMac});
437 return true;
438 }
439 }
440
441 return false;
442 }
Jonathan Hart5e448782013-12-10 12:36:35 -0800443 */
Jonathan Hart1caaa932013-11-04 15:28:28 -0800444
Jonathan Hart7e6df362013-12-10 23:33:59 -0800445 private OFPacketOut constructPacketOut(OFPacketIn pi, IOFSwitch sw) {
Jonathan Hart41d1e912013-11-24 16:50:25 -0800446 OFPacketOut po = new OFPacketOut();
447 po.setInPort(OFPort.OFPP_NONE)
448 .setInPort(pi.getInPort())
Jonathan Hart5e448782013-12-10 12:36:35 -0800449 .setActions(new ArrayList<OFAction>())
450 .setLengthU(OFPacketOut.MINIMUM_LENGTH);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800451
452 if (sw.getBuffers() == 0) {
453 po.setBufferId(OFPacketOut.BUFFER_ID_NONE)
454 .setPacketData(pi.getPacketData())
455 .setLengthU(po.getLengthU() + po.getPacketData().length);
456 }
457 else {
458 po.setBufferId(pi.getBufferId());
459 }
460
Jonathan Hart5e448782013-12-10 12:36:35 -0800461 return po;
462 }
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800463
Jonathan Hart5e448782013-12-10 12:36:35 -0800464 @Override
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800465 public void flowsInstalled(Collection<FlowPath> installedFlowPaths) {
466 for (FlowPath flowPath : installedFlowPaths) {
467 flowInstalled(flowPath);
468 }
469 }
470
471 private void flowInstalled(FlowPath installedFlowPath) {
Jonathan Hart5e448782013-12-10 12:36:35 -0800472 long flowId = installedFlowPath.flowId().value();
Jonathan Hart5e448782013-12-10 12:36:35 -0800473
Jonathan Harte789d6e2013-12-17 17:50:11 -0800474 short outPort =
475 installedFlowPath.flowEntries().get(0).outPort().value();
476
477 MACAddress srcMacAddress = installedFlowPath.flowEntryMatch().srcMac();
478 MACAddress dstMacAddress = installedFlowPath.flowEntryMatch().dstMac();
479
Jonathan Hart7e6df362013-12-10 23:33:59 -0800480 Collection<PacketToPush> packets;
481 synchronized (lock) {
Jonathan Harte789d6e2013-12-17 17:50:11 -0800482 log.debug("Flow {} has been installed, sending queued packets",
483 installedFlowPath.flowId());
484
Jonathan Hart7e6df362013-12-10 23:33:59 -0800485 packets = waitingPackets.removeAll(flowId);
486
Jonathan Harte789d6e2013-12-17 17:50:11 -0800487 // remove pending flows entry
488 Path installedPath = new Path(installedFlowPath.dataPath().srcPort(),
489 installedFlowPath.dataPath().dstPort(),
490 srcMacAddress, dstMacAddress);
491 //pendingFlows.remove(pathToRemove);
492 pendingFlows.get(installedPath).firstHopOutPort = outPort;
Jonathan Hart7e6df362013-12-10 23:33:59 -0800493 }
Jonathan Hart5e448782013-12-10 12:36:35 -0800494
495 for (PacketToPush packet : packets) {
496 IOFSwitch sw = floodlightProvider.getSwitches().get(packet.dpid);
497
Jonathan Harte789d6e2013-12-17 17:50:11 -0800498 sendPacketOut(sw, packet.packet, outPort);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800499 }
500 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800501
502 private void sendPacketOut(IOFSwitch sw, OFPacketOut po, short outPort) {
503 po.getActions().add(new OFActionOutput(outPort));
504 po.setActionsLength((short)
505 (po.getActionsLength() + OFActionOutput.MINIMUM_LENGTH));
506 po.setLengthU(po.getLengthU() + OFActionOutput.MINIMUM_LENGTH);
507
508 flowPusher.add(sw, po);
509 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800510}