blob: daf713478d1620698a6f863d3229d1e3f4cdd20d [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2014-present Open Networking Laboratory
Thomas Vachuska781d18b2014-10-27 10:31:25 -07003 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07004 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
Thomas Vachuska781d18b2014-10-27 10:31:25 -07007 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07008 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
Thomas Vachuska781d18b2014-10-27 10:31:25 -070015 */
Brian O'Connorabafb502014-12-02 22:26:20 -080016package org.onosproject.provider.host.impl;
alshabibe1cf87d2014-10-17 09:23:50 -070017
18import org.apache.felix.scr.annotations.Activate;
19import org.apache.felix.scr.annotations.Component;
20import org.apache.felix.scr.annotations.Deactivate;
Thomas Vachuska33601602014-11-19 03:32:15 -080021import org.apache.felix.scr.annotations.Modified;
22import org.apache.felix.scr.annotations.Property;
alshabibe1cf87d2014-10-17 09:23:50 -070023import org.apache.felix.scr.annotations.Reference;
24import org.apache.felix.scr.annotations.ReferenceCardinality;
Jonathan Harte8600eb2015-01-12 10:30:45 -080025import org.onlab.packet.ARP;
26import org.onlab.packet.Ethernet;
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -080027import org.onlab.packet.ICMP6;
Thomas Vachuskaf845cf62015-03-24 10:13:09 -070028import org.onlab.packet.IPacket;
Jonathan Harte8600eb2015-01-12 10:30:45 -080029import org.onlab.packet.IPv6;
30import org.onlab.packet.IpAddress;
Charles M.C. Chan956cb692015-04-26 18:49:39 +080031import org.onlab.packet.MacAddress;
Jonathan Harte8600eb2015-01-12 10:30:45 -080032import org.onlab.packet.VlanId;
Charles M.C. Chan956cb692015-04-26 18:49:39 +080033import org.onlab.packet.ipv6.IExtensionHeader;
Jonathan Harte8600eb2015-01-12 10:30:45 -080034import org.onlab.packet.ndp.NeighborAdvertisement;
35import org.onlab.packet.ndp.NeighborSolicitation;
Charles M.C. Chan441d7da2015-03-17 21:03:39 +080036import org.onlab.packet.ndp.RouterAdvertisement;
37import org.onlab.packet.ndp.RouterSolicitation;
Jian Lid9b5f552016-03-11 18:15:31 -080038import org.onlab.util.Tools;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070039import org.onosproject.cfg.ComponentConfigService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080040import org.onosproject.core.ApplicationId;
41import org.onosproject.core.CoreService;
Brian O'Connorabafb502014-12-02 22:26:20 -080042import org.onosproject.net.ConnectPoint;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080043import org.onosproject.net.Device;
Brian O'Connorabafb502014-12-02 22:26:20 -080044import org.onosproject.net.Host;
45import org.onosproject.net.HostId;
46import org.onosproject.net.HostLocation;
sdn94b00152016-08-30 02:12:32 -070047import org.onosproject.net.MastershipRole;
Brian O'Connorabafb502014-12-02 22:26:20 -080048import org.onosproject.net.device.DeviceEvent;
49import org.onosproject.net.device.DeviceListener;
50import org.onosproject.net.device.DeviceService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080051import org.onosproject.net.flow.DefaultTrafficSelector;
sdn94b00152016-08-30 02:12:32 -070052import org.onosproject.net.flow.DefaultTrafficTreatment;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080053import org.onosproject.net.flow.TrafficSelector;
sdn94b00152016-08-30 02:12:32 -070054import org.onosproject.net.flow.TrafficTreatment;
Brian O'Connorabafb502014-12-02 22:26:20 -080055import org.onosproject.net.host.DefaultHostDescription;
56import org.onosproject.net.host.HostDescription;
57import org.onosproject.net.host.HostProvider;
58import org.onosproject.net.host.HostProviderRegistry;
59import org.onosproject.net.host.HostProviderService;
60import org.onosproject.net.host.HostService;
sdn94b00152016-08-30 02:12:32 -070061import org.onosproject.net.packet.DefaultOutboundPacket;
62import org.onosproject.net.packet.OutboundPacket;
Brian O'Connorabafb502014-12-02 22:26:20 -080063import org.onosproject.net.packet.PacketContext;
Jonathan Hart3cfce8e2015-01-14 16:43:27 -080064import org.onosproject.net.packet.PacketPriority;
Brian O'Connorabafb502014-12-02 22:26:20 -080065import org.onosproject.net.packet.PacketProcessor;
66import org.onosproject.net.packet.PacketService;
67import org.onosproject.net.provider.AbstractProvider;
68import org.onosproject.net.provider.ProviderId;
69import org.onosproject.net.topology.Topology;
70import org.onosproject.net.topology.TopologyService;
Thomas Vachuska33601602014-11-19 03:32:15 -080071import org.osgi.service.component.ComponentContext;
alshabibe1cf87d2014-10-17 09:23:50 -070072import org.slf4j.Logger;
73
sdn94b00152016-08-30 02:12:32 -070074import java.nio.ByteBuffer;
Thomas Vachuskaf845cf62015-03-24 10:13:09 -070075import java.util.Dictionary;
76import java.util.Set;
Madan Jampania3770c32015-12-11 12:07:41 -080077import java.util.concurrent.ExecutorService;
Thomas Vachuskaf845cf62015-03-24 10:13:09 -070078
Madan Jampania3770c32015-12-11 12:07:41 -080079import static java.util.concurrent.Executors.newSingleThreadScheduledExecutor;
80import static org.onlab.util.Tools.groupedThreads;
Thomas Vachuskaf845cf62015-03-24 10:13:09 -070081import static org.slf4j.LoggerFactory.getLogger;
82
alshabibe1cf87d2014-10-17 09:23:50 -070083/**
Thomas Vachuskaec9c7dd2015-09-03 18:30:04 -070084 * Provider which uses an OpenFlow controller to detect network end-station
85 * hosts.
alshabibe1cf87d2014-10-17 09:23:50 -070086 */
87@Component(immediate = true)
88public class HostLocationProvider extends AbstractProvider implements HostProvider {
89
90 private final Logger log = getLogger(getClass());
91
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080092 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
93 protected CoreService coreService;
94
95 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
alshabibe1cf87d2014-10-17 09:23:50 -070096 protected HostProviderRegistry providerRegistry;
97
98 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -080099 protected PacketService packetService;
alshabibe1cf87d2014-10-17 09:23:50 -0700100
101 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
102 protected TopologyService topologyService;
103
Thomas Vachuska33601602014-11-19 03:32:15 -0800104 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
105 protected HostService hostService;
106
107 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
108 protected DeviceService deviceService;
109
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700110 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
111 protected ComponentConfigService cfgService;
112
alshabibe1cf87d2014-10-17 09:23:50 -0700113 private HostProviderService providerService;
114
115 private final InternalHostProvider processor = new InternalHostProvider();
Thomas Vachuska33601602014-11-19 03:32:15 -0800116 private final DeviceListener deviceListener = new InternalDeviceListener();
117
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800118 private ApplicationId appId;
119
Thomas Vachuska33601602014-11-19 03:32:15 -0800120 @Property(name = "hostRemovalEnabled", boolValue = true,
121 label = "Enable host removal on port/device down events")
122 private boolean hostRemovalEnabled = true;
alshabibe1cf87d2014-10-17 09:23:50 -0700123
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800124 @Property(name = "ipv6NeighborDiscovery", boolValue = false,
Thomas Vachuska27bee092015-06-23 19:03:10 -0700125 label = "Enable using IPv6 Neighbor Discovery by the " +
126 "Host Location Provider; default is false")
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800127 private boolean ipv6NeighborDiscovery = false;
alshabibe1cf87d2014-10-17 09:23:50 -0700128
Jonathan Hart9af322d2016-01-06 17:42:04 -0800129 @Property(name = "requestInterceptsEnabled", boolValue = true,
130 label = "Enable requesting packet intercepts")
131 private boolean requestInterceptsEnabled = true;
132
Madan Jampania3770c32015-12-11 12:07:41 -0800133 protected ExecutorService eventHandler;
134
sdn94b00152016-08-30 02:12:32 -0700135 private static final byte[] SENDER_ADDRESS = IpAddress.valueOf("0.0.0.0").toOctets();
136
alshabibe1cf87d2014-10-17 09:23:50 -0700137 /**
138 * Creates an OpenFlow host provider.
139 */
140 public HostLocationProvider() {
Brian O'Connorabafb502014-12-02 22:26:20 -0800141 super(new ProviderId("of", "org.onosproject.provider.host"));
alshabibe1cf87d2014-10-17 09:23:50 -0700142 }
143
144 @Activate
Thomas Vachuska33601602014-11-19 03:32:15 -0800145 public void activate(ComponentContext context) {
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700146 cfgService.registerProperties(getClass());
147 appId = coreService.registerApplication("org.onosproject.provider.host");
Jonathan Hart9af322d2016-01-06 17:42:04 -0800148 eventHandler = newSingleThreadScheduledExecutor(
HIGUCHI Yutad9e01052016-04-14 09:31:42 -0700149 groupedThreads("onos/host-loc-provider", "event-handler", log));
alshabibe1cf87d2014-10-17 09:23:50 -0700150 providerService = providerRegistry.register(this);
Brian O'Connor3b783262015-07-29 17:49:24 -0700151 packetService.addProcessor(processor, PacketProcessor.advisor(1));
Thomas Vachuska33601602014-11-19 03:32:15 -0800152 deviceService.addListener(deviceListener);
Jonathan Hart9af322d2016-01-06 17:42:04 -0800153
154 modified(context);
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800155
Charles M.C. Chane148de82015-05-06 12:38:21 +0800156 log.info("Started with Application ID {}", appId.id());
157 }
158
159 @Deactivate
160 public void deactivate() {
Charles M.C. Chane148de82015-05-06 12:38:21 +0800161 cfgService.unregisterProperties(getClass(), false);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700162
163 withdrawIntercepts();
164
Charles M.C. Chane148de82015-05-06 12:38:21 +0800165 providerRegistry.unregister(this);
166 packetService.removeProcessor(processor);
167 deviceService.removeListener(deviceListener);
Madan Jampania3770c32015-12-11 12:07:41 -0800168 eventHandler.shutdown();
Charles M.C. Chane148de82015-05-06 12:38:21 +0800169 providerService = null;
170 log.info("Stopped");
171 }
172
173 @Modified
174 public void modified(ComponentContext context) {
Charles M.C. Chane148de82015-05-06 12:38:21 +0800175 readComponentConfiguration(context);
Jonathan Hart9af322d2016-01-06 17:42:04 -0800176
177 if (requestInterceptsEnabled) {
178 requestIntercepts();
179 } else {
180 withdrawIntercepts();
181 }
Charles M.C. Chane148de82015-05-06 12:38:21 +0800182 }
183
184 /**
Thomas Vachuska27bee092015-06-23 19:03:10 -0700185 * Request packet intercepts.
Charles M.C. Chane148de82015-05-06 12:38:21 +0800186 */
Thomas Vachuska27bee092015-06-23 19:03:10 -0700187 private void requestIntercepts() {
188 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
189 selector.matchEthType(Ethernet.TYPE_ARP);
190 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800191
Thomas Vachuska27bee092015-06-23 19:03:10 -0700192 // IPv6 Neighbor Solicitation packet.
193 selector.matchEthType(Ethernet.TYPE_IPV6);
194 selector.matchIPProtocol(IPv6.PROTOCOL_ICMP6);
195 selector.matchIcmpv6Type(ICMP6.NEIGHBOR_SOLICITATION);
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800196 if (ipv6NeighborDiscovery) {
Thomas Vachuska27bee092015-06-23 19:03:10 -0700197 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
198 } else {
199 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800200 }
Thomas Vachuska27bee092015-06-23 19:03:10 -0700201
202 // IPv6 Neighbor Advertisement packet.
203 selector.matchIcmpv6Type(ICMP6.NEIGHBOR_ADVERTISEMENT);
204 if (ipv6NeighborDiscovery) {
205 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
206 } else {
207 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
208 }
209 }
210
211 /**
212 * Withdraw packet intercepts.
213 */
214 private void withdrawIntercepts() {
215 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
216 selector.matchEthType(Ethernet.TYPE_ARP);
Aaron Kruglikov07a923d2015-07-03 13:30:57 -0700217 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700218
219 // IPv6 Neighbor Solicitation packet.
220 selector.matchEthType(Ethernet.TYPE_IPV6);
221 selector.matchIPProtocol(IPv6.PROTOCOL_ICMP6);
222 selector.matchIcmpv6Type(ICMP6.NEIGHBOR_SOLICITATION);
223 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
224
225 // IPv6 Neighbor Advertisement packet.
226 selector.matchIcmpv6Type(ICMP6.NEIGHBOR_ADVERTISEMENT);
227 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800228 }
229
230 /**
231 * Extracts properties from the component configuration context.
232 *
233 * @param context the component context
234 */
235 private void readComponentConfiguration(ComponentContext context) {
236 Dictionary<?, ?> properties = context.getProperties();
237 Boolean flag;
238
Jian Lid9b5f552016-03-11 18:15:31 -0800239 flag = Tools.isPropertyEnabled(properties, "hostRemovalEnabled");
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800240 if (flag == null) {
241 log.info("Host removal on port/device down events is not configured, " +
Thomas Vachuska27bee092015-06-23 19:03:10 -0700242 "using current value of {}", hostRemovalEnabled);
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800243 } else {
244 hostRemovalEnabled = flag;
245 log.info("Configured. Host removal on port/device down events is {}",
246 hostRemovalEnabled ? "enabled" : "disabled");
Thomas Vachuska33601602014-11-19 03:32:15 -0800247 }
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800248
Jian Lid9b5f552016-03-11 18:15:31 -0800249 flag = Tools.isPropertyEnabled(properties, "ipv6NeighborDiscovery");
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800250 if (flag == null) {
251 log.info("Using IPv6 Neighbor Discovery is not configured, " +
Thomas Vachuska27bee092015-06-23 19:03:10 -0700252 "using current value of {}", ipv6NeighborDiscovery);
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800253 } else {
254 ipv6NeighborDiscovery = flag;
255 log.info("Configured. Using IPv6 Neighbor Discovery is {}",
256 ipv6NeighborDiscovery ? "enabled" : "disabled");
257 }
Jonathan Hart9af322d2016-01-06 17:42:04 -0800258
Jian Lid9b5f552016-03-11 18:15:31 -0800259 flag = Tools.isPropertyEnabled(properties, "requestInterceptsEnabled");
Jonathan Hart9af322d2016-01-06 17:42:04 -0800260 if (flag == null) {
261 log.info("Request intercepts is not configured, " +
262 "using current value of {}", requestInterceptsEnabled);
263 } else {
264 requestInterceptsEnabled = flag;
265 log.info("Configured. Request intercepts is {}",
266 requestInterceptsEnabled ? "enabled" : "disabled");
267 }
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800268 }
269
alshabibe1cf87d2014-10-17 09:23:50 -0700270 @Override
271 public void triggerProbe(Host host) {
sdn94b00152016-08-30 02:12:32 -0700272 log.info("Triggering probe on device {} ", host);
273 MastershipRole role = deviceService.getRole(host.location().deviceId());
274 if (role.equals(MastershipRole.MASTER)) {
275 host.ipAddresses().forEach(ip -> {
276 sendProbe(host, ip);
277 });
278 } else {
279 log.info("not the master, master will probe {}");
280 }
281 }
282
283 private void sendProbe(Host host, IpAddress targetIp) {
284 Ethernet probePacket = null;
285 if (targetIp.isIp4()) {
286 // IPv4: Use ARP
287 probePacket = buildArpRequest(targetIp, host);
288 } else {
289 // IPv6: Use Neighbor Discovery
290 //FIX ME need to implement ndp probe
291 log.info("Triggering probe on device {} ", host);
292 }
293
294 TrafficTreatment treatment = DefaultTrafficTreatment.builder().setOutput(host.location().port()).build();
295
296 OutboundPacket outboundPacket = new DefaultOutboundPacket(host.location().deviceId(), treatment,
297 ByteBuffer.wrap(probePacket.serialize()));
298
299 packetService.emit(outboundPacket);
300 }
301
302 /*
303 * This method is using source ip as 0.0.0.0 , to receive the reply even from the sub net hosts.
304 */
305 private Ethernet buildArpRequest(IpAddress targetIp, Host host) {
306
307 ARP arp = new ARP();
308 arp.setHardwareType(ARP.HW_TYPE_ETHERNET)
309 .setHardwareAddressLength((byte) Ethernet.DATALAYER_ADDRESS_LENGTH)
310 .setProtocolType(ARP.PROTO_TYPE_IP)
311 .setProtocolAddressLength((byte) IpAddress.INET_BYTE_LENGTH)
312 .setOpCode(ARP.OP_REQUEST);
313
314 arp.setSenderHardwareAddress(MacAddress.BROADCAST.toBytes())
315 .setSenderProtocolAddress(SENDER_ADDRESS)
316 .setTargetHardwareAddress(MacAddress.BROADCAST.toBytes())
317 .setTargetProtocolAddress(targetIp.toOctets());
318
319 Ethernet ethernet = new Ethernet();
320 ethernet.setEtherType(Ethernet.TYPE_ARP)
321 .setDestinationMACAddress(MacAddress.BROADCAST)
322 .setSourceMACAddress(MacAddress.BROADCAST).setPayload(arp);
323
324 ethernet.setPad(true);
325 return ethernet;
alshabibe1cf87d2014-10-17 09:23:50 -0700326 }
327
328 private class InternalHostProvider implements PacketProcessor {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800329 /**
330 * Update host location only.
331 *
Thomas Vachuska27bee092015-06-23 19:03:10 -0700332 * @param hid host ID
333 * @param mac source Mac address
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800334 * @param vlan VLAN ID
335 * @param hloc host location
336 */
337 private void updateLocation(HostId hid, MacAddress mac,
Thomas Vachuska27bee092015-06-23 19:03:10 -0700338 VlanId vlan, HostLocation hloc) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800339 HostDescription desc = new DefaultHostDescription(mac, vlan, hloc);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700340 try {
Ray Milkeydc083442016-02-22 11:27:57 -0800341 providerService.hostDetected(hid, desc, false);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700342 } catch (IllegalStateException e) {
343 log.debug("Host {} suppressed", hid);
344 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800345 }
Thomas Vachuska27bee092015-06-23 19:03:10 -0700346
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800347 /**
348 * Update host location and IP address.
349 *
Thomas Vachuska27bee092015-06-23 19:03:10 -0700350 * @param hid host ID
351 * @param mac source Mac address
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800352 * @param vlan VLAN ID
353 * @param hloc host location
Thomas Vachuska27bee092015-06-23 19:03:10 -0700354 * @param ip source IP address
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800355 */
356 private void updateLocationIP(HostId hid, MacAddress mac,
357 VlanId vlan, HostLocation hloc,
358 IpAddress ip) {
Thomas Vachuskae7966102015-09-09 17:33:33 -0700359 HostDescription desc = ip.isZero() || ip.isSelfAssigned() ?
Thomas Vachuskaec9c7dd2015-09-03 18:30:04 -0700360 new DefaultHostDescription(mac, vlan, hloc) :
361 new DefaultHostDescription(mac, vlan, hloc, ip);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700362 try {
Ray Milkeydc083442016-02-22 11:27:57 -0800363 providerService.hostDetected(hid, desc, false);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700364 } catch (IllegalStateException e) {
365 log.debug("Host {} suppressed", hid);
366 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800367 }
alshabibe1cf87d2014-10-17 09:23:50 -0700368
369 @Override
370 public void process(PacketContext context) {
alshabib4a179dc2014-10-17 17:17:01 -0700371 if (context == null) {
372 return;
373 }
alshabibe1cf87d2014-10-17 09:23:50 -0700374
Thomas Vachuskaf845cf62015-03-24 10:13:09 -0700375 Ethernet eth = context.inPacket().parsed();
Jonathan Harte8600eb2015-01-12 10:30:45 -0800376 if (eth == null) {
377 return;
378 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800379 MacAddress srcMac = eth.getSourceMAC();
Jonathan Harte8600eb2015-01-12 10:30:45 -0800380
alshabibe1cf87d2014-10-17 09:23:50 -0700381 VlanId vlan = VlanId.vlanId(eth.getVlanID());
382 ConnectPoint heardOn = context.inPacket().receivedFrom();
383
Thomas Vachuskaf845cf62015-03-24 10:13:09 -0700384 // If this arrived on control port, bail out.
385 if (heardOn.port().isLogical()) {
386 return;
387 }
388
alshabibe1cf87d2014-10-17 09:23:50 -0700389 // If this is not an edge port, bail out.
390 Topology topology = topologyService.currentTopology();
391 if (topologyService.isInfrastructure(topology, heardOn)) {
392 return;
393 }
394
Thomas Vachuskaec9c7dd2015-09-03 18:30:04 -0700395 HostLocation hloc = new HostLocation(heardOn, System.currentTimeMillis());
alshabibe1cf87d2014-10-17 09:23:50 -0700396 HostId hid = HostId.hostId(eth.getSourceMAC(), vlan);
397
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800398 // ARP: possible new hosts, update both location and IP
alshabibe1cf87d2014-10-17 09:23:50 -0700399 if (eth.getEtherType() == Ethernet.TYPE_ARP) {
400 ARP arp = (ARP) eth.getPayload();
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -0800401 IpAddress ip = IpAddress.valueOf(IpAddress.Version.INET,
402 arp.getSenderProtocolAddress());
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800403 updateLocationIP(hid, srcMac, vlan, hloc, ip);
alshabibe1cf87d2014-10-17 09:23:50 -0700404
Charles Chan5d5e0622015-09-25 13:00:06 -0700405 // IPv4: update location only
alshabibe1cf87d2014-10-17 09:23:50 -0700406 } else if (eth.getEtherType() == Ethernet.TYPE_IPV4) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800407 updateLocation(hid, srcMac, vlan, hloc);
alshabibe1cf87d2014-10-17 09:23:50 -0700408
Charles Chan5d5e0622015-09-25 13:00:06 -0700409 //
410 // NeighborAdvertisement and NeighborSolicitation: possible
411 // new hosts, update both location and IP.
412 //
413 // IPv6: update location only
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800414 } else if (eth.getEtherType() == Ethernet.TYPE_IPV6) {
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800415 IPv6 ipv6 = (IPv6) eth.getPayload();
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800416 IpAddress ip = IpAddress.valueOf(IpAddress.Version.INET6,
Thomas Vachuska27bee092015-06-23 19:03:10 -0700417 ipv6.getSourceAddress());
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800418
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800419 // skip extension headers
420 IPacket pkt = ipv6;
421 while (pkt.getPayload() != null &&
422 pkt.getPayload() instanceof IExtensionHeader) {
423 pkt = pkt.getPayload();
424 }
425
426 // Neighbor Discovery Protocol
Charles M.C. Chan9148d2d2015-04-27 03:36:39 +0800427 pkt = pkt.getPayload();
428 if (pkt != null && pkt instanceof ICMP6) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800429 pkt = pkt.getPayload();
430 // RouterSolicitation, RouterAdvertisement
Charles M.C. Chan9148d2d2015-04-27 03:36:39 +0800431 if (pkt != null && (pkt instanceof RouterAdvertisement ||
Thomas Vachuska27bee092015-06-23 19:03:10 -0700432 pkt instanceof RouterSolicitation)) {
Charles M.C. Chan441d7da2015-03-17 21:03:39 +0800433 return;
434 }
Charles M.C. Chan9148d2d2015-04-27 03:36:39 +0800435 if (pkt != null && (pkt instanceof NeighborSolicitation ||
Thomas Vachuska27bee092015-06-23 19:03:10 -0700436 pkt instanceof NeighborAdvertisement)) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800437 // Duplicate Address Detection
438 if (ip.isZero()) {
Charles M.C. Chan441d7da2015-03-17 21:03:39 +0800439 return;
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800440 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800441 // NeighborSolicitation, NeighborAdvertisement
442 updateLocationIP(hid, srcMac, vlan, hloc, ip);
443 return;
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800444 }
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800445 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800446
447 // multicast
448 if (eth.isMulticast()) {
449 return;
450 }
451
452 // normal IPv6 packets
453 updateLocation(hid, srcMac, vlan, hloc);
alshabibe1cf87d2014-10-17 09:23:50 -0700454 }
455 }
456 }
Thomas Vachuska33601602014-11-19 03:32:15 -0800457
458 // Auxiliary listener to device events.
459 private class InternalDeviceListener implements DeviceListener {
460 @Override
461 public void event(DeviceEvent event) {
Madan Jampania3770c32015-12-11 12:07:41 -0800462 eventHandler.execute(() -> handleEvent(event));
463 }
464
465 private void handleEvent(DeviceEvent event) {
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800466 Device device = event.subject();
467 switch (event.type()) {
Thomas Vachuska27bee092015-06-23 19:03:10 -0700468 case DEVICE_ADDED:
469 break;
470 case DEVICE_AVAILABILITY_CHANGED:
471 if (hostRemovalEnabled &&
472 !deviceService.isAvailable(device.id())) {
473 removeHosts(hostService.getConnectedHosts(device.id()));
474 }
475 break;
476 case DEVICE_SUSPENDED:
477 case DEVICE_UPDATED:
478 // Nothing to do?
479 break;
480 case DEVICE_REMOVED:
481 if (hostRemovalEnabled) {
482 removeHosts(hostService.getConnectedHosts(device.id()));
483 }
484 break;
485 case PORT_ADDED:
486 break;
487 case PORT_UPDATED:
488 if (hostRemovalEnabled) {
489 ConnectPoint point =
490 new ConnectPoint(device.id(), event.port().number());
491 removeHosts(hostService.getConnectedHosts(point));
492 }
493 break;
494 case PORT_REMOVED:
495 // Nothing to do?
496 break;
497 default:
498 break;
Thomas Vachuska33601602014-11-19 03:32:15 -0800499 }
500 }
501 }
502
503 // Signals host vanish for all specified hosts.
504 private void removeHosts(Set<Host> hosts) {
505 for (Host host : hosts) {
Charles Chan8b0fdaa2016-02-23 18:35:27 -0800506 if (host.providerId().equals(HostLocationProvider.this.id())) {
507 providerService.hostVanished(host.id());
508 }
Thomas Vachuska33601602014-11-19 03:32:15 -0800509 }
510 }
511
alshabibe1cf87d2014-10-17 09:23:50 -0700512}