Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 1 | package net.onrc.onos.core.packetservice; |
| 2 | |
| 3 | import java.util.ArrayList; |
| 4 | import java.util.Collection; |
| 5 | import java.util.HashMap; |
| 6 | import java.util.List; |
| 7 | import java.util.Map; |
| 8 | import java.util.concurrent.CopyOnWriteArrayList; |
| 9 | |
| 10 | import net.floodlightcontroller.core.FloodlightContext; |
| 11 | import net.floodlightcontroller.core.IFloodlightProviderService; |
| 12 | import net.floodlightcontroller.core.IOFMessageListener; |
| 13 | import net.floodlightcontroller.core.IOFSwitch; |
| 14 | import net.floodlightcontroller.core.module.FloodlightModuleContext; |
| 15 | import net.floodlightcontroller.core.module.FloodlightModuleException; |
| 16 | import net.floodlightcontroller.core.module.IFloodlightModule; |
| 17 | import net.floodlightcontroller.core.module.IFloodlightService; |
| 18 | import net.onrc.onos.api.packet.IPacketListener; |
| 19 | import net.onrc.onos.api.packet.IPacketService; |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 20 | import net.onrc.onos.core.datagrid.IDatagridService; |
| 21 | import net.onrc.onos.core.datagrid.IEventChannel; |
| 22 | import net.onrc.onos.core.datagrid.IEventChannelListener; |
| 23 | import net.onrc.onos.core.flowprogrammer.IFlowPusherService; |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 24 | import net.onrc.onos.core.packet.Ethernet; |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 25 | import net.onrc.onos.core.topology.ITopologyService; |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 26 | import net.onrc.onos.core.topology.Port; |
| 27 | import net.onrc.onos.core.topology.Switch; |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 28 | import net.onrc.onos.core.topology.Topology; |
Yuta HIGUCHI | 8f3dfa3 | 2014-06-25 00:14:25 -0700 | [diff] [blame] | 29 | import net.onrc.onos.core.util.Dpid; |
| 30 | import net.onrc.onos.core.util.PortNumber; |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 31 | import net.onrc.onos.core.util.SwitchPort; |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 32 | |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 33 | import org.projectfloodlight.openflow.protocol.OFFactory; |
| 34 | import org.projectfloodlight.openflow.protocol.OFMessage; |
| 35 | import org.projectfloodlight.openflow.protocol.OFPacketIn; |
| 36 | import org.projectfloodlight.openflow.protocol.OFPacketOut; |
| 37 | import org.projectfloodlight.openflow.protocol.OFPortDesc; |
| 38 | import org.projectfloodlight.openflow.protocol.OFType; |
| 39 | import org.projectfloodlight.openflow.protocol.action.OFAction; |
| 40 | import org.projectfloodlight.openflow.types.OFPort; |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 41 | import org.slf4j.Logger; |
| 42 | import org.slf4j.LoggerFactory; |
| 43 | |
| 44 | import com.google.common.collect.HashMultimap; |
| 45 | import com.google.common.collect.Multimap; |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 46 | |
| 47 | public class PacketModule implements IOFMessageListener, IPacketService, |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 48 | IFloodlightModule { |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 49 | private static final Logger log = LoggerFactory.getLogger(PacketModule.class); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 50 | |
| 51 | private final CopyOnWriteArrayList<IPacketListener> listeners; |
| 52 | |
| 53 | private IFloodlightProviderService floodlightProvider; |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 54 | private Topology topology; |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 55 | private IDatagridService datagrid; |
| 56 | private IFlowPusherService flowPusher; |
| 57 | |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 58 | private IEventChannel<Long, PacketOutNotification> packetOutEventChannel; |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 59 | |
| 60 | private static final String PACKET_OUT_CHANNEL_NAME = |
| 61 | "onos.packet_out"; |
| 62 | |
| 63 | private PacketOutEventHandler packetOutEventHandler = |
| 64 | new PacketOutEventHandler(); |
| 65 | |
| 66 | private class PacketOutEventHandler implements |
| 67 | IEventChannelListener<Long, PacketOutNotification> { |
| 68 | |
| 69 | @Override |
| 70 | public void entryAdded(PacketOutNotification value) { |
Jonathan Hart | f5bd258 | 2014-04-09 17:43:41 -0700 | [diff] [blame] | 71 | Multimap<Long, Short> localPorts = HashMultimap.create(); |
| 72 | for (IOFSwitch sw : floodlightProvider.getSwitches().values()) { |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 73 | for (OFPortDesc port : sw.getEnabledPorts()) { |
| 74 | // XXX S fix this to int |
| 75 | localPorts.put(sw.getId(), port.getPortNo().getShortPortNumber()); |
Jonathan Hart | f5bd258 | 2014-04-09 17:43:41 -0700 | [diff] [blame] | 76 | } |
| 77 | } |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 78 | Multimap<Long, Short> outPorts = value.calculateOutPorts( |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 79 | localPorts, topology); |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 80 | sendPacketToSwitches(outPorts, value.getPacketData()); |
| 81 | } |
| 82 | |
| 83 | @Override |
| 84 | public void entryUpdated(PacketOutNotification value) { |
| 85 | entryAdded(value); |
| 86 | } |
| 87 | |
| 88 | @Override |
| 89 | public void entryRemoved(PacketOutNotification value) { |
| 90 | // Not used |
| 91 | } |
| 92 | } |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 93 | |
| 94 | public PacketModule() { |
| 95 | listeners = new CopyOnWriteArrayList<>(); |
| 96 | } |
| 97 | |
| 98 | @Override |
| 99 | public void registerPacketListener(IPacketListener listener) { |
| 100 | listeners.addIfAbsent(listener); |
| 101 | } |
| 102 | |
| 103 | @Override |
Jonathan Hart | e3702f2 | 2014-04-29 02:56:56 -0700 | [diff] [blame] | 104 | public void sendPacket(Ethernet eth, SwitchPort switchPort) { |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 105 | SinglePacketOutNotification notification = |
| 106 | new SinglePacketOutNotification(eth.serialize(), 0, |
Pavlin Radoslavov | 3d322f4 | 2014-08-18 14:58:55 -0700 | [diff] [blame] | 107 | switchPort.getDpid().value(), switchPort.getPortNumber().shortValue()); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 108 | |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 109 | // TODO We shouldn't care what the destination MAC is |
| 110 | long dstMac = eth.getDestinationMAC().toLong(); |
| 111 | packetOutEventChannel.addTransientEntry(dstMac, notification); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 112 | } |
| 113 | |
| 114 | @Override |
Jonathan Hart | e3702f2 | 2014-04-29 02:56:56 -0700 | [diff] [blame] | 115 | public void sendPacket(Ethernet eth, List<SwitchPort> switchPorts) { |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 116 | // TODO Auto-generated method stub |
Jonathan Hart | e3702f2 | 2014-04-29 02:56:56 -0700 | [diff] [blame] | 117 | throw new UnsupportedOperationException("Not yet implemented"); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 118 | } |
| 119 | |
| 120 | @Override |
Jonathan Hart | e3702f2 | 2014-04-29 02:56:56 -0700 | [diff] [blame] | 121 | public void broadcastPacketOutEdge(Ethernet eth) { |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 122 | // TODO Auto-generated method stub |
Jonathan Hart | e3702f2 | 2014-04-29 02:56:56 -0700 | [diff] [blame] | 123 | throw new UnsupportedOperationException("Not yet implemented"); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 124 | } |
| 125 | |
| 126 | @Override |
Jonathan Hart | e3702f2 | 2014-04-29 02:56:56 -0700 | [diff] [blame] | 127 | public void broadcastPacketOutEdge(Ethernet eth, SwitchPort inSwitchPort) { |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 128 | BroadcastPacketOutNotification notification = |
| 129 | new BroadcastPacketOutNotification(eth.serialize(), 0, |
Pavlin Radoslavov | 3d322f4 | 2014-08-18 14:58:55 -0700 | [diff] [blame] | 130 | inSwitchPort.getDpid().value(), inSwitchPort.getPortNumber().shortValue()); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 131 | |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 132 | long dstMac = eth.getDestinationMAC().toLong(); |
| 133 | packetOutEventChannel.addTransientEntry(dstMac, notification); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 134 | } |
| 135 | |
| 136 | @Override |
| 137 | public String getName() { |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 138 | return "packetmodule"; |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 139 | } |
| 140 | |
| 141 | @Override |
| 142 | public boolean isCallbackOrderingPrereq(OFType type, String name) { |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 143 | return false; |
| 144 | } |
| 145 | |
| 146 | @Override |
| 147 | public boolean isCallbackOrderingPostreq(OFType type, String name) { |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 148 | return false; |
| 149 | } |
| 150 | |
| 151 | @Override |
| 152 | public Command receive(IOFSwitch sw, OFMessage msg, |
| 153 | FloodlightContext cntx) { |
| 154 | if (!(msg instanceof OFPacketIn)) { |
| 155 | return Command.CONTINUE; |
| 156 | } |
| 157 | |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 158 | Ethernet eth = IFloodlightProviderService.bcStore. |
| 159 | get(cntx, IFloodlightProviderService.CONTEXT_PI_PAYLOAD); |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 160 | short inport = (short) cntx.getStorage() |
| 161 | .get(IFloodlightProviderService.CONTEXT_PI_INPORT); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 162 | |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 163 | Switch topologySwitch; |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 164 | Port inPort; |
| 165 | try { |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 166 | topology.acquireReadLock(); |
| 167 | Dpid dpid = new Dpid(sw.getId()); |
| 168 | PortNumber p = new PortNumber(inport); |
Yuta HIGUCHI | 8f3dfa3 | 2014-06-25 00:14:25 -0700 | [diff] [blame] | 169 | topologySwitch = topology.getSwitch(dpid); |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 170 | inPort = topology.getPort(dpid, p); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 171 | } finally { |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 172 | topology.releaseReadLock(); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 173 | } |
| 174 | |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 175 | if (topologySwitch == null || inPort == null) { |
Jonathan Hart | f5bd258 | 2014-04-09 17:43:41 -0700 | [diff] [blame] | 176 | // We can't send packets for switches or ports that aren't in the |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 177 | // topology yet |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 178 | return Command.CONTINUE; |
| 179 | } |
| 180 | |
| 181 | for (IPacketListener listener : listeners) { |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 182 | listener.receive(topologySwitch, inPort, eth); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 183 | } |
| 184 | |
| 185 | return Command.CONTINUE; |
| 186 | } |
| 187 | |
| 188 | @Override |
| 189 | public Collection<Class<? extends IFloodlightService>> getModuleServices() { |
| 190 | List<Class<? extends IFloodlightService>> services = new ArrayList<>(); |
| 191 | services.add(IPacketService.class); |
| 192 | return services; |
| 193 | } |
| 194 | |
| 195 | @Override |
| 196 | public Map<Class<? extends IFloodlightService>, IFloodlightService> |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 197 | getServiceImpls() { |
| 198 | |
| 199 | Map<Class<? extends IFloodlightService>, IFloodlightService> serviceImpls = new HashMap<>(); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 200 | serviceImpls.put(IPacketService.class, this); |
| 201 | return serviceImpls; |
| 202 | } |
| 203 | |
| 204 | @Override |
| 205 | public Collection<Class<? extends IFloodlightService>> getModuleDependencies() { |
| 206 | List<Class<? extends IFloodlightService>> dependencies = new ArrayList<>(); |
| 207 | dependencies.add(IFloodlightProviderService.class); |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 208 | dependencies.add(ITopologyService.class); |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 209 | dependencies.add(IDatagridService.class); |
| 210 | dependencies.add(IFlowPusherService.class); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 211 | return dependencies; |
| 212 | } |
| 213 | |
| 214 | @Override |
| 215 | public void init(FloodlightModuleContext context) |
| 216 | throws FloodlightModuleException { |
| 217 | floodlightProvider = |
| 218 | context.getServiceImpl(IFloodlightProviderService.class); |
Jonathan Hart | e37e4e2 | 2014-05-13 19:12:02 -0700 | [diff] [blame] | 219 | topology = context.getServiceImpl(ITopologyService.class) |
| 220 | .getTopology(); |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 221 | datagrid = context.getServiceImpl(IDatagridService.class); |
| 222 | flowPusher = context.getServiceImpl(IFlowPusherService.class); |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 223 | } |
| 224 | |
| 225 | @Override |
| 226 | public void startUp(FloodlightModuleContext context) { |
| 227 | floodlightProvider.addOFMessageListener(OFType.PACKET_IN, this); |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 228 | packetOutEventChannel = datagrid.addListener(PACKET_OUT_CHANNEL_NAME, |
| 229 | packetOutEventHandler, |
| 230 | Long.class, |
| 231 | PacketOutNotification.class); |
| 232 | } |
| 233 | |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 234 | private void sendPacketToSwitches(Multimap<Long, Short> outPorts, |
| 235 | byte[] packetData) { |
| 236 | for (Long dpid : outPorts.keySet()) { |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 237 | IOFSwitch sw = floodlightProvider.getSwitches().get(dpid); |
| 238 | |
| 239 | if (sw == null) { |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 240 | log.warn("Switch {} not found when sending packet", dpid); |
| 241 | continue; |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 242 | } |
| 243 | |
Jonathan Hart | a213bce | 2014-08-11 15:44:07 -0700 | [diff] [blame] | 244 | OFFactory factory = sw.getFactory(); |
| 245 | |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 246 | List<OFAction> actions = new ArrayList<>(); |
| 247 | for (Short port : outPorts.get(dpid)) { |
| 248 | actions.add(factory.actions().output(OFPort.of(port), Short.MAX_VALUE)); |
| 249 | } |
| 250 | |
| 251 | OFPacketOut po = factory.buildPacketOut() |
| 252 | .setData(packetData) |
| 253 | .setActions(actions) |
| 254 | .build(); |
| 255 | |
Jonathan Hart | e6e6373 | 2014-04-16 14:29:49 -0700 | [diff] [blame] | 256 | flowPusher.add(sw, po); |
| 257 | } |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 258 | } |
Brian O'Connor | c67f9fa | 2014-08-07 18:17:46 -0700 | [diff] [blame] | 259 | |
Jonathan Hart | 1f75cae | 2014-04-09 17:24:09 -0700 | [diff] [blame] | 260 | } |