blob: d723ce1963c5c88b9dd389e0880dd84b14914afe [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.lldp.impl;
alshabib7911a052014-10-16 17:49:37 -070017
Thomas Vachuska6f94ded2015-02-21 14:02:38 -080018import com.google.common.collect.ImmutableMap;
19import com.google.common.collect.ImmutableSet;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070020import com.google.common.collect.Maps;
alshabib7911a052014-10-16 17:49:37 -070021import org.apache.felix.scr.annotations.Activate;
22import org.apache.felix.scr.annotations.Component;
23import org.apache.felix.scr.annotations.Deactivate;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080024import org.apache.felix.scr.annotations.Modified;
25import org.apache.felix.scr.annotations.Property;
alshabib7911a052014-10-16 17:49:37 -070026import org.apache.felix.scr.annotations.Reference;
27import org.apache.felix.scr.annotations.ReferenceCardinality;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080028import org.onlab.packet.Ethernet;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070029import org.onosproject.cfg.ComponentConfigService;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070030import org.onosproject.cluster.ClusterService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080031import org.onosproject.core.ApplicationId;
32import org.onosproject.core.CoreService;
Brian O'Connorabafb502014-12-02 22:26:20 -080033import org.onosproject.mastership.MastershipEvent;
34import org.onosproject.mastership.MastershipListener;
35import org.onosproject.mastership.MastershipService;
36import org.onosproject.net.ConnectPoint;
37import org.onosproject.net.Device;
38import org.onosproject.net.DeviceId;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070039import org.onosproject.net.LinkKey;
Brian O'Connorabafb502014-12-02 22:26:20 -080040import org.onosproject.net.Port;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070041import org.onosproject.net.config.NetworkConfigRegistry;
Brian O'Connorabafb502014-12-02 22:26:20 -080042import org.onosproject.net.device.DeviceEvent;
43import org.onosproject.net.device.DeviceListener;
44import org.onosproject.net.device.DeviceService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080045import org.onosproject.net.flow.DefaultTrafficSelector;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080046import org.onosproject.net.flow.TrafficSelector;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070047import org.onosproject.net.link.DefaultLinkDescription;
Brian O'Connorabafb502014-12-02 22:26:20 -080048import org.onosproject.net.link.LinkProvider;
49import org.onosproject.net.link.LinkProviderRegistry;
50import org.onosproject.net.link.LinkProviderService;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070051import org.onosproject.net.link.LinkService;
Brian O'Connorabafb502014-12-02 22:26:20 -080052import org.onosproject.net.packet.PacketContext;
Jonathan Hart3cfce8e2015-01-14 16:43:27 -080053import org.onosproject.net.packet.PacketPriority;
Brian O'Connorabafb502014-12-02 22:26:20 -080054import org.onosproject.net.packet.PacketProcessor;
55import org.onosproject.net.packet.PacketService;
56import org.onosproject.net.provider.AbstractProvider;
57import org.onosproject.net.provider.ProviderId;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080058import org.osgi.service.component.ComponentContext;
alshabib7911a052014-10-16 17:49:37 -070059import org.slf4j.Logger;
60
Thomas Vachuska6f94ded2015-02-21 14:02:38 -080061import java.io.IOException;
62import java.util.Dictionary;
63import java.util.EnumSet;
64import java.util.Map;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070065import java.util.Optional;
Thomas Vachuska05453c92015-09-09 14:40:49 -070066import java.util.Properties;
Thomas Vachuska6f94ded2015-02-21 14:02:38 -080067import java.util.concurrent.ConcurrentHashMap;
68import java.util.concurrent.ScheduledExecutorService;
69
Thomas Vachuska05453c92015-09-09 14:40:49 -070070import static com.google.common.base.Strings.isNullOrEmpty;
Thomas Vachuska6f94ded2015-02-21 14:02:38 -080071import static java.util.concurrent.Executors.newSingleThreadScheduledExecutor;
72import static java.util.concurrent.TimeUnit.SECONDS;
Thomas Vachuska347cc872015-09-23 10:25:29 -070073import static org.onlab.packet.Ethernet.TYPE_BSN;
74import static org.onlab.packet.Ethernet.TYPE_LLDP;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070075import static org.onlab.util.Tools.get;
Thomas Vachuska6f94ded2015-02-21 14:02:38 -080076import static org.onlab.util.Tools.groupedThreads;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070077import static org.onosproject.net.Link.Type.DIRECT;
Thomas Vachuska6f94ded2015-02-21 14:02:38 -080078import static org.slf4j.LoggerFactory.getLogger;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080079
alshabib7911a052014-10-16 17:49:37 -070080/**
Thomas Vachuska05453c92015-09-09 14:40:49 -070081 * Provider which uses LLDP and BDDP packets to detect network infrastructure links.
alshabib7911a052014-10-16 17:49:37 -070082 */
83@Component(immediate = true)
Jonathan Hartb35540a2015-11-17 09:30:56 -080084public class LldpLinkProvider extends AbstractProvider implements LinkProvider {
alshabib7911a052014-10-16 17:49:37 -070085
Thomas Vachuskafc52fec2015-05-18 19:13:56 -070086 private static final String PROVIDER_NAME = "org.onosproject.provider.lldp";
87
Thomas Vachuska05453c92015-09-09 14:40:49 -070088 private static final String FORMAT =
89 "Settings: enabled={}, useBDDP={}, probeRate={}, " +
90 "staleLinkAge={}, lldpSuppression={}";
Yuta HIGUCHI41289382014-12-19 17:47:12 -080091
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070092 // When a Device/Port has this annotation, do not send out LLDP/BDDP
93 public static final String NO_LLDP = "no-lldp";
94
95
alshabib7911a052014-10-16 17:49:37 -070096 private final Logger log = getLogger(getClass());
97
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080098 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
99 protected CoreService coreService;
100
101 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
alshabib7911a052014-10-16 17:49:37 -0700102 protected LinkProviderRegistry providerRegistry;
103
104 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
105 protected DeviceService deviceService;
106
107 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700108 protected LinkService linkService;
109
110 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Marc De Leenheer8b3e80b2015-03-06 14:27:03 -0800111 protected PacketService packetService;
alshabib7911a052014-10-16 17:49:37 -0700112
alshabib875d6262014-10-17 16:19:40 -0700113 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
114 protected MastershipService masterService;
115
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700116 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
117 protected ComponentConfigService cfgService;
118
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700119 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
120 protected ClusterService clusterService;
121
122 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
123 protected NetworkConfigRegistry cfgRegistry;
124
alshabib7911a052014-10-16 17:49:37 -0700125 private LinkProviderService providerService;
126
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800127 private ScheduledExecutorService executor;
128
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700129 // TODO: Add sanity checking for the configurable params based on the delays
130 private static final long DEVICE_SYNC_DELAY = 5;
131 private static final long LINK_PRUNER_DELAY = 3;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700132
133 private static final String PROP_ENABLED = "enabled";
134 @Property(name = PROP_ENABLED, boolValue = true,
135 label = "If false, link discovery is disabled")
136 private boolean enabled = false;
137
138 private static final String PROP_USE_BDDP = "useBDDP";
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700139 @Property(name = PROP_USE_BDDP, boolValue = true,
140 label = "Use BDDP for link discovery")
Jonathan Hartb35540a2015-11-17 09:30:56 -0800141 private boolean useBddp = true;
alshabib7911a052014-10-16 17:49:37 -0700142
Thomas Vachuska05453c92015-09-09 14:40:49 -0700143 private static final String PROP_PROBE_RATE = "probeRate";
144 private static final int DEFAULT_PROBE_RATE = 3_000;
145 @Property(name = PROP_PROBE_RATE, intValue = DEFAULT_PROBE_RATE,
146 label = "LLDP and BDDP probe rate specified in millis")
147 private int probeRate = DEFAULT_PROBE_RATE;
Saurav Dasc313c402015-02-27 10:09:47 -0800148
Thomas Vachuska05453c92015-09-09 14:40:49 -0700149 private static final String PROP_STALE_LINK_AGE = "staleLinkAge";
150 private static final int DEFAULT_STALE_LINK_AGE = 10_000;
151 @Property(name = PROP_STALE_LINK_AGE, intValue = DEFAULT_STALE_LINK_AGE,
152 label = "Number of millis beyond which links will be considered stale")
153 private int staleLinkAge = DEFAULT_STALE_LINK_AGE;
alshabib7911a052014-10-16 17:49:37 -0700154
Thomas Vachuska05453c92015-09-09 14:40:49 -0700155 // FIXME: convert to use network config subsystem instead
156 private static final String PROP_LLDP_SUPPRESSION = "lldpSuppression";
157 private static final String DEFAULT_LLDP_SUPPRESSION_CONFIG = "../config/lldp_suppression.json";
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700158 @Property(name = PROP_LLDP_SUPPRESSION, value = DEFAULT_LLDP_SUPPRESSION_CONFIG,
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800159 label = "Path to LLDP suppression configuration file")
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700160 private String lldpSuppression = DEFAULT_LLDP_SUPPRESSION_CONFIG;
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800161
Thomas Vachuska05453c92015-09-09 14:40:49 -0700162 private final DiscoveryContext context = new InternalDiscoveryContext();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800163 private final InternalRoleListener roleListener = new InternalRoleListener();
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700164 private final InternalDeviceListener deviceListener = new InternalDeviceListener();
165 private final InternalPacketProcessor packetProcessor = new InternalPacketProcessor();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800166
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700167 // Device link discovery helpers.
alshabib7911a052014-10-16 17:49:37 -0700168 protected final Map<DeviceId, LinkDiscovery> discoverers = new ConcurrentHashMap<>();
169
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700170 // Most recent time a tracked link was seen; links are tracked if their
171 // destination connection point is mastered by this controller instance.
172 private final Map<LinkKey, Long> linkTimes = Maps.newConcurrentMap();
173
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800174 private SuppressionRules rules;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800175 private ApplicationId appId;
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800176
alshabib7911a052014-10-16 17:49:37 -0700177 /**
178 * Creates an OpenFlow link provider.
179 */
Jonathan Hartb35540a2015-11-17 09:30:56 -0800180 public LldpLinkProvider() {
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700181 super(new ProviderId("lldp", PROVIDER_NAME));
alshabib7911a052014-10-16 17:49:37 -0700182 }
183
184 @Activate
Saurav Dasc313c402015-02-27 10:09:47 -0800185 public void activate(ComponentContext context) {
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700186 cfgService.registerProperties(getClass());
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700187 appId = coreService.registerApplication(PROVIDER_NAME);
Saurav Dasc313c402015-02-27 10:09:47 -0800188 modified(context);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700189 log.info("Started");
190 }
191
192 @Deactivate
193 public void deactivate() {
194 cfgService.unregisterProperties(getClass(), false);
195 disable();
196 log.info("Stopped");
197 }
198
199 @Modified
200 public void modified(ComponentContext context) {
201 Dictionary<?, ?> properties = context != null ? context.getProperties() : new Properties();
202
203 boolean newEnabled, newUseBddp;
204 int newProbeRate, newStaleLinkAge;
205 String newLldpSuppression;
206 try {
207 String s = get(properties, PROP_ENABLED);
208 newEnabled = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
209
210 s = get(properties, PROP_USE_BDDP);
211 newUseBddp = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
212
213 s = get(properties, PROP_PROBE_RATE);
214 newProbeRate = isNullOrEmpty(s) ? probeRate : Integer.parseInt(s.trim());
215
216 s = get(properties, PROP_STALE_LINK_AGE);
217 newStaleLinkAge = isNullOrEmpty(s) ? staleLinkAge : Integer.parseInt(s.trim());
218
219 s = get(properties, PROP_LLDP_SUPPRESSION);
220 newLldpSuppression = isNullOrEmpty(s) ? DEFAULT_LLDP_SUPPRESSION_CONFIG : s;
221
222 } catch (NumberFormatException e) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700223 log.warn("Component configuration had invalid values", e);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700224 newEnabled = enabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800225 newUseBddp = useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700226 newProbeRate = probeRate;
227 newStaleLinkAge = staleLinkAge;
228 newLldpSuppression = lldpSuppression;
Saurav Dasc313c402015-02-27 10:09:47 -0800229 }
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800230
Thomas Vachuska05453c92015-09-09 14:40:49 -0700231 boolean wasEnabled = enabled;
232
233 enabled = newEnabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800234 useBddp = newUseBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700235 probeRate = newProbeRate;
236 staleLinkAge = newStaleLinkAge;
237 lldpSuppression = newLldpSuppression;
238
239 if (!wasEnabled && enabled) {
240 enable();
241 } else if (wasEnabled && !enabled) {
242 disable();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700243 } else {
244 // reflect changes in suppression rules to discovery helpers
245 // FIXME: After migrating to Network Configuration Subsystem,
246 // it should be possible to update only changed subset
247 if (enabled) {
248 // update all discovery helper state
249 loadDevices();
250 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700251 }
252
Jonathan Hartb35540a2015-11-17 09:30:56 -0800253 log.info(FORMAT, enabled, useBddp, probeRate, staleLinkAge, lldpSuppression);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700254 }
255
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700256 /**
257 * Enables link discovery processing.
258 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700259 private void enable() {
alshabib7911a052014-10-16 17:49:37 -0700260 providerService = providerRegistry.register(this);
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800261 masterService.addListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700262 deviceService.addListener(deviceListener);
263 packetService.addProcessor(packetProcessor, PacketProcessor.advisor(0));
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800264
Jon Hall3edc08a2015-09-14 16:59:07 -0700265 loadSuppressionRules();
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700266 loadDevices();
Thomas Vachuska05453c92015-09-09 14:40:49 -0700267
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700268 executor = newSingleThreadScheduledExecutor(groupedThreads("onos/link", "discovery-%d"));
269 executor.scheduleAtFixedRate(new SyncDeviceInfoTask(),
270 DEVICE_SYNC_DELAY, DEVICE_SYNC_DELAY, SECONDS);
271 executor.scheduleAtFixedRate(new LinkPrunerTask(),
272 LINK_PRUNER_DELAY, LINK_PRUNER_DELAY, SECONDS);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700273
Thomas Vachuska05453c92015-09-09 14:40:49 -0700274 requestIntercepts();
275 }
276
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700277 /**
278 * Disables link discovery processing.
279 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700280 private void disable() {
281 withdrawIntercepts();
282
283 providerRegistry.unregister(this);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700284 masterService.removeListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700285 deviceService.removeListener(deviceListener);
286 packetService.removeProcessor(packetProcessor);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700287
288 if (executor != null) {
289 executor.shutdownNow();
290 }
291 discoverers.values().forEach(LinkDiscovery::stop);
292 discoverers.clear();
293
294 providerService = null;
295 }
296
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700297 /**
298 * Loads available devices and registers their ports to be probed.
299 */
300 private void loadDevices() {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700301 deviceService.getAvailableDevices()
302 .forEach(d -> updateDevice(d)
303 .ifPresent(ld -> updatePorts(ld, d.id())));
304 }
305
306 /**
307 * Updates discovery helper for specified device.
308 *
309 * Adds and starts a discovery helper for specified device if enabled,
310 * calls {@link #removeDevice(DeviceId)} otherwise.
311 *
312 * @param device device to add
313 * @return discovery helper if discovery is enabled for the device
314 */
315 private Optional<LinkDiscovery> updateDevice(Device device) {
316 if (rules.isSuppressed(device)) {
317 log.trace("LinkDiscovery from {} disabled by configuration", device.id());
318 removeDevice(device.id());
319 return Optional.empty();
320 }
321 LinkDiscovery ld = discoverers.computeIfAbsent(device.id(),
322 did -> new LinkDiscovery(device, context));
323 if (ld.isStopped()) {
324 ld.start();
325 }
326 return Optional.of(ld);
327 }
328
329 /**
330 * Removes after stopping discovery helper for specified device.
331 * @param deviceId device to remove
332 */
333 private void removeDevice(final DeviceId deviceId) {
334 discoverers.computeIfPresent(deviceId, (did, ld) -> {
335 ld.stop();
336 providerService.linksVanished(deviceId);
337 return null;
338 });
339
340 }
341
342 /**
343 * Updates ports of the specified device to the specified discovery helper.
344 */
345 private void updatePorts(LinkDiscovery discoverer, DeviceId deviceId) {
346 deviceService.getPorts(deviceId).forEach(p -> updatePort(discoverer, p));
347 }
348
349 /**
350 * Updates discovery helper state of the specified port.
351 *
352 * Adds a port to the discovery helper if up and discovery is enabled,
353 * or calls {@link #removePort(Port)} otherwise.
354 */
355 private void updatePort(LinkDiscovery discoverer, Port port) {
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800356 if (port.number().isLogical()) {
357 // silently ignore logical ports
358 return;
359 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700360 if (rules.isSuppressed(port)) {
361 log.trace("LinkDiscovery from {} disabled by configuration", port);
362 removePort(port);
363 return;
364 }
365
366 // check if enabled and turn off discovery?
367 if (!port.isEnabled()) {
368 removePort(port);
369 return;
370 }
371
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800372 discoverer.addPort(port);
alshabib7911a052014-10-16 17:49:37 -0700373 }
374
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700375 /**
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700376 * Removes a port from the specified discovery helper.
377 * @param port the port
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700378 */
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700379 private void removePort(Port port) {
380 if (port.element() instanceof Device) {
381 Device d = (Device) port.element();
382 LinkDiscovery ld = discoverers.get(d.id());
383 if (ld != null) {
384 ld.removePort(port.number());
Jonathan Hart45066bc2015-07-28 11:18:34 -0700385 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700386
387 ConnectPoint point = new ConnectPoint(d.id(), port.number());
388 providerService.linksVanished(point);
389 } else {
390 log.warn("Attempted to remove non-Device port", port);
Jonathan Hart45066bc2015-07-28 11:18:34 -0700391 }
392 }
393
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700394 /**
395 * Loads LLDP suppression rules.
396 */
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800397 private void loadSuppressionRules() {
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700398 // FIXME: convert to use network configuration
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700399 SuppressionRulesStore store = new SuppressionRulesStore(lldpSuppression);
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800400 try {
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700401 log.info("Reading suppression rules from {}", lldpSuppression);
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800402 rules = store.read();
403 } catch (IOException e) {
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700404 log.info("Failed to load {}, using built-in rules", lldpSuppression);
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800405 // default rule to suppress ROADM to maintain compatibility
406 rules = new SuppressionRules(ImmutableSet.of(),
407 EnumSet.of(Device.Type.ROADM),
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700408 ImmutableMap.of(NO_LLDP, SuppressionRules.ANY_VALUE));
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800409 }
410
411 // should refresh discoverers when we need dynamic reconfiguration
412 }
413
Charles M.C. Chane148de82015-05-06 12:38:21 +0800414 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700415 * Requests packet intercepts.
Charles M.C. Chane148de82015-05-06 12:38:21 +0800416 */
Thomas Vachuska27bee092015-06-23 19:03:10 -0700417 private void requestIntercepts() {
418 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700419 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700420 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800421
Thomas Vachuska347cc872015-09-23 10:25:29 -0700422 selector.matchEthType(TYPE_BSN);
Jonathan Hartb35540a2015-11-17 09:30:56 -0800423 if (useBddp) {
Thomas Vachuska27bee092015-06-23 19:03:10 -0700424 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
425 } else {
426 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800427 }
428 }
429
Thomas Vachuska27bee092015-06-23 19:03:10 -0700430 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700431 * Withdraws packet intercepts.
Thomas Vachuska27bee092015-06-23 19:03:10 -0700432 */
433 private void withdrawIntercepts() {
434 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700435 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700436 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Thomas Vachuska347cc872015-09-23 10:25:29 -0700437 selector.matchEthType(TYPE_BSN);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700438 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
439 }
440
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700441 /**
442 * Processes device mastership role changes.
443 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800444 private class InternalRoleListener implements MastershipListener {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800445 @Override
446 public void event(MastershipEvent event) {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800447 if (MastershipEvent.Type.BACKUPS_CHANGED.equals(event.type())) {
448 // only need new master events
449 return;
450 }
451
452 DeviceId deviceId = event.subject();
453 Device device = deviceService.getDevice(deviceId);
454 if (device == null) {
Thomas Vachuska3358af22015-05-19 18:40:34 -0700455 log.debug("Device {} doesn't exist, or isn't there yet", deviceId);
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800456 return;
457 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700458 if (clusterService.getLocalNode().id().equals(event.roleInfo().master())) {
459 updateDevice(device).ifPresent(ld -> updatePorts(ld, device.id()));
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800460 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800461 }
462
463 }
alshabib7911a052014-10-16 17:49:37 -0700464
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700465 /**
466 * Processes device events.
467 */
468 private class InternalDeviceListener implements DeviceListener {
alshabib7911a052014-10-16 17:49:37 -0700469 @Override
470 public void event(DeviceEvent event) {
alshabib7911a052014-10-16 17:49:37 -0700471 Device device = event.subject();
alshabibacd91832014-10-17 14:38:41 -0700472 Port port = event.port();
alshabibdfc7afb2014-10-21 20:13:27 -0700473 if (device == null) {
474 log.error("Device is null.");
475 return;
476 }
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700477 log.trace("{} {} {}", event.type(), event.subject(), event);
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700478 final DeviceId deviceId = device.id();
alshabib7911a052014-10-16 17:49:37 -0700479 switch (event.type()) {
480 case DEVICE_ADDED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700481 case DEVICE_UPDATED:
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700482 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700483 break;
484 case PORT_ADDED:
485 case PORT_UPDATED:
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700486 if (port.isEnabled()) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700487 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
alshabib7911a052014-10-16 17:49:37 -0700488 } else {
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700489 log.debug("Port down {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700490 removePort(port);
alshabib7911a052014-10-16 17:49:37 -0700491 }
492 break;
493 case PORT_REMOVED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700494 log.debug("Port removed {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700495 removePort(port);
alshabib7911a052014-10-16 17:49:37 -0700496 break;
497 case DEVICE_REMOVED:
498 case DEVICE_SUSPENDED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700499 log.debug("Device removed {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700500 removeDevice(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700501 break;
502 case DEVICE_AVAILABILITY_CHANGED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700503 if (deviceService.isAvailable(deviceId)) {
504 log.debug("Device up {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700505 updateDevice(device);
alshabib7911a052014-10-16 17:49:37 -0700506 } else {
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700507 log.debug("Device down {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700508 removeDevice(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700509 }
510 break;
Jonathan Hart9de692c2015-04-23 11:45:47 -0700511 case PORT_STATS_UPDATED:
512 break;
alshabib7911a052014-10-16 17:49:37 -0700513 default:
514 log.debug("Unknown event {}", event);
515 }
516 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700517 }
alshabib7911a052014-10-16 17:49:37 -0700518
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700519 /**
520 * Processes incoming packets.
521 */
522 private class InternalPacketProcessor implements PacketProcessor {
alshabib7911a052014-10-16 17:49:37 -0700523 @Override
524 public void process(PacketContext context) {
Thomas Vachuska347cc872015-09-23 10:25:29 -0700525 if (context == null || context.isHandled()) {
alshabib4a179dc2014-10-17 17:17:01 -0700526 return;
527 }
Thomas Vachuska347cc872015-09-23 10:25:29 -0700528
529 Ethernet eth = context.inPacket().parsed();
530 if (eth == null || (eth.getEtherType() != TYPE_LLDP && eth.getEtherType() != TYPE_BSN)) {
531 return;
532 }
533
Thomas Vachuska96f3ea72015-09-08 13:50:12 -0700534 LinkDiscovery ld = discoverers.get(context.inPacket().receivedFrom().deviceId());
alshabib7911a052014-10-16 17:49:37 -0700535 if (ld == null) {
536 return;
537 }
538
Jonathan Hartb35540a2015-11-17 09:30:56 -0800539 if (ld.handleLldp(context)) {
alshabib7911a052014-10-16 17:49:37 -0700540 context.block();
541 }
542 }
543 }
544
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700545 /**
546 * Auxiliary task to keep device ports up to date.
547 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800548 private final class SyncDeviceInfoTask implements Runnable {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800549 @Override
550 public void run() {
551 if (Thread.currentThread().isInterrupted()) {
552 log.info("Interrupted, quitting");
553 return;
554 }
555 // check what deviceService sees, to see if we are missing anything
556 try {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700557 loadDevices();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800558 } catch (Exception e) {
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700559 // Catch all exceptions to avoid task being suppressed
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800560 log.error("Exception thrown during synchronization process", e);
561 }
562 }
563 }
564
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700565 /**
566 * Auxiliary task for pruning stale links.
567 */
568 private class LinkPrunerTask implements Runnable {
569 @Override
570 public void run() {
571 if (Thread.currentThread().isInterrupted()) {
572 log.info("Interrupted, quitting");
573 return;
574 }
575
576 try {
577 // TODO: There is still a slight possibility of mastership
578 // change occurring right with link going stale. This will
579 // result in the stale link not being pruned.
580 Maps.filterEntries(linkTimes, e -> {
581 if (!masterService.isLocalMaster(e.getKey().dst().deviceId())) {
582 return true;
583 }
584 if (isStale(e.getValue())) {
585 providerService.linkVanished(new DefaultLinkDescription(e.getKey().src(),
586 e.getKey().dst(),
587 DIRECT));
588 return true;
589 }
590 return false;
591 }).clear();
592
593 } catch (Exception e) {
594 // Catch all exceptions to avoid task being suppressed
595 log.error("Exception thrown during link pruning process", e);
596 }
597 }
598
599 private boolean isStale(long lastSeen) {
600 return lastSeen < System.currentTimeMillis() - staleLinkAge;
601 }
602 }
603
604 /**
605 * Provides processing context for the device link discovery helpers.
606 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700607 private class InternalDiscoveryContext implements DiscoveryContext {
608 @Override
609 public MastershipService mastershipService() {
610 return masterService;
611 }
612
613 @Override
614 public LinkProviderService providerService() {
615 return providerService;
616 }
617
618 @Override
619 public PacketService packetService() {
620 return packetService;
621 }
622
623 @Override
624 public long probeRate() {
625 return probeRate;
626 }
627
628 @Override
Jonathan Hartb35540a2015-11-17 09:30:56 -0800629 public boolean useBddp() {
630 return useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700631 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700632
633 @Override
634 public void touchLink(LinkKey key) {
635 linkTimes.put(key, System.currentTimeMillis());
636 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700637 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700638
alshabib7911a052014-10-16 17:49:37 -0700639}