blob: bf88f572dbd22eb597d3bd6c5b5fb6121af3572f [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
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -080018import static com.google.common.base.Strings.isNullOrEmpty;
19import static java.util.concurrent.Executors.newSingleThreadScheduledExecutor;
20import static java.util.concurrent.TimeUnit.SECONDS;
21import static org.onlab.packet.Ethernet.TYPE_BSN;
22import static org.onlab.packet.Ethernet.TYPE_LLDP;
23import static org.onlab.util.Tools.get;
24import static org.onlab.util.Tools.groupedThreads;
25import static org.onosproject.net.Link.Type.DIRECT;
26import static org.onosproject.net.config.basics.SubjectFactories.APP_SUBJECT_FACTORY;
27import static org.onosproject.net.config.basics.SubjectFactories.CONNECT_POINT_SUBJECT_FACTORY;
28import static org.onosproject.net.config.basics.SubjectFactories.DEVICE_SUBJECT_FACTORY;
29import static org.slf4j.LoggerFactory.getLogger;
30
31import java.util.Dictionary;
32import java.util.EnumSet;
33import java.util.Map;
34import java.util.Optional;
35import java.util.Properties;
36import java.util.Set;
37import java.util.concurrent.ConcurrentHashMap;
38import java.util.concurrent.ScheduledExecutorService;
39
alshabib7911a052014-10-16 17:49:37 -070040import org.apache.felix.scr.annotations.Activate;
41import org.apache.felix.scr.annotations.Component;
42import org.apache.felix.scr.annotations.Deactivate;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080043import org.apache.felix.scr.annotations.Modified;
44import org.apache.felix.scr.annotations.Property;
alshabib7911a052014-10-16 17:49:37 -070045import org.apache.felix.scr.annotations.Reference;
46import org.apache.felix.scr.annotations.ReferenceCardinality;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080047import org.onlab.packet.Ethernet;
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -080048import org.onlab.util.Tools;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070049import org.onosproject.cfg.ComponentConfigService;
Ayaka Koshibe12c8c082015-12-08 12:48:46 -080050import org.onosproject.cluster.ClusterMetadataService;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070051import org.onosproject.cluster.ClusterService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080052import org.onosproject.core.ApplicationId;
53import org.onosproject.core.CoreService;
Brian O'Connorabafb502014-12-02 22:26:20 -080054import org.onosproject.mastership.MastershipEvent;
55import org.onosproject.mastership.MastershipListener;
56import org.onosproject.mastership.MastershipService;
57import org.onosproject.net.ConnectPoint;
58import org.onosproject.net.Device;
59import org.onosproject.net.DeviceId;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070060import org.onosproject.net.LinkKey;
Brian O'Connorabafb502014-12-02 22:26:20 -080061import org.onosproject.net.Port;
Naoki Shiota399a0b32015-11-15 20:36:13 -060062import org.onosproject.net.config.ConfigFactory;
63import org.onosproject.net.config.NetworkConfigEvent;
64import org.onosproject.net.config.NetworkConfigListener;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070065import org.onosproject.net.config.NetworkConfigRegistry;
Brian O'Connorabafb502014-12-02 22:26:20 -080066import org.onosproject.net.device.DeviceEvent;
HIGUCHI Yuta1979f552015-12-28 21:24:26 -080067import org.onosproject.net.device.DeviceEvent.Type;
Brian O'Connorabafb502014-12-02 22:26:20 -080068import org.onosproject.net.device.DeviceListener;
69import org.onosproject.net.device.DeviceService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080070import org.onosproject.net.flow.DefaultTrafficSelector;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080071import org.onosproject.net.flow.TrafficSelector;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070072import org.onosproject.net.link.DefaultLinkDescription;
Brian O'Connorabafb502014-12-02 22:26:20 -080073import org.onosproject.net.link.LinkProvider;
74import org.onosproject.net.link.LinkProviderRegistry;
75import org.onosproject.net.link.LinkProviderService;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070076import org.onosproject.net.link.LinkService;
Brian O'Connorabafb502014-12-02 22:26:20 -080077import org.onosproject.net.packet.PacketContext;
Jonathan Hart3cfce8e2015-01-14 16:43:27 -080078import org.onosproject.net.packet.PacketPriority;
Brian O'Connorabafb502014-12-02 22:26:20 -080079import org.onosproject.net.packet.PacketProcessor;
80import org.onosproject.net.packet.PacketService;
81import org.onosproject.net.provider.AbstractProvider;
82import org.onosproject.net.provider.ProviderId;
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -080083import org.onosproject.store.service.ConsistentMapException;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080084import org.osgi.service.component.ComponentContext;
alshabib7911a052014-10-16 17:49:37 -070085import org.slf4j.Logger;
86
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -080087import com.google.common.collect.ImmutableMap;
88import com.google.common.collect.ImmutableSet;
89import com.google.common.collect.Maps;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080090
alshabib7911a052014-10-16 17:49:37 -070091/**
Thomas Vachuska05453c92015-09-09 14:40:49 -070092 * Provider which uses LLDP and BDDP packets to detect network infrastructure links.
alshabib7911a052014-10-16 17:49:37 -070093 */
94@Component(immediate = true)
Jonathan Hartb35540a2015-11-17 09:30:56 -080095public class LldpLinkProvider extends AbstractProvider implements LinkProvider {
alshabib7911a052014-10-16 17:49:37 -070096
Thomas Vachuskafc52fec2015-05-18 19:13:56 -070097 private static final String PROVIDER_NAME = "org.onosproject.provider.lldp";
98
Thomas Vachuska05453c92015-09-09 14:40:49 -070099 private static final String FORMAT =
100 "Settings: enabled={}, useBDDP={}, probeRate={}, " +
Naoki Shiota399a0b32015-11-15 20:36:13 -0600101 "staleLinkAge={}";
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800102
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700103 // When a Device/Port has this annotation, do not send out LLDP/BDDP
104 public static final String NO_LLDP = "no-lldp";
105
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -0800106 private static final int MAX_RETRIES = 5;
107 private static final int RETRY_DELAY = 1_000; // millis
108
alshabib7911a052014-10-16 17:49:37 -0700109 private final Logger log = getLogger(getClass());
110
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800111 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
112 protected CoreService coreService;
113
114 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
alshabib7911a052014-10-16 17:49:37 -0700115 protected LinkProviderRegistry providerRegistry;
116
117 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
118 protected DeviceService deviceService;
119
120 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700121 protected LinkService linkService;
122
123 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Marc De Leenheer8b3e80b2015-03-06 14:27:03 -0800124 protected PacketService packetService;
alshabib7911a052014-10-16 17:49:37 -0700125
alshabib875d6262014-10-17 16:19:40 -0700126 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
127 protected MastershipService masterService;
128
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700129 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
130 protected ComponentConfigService cfgService;
131
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700132 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
133 protected ClusterService clusterService;
134
135 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
136 protected NetworkConfigRegistry cfgRegistry;
137
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800138 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
139 protected ClusterMetadataService clusterMetadataService;
140
alshabib7911a052014-10-16 17:49:37 -0700141 private LinkProviderService providerService;
142
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800143 private ScheduledExecutorService executor;
144
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700145 // TODO: Add sanity checking for the configurable params based on the delays
146 private static final long DEVICE_SYNC_DELAY = 5;
147 private static final long LINK_PRUNER_DELAY = 3;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700148
149 private static final String PROP_ENABLED = "enabled";
150 @Property(name = PROP_ENABLED, boolValue = true,
151 label = "If false, link discovery is disabled")
152 private boolean enabled = false;
153
154 private static final String PROP_USE_BDDP = "useBDDP";
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700155 @Property(name = PROP_USE_BDDP, boolValue = true,
156 label = "Use BDDP for link discovery")
Jonathan Hartb35540a2015-11-17 09:30:56 -0800157 private boolean useBddp = true;
alshabib7911a052014-10-16 17:49:37 -0700158
Thomas Vachuska05453c92015-09-09 14:40:49 -0700159 private static final String PROP_PROBE_RATE = "probeRate";
Thomas Vachuska9e51fd02016-01-04 16:51:28 -0800160 private static final int DEFAULT_PROBE_RATE = 3000;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700161 @Property(name = PROP_PROBE_RATE, intValue = DEFAULT_PROBE_RATE,
162 label = "LLDP and BDDP probe rate specified in millis")
163 private int probeRate = DEFAULT_PROBE_RATE;
Saurav Dasc313c402015-02-27 10:09:47 -0800164
Thomas Vachuska05453c92015-09-09 14:40:49 -0700165 private static final String PROP_STALE_LINK_AGE = "staleLinkAge";
Thomas Vachuska9e51fd02016-01-04 16:51:28 -0800166 private static final int DEFAULT_STALE_LINK_AGE = 10000;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700167 @Property(name = PROP_STALE_LINK_AGE, intValue = DEFAULT_STALE_LINK_AGE,
168 label = "Number of millis beyond which links will be considered stale")
169 private int staleLinkAge = DEFAULT_STALE_LINK_AGE;
alshabib7911a052014-10-16 17:49:37 -0700170
Thomas Vachuska05453c92015-09-09 14:40:49 -0700171 private final DiscoveryContext context = new InternalDiscoveryContext();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800172 private final InternalRoleListener roleListener = new InternalRoleListener();
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700173 private final InternalDeviceListener deviceListener = new InternalDeviceListener();
174 private final InternalPacketProcessor packetProcessor = new InternalPacketProcessor();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800175
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700176 // Device link discovery helpers.
alshabib7911a052014-10-16 17:49:37 -0700177 protected final Map<DeviceId, LinkDiscovery> discoverers = new ConcurrentHashMap<>();
178
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700179 // Most recent time a tracked link was seen; links are tracked if their
180 // destination connection point is mastered by this controller instance.
181 private final Map<LinkKey, Long> linkTimes = Maps.newConcurrentMap();
182
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800183 private ApplicationId appId;
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800184
Naoki Shiota399a0b32015-11-15 20:36:13 -0600185 static final SuppressionRules DEFAULT_RULES
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800186 = new SuppressionRules(EnumSet.of(Device.Type.ROADM),
Naoki Shiota399a0b32015-11-15 20:36:13 -0600187 ImmutableMap.of(NO_LLDP, SuppressionRules.ANY_VALUE));
188
189 private SuppressionRules rules = LldpLinkProvider.DEFAULT_RULES;
190
191 public static final String CONFIG_KEY = "suppression";
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800192 public static final String FEATURE_NAME = "linkDiscovery";
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800193 public static final String FINGERPRINT_FEATURE_NAME = "fingerprint";
Naoki Shiota399a0b32015-11-15 20:36:13 -0600194
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800195 private final Set<ConfigFactory<?, ?>> factories = ImmutableSet.of(
Naoki Shiota399a0b32015-11-15 20:36:13 -0600196 new ConfigFactory<ApplicationId, SuppressionConfig>(APP_SUBJECT_FACTORY,
197 SuppressionConfig.class,
198 CONFIG_KEY) {
199 @Override
200 public SuppressionConfig createConfig() {
201 return new SuppressionConfig();
202 }
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800203 },
204 new ConfigFactory<DeviceId, LinkDiscoveryFromDevice>(DEVICE_SUBJECT_FACTORY,
205 LinkDiscoveryFromDevice.class, FEATURE_NAME) {
206 @Override
207 public LinkDiscoveryFromDevice createConfig() {
208 return new LinkDiscoveryFromDevice();
209 }
210 },
211 new ConfigFactory<ConnectPoint, LinkDiscoveryFromPort>(CONNECT_POINT_SUBJECT_FACTORY,
212 LinkDiscoveryFromPort.class, FEATURE_NAME) {
213 @Override
214 public LinkDiscoveryFromPort createConfig() {
215 return new LinkDiscoveryFromPort();
216 }
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800217 },
218 new ConfigFactory<DeviceId, FingerprintProbeFromDevice>(DEVICE_SUBJECT_FACTORY,
219 FingerprintProbeFromDevice.class, FINGERPRINT_FEATURE_NAME) {
220 @Override
221 public FingerprintProbeFromDevice createConfig() {
222 return new FingerprintProbeFromDevice();
223 }
Naoki Shiota399a0b32015-11-15 20:36:13 -0600224 }
225 );
226
227 private final InternalConfigListener cfgListener = new InternalConfigListener();
228
229
alshabib7911a052014-10-16 17:49:37 -0700230 /**
231 * Creates an OpenFlow link provider.
232 */
Jonathan Hartb35540a2015-11-17 09:30:56 -0800233 public LldpLinkProvider() {
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700234 super(new ProviderId("lldp", PROVIDER_NAME));
alshabib7911a052014-10-16 17:49:37 -0700235 }
236
237 @Activate
Saurav Dasc313c402015-02-27 10:09:47 -0800238 public void activate(ComponentContext context) {
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700239 cfgService.registerProperties(getClass());
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700240 appId = coreService.registerApplication(PROVIDER_NAME);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600241
242 cfgRegistry.addListener(cfgListener);
243 factories.forEach(cfgRegistry::registerConfigFactory);
244
Thomas Vachuska762a2d82016-01-04 10:25:20 -0800245 SuppressionConfig cfg =
246 Tools.retryable(() -> cfgRegistry.getConfig(appId, SuppressionConfig.class),
247 ConsistentMapException.class, MAX_RETRIES, RETRY_DELAY).get();
Naoki Shiota399a0b32015-11-15 20:36:13 -0600248 if (cfg == null) {
249 // If no configuration is found, register default.
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -0800250 cfg = Tools.retryable(this::setDefaultSuppressionConfig,
251 ConsistentMapException.class,
252 MAX_RETRIES, RETRY_DELAY).get();
Naoki Shiota399a0b32015-11-15 20:36:13 -0600253 }
254 cfgListener.reconfigureSuppressionRules(cfg);
255
Saurav Dasc313c402015-02-27 10:09:47 -0800256 modified(context);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700257 log.info("Started");
258 }
259
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -0800260 private SuppressionConfig setDefaultSuppressionConfig() {
261 SuppressionConfig cfg = cfgRegistry.addConfig(appId, SuppressionConfig.class);
262 cfg.deviceTypes(DEFAULT_RULES.getSuppressedDeviceType())
263 .annotation(DEFAULT_RULES.getSuppressedAnnotation())
264 .apply();
265 return cfg;
266 }
267
Thomas Vachuska05453c92015-09-09 14:40:49 -0700268 @Deactivate
269 public void deactivate() {
Naoki Shiota399a0b32015-11-15 20:36:13 -0600270 cfgRegistry.removeListener(cfgListener);
271 factories.forEach(cfgRegistry::unregisterConfigFactory);
272
Thomas Vachuska05453c92015-09-09 14:40:49 -0700273 cfgService.unregisterProperties(getClass(), false);
274 disable();
275 log.info("Stopped");
276 }
277
278 @Modified
279 public void modified(ComponentContext context) {
280 Dictionary<?, ?> properties = context != null ? context.getProperties() : new Properties();
281
282 boolean newEnabled, newUseBddp;
283 int newProbeRate, newStaleLinkAge;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700284 try {
285 String s = get(properties, PROP_ENABLED);
286 newEnabled = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
287
288 s = get(properties, PROP_USE_BDDP);
289 newUseBddp = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
290
291 s = get(properties, PROP_PROBE_RATE);
292 newProbeRate = isNullOrEmpty(s) ? probeRate : Integer.parseInt(s.trim());
293
294 s = get(properties, PROP_STALE_LINK_AGE);
295 newStaleLinkAge = isNullOrEmpty(s) ? staleLinkAge : Integer.parseInt(s.trim());
296
Thomas Vachuska05453c92015-09-09 14:40:49 -0700297 } catch (NumberFormatException e) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700298 log.warn("Component configuration had invalid values", e);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700299 newEnabled = enabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800300 newUseBddp = useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700301 newProbeRate = probeRate;
302 newStaleLinkAge = staleLinkAge;
Saurav Dasc313c402015-02-27 10:09:47 -0800303 }
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800304
Thomas Vachuska05453c92015-09-09 14:40:49 -0700305 boolean wasEnabled = enabled;
306
307 enabled = newEnabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800308 useBddp = newUseBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700309 probeRate = newProbeRate;
310 staleLinkAge = newStaleLinkAge;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700311
312 if (!wasEnabled && enabled) {
313 enable();
314 } else if (wasEnabled && !enabled) {
315 disable();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700316 } else {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700317 if (enabled) {
318 // update all discovery helper state
319 loadDevices();
320 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700321 }
322
Naoki Shiota399a0b32015-11-15 20:36:13 -0600323 log.info(FORMAT, enabled, useBddp, probeRate, staleLinkAge);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700324 }
325
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700326 /**
327 * Enables link discovery processing.
328 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700329 private void enable() {
alshabib7911a052014-10-16 17:49:37 -0700330 providerService = providerRegistry.register(this);
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800331 masterService.addListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700332 deviceService.addListener(deviceListener);
333 packetService.addProcessor(packetProcessor, PacketProcessor.advisor(0));
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800334
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700335 loadDevices();
Thomas Vachuska05453c92015-09-09 14:40:49 -0700336
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700337 executor = newSingleThreadScheduledExecutor(groupedThreads("onos/link", "discovery-%d"));
338 executor.scheduleAtFixedRate(new SyncDeviceInfoTask(),
339 DEVICE_SYNC_DELAY, DEVICE_SYNC_DELAY, SECONDS);
340 executor.scheduleAtFixedRate(new LinkPrunerTask(),
341 LINK_PRUNER_DELAY, LINK_PRUNER_DELAY, SECONDS);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700342
Thomas Vachuska05453c92015-09-09 14:40:49 -0700343 requestIntercepts();
344 }
345
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700346 /**
347 * Disables link discovery processing.
348 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700349 private void disable() {
350 withdrawIntercepts();
351
352 providerRegistry.unregister(this);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700353 masterService.removeListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700354 deviceService.removeListener(deviceListener);
355 packetService.removeProcessor(packetProcessor);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700356
Naoki Shiota399a0b32015-11-15 20:36:13 -0600357
Thomas Vachuska05453c92015-09-09 14:40:49 -0700358 if (executor != null) {
359 executor.shutdownNow();
360 }
361 discoverers.values().forEach(LinkDiscovery::stop);
362 discoverers.clear();
363
364 providerService = null;
365 }
366
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700367 /**
368 * Loads available devices and registers their ports to be probed.
369 */
370 private void loadDevices() {
Naoki Shiota399a0b32015-11-15 20:36:13 -0600371 if (!enabled) {
372 return;
373 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700374 deviceService.getAvailableDevices()
375 .forEach(d -> updateDevice(d)
376 .ifPresent(ld -> updatePorts(ld, d.id())));
377 }
378
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800379 private boolean isBlacklisted(DeviceId did) {
380 LinkDiscoveryFromDevice cfg = cfgRegistry.getConfig(did, LinkDiscoveryFromDevice.class);
381 if (cfg == null) {
382 return false;
383 }
384 return !cfg.enabled();
385 }
386
387 private boolean isBlacklisted(ConnectPoint cp) {
388 // if parent device is blacklisted, so is the port
389 if (isBlacklisted(cp.deviceId())) {
390 return true;
391 }
392 LinkDiscoveryFromPort cfg = cfgRegistry.getConfig(cp, LinkDiscoveryFromPort.class);
393 if (cfg == null) {
394 return false;
395 }
396 return !cfg.enabled();
397 }
398
399 private boolean isBlacklisted(Port port) {
400 return isBlacklisted(new ConnectPoint(port.element().id(), port.number()));
401 }
402
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800403 private boolean isFingerprinted(DeviceId did) {
404 FingerprintProbeFromDevice cfg = cfgRegistry.getConfig(did, FingerprintProbeFromDevice.class);
405 if (cfg == null) {
406 return false;
407 }
408 return cfg.enabled();
409 }
410
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700411 /**
412 * Updates discovery helper for specified device.
413 *
414 * Adds and starts a discovery helper for specified device if enabled,
415 * calls {@link #removeDevice(DeviceId)} otherwise.
416 *
417 * @param device device to add
418 * @return discovery helper if discovery is enabled for the device
419 */
420 private Optional<LinkDiscovery> updateDevice(Device device) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800421 if (device == null) {
422 return Optional.empty();
423 }
424 if (rules.isSuppressed(device) || isBlacklisted(device.id())) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700425 log.trace("LinkDiscovery from {} disabled by configuration", device.id());
426 removeDevice(device.id());
427 return Optional.empty();
428 }
429 LinkDiscovery ld = discoverers.computeIfAbsent(device.id(),
430 did -> new LinkDiscovery(device, context));
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800431 if (isFingerprinted(device.id())) {
432 ld.enableFingerprint();
433 } else {
434 ld.disableFingerprint();
435 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700436 if (ld.isStopped()) {
437 ld.start();
438 }
439 return Optional.of(ld);
440 }
441
442 /**
443 * Removes after stopping discovery helper for specified device.
444 * @param deviceId device to remove
445 */
446 private void removeDevice(final DeviceId deviceId) {
447 discoverers.computeIfPresent(deviceId, (did, ld) -> {
448 ld.stop();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700449 return null;
450 });
451
452 }
453
454 /**
455 * Updates ports of the specified device to the specified discovery helper.
456 */
457 private void updatePorts(LinkDiscovery discoverer, DeviceId deviceId) {
458 deviceService.getPorts(deviceId).forEach(p -> updatePort(discoverer, p));
459 }
460
461 /**
462 * Updates discovery helper state of the specified port.
463 *
464 * Adds a port to the discovery helper if up and discovery is enabled,
465 * or calls {@link #removePort(Port)} otherwise.
466 */
467 private void updatePort(LinkDiscovery discoverer, Port port) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800468 if (port == null) {
469 return;
470 }
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800471 if (port.number().isLogical()) {
472 // silently ignore logical ports
473 return;
474 }
Naoki Shiota399a0b32015-11-15 20:36:13 -0600475
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800476 if (rules.isSuppressed(port) || isBlacklisted(port)) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700477 log.trace("LinkDiscovery from {} disabled by configuration", port);
478 removePort(port);
479 return;
480 }
481
482 // check if enabled and turn off discovery?
483 if (!port.isEnabled()) {
484 removePort(port);
485 return;
486 }
487
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800488 discoverer.addPort(port);
alshabib7911a052014-10-16 17:49:37 -0700489 }
490
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700491 /**
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700492 * Removes a port from the specified discovery helper.
493 * @param port the port
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700494 */
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700495 private void removePort(Port port) {
496 if (port.element() instanceof Device) {
497 Device d = (Device) port.element();
498 LinkDiscovery ld = discoverers.get(d.id());
499 if (ld != null) {
500 ld.removePort(port.number());
Jonathan Hart45066bc2015-07-28 11:18:34 -0700501 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700502 } else {
503 log.warn("Attempted to remove non-Device port", port);
Jonathan Hart45066bc2015-07-28 11:18:34 -0700504 }
505 }
506
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700507 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700508 * Requests packet intercepts.
Charles M.C. Chane148de82015-05-06 12:38:21 +0800509 */
Thomas Vachuska27bee092015-06-23 19:03:10 -0700510 private void requestIntercepts() {
511 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700512 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700513 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800514
Thomas Vachuska347cc872015-09-23 10:25:29 -0700515 selector.matchEthType(TYPE_BSN);
Jonathan Hartb35540a2015-11-17 09:30:56 -0800516 if (useBddp) {
Thomas Vachuska27bee092015-06-23 19:03:10 -0700517 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
518 } else {
519 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800520 }
521 }
522
Thomas Vachuska27bee092015-06-23 19:03:10 -0700523 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700524 * Withdraws packet intercepts.
Thomas Vachuska27bee092015-06-23 19:03:10 -0700525 */
526 private void withdrawIntercepts() {
527 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700528 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700529 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Thomas Vachuska347cc872015-09-23 10:25:29 -0700530 selector.matchEthType(TYPE_BSN);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700531 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
532 }
533
Naoki Shiota399a0b32015-11-15 20:36:13 -0600534 protected SuppressionRules rules() {
535 return rules;
536 }
537
538 protected void updateRules(SuppressionRules newRules) {
539 if (!rules.equals(newRules)) {
540 rules = newRules;
541 loadDevices();
542 }
543 }
544
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700545 /**
546 * Processes device mastership role changes.
547 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800548 private class InternalRoleListener implements MastershipListener {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800549 @Override
550 public void event(MastershipEvent event) {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800551 if (MastershipEvent.Type.BACKUPS_CHANGED.equals(event.type())) {
552 // only need new master events
553 return;
554 }
555
556 DeviceId deviceId = event.subject();
557 Device device = deviceService.getDevice(deviceId);
558 if (device == null) {
Thomas Vachuska3358af22015-05-19 18:40:34 -0700559 log.debug("Device {} doesn't exist, or isn't there yet", deviceId);
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800560 return;
561 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700562 if (clusterService.getLocalNode().id().equals(event.roleInfo().master())) {
563 updateDevice(device).ifPresent(ld -> updatePorts(ld, device.id()));
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800564 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800565 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800566 }
alshabib7911a052014-10-16 17:49:37 -0700567
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700568 /**
569 * Processes device events.
570 */
571 private class InternalDeviceListener implements DeviceListener {
alshabib7911a052014-10-16 17:49:37 -0700572 @Override
573 public void event(DeviceEvent event) {
HIGUCHI Yuta1979f552015-12-28 21:24:26 -0800574 if (event.type() == Type.PORT_STATS_UPDATED) {
575 return;
576 }
alshabib7911a052014-10-16 17:49:37 -0700577 Device device = event.subject();
alshabibacd91832014-10-17 14:38:41 -0700578 Port port = event.port();
alshabibdfc7afb2014-10-21 20:13:27 -0700579 if (device == null) {
580 log.error("Device is null.");
581 return;
582 }
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700583 log.trace("{} {} {}", event.type(), event.subject(), event);
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700584 final DeviceId deviceId = device.id();
alshabib7911a052014-10-16 17:49:37 -0700585 switch (event.type()) {
586 case DEVICE_ADDED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700587 case DEVICE_UPDATED:
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700588 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700589 break;
590 case PORT_ADDED:
591 case PORT_UPDATED:
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700592 if (port.isEnabled()) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700593 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
alshabib7911a052014-10-16 17:49:37 -0700594 } else {
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700595 log.debug("Port down {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700596 removePort(port);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600597 providerService.linksVanished(new ConnectPoint(port.element().id(),
598 port.number()));
alshabib7911a052014-10-16 17:49:37 -0700599 }
600 break;
601 case PORT_REMOVED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700602 log.debug("Port removed {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700603 removePort(port);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600604 providerService.linksVanished(new ConnectPoint(port.element().id(),
605 port.number()));
alshabib7911a052014-10-16 17:49:37 -0700606 break;
607 case DEVICE_REMOVED:
608 case DEVICE_SUSPENDED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700609 log.debug("Device removed {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700610 removeDevice(deviceId);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600611 providerService.linksVanished(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700612 break;
613 case DEVICE_AVAILABILITY_CHANGED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700614 if (deviceService.isAvailable(deviceId)) {
615 log.debug("Device up {}", deviceId);
alshabibe3af2652015-12-01 23:05:34 -0800616 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700617 } else {
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700618 log.debug("Device down {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700619 removeDevice(deviceId);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600620 providerService.linksVanished(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700621 }
622 break;
Jonathan Hart9de692c2015-04-23 11:45:47 -0700623 case PORT_STATS_UPDATED:
624 break;
alshabib7911a052014-10-16 17:49:37 -0700625 default:
626 log.debug("Unknown event {}", event);
627 }
628 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700629 }
alshabib7911a052014-10-16 17:49:37 -0700630
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700631 /**
632 * Processes incoming packets.
633 */
634 private class InternalPacketProcessor implements PacketProcessor {
alshabib7911a052014-10-16 17:49:37 -0700635 @Override
636 public void process(PacketContext context) {
Thomas Vachuska347cc872015-09-23 10:25:29 -0700637 if (context == null || context.isHandled()) {
alshabib4a179dc2014-10-17 17:17:01 -0700638 return;
639 }
Thomas Vachuska347cc872015-09-23 10:25:29 -0700640
641 Ethernet eth = context.inPacket().parsed();
642 if (eth == null || (eth.getEtherType() != TYPE_LLDP && eth.getEtherType() != TYPE_BSN)) {
643 return;
644 }
645
Thomas Vachuska96f3ea72015-09-08 13:50:12 -0700646 LinkDiscovery ld = discoverers.get(context.inPacket().receivedFrom().deviceId());
alshabib7911a052014-10-16 17:49:37 -0700647 if (ld == null) {
648 return;
649 }
650
Jonathan Hartb35540a2015-11-17 09:30:56 -0800651 if (ld.handleLldp(context)) {
alshabib7911a052014-10-16 17:49:37 -0700652 context.block();
653 }
654 }
655 }
656
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700657 /**
658 * Auxiliary task to keep device ports up to date.
659 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800660 private final class SyncDeviceInfoTask implements Runnable {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800661 @Override
662 public void run() {
663 if (Thread.currentThread().isInterrupted()) {
664 log.info("Interrupted, quitting");
665 return;
666 }
667 // check what deviceService sees, to see if we are missing anything
668 try {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700669 loadDevices();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800670 } catch (Exception e) {
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700671 // Catch all exceptions to avoid task being suppressed
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800672 log.error("Exception thrown during synchronization process", e);
673 }
674 }
675 }
676
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700677 /**
678 * Auxiliary task for pruning stale links.
679 */
680 private class LinkPrunerTask implements Runnable {
681 @Override
682 public void run() {
683 if (Thread.currentThread().isInterrupted()) {
684 log.info("Interrupted, quitting");
685 return;
686 }
687
688 try {
689 // TODO: There is still a slight possibility of mastership
690 // change occurring right with link going stale. This will
691 // result in the stale link not being pruned.
692 Maps.filterEntries(linkTimes, e -> {
693 if (!masterService.isLocalMaster(e.getKey().dst().deviceId())) {
694 return true;
695 }
696 if (isStale(e.getValue())) {
697 providerService.linkVanished(new DefaultLinkDescription(e.getKey().src(),
698 e.getKey().dst(),
699 DIRECT));
700 return true;
701 }
702 return false;
703 }).clear();
704
705 } catch (Exception e) {
706 // Catch all exceptions to avoid task being suppressed
707 log.error("Exception thrown during link pruning process", e);
708 }
709 }
710
711 private boolean isStale(long lastSeen) {
712 return lastSeen < System.currentTimeMillis() - staleLinkAge;
713 }
714 }
715
716 /**
717 * Provides processing context for the device link discovery helpers.
718 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700719 private class InternalDiscoveryContext implements DiscoveryContext {
720 @Override
721 public MastershipService mastershipService() {
722 return masterService;
723 }
724
725 @Override
726 public LinkProviderService providerService() {
727 return providerService;
728 }
729
730 @Override
731 public PacketService packetService() {
732 return packetService;
733 }
734
735 @Override
736 public long probeRate() {
737 return probeRate;
738 }
739
740 @Override
Jonathan Hartb35540a2015-11-17 09:30:56 -0800741 public boolean useBddp() {
742 return useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700743 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700744
745 @Override
746 public void touchLink(LinkKey key) {
747 linkTimes.put(key, System.currentTimeMillis());
748 }
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800749
750 @Override
751 public String fingerprint() {
752 return clusterMetadataService.getClusterMetadata().getName();
753 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700754 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700755
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800756 static final EnumSet<NetworkConfigEvent.Type> CONFIG_CHANGED
757 = EnumSet.of(NetworkConfigEvent.Type.CONFIG_ADDED,
758 NetworkConfigEvent.Type.CONFIG_UPDATED,
759 NetworkConfigEvent.Type.CONFIG_REMOVED);
760
Naoki Shiota399a0b32015-11-15 20:36:13 -0600761 private class InternalConfigListener implements NetworkConfigListener {
762
763 private synchronized void reconfigureSuppressionRules(SuppressionConfig cfg) {
764 if (cfg == null) {
765 log.error("Suppression Config is null.");
766 return;
767 }
768
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800769 SuppressionRules newRules = new SuppressionRules(cfg.deviceTypes(),
Naoki Shiota399a0b32015-11-15 20:36:13 -0600770 cfg.annotation());
771
772 updateRules(newRules);
773 }
774
775 @Override
776 public void event(NetworkConfigEvent event) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800777 if (event.configClass() == LinkDiscoveryFromDevice.class &&
778 CONFIG_CHANGED.contains(event.type())) {
779
780 if (event.subject() instanceof DeviceId) {
781 final DeviceId did = (DeviceId) event.subject();
782 Device device = deviceService.getDevice(did);
783 updateDevice(device).ifPresent(ld -> updatePorts(ld, did));
784 }
785
786 } else if (event.configClass() == LinkDiscoveryFromPort.class &&
787 CONFIG_CHANGED.contains(event.type())) {
788
789 if (event.subject() instanceof ConnectPoint) {
790 ConnectPoint cp = (ConnectPoint) event.subject();
791 if (cp.elementId() instanceof DeviceId) {
792 final DeviceId did = (DeviceId) cp.elementId();
793 Device device = deviceService.getDevice(did);
794 Port port = deviceService.getPort(did, cp.port());
795 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
796 }
797 }
798
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800799 } else if (event.configClass() == FingerprintProbeFromDevice.class &&
800 CONFIG_CHANGED.contains(event.type())) {
801
802 if (event.subject() instanceof DeviceId) {
803 final DeviceId did = (DeviceId) event.subject();
804 Device device = deviceService.getDevice(did);
805 updateDevice(device);
806 }
807
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800808 } else if (event.configClass().equals(SuppressionConfig.class) &&
Naoki Shiota399a0b32015-11-15 20:36:13 -0600809 (event.type() == NetworkConfigEvent.Type.CONFIG_ADDED ||
810 event.type() == NetworkConfigEvent.Type.CONFIG_UPDATED)) {
811 SuppressionConfig cfg = cfgRegistry.getConfig(appId, SuppressionConfig.class);
812 reconfigureSuppressionRules(cfg);
813 log.trace("Network config reconfigured");
814 }
815 }
816 }
alshabib7911a052014-10-16 17:49:37 -0700817}