blob: 3778632aed51b83aa7efa4fa9164d66144821962 [file] [log] [blame]
Jonathan Hart1caaa932013-11-04 15:28:28 -08001package net.onrc.onos.ofcontroller.forwarding;
2
Jonathan Hart41d1e912013-11-24 16:50:25 -08003import java.util.ArrayList;
Jonathan Harte93aed42013-12-05 18:39:50 -08004import 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 Hart41d1e912013-11-24 16:50:25 -08007import java.util.List;
Jonathan Harte93aed42013-12-05 18:39:50 -08008import java.util.Map;
TeruU417fe022014-02-04 12:59:30 -08009import java.util.concurrent.Executors;
10import java.util.concurrent.ScheduledExecutorService;
11import java.util.concurrent.TimeUnit;
Jonathan Hart1caaa932013-11-04 15:28:28 -080012
13import net.floodlightcontroller.core.FloodlightContext;
14import net.floodlightcontroller.core.IFloodlightProviderService;
15import net.floodlightcontroller.core.IOFMessageListener;
16import net.floodlightcontroller.core.IOFSwitch;
Jonathan Harte93aed42013-12-05 18:39:50 -080017import net.floodlightcontroller.core.module.FloodlightModuleContext;
18import net.floodlightcontroller.core.module.IFloodlightModule;
19import net.floodlightcontroller.core.module.IFloodlightService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080020import net.floodlightcontroller.packet.Ethernet;
21import net.floodlightcontroller.util.MACAddress;
Jonathan Hartdc3ad702013-11-14 11:34:59 -080022import net.onrc.onos.datagrid.IDatagridService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080023import net.onrc.onos.ofcontroller.core.IDeviceStorage;
24import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.IDeviceObject;
25import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.IPortObject;
26import net.onrc.onos.ofcontroller.core.INetMapTopologyObjects.ISwitchObject;
27import net.onrc.onos.ofcontroller.core.internal.DeviceStorageImpl;
Jonathan Hartd857ad62013-12-14 18:08:17 -080028import net.onrc.onos.ofcontroller.devicemanager.IOnosDeviceService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080029import net.onrc.onos.ofcontroller.flowmanager.IFlowService;
Jonathan Hart7e6df362013-12-10 23:33:59 -080030import net.onrc.onos.ofcontroller.flowprogrammer.IFlowPusherService;
Jonathan Hart7804bea2014-01-07 10:50:52 -080031import net.onrc.onos.ofcontroller.proxyarp.BroadcastPacketOutNotification;
Jonathan Hart0444d932014-01-22 15:06:17 -080032import net.onrc.onos.ofcontroller.proxyarp.IProxyArpService;
Jonathan Hart1caaa932013-11-04 15:28:28 -080033import net.onrc.onos.ofcontroller.topology.TopologyManager;
34import net.onrc.onos.ofcontroller.util.CallerId;
35import net.onrc.onos.ofcontroller.util.DataPath;
36import net.onrc.onos.ofcontroller.util.Dpid;
Jonathan Hart0444d932014-01-22 15:06:17 -080037import net.onrc.onos.ofcontroller.util.FlowEntry;
Jonathan Hart4fb16d82013-11-07 22:41:32 -080038import net.onrc.onos.ofcontroller.util.FlowEntryMatch;
Jonathan Hart1caaa932013-11-04 15:28:28 -080039import net.onrc.onos.ofcontroller.util.FlowId;
40import net.onrc.onos.ofcontroller.util.FlowPath;
41import net.onrc.onos.ofcontroller.util.FlowPathType;
42import net.onrc.onos.ofcontroller.util.FlowPathUserState;
43import net.onrc.onos.ofcontroller.util.Port;
44import net.onrc.onos.ofcontroller.util.SwitchPort;
45
46import org.openflow.protocol.OFMessage;
47import org.openflow.protocol.OFPacketIn;
Jonathan Hart41d1e912013-11-24 16:50:25 -080048import org.openflow.protocol.OFPacketOut;
49import org.openflow.protocol.OFPort;
Jonathan Hart1caaa932013-11-04 15:28:28 -080050import org.openflow.protocol.OFType;
Jonathan Hart41d1e912013-11-24 16:50:25 -080051import org.openflow.protocol.action.OFAction;
52import org.openflow.protocol.action.OFActionOutput;
Jonathan Hart1caaa932013-11-04 15:28:28 -080053import org.openflow.util.HexString;
54import org.slf4j.Logger;
55import org.slf4j.LoggerFactory;
56
Jonathan Hartd857ad62013-12-14 18:08:17 -080057import com.google.common.collect.LinkedListMultimap;
58import com.google.common.collect.ListMultimap;
Jonathan Hart5e448782013-12-10 12:36:35 -080059
60public class Forwarding implements IOFMessageListener, IFloodlightModule,
61 IForwardingService {
Jonathan Hart1caaa932013-11-04 15:28:28 -080062 private final static Logger log = LoggerFactory.getLogger(Forwarding.class);
TeruU417fe022014-02-04 12:59:30 -080063
Jonathan Hart7e6df362013-12-10 23:33:59 -080064 private final int IDLE_TIMEOUT = 5; // seconds
65 private final int HARD_TIMEOUT = 0; // seconds
TeruU417fe022014-02-04 12:59:30 -080066 private final int SLEEP_TIME_FOR_DB_DEVICE_INSTALLED = 100; // milliseconds
67 private final static int NUMBER_OF_THREAD_FOR_EXECUTOR = 1;
68
69 private final static ScheduledExecutorService executor = Executors.newScheduledThreadPool(NUMBER_OF_THREAD_FOR_EXECUTOR);
Jonathan Hart0444d932014-01-22 15:06:17 -080070
71 private final CallerId callerId = new CallerId("Forwarding");
Jonathan Hart7e6df362013-12-10 23:33:59 -080072
Jonathan Hart1caaa932013-11-04 15:28:28 -080073 private IFloodlightProviderService floodlightProvider;
74 private IFlowService flowService;
Jonathan Hart7e6df362013-12-10 23:33:59 -080075 private IFlowPusherService flowPusher;
Jonathan Hart17672992013-12-12 16:15:16 -080076 private IDatagridService datagrid;
Jonathan Hart1caaa932013-11-04 15:28:28 -080077
78 private IDeviceStorage deviceStorage;
79 private TopologyManager topologyService;
80
Jonathan Harte789d6e2013-12-17 17:50:11 -080081 // TODO it seems there is a Guava collection that will time out entries.
82 // We should see if this will work here.
83 private Map<Path, PushedFlow> pendingFlows;
Jonathan Hartd857ad62013-12-14 18:08:17 -080084 private ListMultimap<Long, PacketToPush> waitingPackets;
Jonathan Hart5e448782013-12-10 12:36:35 -080085
Jonathan Hart7e6df362013-12-10 23:33:59 -080086 private final Object lock = new Object();
87
Jonathan Harte789d6e2013-12-17 17:50:11 -080088 private class PacketToPush {
Jonathan Hart5e448782013-12-10 12:36:35 -080089 public final OFPacketOut packet;
90 public final long dpid;
Jonathan Hart1caaa932013-11-04 15:28:28 -080091
Jonathan Hart5e448782013-12-10 12:36:35 -080092 public PacketToPush(OFPacketOut packet, long dpid) {
93 this.packet = packet;
94 this.dpid = dpid;
95 }
Jonathan Hart1caaa932013-11-04 15:28:28 -080096 }
97
Jonathan Harte789d6e2013-12-17 17:50:11 -080098 private class PushedFlow {
99 public final long flowId;
Jonathan Hart0444d932014-01-22 15:06:17 -0800100 public boolean installed = false;
TeruU417fe022014-02-04 12:59:30 -0800101 public short firstOutPort;
Jonathan Harte789d6e2013-12-17 17:50:11 -0800102
103 public PushedFlow(long flowId) {
104 this.flowId = flowId;
Jonathan Harte789d6e2013-12-17 17:50:11 -0800105 }
106 }
107
108 private final class Path {
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 Hart0444d932014-01-22 15:06:17 -0800112 public Path(MACAddress srcMac, MACAddress dstMac) {
Jonathan Harte789d6e2013-12-17 17:50:11 -0800113 this.srcMac = srcMac;
114 this.dstMac = dstMac;
Jonathan Hart5e448782013-12-10 12:36:35 -0800115 }
116
117 @Override
118 public boolean equals(Object other) {
119 if (!(other instanceof Path)) {
120 return false;
121 }
122
123 Path otherPath = (Path) other;
Jonathan Hart0444d932014-01-22 15:06:17 -0800124 return srcMac.equals(otherPath.srcMac) &&
Jonathan Harte789d6e2013-12-17 17:50:11 -0800125 dstMac.equals(otherPath.dstMac);
Jonathan Hart5e448782013-12-10 12:36:35 -0800126 }
127
128 @Override
129 public int hashCode() {
130 int hash = 17;
Jonathan Harte789d6e2013-12-17 17:50:11 -0800131 hash = 31 * hash + srcMac.hashCode();
132 hash = 31 * hash + dstMac.hashCode();
Jonathan Hart5e448782013-12-10 12:36:35 -0800133 return hash;
134 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800135
136 @Override
137 public String toString() {
Jonathan Hart0444d932014-01-22 15:06:17 -0800138 return "(" + srcMac + ") => (" + dstMac + ")";
Jonathan Harte789d6e2013-12-17 17:50:11 -0800139 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800140 }
141
Jonathan Harte93aed42013-12-05 18:39:50 -0800142 @Override
143 public Collection<Class<? extends IFloodlightService>> getModuleServices() {
Jonathan Hartd33a6cf2013-12-10 14:29:08 -0800144 List<Class<? extends IFloodlightService>> services =
145 new ArrayList<Class<? extends IFloodlightService>>(1);
146 services.add(IForwardingService.class);
147 return services;
Jonathan Harte93aed42013-12-05 18:39:50 -0800148 }
149
150 @Override
151 public Map<Class<? extends IFloodlightService>, IFloodlightService> getServiceImpls() {
Jonathan Hartd33a6cf2013-12-10 14:29:08 -0800152 Map<Class<? extends IFloodlightService>, IFloodlightService> impls =
153 new HashMap<Class<? extends IFloodlightService>, IFloodlightService>(1);
154 impls.put(IForwardingService.class, this);
155 return impls;
Jonathan Harte93aed42013-12-05 18:39:50 -0800156 }
157
158 @Override
159 public Collection<Class<? extends IFloodlightService>> getModuleDependencies() {
160 List<Class<? extends IFloodlightService>> dependencies =
161 new ArrayList<Class<? extends IFloodlightService>>();
162 dependencies.add(IFloodlightProviderService.class);
163 dependencies.add(IFlowService.class);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800164 dependencies.add(IFlowPusherService.class);
Jonathan Hartd857ad62013-12-14 18:08:17 -0800165 dependencies.add(IOnosDeviceService.class);
Jonathan Hart7804bea2014-01-07 10:50:52 -0800166 // We don't use the IProxyArpService directly, but reactive forwarding
167 // requires it to be loaded and answering ARP requests
168 dependencies.add(IProxyArpService.class);
Jonathan Harte93aed42013-12-05 18:39:50 -0800169 return dependencies;
170 }
171
172 @Override
173 public void init(FloodlightModuleContext context) {
Jonathan Hart7e6df362013-12-10 23:33:59 -0800174 floodlightProvider =
Jonathan Harte93aed42013-12-05 18:39:50 -0800175 context.getServiceImpl(IFloodlightProviderService.class);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800176 flowService = context.getServiceImpl(IFlowService.class);
177 flowPusher = context.getServiceImpl(IFlowPusherService.class);
Jonathan Hart17672992013-12-12 16:15:16 -0800178 datagrid = context.getServiceImpl(IDatagridService.class);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800179
180 floodlightProvider.addOFMessageListener(OFType.PACKET_IN, this);
Jonathan Hart0444d932014-01-22 15:06:17 -0800181
Jonathan Harte789d6e2013-12-17 17:50:11 -0800182 pendingFlows = new HashMap<Path, PushedFlow>();
Jonathan Hartd857ad62013-12-14 18:08:17 -0800183 waitingPackets = LinkedListMultimap.create();
Jonathan Hart5e448782013-12-10 12:36:35 -0800184
Jonathan Hart1caaa932013-11-04 15:28:28 -0800185 deviceStorage = new DeviceStorageImpl();
yoshitomob292c622013-11-23 14:35:58 -0800186 deviceStorage.init("","");
Jonathan Hart1caaa932013-11-04 15:28:28 -0800187 topologyService = new TopologyManager();
yoshitomob292c622013-11-23 14:35:58 -0800188 topologyService.init("","");
Jonathan Hart1caaa932013-11-04 15:28:28 -0800189 }
190
Jonathan Harte93aed42013-12-05 18:39:50 -0800191 @Override
192 public void startUp(FloodlightModuleContext context) {
Jonathan Hart1caaa932013-11-04 15:28:28 -0800193 // no-op
194 }
195
196 @Override
197 public String getName() {
198 return "onosforwarding";
199 }
200
201 @Override
202 public boolean isCallbackOrderingPrereq(OFType type, String name) {
203 return (type == OFType.PACKET_IN) &&
Jonathan Hartd857ad62013-12-14 18:08:17 -0800204 (name.equals("devicemanager") || name.equals("proxyarpmanager")
205 || name.equals("onosdevicemanager"));
Jonathan Hart1caaa932013-11-04 15:28:28 -0800206 }
207
208 @Override
209 public boolean isCallbackOrderingPostreq(OFType type, String name) {
210 return false;
211 }
212
213 @Override
214 public Command receive(
215 IOFSwitch sw, OFMessage msg, FloodlightContext cntx) {
216
217 if (msg.getType() != OFType.PACKET_IN) {
218 return Command.CONTINUE;
219 }
220
221 OFPacketIn pi = (OFPacketIn) msg;
222
223 Ethernet eth = IFloodlightProviderService.bcStore.
224 get(cntx, IFloodlightProviderService.CONTEXT_PI_PAYLOAD);
225
Jonathan Hart17672992013-12-12 16:15:16 -0800226 if (eth.getEtherType() != Ethernet.TYPE_IPv4) {
Jonathan Hart1caaa932013-11-04 15:28:28 -0800227 return Command.CONTINUE;
228 }
229
Jonathan Hart17672992013-12-12 16:15:16 -0800230 if (eth.isBroadcast() || eth.isMulticast()) {
231 handleBroadcast(sw, pi, eth);
Jonathan Hart17672992013-12-12 16:15:16 -0800232 }
233 else {
234 // Unicast
235 handlePacketIn(sw, pi, eth);
236 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800237
238 return Command.STOP;
239 }
240
Jonathan Hart17672992013-12-12 16:15:16 -0800241 private void handleBroadcast(IOFSwitch sw, OFPacketIn pi, Ethernet eth) {
242 if (log.isTraceEnabled()) {
243 log.trace("Sending broadcast packet to other ONOS instances");
244 }
Jonathan Hart7804bea2014-01-07 10:50:52 -0800245
246 datagrid.sendPacketOutNotification(new BroadcastPacketOutNotification(
Naoki Shiota78e403c2014-02-20 17:13:36 -0800247 eth.serialize(), null, sw.getId(), pi.getInPort()));
Jonathan Hart17672992013-12-12 16:15:16 -0800248 }
249
TeruU417fe022014-02-04 12:59:30 -0800250 private void handlePacketIn(IOFSwitch sw, OFPacketIn pi, Ethernet eth){
251 log.debug("Start handlePacketIn swId {}, portId {}", sw.getId(), pi.getInPort());
252
Jonathan Hart5e448782013-12-10 12:36:35 -0800253 String destinationMac =
254 HexString.toHexString(eth.getDestinationMACAddress());
Jonathan Hart1caaa932013-11-04 15:28:28 -0800255
TeruUd1ba0e22014-02-10 11:44:15 -0800256 //FIXME getDeviceByMac() is a blocking call, so it may be better way to handle it to avoid the condition.
257 try{
258 IDeviceObject deviceObject = deviceStorage.getDeviceByMac(
Jonathan Hart1caaa932013-11-04 15:28:28 -0800259 destinationMac);
TeruUd1ba0e22014-02-10 11:44:15 -0800260
261 if (deviceObject == null) {
262 log.debug("No device entry found for {}",
263 destinationMac);
264
265 //Device is not in the DB, so wait it until the device is added.
266 executor.schedule(new WaitDeviceArp(sw, pi, eth), SLEEP_TIME_FOR_DB_DEVICE_INSTALLED, TimeUnit.MILLISECONDS);
267 return;
268 }
269
270 continueHandlePacketIn(sw, pi, eth, deviceObject);
271 } finally {
272 deviceStorage.rollback();
Jonathan Hart1caaa932013-11-04 15:28:28 -0800273 }
TeruU417fe022014-02-04 12:59:30 -0800274 }
275
276 private class WaitDeviceArp implements Runnable {
277 IOFSwitch sw;
278 OFPacketIn pi;
279 Ethernet eth;
280
281 public WaitDeviceArp(IOFSwitch sw, OFPacketIn pi, Ethernet eth) {
282 super();
283 this.sw = sw;
284 this.pi = pi;
285 this.eth = eth;
286 }
287
288 @Override
289 public void run() {
TeruUd1ba0e22014-02-10 11:44:15 -0800290 try {
TeruU417fe022014-02-04 12:59:30 -0800291 IDeviceObject deviceObject = deviceStorage.getDeviceByMac(HexString.toHexString(eth.getDestinationMACAddress()));
TeruUd1ba0e22014-02-10 11:44:15 -0800292 if(deviceObject == null){
293 log.debug("wait {}ms and device was not found. Send broadcast packet and the thread finish.", SLEEP_TIME_FOR_DB_DEVICE_INSTALLED);
294 handleBroadcast(sw, pi, eth);
295 return;
296 }
297 log.debug("wait {}ms and device {} was found, continue",SLEEP_TIME_FOR_DB_DEVICE_INSTALLED, deviceObject.getMACAddress());
298 continueHandlePacketIn(sw, pi, eth, deviceObject);
299 } finally {
300 deviceStorage.rollback();
301 }
TeruU417fe022014-02-04 12:59:30 -0800302 }
303 }
304
305 private void continueHandlePacketIn(IOFSwitch sw, OFPacketIn pi, Ethernet eth, IDeviceObject deviceObject) {
306 log.debug("Start continuehandlePacketIn");
307
Jonathan Hart5e448782013-12-10 12:36:35 -0800308 Iterator<IPortObject> ports = deviceObject.getAttachedPorts().iterator();
Jonathan Hart1caaa932013-11-04 15:28:28 -0800309 if (!ports.hasNext()) {
Jonathan Hart18ad9502013-12-15 18:28:00 -0800310 log.debug("No attachment point found for device {} - broadcasting packet",
TeruU417fe022014-02-04 12:59:30 -0800311 deviceObject.getMACAddress());
Jonathan Hart18ad9502013-12-15 18:28:00 -0800312 handleBroadcast(sw, pi, eth);
TeruU417fe022014-02-04 12:59:30 -0800313 return;
Jonathan Hart1caaa932013-11-04 15:28:28 -0800314 }
TeruU417fe022014-02-04 12:59:30 -0800315
316 //This code assumes the device has only one port. It should be problem.
Jonathan Hart1caaa932013-11-04 15:28:28 -0800317 IPortObject portObject = ports.next();
TeruU417fe022014-02-04 12:59:30 -0800318
Jonathan Hart1caaa932013-11-04 15:28:28 -0800319 short destinationPort = portObject.getNumber();
320 ISwitchObject switchObject = portObject.getSwitch();
321 long destinationDpid = HexString.toLong(switchObject.getDPID());
322
Jonathan Hart41d1e912013-11-24 16:50:25 -0800323 // TODO SwitchPort, Dpid and Port should probably be immutable
Jonathan Hart1caaa932013-11-04 15:28:28 -0800324 SwitchPort srcSwitchPort = new SwitchPort(
325 new Dpid(sw.getId()), new Port(pi.getInPort()));
326 SwitchPort dstSwitchPort = new SwitchPort(
327 new Dpid(destinationDpid), new Port(destinationPort));
Jonathan Hartdc3ad702013-11-14 11:34:59 -0800328
329 MACAddress srcMacAddress = MACAddress.valueOf(eth.getSourceMACAddress());
330 MACAddress dstMacAddress = MACAddress.valueOf(eth.getDestinationMACAddress());
Jonathan Hart1caaa932013-11-04 15:28:28 -0800331
Jonathan Hart7e6df362013-12-10 23:33:59 -0800332 FlowPath flowPath, reverseFlowPath;
Jonathan Hart5e448782013-12-10 12:36:35 -0800333
Jonathan Hart7e6df362013-12-10 23:33:59 -0800334 synchronized (lock) {
TeruU417fe022014-02-04 12:59:30 -0800335 //TODO check concurrency
336 Path pathspec = new Path(srcMacAddress, dstMacAddress);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800337 PushedFlow existingFlow = pendingFlows.get(pathspec);
Jonathan Hart0444d932014-01-22 15:06:17 -0800338
339 if (existingFlow != null) {
340 // We've already installed a flow for this pair of MAC addresses
TeruU417fe022014-02-04 12:59:30 -0800341 log.debug("Found existing same pathspec {}, Flow ID is {}",
342 pathspec, HexString.toHexString(existingFlow.flowId));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800343 OFPacketOut po = constructPacketOut(pi, sw);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800344
Jonathan Hart0444d932014-01-22 15:06:17 -0800345 // Find the correct port here. We just assume the PI is from
346 // the first hop switch, but this is definitely not always
347 // the case. We'll have to retrieve the flow from HZ every time
348 // because it could change (be rerouted) sometimes.
349 if (existingFlow.installed) {
Jonathan Harte789d6e2013-12-17 17:50:11 -0800350 // Flow has been sent to the switches so it is safe to
351 // send a packet out now
Jonathan Hart0444d932014-01-22 15:06:17 -0800352 FlowPath flow = datagrid.getFlow(new FlowId(existingFlow.flowId));
353 FlowEntry flowEntryForThisSwitch = null;
Jonathan Hart84198d32014-01-22 17:14:37 -0800354
355 if (flow != null) {
356 for (FlowEntry flowEntry : flow.flowEntries()) {
357 if (flowEntry.dpid().equals(new Dpid(sw.getId()))) {
358 flowEntryForThisSwitch = flowEntry;
359 break;
360 }
Jonathan Hart0444d932014-01-22 15:06:17 -0800361 }
362 }
363
364 if (flowEntryForThisSwitch == null) {
365 // If we don't find a flow entry for that switch, then we're
366 // in the middle of a rerouting (or something's gone wrong).
367 // This packet will be dropped as a victim of the rerouting.
368 log.debug("Dropping packet on flow {} between {}-{}, flow path {}",
369 new Object[] {new FlowId(existingFlow.flowId),
370 srcMacAddress, dstMacAddress, flow});
371 }
372 else {
TeruU417fe022014-02-04 12:59:30 -0800373 log.debug("Sending packet out from sw {}, outport{}", sw, flowEntryForThisSwitch.outPort().value());
Jonathan Hart0444d932014-01-22 15:06:17 -0800374 sendPacketOut(sw, po, flowEntryForThisSwitch.outPort().value());
375 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800376 }
377 else {
TeruU417fe022014-02-04 12:59:30 -0800378 //log.debug("Existing Flow ID {} is not installed. Continue to overwrite.",Long.toHexString(existingFlow.flowId) );
379 // Flow has not yet been installed to switches so save the
Jonathan Harte789d6e2013-12-17 17:50:11 -0800380 // packet out for later
TeruU417fe022014-02-04 12:59:30 -0800381 log.debug("Put a packet into the waitng list. flowId {}", Long.toHexString(existingFlow.flowId));
382 waitingPackets.put(existingFlow.flowId, new PacketToPush(po, sw.getId()));
Jonathan Harte789d6e2013-12-17 17:50:11 -0800383 }
Jonathan Hart41d1e912013-11-24 16:50:25 -0800384 return;
385 }
Jonathan Hart0444d932014-01-22 15:06:17 -0800386
Jonathan Hart7e6df362013-12-10 23:33:59 -0800387 log.debug("Adding new flow between {} at {} and {} at {}",
388 new Object[]{srcMacAddress, srcSwitchPort, dstMacAddress, dstSwitchPort});
Jonathan Hart48c2d312013-12-05 19:09:59 -0800389
Jonathan Hart7e6df362013-12-10 23:33:59 -0800390 DataPath datapath = new DataPath();
391 datapath.setSrcPort(srcSwitchPort);
392 datapath.setDstPort(dstSwitchPort);
393
394 flowPath = new FlowPath();
Jonathan Hart0444d932014-01-22 15:06:17 -0800395 flowPath.setInstallerId(new CallerId(callerId));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800396
397 flowPath.setFlowPathType(FlowPathType.FP_TYPE_SHORTEST_PATH);
398 flowPath.setFlowPathUserState(FlowPathUserState.FP_USER_ADD);
399 flowPath.setFlowEntryMatch(new FlowEntryMatch());
400 flowPath.setIdleTimeout(IDLE_TIMEOUT);
401 flowPath.setHardTimeout(HARD_TIMEOUT);
402 flowPath.flowEntryMatch().enableSrcMac(srcMacAddress);
403 flowPath.flowEntryMatch().enableDstMac(dstMacAddress);
404 flowPath.flowEntryMatch().enableEthernetFrameType(Ethernet.TYPE_IPv4);
405 flowPath.setDataPath(datapath);
406
407
408 DataPath reverseDataPath = new DataPath();
409 // Reverse the ports for the reverse path
410 reverseDataPath.setSrcPort(dstSwitchPort);
411 reverseDataPath.setDstPort(srcSwitchPort);
412
413 // TODO implement copy constructor for FlowPath
414 reverseFlowPath = new FlowPath();
Jonathan Hart0444d932014-01-22 15:06:17 -0800415 reverseFlowPath.setInstallerId(new CallerId(callerId));
Jonathan Hart7e6df362013-12-10 23:33:59 -0800416 reverseFlowPath.setFlowPathType(FlowPathType.FP_TYPE_SHORTEST_PATH);
417 reverseFlowPath.setFlowPathUserState(FlowPathUserState.FP_USER_ADD);
418 reverseFlowPath.setIdleTimeout(IDLE_TIMEOUT);
419 reverseFlowPath.setHardTimeout(HARD_TIMEOUT);
420 reverseFlowPath.setFlowEntryMatch(new FlowEntryMatch());
421 // Reverse the MAC addresses for the reverse path
422 reverseFlowPath.flowEntryMatch().enableSrcMac(dstMacAddress);
423 reverseFlowPath.flowEntryMatch().enableDstMac(srcMacAddress);
424 reverseFlowPath.flowEntryMatch().enableEthernetFrameType(Ethernet.TYPE_IPv4);
425 reverseFlowPath.setDataPath(reverseDataPath);
Jonathan Hart0444d932014-01-22 15:06:17 -0800426
Jonathan Hart7e6df362013-12-10 23:33:59 -0800427 FlowId flowId = new FlowId(flowService.getNextFlowEntryId());
428 FlowId reverseFlowId = new FlowId(flowService.getNextFlowEntryId());
429
430 flowPath.setFlowId(flowId);
431 reverseFlowPath.setFlowId(reverseFlowId);
432
433 OFPacketOut po = constructPacketOut(pi, sw);
Jonathan Hart0444d932014-01-22 15:06:17 -0800434 Path reversePathSpec = new Path(dstMacAddress, srcMacAddress);
Jonathan Hart7e6df362013-12-10 23:33:59 -0800435
436 // Add to waiting lists
Jonathan Harte789d6e2013-12-17 17:50:11 -0800437 pendingFlows.put(pathspec, new PushedFlow(flowId.value()));
TeruU417fe022014-02-04 12:59:30 -0800438 log.debug("Put a Path {} in the pending flow, Flow ID {}", pathspec, flowId);
Jonathan Harte789d6e2013-12-17 17:50:11 -0800439 pendingFlows.put(reversePathSpec, new PushedFlow(reverseFlowId.value()));
TeruU417fe022014-02-04 12:59:30 -0800440 log.debug("Put a Path {} in the pending flow, Flow ID {}", reversePathSpec, reverseFlowId);
441 PacketToPush pp = new PacketToPush(po, sw.getId());
442 waitingPackets.put(flowId.value(), pp);
443
444 log.debug("Put a Packet in the wating list. relatedflowId {}, realatedReversedFlowId {}",
445 flowId, reverseFlowId);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800446 }
Jonathan Hart41d1e912013-11-24 16:50:25 -0800447
TeruU417fe022014-02-04 12:59:30 -0800448 log.debug("Adding reverse {} to {}. Flow ID {}", new Object[] {
Jonathan Hart0444d932014-01-22 15:06:17 -0800449 dstMacAddress, srcMacAddress, reverseFlowPath.flowId()});
Jonathan Hart5e448782013-12-10 12:36:35 -0800450 flowService.addFlow(reverseFlowPath);
TeruU417fe022014-02-04 12:59:30 -0800451 log.debug("Adding forward {} to {}. Flow ID {}", new Object[] {
Jonathan Hart0444d932014-01-22 15:06:17 -0800452 srcMacAddress, dstMacAddress, flowPath.flowId()});
Jonathan Hart5e448782013-12-10 12:36:35 -0800453 flowService.addFlow(flowPath);
Jonathan Hart1caaa932013-11-04 15:28:28 -0800454 }
Jonathan Hart1caaa932013-11-04 15:28:28 -0800455
Jonathan Hart7e6df362013-12-10 23:33:59 -0800456 private OFPacketOut constructPacketOut(OFPacketIn pi, IOFSwitch sw) {
Jonathan Hart41d1e912013-11-24 16:50:25 -0800457 OFPacketOut po = new OFPacketOut();
458 po.setInPort(OFPort.OFPP_NONE)
459 .setInPort(pi.getInPort())
Jonathan Hart5e448782013-12-10 12:36:35 -0800460 .setActions(new ArrayList<OFAction>())
461 .setLengthU(OFPacketOut.MINIMUM_LENGTH);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800462
463 if (sw.getBuffers() == 0) {
464 po.setBufferId(OFPacketOut.BUFFER_ID_NONE)
465 .setPacketData(pi.getPacketData())
466 .setLengthU(po.getLengthU() + po.getPacketData().length);
467 }
468 else {
469 po.setBufferId(pi.getBufferId());
470 }
471
Jonathan Hart5e448782013-12-10 12:36:35 -0800472 return po;
473 }
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800474
Jonathan Hart5e448782013-12-10 12:36:35 -0800475 @Override
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800476 public void flowsInstalled(Collection<FlowPath> installedFlowPaths) {
477 for (FlowPath flowPath : installedFlowPaths) {
478 flowInstalled(flowPath);
479 }
480 }
Jonathan Hart0444d932014-01-22 15:06:17 -0800481
482 @Override
483 public void flowRemoved(FlowPath removedFlowPath) {
TeruU417fe022014-02-04 12:59:30 -0800484 if(log.isDebugEnabled()){
485 log.debug("Flow {} was removed, having {} queued packets",
486 removedFlowPath.flowId(), waitingPackets.get(removedFlowPath.flowId().value()).size());
487 }
488
489
Jonathan Hart0444d932014-01-22 15:06:17 -0800490 if (!removedFlowPath.installerId().equals(callerId)) {
491 // Not our flow path, ignore
492 return;
493 }
TeruU417fe022014-02-04 12:59:30 -0800494
Jonathan Hart0444d932014-01-22 15:06:17 -0800495 MACAddress srcMacAddress = removedFlowPath.flowEntryMatch().srcMac();
496 MACAddress dstMacAddress = removedFlowPath.flowEntryMatch().dstMac();
497
498 Path removedPath = new Path(srcMacAddress, dstMacAddress);
499
500 synchronized (lock) {
Jonathan Hart0444d932014-01-22 15:06:17 -0800501 // There *shouldn't* be any packets queued if the flow has
502 // just been removed.
503 List<PacketToPush> packets =
504 waitingPackets.removeAll(removedFlowPath.flowId().value());
505 if (!packets.isEmpty()) {
TeruU417fe022014-02-04 12:59:30 -0800506 log.warn("Removed flow {} has packets queued.", removedFlowPath.flowId());
Jonathan Hart0444d932014-01-22 15:06:17 -0800507 }
TeruU417fe022014-02-04 12:59:30 -0800508 pendingFlows.remove(removedPath);
509 log.debug("Removed from the pendingFlow: Path {}, Flow ID {}", removedPath, removedFlowPath.flowId());
Jonathan Hart0444d932014-01-22 15:06:17 -0800510 }
511 }
Pavlin Radoslavov7208e9a2013-12-11 14:31:07 -0800512
TeruU417fe022014-02-04 12:59:30 -0800513 private void flowInstalled(FlowPath installedFlowPath) {
514 log.debug("Flow {} was installed", installedFlowPath.flowId());
Jonathan Hart5e448782013-12-10 12:36:35 -0800515
Jonathan Hart0444d932014-01-22 15:06:17 -0800516 if (!installedFlowPath.installerId().equals(callerId)) {
517 // Not our flow path, ignore
518 return;
TeruU417fe022014-02-04 12:59:30 -0800519 }
520
521 if(installedFlowPath.flowEntries().isEmpty()){
522 //If there is no flowEntry, ignore
523 log.warn("There is no flowEntry in the installedFlowPath id {}.return.", installedFlowPath.flowId());
524 return;
Jonathan Hart0444d932014-01-22 15:06:17 -0800525 }
526
TeruU417fe022014-02-04 12:59:30 -0800527 MACAddress srcMacAddress = installedFlowPath.flowEntryMatch().srcMac();
528 MACAddress dstMacAddress = installedFlowPath.flowEntryMatch().dstMac();
529 Path installedPath = new Path(srcMacAddress, dstMacAddress);
530 Path reversedInstalledPath = new Path(dstMacAddress, srcMacAddress);
531
Jonathan Hart0444d932014-01-22 15:06:17 -0800532 // TODO waiting packets should time out. We could request a path that
533 // can't be installed right now because of a network partition. The path
534 // may eventually be installed, but we may have received thousands of
535 // packets in the meantime and probably don't want to send very old packets.
Jonathan Harte789d6e2013-12-17 17:50:11 -0800536
TeruU417fe022014-02-04 12:59:30 -0800537 List<PacketToPush> packets;
538 List<PacketToPush> reversedPackets;
539 Short outPort = installedFlowPath.flowEntries().get(0).outPort().value();
540
541 PushedFlow existingFlow;
542 PushedFlow reversedExistingFlow;
Jonathan Harte789d6e2013-12-17 17:50:11 -0800543
Jonathan Hart7e6df362013-12-10 23:33:59 -0800544 synchronized (lock) {
TeruU417fe022014-02-04 12:59:30 -0800545 existingFlow = pendingFlows.get(installedPath);
546 reversedExistingFlow = pendingFlows.get(reversedInstalledPath);
547
Jonathan Hart0444d932014-01-22 15:06:17 -0800548 if (existingFlow != null) {
549 existingFlow.installed = true;
TeruU417fe022014-02-04 12:59:30 -0800550 existingFlow.firstOutPort = outPort;
551 } else {
552 log.debug("ExistingFlow {} is null", installedPath);
553 return;
554 }
555
556 if(reversedExistingFlow == null) {
557 log.debug("ReversedExistingFlow {} is null", reversedInstalledPath);
558 return;
559 }
560
561 //Check both existing flow and reversedExisting flow are installed status.
562 if(reversedExistingFlow.installed){
563 packets = waitingPackets.removeAll(existingFlow.flowId);
564 if(log.isDebugEnabled()){
565 log.debug("removed my packets {} to push from waitingPackets. outPort {} size {}",
566 Long.toHexString(existingFlow.flowId), existingFlow.firstOutPort, packets.size());
567 }
568 reversedPackets = waitingPackets.removeAll(reversedExistingFlow.flowId);
569 if(log.isDebugEnabled()){
570 log.debug("removed my reversed packets {} to push from waitingPackets. outPort {} size {}",
571 Long.toHexString(reversedExistingFlow.flowId), reversedExistingFlow.firstOutPort, reversedPackets.size());
572 }
573 }else{
574 log.debug("Forward or reverse flows hasn't been pushed yet. return");
575 return;
Jonathan Hart0444d932014-01-22 15:06:17 -0800576 }
Jonathan Hart7e6df362013-12-10 23:33:59 -0800577 }
TeruU417fe022014-02-04 12:59:30 -0800578
Jonathan Hart5e448782013-12-10 12:36:35 -0800579 for (PacketToPush packet : packets) {
TeruU417fe022014-02-04 12:59:30 -0800580 log.debug("Start packetToPush to sw {}, outPort {}", packet.dpid, existingFlow.firstOutPort);
Jonathan Hart5e448782013-12-10 12:36:35 -0800581 IOFSwitch sw = floodlightProvider.getSwitches().get(packet.dpid);
TeruU417fe022014-02-04 12:59:30 -0800582 sendPacketOut(sw, packet.packet, existingFlow.firstOutPort);
583 }
584
585 for (PacketToPush packet : reversedPackets) {
586 log.debug("Start packetToPush to sw {}, outPort {}", packet.dpid, reversedExistingFlow.firstOutPort);
587 IOFSwitch sw = floodlightProvider.getSwitches().get(packet.dpid);
588 sendPacketOut(sw, packet.packet, reversedExistingFlow.firstOutPort);
Jonathan Hart41d1e912013-11-24 16:50:25 -0800589 }
590 }
Jonathan Harte789d6e2013-12-17 17:50:11 -0800591
592 private void sendPacketOut(IOFSwitch sw, OFPacketOut po, short outPort) {
593 po.getActions().add(new OFActionOutput(outPort));
594 po.setActionsLength((short)
595 (po.getActionsLength() + OFActionOutput.MINIMUM_LENGTH));
596 po.setLengthU(po.getLengthU() + OFActionOutput.MINIMUM_LENGTH);
597
598 flowPusher.add(sw, po);
599 }
Jonathan Hart0444d932014-01-22 15:06:17 -0800600
Jonathan Hart1caaa932013-11-04 15:28:28 -0800601}