blob: b7762805a6888ff0bd593dac6f73d44d9e6fad28 [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Ray Milkey34c95902015-04-15 09:47:53 -07002 * Copyright 2014-2015 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;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070038import org.onosproject.cfg.ComponentConfigService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080039import org.onosproject.core.ApplicationId;
40import org.onosproject.core.CoreService;
Brian O'Connorabafb502014-12-02 22:26:20 -080041import org.onosproject.net.ConnectPoint;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080042import org.onosproject.net.Device;
Brian O'Connorabafb502014-12-02 22:26:20 -080043import org.onosproject.net.Host;
44import org.onosproject.net.HostId;
45import org.onosproject.net.HostLocation;
46import org.onosproject.net.device.DeviceEvent;
47import org.onosproject.net.device.DeviceListener;
48import org.onosproject.net.device.DeviceService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080049import org.onosproject.net.flow.DefaultTrafficSelector;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080050import org.onosproject.net.flow.TrafficSelector;
Brian O'Connorabafb502014-12-02 22:26:20 -080051import org.onosproject.net.host.DefaultHostDescription;
52import org.onosproject.net.host.HostDescription;
53import org.onosproject.net.host.HostProvider;
54import org.onosproject.net.host.HostProviderRegistry;
55import org.onosproject.net.host.HostProviderService;
56import org.onosproject.net.host.HostService;
57import org.onosproject.net.packet.PacketContext;
Jonathan Hart3cfce8e2015-01-14 16:43:27 -080058import org.onosproject.net.packet.PacketPriority;
Brian O'Connorabafb502014-12-02 22:26:20 -080059import org.onosproject.net.packet.PacketProcessor;
60import org.onosproject.net.packet.PacketService;
61import org.onosproject.net.provider.AbstractProvider;
62import org.onosproject.net.provider.ProviderId;
63import org.onosproject.net.topology.Topology;
64import org.onosproject.net.topology.TopologyService;
Thomas Vachuska33601602014-11-19 03:32:15 -080065import org.osgi.service.component.ComponentContext;
alshabibe1cf87d2014-10-17 09:23:50 -070066import org.slf4j.Logger;
67
Thomas Vachuskaf845cf62015-03-24 10:13:09 -070068import java.util.Dictionary;
69import java.util.Set;
70
71import static com.google.common.base.Strings.isNullOrEmpty;
72import static org.slf4j.LoggerFactory.getLogger;
73
alshabibe1cf87d2014-10-17 09:23:50 -070074/**
75 * Provider which uses an OpenFlow controller to detect network
76 * end-station hosts.
77 */
78@Component(immediate = true)
79public class HostLocationProvider extends AbstractProvider implements HostProvider {
80
81 private final Logger log = getLogger(getClass());
82
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080083 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
84 protected CoreService coreService;
85
86 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
alshabibe1cf87d2014-10-17 09:23:50 -070087 protected HostProviderRegistry providerRegistry;
88
89 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -080090 protected PacketService packetService;
alshabibe1cf87d2014-10-17 09:23:50 -070091
92 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
93 protected TopologyService topologyService;
94
Thomas Vachuska33601602014-11-19 03:32:15 -080095 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
96 protected HostService hostService;
97
98 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
99 protected DeviceService deviceService;
100
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700101 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
102 protected ComponentConfigService cfgService;
103
alshabibe1cf87d2014-10-17 09:23:50 -0700104 private HostProviderService providerService;
105
106 private final InternalHostProvider processor = new InternalHostProvider();
Thomas Vachuska33601602014-11-19 03:32:15 -0800107 private final DeviceListener deviceListener = new InternalDeviceListener();
108
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800109 private ApplicationId appId;
110
Thomas Vachuska33601602014-11-19 03:32:15 -0800111 @Property(name = "hostRemovalEnabled", boolValue = true,
112 label = "Enable host removal on port/device down events")
113 private boolean hostRemovalEnabled = true;
alshabibe1cf87d2014-10-17 09:23:50 -0700114
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800115 @Property(name = "ipv6NeighborDiscovery", boolValue = false,
116 label = "Enable using IPv6 Neighbor Discovery by the " +
117 "Host Location Provider; default is false")
118 private boolean ipv6NeighborDiscovery = false;
alshabibe1cf87d2014-10-17 09:23:50 -0700119
120 /**
121 * Creates an OpenFlow host provider.
122 */
123 public HostLocationProvider() {
Brian O'Connorabafb502014-12-02 22:26:20 -0800124 super(new ProviderId("of", "org.onosproject.provider.host"));
alshabibe1cf87d2014-10-17 09:23:50 -0700125 }
126
127 @Activate
Thomas Vachuska33601602014-11-19 03:32:15 -0800128 public void activate(ComponentContext context) {
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700129 cfgService.registerProperties(getClass());
130 appId = coreService.registerApplication("org.onosproject.provider.host");
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800131
alshabibe1cf87d2014-10-17 09:23:50 -0700132 providerService = providerRegistry.register(this);
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -0800133 packetService.addProcessor(processor, 1);
Thomas Vachuska33601602014-11-19 03:32:15 -0800134 deviceService.addListener(deviceListener);
Charles M.C. Chane148de82015-05-06 12:38:21 +0800135 readComponentConfiguration(context);
136 requestPackests();
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800137
Charles M.C. Chane148de82015-05-06 12:38:21 +0800138 log.info("Started with Application ID {}", appId.id());
139 }
140
141 @Deactivate
142 public void deactivate() {
143 // TODO revoke all packet requests when deactivate
144 cfgService.unregisterProperties(getClass(), false);
145 providerRegistry.unregister(this);
146 packetService.removeProcessor(processor);
147 deviceService.removeListener(deviceListener);
148 providerService = null;
149 log.info("Stopped");
150 }
151
152 @Modified
153 public void modified(ComponentContext context) {
154 // TODO revoke unnecessary packet requests when config being modified
155 readComponentConfiguration(context);
156 requestPackests();
157 }
158
159 /**
160 * Request packet in via PacketService.
161 */
162 private void requestPackests() {
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800163 TrafficSelector.Builder selectorBuilder =
164 DefaultTrafficSelector.builder();
165 selectorBuilder.matchEthType(Ethernet.TYPE_ARP);
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -0800166 packetService.requestPackets(selectorBuilder.build(),
Charles M.C. Chane148de82015-05-06 12:38:21 +0800167 PacketPriority.CONTROL, appId);
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800168
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800169 if (ipv6NeighborDiscovery) {
170 // IPv6 Neighbor Solicitation packet.
171 selectorBuilder = DefaultTrafficSelector.builder();
172 selectorBuilder.matchEthType(Ethernet.TYPE_IPV6);
173 selectorBuilder.matchIPProtocol(IPv6.PROTOCOL_ICMP6);
174 selectorBuilder.matchIcmpv6Type(ICMP6.NEIGHBOR_SOLICITATION);
175 packetService.requestPackets(selectorBuilder.build(),
176 PacketPriority.CONTROL, appId);
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -0800177
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800178 // IPv6 Neighbor Advertisement packet.
179 selectorBuilder = DefaultTrafficSelector.builder();
180 selectorBuilder.matchEthType(Ethernet.TYPE_IPV6);
181 selectorBuilder.matchIPProtocol(IPv6.PROTOCOL_ICMP6);
182 selectorBuilder.matchIcmpv6Type(ICMP6.NEIGHBOR_ADVERTISEMENT);
183 packetService.requestPackets(selectorBuilder.build(),
184 PacketPriority.CONTROL, appId);
185 }
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800186 }
187
188 /**
189 * Extracts properties from the component configuration context.
190 *
191 * @param context the component context
192 */
193 private void readComponentConfiguration(ComponentContext context) {
194 Dictionary<?, ?> properties = context.getProperties();
195 Boolean flag;
196
197 flag = isPropertyEnabled(properties, "hostRemovalEnabled");
198 if (flag == null) {
199 log.info("Host removal on port/device down events is not configured, " +
200 "using current value of {}", hostRemovalEnabled);
201 } else {
202 hostRemovalEnabled = flag;
203 log.info("Configured. Host removal on port/device down events is {}",
204 hostRemovalEnabled ? "enabled" : "disabled");
Thomas Vachuska33601602014-11-19 03:32:15 -0800205 }
Pavlin Radoslavov93b606b2015-02-25 17:28:39 -0800206
207 flag = isPropertyEnabled(properties, "ipv6NeighborDiscovery");
208 if (flag == null) {
209 log.info("Using IPv6 Neighbor Discovery is not configured, " +
210 "using current value of {}", ipv6NeighborDiscovery);
211 } else {
212 ipv6NeighborDiscovery = flag;
213 log.info("Configured. Using IPv6 Neighbor Discovery is {}",
214 ipv6NeighborDiscovery ? "enabled" : "disabled");
215 }
216 }
217
218 /**
219 * Check property name is defined and set to true.
220 *
221 * @param properties properties to be looked up
222 * @param propertyName the name of the property to look up
223 * @return value when the propertyName is defined or return null
224 */
225 private static Boolean isPropertyEnabled(Dictionary<?, ?> properties,
226 String propertyName) {
227 Boolean value = null;
228 try {
229 String s = (String) properties.get(propertyName);
230 value = isNullOrEmpty(s) ? null : s.trim().equals("true");
231 } catch (ClassCastException e) {
232 // No propertyName defined.
233 value = null;
234 }
235 return value;
Thomas Vachuska33601602014-11-19 03:32:15 -0800236 }
237
alshabibe1cf87d2014-10-17 09:23:50 -0700238 @Override
239 public void triggerProbe(Host host) {
240 log.info("Triggering probe on device {}", host);
241 }
242
243 private class InternalHostProvider implements PacketProcessor {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800244 /**
245 * Update host location only.
246 *
247 * @param hid host ID
248 * @param mac source Mac address
249 * @param vlan VLAN ID
250 * @param hloc host location
251 */
252 private void updateLocation(HostId hid, MacAddress mac,
253 VlanId vlan, HostLocation hloc) {
254 HostDescription desc = new DefaultHostDescription(mac, vlan, hloc);
255 providerService.hostDetected(hid, desc);
256 }
257 /**
258 * Update host location and IP address.
259 *
260 * @param hid host ID
261 * @param mac source Mac address
262 * @param vlan VLAN ID
263 * @param hloc host location
264 * @param ip source IP address
265 */
266 private void updateLocationIP(HostId hid, MacAddress mac,
267 VlanId vlan, HostLocation hloc,
268 IpAddress ip) {
269 HostDescription desc = new DefaultHostDescription(mac, vlan, hloc, ip);
270 providerService.hostDetected(hid, desc);
271 }
alshabibe1cf87d2014-10-17 09:23:50 -0700272
273 @Override
274 public void process(PacketContext context) {
alshabib4a179dc2014-10-17 17:17:01 -0700275 if (context == null) {
276 return;
277 }
alshabibe1cf87d2014-10-17 09:23:50 -0700278
Thomas Vachuskaf845cf62015-03-24 10:13:09 -0700279 Ethernet eth = context.inPacket().parsed();
Jonathan Harte8600eb2015-01-12 10:30:45 -0800280 if (eth == null) {
281 return;
282 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800283 MacAddress srcMac = eth.getSourceMAC();
Jonathan Harte8600eb2015-01-12 10:30:45 -0800284
alshabibe1cf87d2014-10-17 09:23:50 -0700285 VlanId vlan = VlanId.vlanId(eth.getVlanID());
286 ConnectPoint heardOn = context.inPacket().receivedFrom();
287
Thomas Vachuskaf845cf62015-03-24 10:13:09 -0700288 // If this arrived on control port, bail out.
289 if (heardOn.port().isLogical()) {
290 return;
291 }
292
alshabibe1cf87d2014-10-17 09:23:50 -0700293 // If this is not an edge port, bail out.
294 Topology topology = topologyService.currentTopology();
295 if (topologyService.isInfrastructure(topology, heardOn)) {
296 return;
297 }
298
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -0800299 HostLocation hloc =
300 new HostLocation(heardOn, System.currentTimeMillis());
alshabibe1cf87d2014-10-17 09:23:50 -0700301
302 HostId hid = HostId.hostId(eth.getSourceMAC(), vlan);
303
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800304 // ARP: possible new hosts, update both location and IP
alshabibe1cf87d2014-10-17 09:23:50 -0700305 if (eth.getEtherType() == Ethernet.TYPE_ARP) {
306 ARP arp = (ARP) eth.getPayload();
Pavlin Radoslavovd6612f92015-02-23 13:53:32 -0800307 IpAddress ip = IpAddress.valueOf(IpAddress.Version.INET,
308 arp.getSenderProtocolAddress());
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800309 updateLocationIP(hid, srcMac, vlan, hloc, ip);
alshabibe1cf87d2014-10-17 09:23:50 -0700310
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800311 // IPv4: update location only
alshabibe1cf87d2014-10-17 09:23:50 -0700312 } else if (eth.getEtherType() == Ethernet.TYPE_IPV4) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800313 updateLocation(hid, srcMac, vlan, hloc);
alshabibe1cf87d2014-10-17 09:23:50 -0700314
Charles M.C. Chan441d7da2015-03-17 21:03:39 +0800315 //
316 // NeighborAdvertisement and NeighborSolicitation: possible
317 // new hosts, update both location and IP.
318 //
319 // IPv6: update location only
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800320 } else if (eth.getEtherType() == Ethernet.TYPE_IPV6) {
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800321 IPv6 ipv6 = (IPv6) eth.getPayload();
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800322 IpAddress ip = IpAddress.valueOf(IpAddress.Version.INET6,
323 ipv6.getSourceAddress());
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800324
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800325 // skip extension headers
326 IPacket pkt = ipv6;
327 while (pkt.getPayload() != null &&
328 pkt.getPayload() instanceof IExtensionHeader) {
329 pkt = pkt.getPayload();
330 }
331
332 // Neighbor Discovery Protocol
Charles M.C. Chan9148d2d2015-04-27 03:36:39 +0800333 pkt = pkt.getPayload();
334 if (pkt != null && pkt instanceof ICMP6) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800335 pkt = pkt.getPayload();
336 // RouterSolicitation, RouterAdvertisement
Charles M.C. Chan9148d2d2015-04-27 03:36:39 +0800337 if (pkt != null && (pkt instanceof RouterAdvertisement ||
338 pkt instanceof RouterSolicitation)) {
Charles M.C. Chan441d7da2015-03-17 21:03:39 +0800339 return;
340 }
Charles M.C. Chan9148d2d2015-04-27 03:36:39 +0800341 if (pkt != null && (pkt instanceof NeighborSolicitation ||
342 pkt instanceof NeighborAdvertisement)) {
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800343 // Duplicate Address Detection
344 if (ip.isZero()) {
Charles M.C. Chan441d7da2015-03-17 21:03:39 +0800345 return;
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800346 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800347 // NeighborSolicitation, NeighborAdvertisement
348 updateLocationIP(hid, srcMac, vlan, hloc, ip);
349 return;
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800350 }
Charles M.C. Chan7fee36a2014-12-31 00:19:59 +0800351 }
Charles M.C. Chan956cb692015-04-26 18:49:39 +0800352
353 // multicast
354 if (eth.isMulticast()) {
355 return;
356 }
357
358 // normal IPv6 packets
359 updateLocation(hid, srcMac, vlan, hloc);
alshabibe1cf87d2014-10-17 09:23:50 -0700360 }
361 }
362 }
Thomas Vachuska33601602014-11-19 03:32:15 -0800363
364 // Auxiliary listener to device events.
365 private class InternalDeviceListener implements DeviceListener {
366 @Override
367 public void event(DeviceEvent event) {
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800368 Device device = event.subject();
369 switch (event.type()) {
370 case DEVICE_ADDED:
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800371 break;
372 case DEVICE_AVAILABILITY_CHANGED:
373 if (hostRemovalEnabled &&
374 !deviceService.isAvailable(device.id())) {
375 removeHosts(hostService.getConnectedHosts(device.id()));
Thomas Vachuska33601602014-11-19 03:32:15 -0800376 }
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800377 break;
378 case DEVICE_SUSPENDED:
379 case DEVICE_UPDATED:
380 // Nothing to do?
381 break;
382 case DEVICE_REMOVED:
383 if (hostRemovalEnabled) {
384 removeHosts(hostService.getConnectedHosts(device.id()));
385 }
386 break;
387 case PORT_ADDED:
388 break;
389 case PORT_UPDATED:
390 if (hostRemovalEnabled) {
391 ConnectPoint point =
392 new ConnectPoint(device.id(), event.port().number());
393 removeHosts(hostService.getConnectedHosts(point));
394 }
395 break;
396 case PORT_REMOVED:
397 // Nothing to do?
398 break;
399 default:
400 break;
Thomas Vachuska33601602014-11-19 03:32:15 -0800401 }
402 }
403 }
404
405 // Signals host vanish for all specified hosts.
406 private void removeHosts(Set<Host> hosts) {
407 for (Host host : hosts) {
408 providerService.hostVanished(host.id());
409 }
410 }
411
alshabibe1cf87d2014-10-17 09:23:50 -0700412}