blob: c41b8698b1c404d4cfe0925f9833a53e5de936b0 [file] [log] [blame]
Rusty Eddy4ae5aa82015-12-15 12:58:27 -08001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2016-present Open Networking Foundation
Rusty Eddy4ae5aa82015-12-15 12:58:27 -08003 *
4 * 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
7 *
8 * 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.
15 */
16package org.onosproject.pim.impl;
17
Jonathan Hart5af5f142016-01-28 18:45:27 -080018import com.google.common.collect.ImmutableSet;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080019import com.google.common.collect.Maps;
20import org.apache.felix.scr.annotations.Activate;
21import org.apache.felix.scr.annotations.Component;
22import org.apache.felix.scr.annotations.Deactivate;
23import org.apache.felix.scr.annotations.Reference;
24import org.apache.felix.scr.annotations.ReferenceCardinality;
25import org.apache.felix.scr.annotations.Service;
Jonathan Hart54119bb2016-02-06 18:48:27 -080026import org.onlab.util.SafeRecurringTask;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080027import org.onosproject.incubator.net.intf.Interface;
Jonathan Hart5af5f142016-01-28 18:45:27 -080028import org.onosproject.incubator.net.intf.InterfaceEvent;
29import org.onosproject.incubator.net.intf.InterfaceListener;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080030import org.onosproject.incubator.net.intf.InterfaceService;
Jonathan Hart92ca5d32016-04-14 18:05:52 -070031import org.onosproject.incubator.net.routing.Route;
32import org.onosproject.incubator.net.routing.RouteService;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080033import org.onosproject.net.ConnectPoint;
Jonathan Hart00cddda2016-02-16 10:30:37 -080034import org.onosproject.net.Host;
Jonathan Hart5af5f142016-01-28 18:45:27 -080035import org.onosproject.net.config.ConfigFactory;
36import org.onosproject.net.config.NetworkConfigEvent;
37import org.onosproject.net.config.NetworkConfigListener;
38import org.onosproject.net.config.NetworkConfigRegistry;
39import org.onosproject.net.config.basics.SubjectFactories;
Jonathan Hart00cddda2016-02-16 10:30:37 -080040import org.onosproject.net.host.HostService;
41import org.onosproject.net.mcast.McastEvent;
42import org.onosproject.net.mcast.McastListener;
43import org.onosproject.net.mcast.McastRoute;
44import org.onosproject.net.mcast.MulticastRouteService;
Jonathan Hart36fd31e2016-01-28 15:55:31 -080045import org.onosproject.net.packet.PacketService;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080046import org.slf4j.Logger;
Jonathan Hart5af5f142016-01-28 18:45:27 -080047
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080048import java.util.Map;
Jonathan Hart5af5f142016-01-28 18:45:27 -080049import java.util.Set;
Rusty Eddy4d5a92f2016-01-25 17:12:14 -080050import java.util.concurrent.Executors;
51import java.util.concurrent.ScheduledExecutorService;
52import java.util.concurrent.TimeUnit;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080053
54import static org.slf4j.LoggerFactory.getLogger;
55
56/**
57 * Manages PIMInterfaces.
58 *
59 * TODO: Do we need to add a ServiceListener?
60 */
61@Component(immediate = true)
62@Service
Jonathan Hartfbfe2a82016-03-29 11:36:33 -070063public class PimInterfaceManager implements PimInterfaceService {
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080064
65 private final Logger log = getLogger(getClass());
66
Jonathan Hart5af5f142016-01-28 18:45:27 -080067 private static final Class<PimInterfaceConfig> PIM_INTERFACE_CONFIG_CLASS = PimInterfaceConfig.class;
68 private static final String PIM_INTERFACE_CONFIG_KEY = "pimInterface";
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080069
Jonathan Hart6be70952016-02-12 21:11:26 -080070 public static final int DEFAULT_HELLO_INTERVAL = 30; // seconds
71
72 private static final int DEFAULT_TASK_PERIOD_MS = 250;
Jonathan Hart54119bb2016-02-06 18:48:27 -080073
74 // Create a Scheduled Executor service for recurring tasks
75 private final ScheduledExecutorService scheduledExecutorService =
Rusty Eddy4d5a92f2016-01-25 17:12:14 -080076 Executors.newScheduledThreadPool(1);
77
Jonathan Hart6be70952016-02-12 21:11:26 -080078 private final long initialHelloDelay = 1000;
Rusty Eddy4d5a92f2016-01-25 17:12:14 -080079
Jonathan Hart6be70952016-02-12 21:11:26 -080080 private final long pimHelloPeriod = DEFAULT_TASK_PERIOD_MS;
Rusty Eddy4ae5aa82015-12-15 12:58:27 -080081
Jonathan Hart6be70952016-02-12 21:11:26 -080082 private final int timeoutTaskPeriod = DEFAULT_TASK_PERIOD_MS;
Jonathan Hart54119bb2016-02-06 18:48:27 -080083
Jonathan Hart00cddda2016-02-16 10:30:37 -080084 private final int joinTaskPeriod = 10000;
85
Jonathan Hart36fd31e2016-01-28 15:55:31 -080086 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
87 protected PacketService packetService;
88
Jonathan Hart5af5f142016-01-28 18:45:27 -080089 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
90 protected NetworkConfigRegistry networkConfig;
91
92 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
93 protected InterfaceService interfaceService;
94
Jonathan Hart00cddda2016-02-16 10:30:37 -080095 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
96 protected HostService hostService;
97
98 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
99 protected MulticastRouteService multicastRouteService;
100
101 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700102 protected RouteService unicastRouteService;
Jonathan Hart00cddda2016-02-16 10:30:37 -0800103
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800104 // Store PIM Interfaces in a map key'd by ConnectPoint
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700105 private final Map<ConnectPoint, PimInterface> pimInterfaces = Maps.newConcurrentMap();
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800106
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700107 private final Map<McastRoute, PimInterface> routes = Maps.newConcurrentMap();
Jonathan Hart00cddda2016-02-16 10:30:37 -0800108
Jonathan Hart5af5f142016-01-28 18:45:27 -0800109 private final InternalNetworkConfigListener configListener =
110 new InternalNetworkConfigListener();
111 private final InternalInterfaceListener interfaceListener =
112 new InternalInterfaceListener();
Jonathan Hart00cddda2016-02-16 10:30:37 -0800113 private final InternalMulticastListener multicastListener =
114 new InternalMulticastListener();
Jonathan Hart5af5f142016-01-28 18:45:27 -0800115
116 private final ConfigFactory<ConnectPoint, PimInterfaceConfig> pimConfigFactory
117 = new ConfigFactory<ConnectPoint, PimInterfaceConfig>(
118 SubjectFactories.CONNECT_POINT_SUBJECT_FACTORY, PIM_INTERFACE_CONFIG_CLASS,
119 PIM_INTERFACE_CONFIG_KEY) {
120
121 @Override
122 public PimInterfaceConfig createConfig() {
123 return new PimInterfaceConfig();
124 }
125 };
126
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800127 @Activate
128 public void activate() {
Jonathan Hart5af5f142016-01-28 18:45:27 -0800129 networkConfig.registerConfigFactory(pimConfigFactory);
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800130
Jonathan Hart5af5f142016-01-28 18:45:27 -0800131 // Create PIM Interfaces for each of the existing configured interfaces.
132 Set<ConnectPoint> subjects = networkConfig.getSubjects(
133 ConnectPoint.class, PIM_INTERFACE_CONFIG_CLASS);
134 for (ConnectPoint cp : subjects) {
135 PimInterfaceConfig config = networkConfig.getConfig(cp, PIM_INTERFACE_CONFIG_CLASS);
136 updateInterface(config);
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800137 }
Rusty Eddy4d5a92f2016-01-25 17:12:14 -0800138
Jonathan Hart5af5f142016-01-28 18:45:27 -0800139 networkConfig.addListener(configListener);
140 interfaceService.addListener(interfaceListener);
Jonathan Hart00cddda2016-02-16 10:30:37 -0800141 multicastRouteService.addListener(multicastListener);
142
143 multicastRouteService.getRoutes().forEach(this::addRoute);
Jonathan Hart5af5f142016-01-28 18:45:27 -0800144
Rusty Eddy4d5a92f2016-01-25 17:12:14 -0800145 // Schedule the periodic hello sender.
Jonathan Hart54119bb2016-02-06 18:48:27 -0800146 scheduledExecutorService.scheduleAtFixedRate(
147 SafeRecurringTask.wrap(
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700148 () -> pimInterfaces.values().forEach(PimInterface::sendHello)),
Jonathan Hart6be70952016-02-12 21:11:26 -0800149 initialHelloDelay, pimHelloPeriod, TimeUnit.MILLISECONDS);
Jonathan Hart54119bb2016-02-06 18:48:27 -0800150
151 // Schedule task to periodically time out expired neighbors
152 scheduledExecutorService.scheduleAtFixedRate(
153 SafeRecurringTask.wrap(
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700154 () -> pimInterfaces.values().forEach(PimInterface::checkNeighborTimeouts)),
Jonathan Hart54119bb2016-02-06 18:48:27 -0800155 0, timeoutTaskPeriod, TimeUnit.MILLISECONDS);
Jonathan Hart5af5f142016-01-28 18:45:27 -0800156
Jonathan Hart00cddda2016-02-16 10:30:37 -0800157 scheduledExecutorService.scheduleAtFixedRate(
158 SafeRecurringTask.wrap(
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700159 () -> pimInterfaces.values().forEach(PimInterface::sendJoins)),
Jonathan Hart00cddda2016-02-16 10:30:37 -0800160 0, joinTaskPeriod, TimeUnit.MILLISECONDS);
161
Jonathan Hart5af5f142016-01-28 18:45:27 -0800162 log.info("Started");
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800163 }
164
165 @Deactivate
166 public void deactivate() {
Jonathan Hart5af5f142016-01-28 18:45:27 -0800167 interfaceService.removeListener(interfaceListener);
168 networkConfig.removeListener(configListener);
Jonathan Hart00cddda2016-02-16 10:30:37 -0800169 multicastRouteService.removeListener(multicastListener);
Jonathan Hart5af5f142016-01-28 18:45:27 -0800170 networkConfig.unregisterConfigFactory(pimConfigFactory);
Rusty Eddy4d5a92f2016-01-25 17:12:14 -0800171
172 // Shutdown the periodic hello task.
Jonathan Hart54119bb2016-02-06 18:48:27 -0800173 scheduledExecutorService.shutdown();
Rusty Eddy4d5a92f2016-01-25 17:12:14 -0800174
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800175 log.info("Stopped");
176 }
177
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800178 @Override
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700179 public PimInterface getPimInterface(ConnectPoint cp) {
180 PimInterface pi = pimInterfaces.get(cp);
Jonathan Hart7f4bc522016-02-20 11:32:43 -0800181 if (pi == null && log.isTraceEnabled()) {
182 log.trace("We have been asked for an Interface we don't have: {}", cp);
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800183 }
184 return pi;
185 }
Jonathan Hart5af5f142016-01-28 18:45:27 -0800186
187 @Override
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700188 public Set<PimInterface> getPimInterfaces() {
Jonathan Hart5af5f142016-01-28 18:45:27 -0800189 return ImmutableSet.copyOf(pimInterfaces.values());
190 }
191
192 private void updateInterface(PimInterfaceConfig config) {
193 ConnectPoint cp = config.subject();
194
195 if (!config.isEnabled()) {
196 removeInterface(cp);
197 return;
198 }
199
200 String intfName = config.getInterfaceName();
201 Interface intf = interfaceService.getInterfaceByName(cp, intfName);
202
203 if (intf == null) {
204 log.debug("Interface configuration missing: {}", config.getInterfaceName());
205 return;
206 }
207
208
209 log.debug("Updating Interface for " + intf.connectPoint().toString());
210 pimInterfaces.computeIfAbsent(cp, k -> buildPimInterface(config, intf));
211 }
212
213 private void removeInterface(ConnectPoint cp) {
214 pimInterfaces.remove(cp);
215 }
216
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700217 private PimInterface buildPimInterface(PimInterfaceConfig config, Interface intf) {
218 PimInterface.Builder builder = PimInterface.builder()
Jonathan Hart5af5f142016-01-28 18:45:27 -0800219 .withPacketService(packetService)
220 .withInterface(intf);
221
Sho SHIMIZUef7e2902016-02-12 18:38:29 -0800222 config.getHelloInterval().ifPresent(builder::withHelloInterval);
223 config.getHoldTime().ifPresent(builder::withHoldTime);
224 config.getPriority().ifPresent(builder::withPriority);
225 config.getPropagationDelay().ifPresent(builder::withPropagationDelay);
226 config.getOverrideInterval().ifPresent(builder::withOverrideInterval);
Jonathan Hart5af5f142016-01-28 18:45:27 -0800227
228 return builder.build();
229 }
230
Jonathan Hart00cddda2016-02-16 10:30:37 -0800231 private void addRoute(McastRoute route) {
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700232 PimInterface pimInterface = getSourceInterface(route);
Jonathan Hart00cddda2016-02-16 10:30:37 -0800233
234 if (pimInterface == null) {
235 return;
236 }
237
Jonathan Hart32be5172016-05-24 21:13:36 -0700238 multicastRouteService.addSource(route, pimInterface.getInterface().connectPoint());
239
Jonathan Hart00cddda2016-02-16 10:30:37 -0800240 routes.put(route, pimInterface);
241 }
242
243 private void removeRoute(McastRoute route) {
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700244 PimInterface pimInterface = routes.remove(route);
Jonathan Hart00cddda2016-02-16 10:30:37 -0800245
246 if (pimInterface == null) {
247 return;
248 }
249
250 pimInterface.removeRoute(route);
251 }
252
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700253 private PimInterface getSourceInterface(McastRoute route) {
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700254 Route unicastRoute = unicastRouteService.longestPrefixMatch(route.source());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800255
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700256 if (unicastRoute == null) {
Jonathan Hart00cddda2016-02-16 10:30:37 -0800257 log.warn("No route to source {}", route.source());
258 return null;
259 }
260
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700261 Interface intf = interfaceService.getMatchingInterface(unicastRoute.nextHop());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800262
263 if (intf == null) {
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700264 log.warn("No interface with route to next hop {}", unicastRoute.nextHop());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800265 return null;
266 }
267
Jonathan Hartfbfe2a82016-03-29 11:36:33 -0700268 PimInterface pimInterface = pimInterfaces.get(intf.connectPoint());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800269
270 if (pimInterface == null) {
271 log.warn("PIM is not enabled on interface {}", intf);
272 return null;
273 }
274
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700275 Set<Host> hosts = hostService.getHostsByIp(unicastRoute.nextHop());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800276 Host host = null;
277 for (Host h : hosts) {
278 if (h.vlan().equals(intf.vlan())) {
279 host = h;
280 }
281 }
282 if (host == null) {
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700283 log.warn("Next hop host entry not found: {}", unicastRoute.nextHop());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800284 return null;
285 }
286
Jonathan Hart92ca5d32016-04-14 18:05:52 -0700287 pimInterface.addRoute(route, unicastRoute.nextHop(), host.mac());
Jonathan Hart00cddda2016-02-16 10:30:37 -0800288
289 return pimInterface;
290 }
291
Jonathan Hart5af5f142016-01-28 18:45:27 -0800292 /**
293 * Listener for network config events.
294 */
295 private class InternalNetworkConfigListener implements NetworkConfigListener {
296
297 @Override
298 public void event(NetworkConfigEvent event) {
299 if (event.configClass() != PIM_INTERFACE_CONFIG_CLASS) {
300 return;
301 }
302
303 switch (event.type()) {
304 case CONFIG_REGISTERED:
305 case CONFIG_UNREGISTERED:
306 break;
307 case CONFIG_ADDED:
308 case CONFIG_UPDATED:
309 ConnectPoint cp = (ConnectPoint) event.subject();
310 PimInterfaceConfig config = networkConfig.getConfig(
311 cp, PIM_INTERFACE_CONFIG_CLASS);
312
313 updateInterface(config);
314 break;
315 case CONFIG_REMOVED:
316 removeInterface((ConnectPoint) event.subject());
317 break;
318 default:
319 break;
320 }
321 }
322 }
323
324 /**
325 * Listener for interface events.
326 */
327 private class InternalInterfaceListener implements InterfaceListener {
328
329 @Override
330 public void event(InterfaceEvent event) {
331 switch (event.type()) {
332 case INTERFACE_ADDED:
333 PimInterfaceConfig config = networkConfig.getConfig(
334 event.subject().connectPoint(), PIM_INTERFACE_CONFIG_CLASS);
335
336 if (config != null) {
337 updateInterface(config);
338 }
339 break;
340 case INTERFACE_UPDATED:
341 break;
342 case INTERFACE_REMOVED:
343 removeInterface(event.subject().connectPoint());
344 break;
345 default:
346 break;
347
348 }
349 }
350 }
Jonathan Hart00cddda2016-02-16 10:30:37 -0800351
352 /**
353 * Listener for multicast route events.
354 */
355 private class InternalMulticastListener implements McastListener {
356 @Override
357 public void event(McastEvent event) {
358 switch (event.type()) {
359 case ROUTE_ADDED:
360 addRoute(event.subject().route());
361 break;
362 case ROUTE_REMOVED:
363 removeRoute(event.subject().route());
364 break;
365 case SOURCE_ADDED:
366 case SINK_ADDED:
367 case SINK_REMOVED:
368 default:
369 break;
370 }
371 }
372 }
Rusty Eddy4ae5aa82015-12-15 12:58:27 -0800373}