blob: 80affc19d5b7da70ebea7d4a8621025ae0205397 [file] [log] [blame]
Charles Chanc91c8782016-03-30 17:54:24 -07001/*
Brian O'Connor0947d7e2017-08-03 21:12:30 -07002 * Copyright 2016-present Open Networking Foundation
Charles Chanc91c8782016-03-30 17:54:24 -07003 *
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 */
16
17package org.onosproject.segmentrouting;
18
19import com.google.common.collect.ImmutableSet;
20import com.google.common.collect.Lists;
21import com.google.common.collect.Sets;
22import org.onlab.packet.Ethernet;
23import org.onlab.packet.IpAddress;
24import org.onlab.packet.IpPrefix;
25import org.onlab.packet.MacAddress;
26import org.onlab.packet.VlanId;
27import org.onlab.util.KryoNamespace;
28import org.onosproject.core.ApplicationId;
29import org.onosproject.core.CoreService;
Ray Milkeyae0068a2017-08-15 11:02:29 -070030import org.onosproject.net.config.basics.McastConfig;
Charles Chanc91c8782016-03-30 17:54:24 -070031import org.onosproject.net.ConnectPoint;
32import org.onosproject.net.DeviceId;
33import org.onosproject.net.Link;
34import org.onosproject.net.Path;
35import org.onosproject.net.PortNumber;
36import org.onosproject.net.flow.DefaultTrafficSelector;
37import org.onosproject.net.flow.DefaultTrafficTreatment;
38import org.onosproject.net.flow.TrafficSelector;
39import org.onosproject.net.flow.TrafficTreatment;
40import org.onosproject.net.flow.criteria.Criteria;
41import org.onosproject.net.flow.instructions.Instructions.OutputInstruction;
42import org.onosproject.net.flowobjective.DefaultFilteringObjective;
43import org.onosproject.net.flowobjective.DefaultForwardingObjective;
44import org.onosproject.net.flowobjective.DefaultNextObjective;
Charles Chan72779502016-04-23 17:36:10 -070045import org.onosproject.net.flowobjective.DefaultObjectiveContext;
Charles Chanc91c8782016-03-30 17:54:24 -070046import org.onosproject.net.flowobjective.FilteringObjective;
47import org.onosproject.net.flowobjective.ForwardingObjective;
48import org.onosproject.net.flowobjective.NextObjective;
Charles Chan72779502016-04-23 17:36:10 -070049import org.onosproject.net.flowobjective.ObjectiveContext;
Charles Chanc91c8782016-03-30 17:54:24 -070050import org.onosproject.net.mcast.McastEvent;
51import org.onosproject.net.mcast.McastRouteInfo;
52import org.onosproject.net.topology.TopologyService;
Charles Chan370a65b2016-05-10 17:29:47 -070053import org.onosproject.segmentrouting.config.SegmentRoutingAppConfig;
Charles Chan72779502016-04-23 17:36:10 -070054import org.onosproject.segmentrouting.storekey.McastStoreKey;
Charles Chanc91c8782016-03-30 17:54:24 -070055import org.onosproject.store.serializers.KryoNamespaces;
56import org.onosproject.store.service.ConsistentMap;
57import org.onosproject.store.service.Serializer;
58import org.onosproject.store.service.StorageService;
59import org.slf4j.Logger;
60import org.slf4j.LoggerFactory;
61
62import java.util.Collection;
63import java.util.Collections;
64import java.util.List;
Charles Chan72779502016-04-23 17:36:10 -070065import java.util.Map;
Charles Chanc91c8782016-03-30 17:54:24 -070066import java.util.Optional;
67import java.util.Set;
Charles Chan72779502016-04-23 17:36:10 -070068import java.util.stream.Collectors;
69
70import static com.google.common.base.Preconditions.checkState;
Charles Chan10b0fb72017-02-02 16:20:42 -080071import static org.onosproject.segmentrouting.SegmentRoutingManager.INTERNAL_VLAN;
Charles Chanc91c8782016-03-30 17:54:24 -070072
73/**
Charles Chan1eaf4802016-04-18 13:44:03 -070074 * Handles multicast-related events.
Charles Chanc91c8782016-03-30 17:54:24 -070075 */
Charles Chan1eaf4802016-04-18 13:44:03 -070076public class McastHandler {
77 private static final Logger log = LoggerFactory.getLogger(McastHandler.class);
Charles Chanc91c8782016-03-30 17:54:24 -070078 private final SegmentRoutingManager srManager;
79 private final ApplicationId coreAppId;
Charles Chan82f19972016-05-17 13:13:55 -070080 private final StorageService storageService;
81 private final TopologyService topologyService;
Charles Chan72779502016-04-23 17:36:10 -070082 private final ConsistentMap<McastStoreKey, NextObjective> mcastNextObjStore;
83 private final KryoNamespace.Builder mcastKryo;
84 private final ConsistentMap<McastStoreKey, McastRole> mcastRoleStore;
85
86 /**
87 * Role in the multicast tree.
88 */
89 public enum McastRole {
90 /**
91 * The device is the ingress device of this group.
92 */
93 INGRESS,
94 /**
95 * The device is the transit device of this group.
96 */
97 TRANSIT,
98 /**
99 * The device is the egress device of this group.
100 */
101 EGRESS
102 }
Charles Chanc91c8782016-03-30 17:54:24 -0700103
104 /**
105 * Constructs the McastEventHandler.
106 *
107 * @param srManager Segment Routing manager
108 */
Charles Chan1eaf4802016-04-18 13:44:03 -0700109 public McastHandler(SegmentRoutingManager srManager) {
Charles Chanc91c8782016-03-30 17:54:24 -0700110 coreAppId = srManager.coreService.getAppId(CoreService.CORE_APP_NAME);
Charles Chanc91c8782016-03-30 17:54:24 -0700111 this.srManager = srManager;
112 this.storageService = srManager.storageService;
113 this.topologyService = srManager.topologyService;
Charles Chan72779502016-04-23 17:36:10 -0700114 mcastKryo = new KryoNamespace.Builder()
Charles Chanc91c8782016-03-30 17:54:24 -0700115 .register(KryoNamespaces.API)
Charles Chan72779502016-04-23 17:36:10 -0700116 .register(McastStoreKey.class)
117 .register(McastRole.class);
Charles Chanc91c8782016-03-30 17:54:24 -0700118 mcastNextObjStore = storageService
Charles Chan72779502016-04-23 17:36:10 -0700119 .<McastStoreKey, NextObjective>consistentMapBuilder()
Charles Chanc91c8782016-03-30 17:54:24 -0700120 .withName("onos-mcast-nextobj-store")
Charles Chan4922a172016-05-23 16:45:45 -0700121 .withSerializer(Serializer.using(mcastKryo.build("McastHandler-NextObj")))
Charles Chanc91c8782016-03-30 17:54:24 -0700122 .build();
Charles Chan72779502016-04-23 17:36:10 -0700123 mcastRoleStore = storageService
124 .<McastStoreKey, McastRole>consistentMapBuilder()
125 .withName("onos-mcast-role-store")
Charles Chan4922a172016-05-23 16:45:45 -0700126 .withSerializer(Serializer.using(mcastKryo.build("McastHandler-Role")))
Charles Chan72779502016-04-23 17:36:10 -0700127 .build();
128 }
129
130 /**
131 * Read initial multicast from mcast store.
132 */
Charles Chan82f19972016-05-17 13:13:55 -0700133 protected void init() {
Charles Chan72779502016-04-23 17:36:10 -0700134 srManager.multicastRouteService.getRoutes().forEach(mcastRoute -> {
135 ConnectPoint source = srManager.multicastRouteService.fetchSource(mcastRoute);
136 Set<ConnectPoint> sinks = srManager.multicastRouteService.fetchSinks(mcastRoute);
137 sinks.forEach(sink -> {
138 processSinkAddedInternal(source, sink, mcastRoute.group());
139 });
140 });
Charles Chanc91c8782016-03-30 17:54:24 -0700141 }
142
143 /**
144 * Processes the SOURCE_ADDED event.
145 *
146 * @param event McastEvent with SOURCE_ADDED type
147 */
148 protected void processSourceAdded(McastEvent event) {
149 log.info("processSourceAdded {}", event);
150 McastRouteInfo mcastRouteInfo = event.subject();
151 if (!mcastRouteInfo.isComplete()) {
152 log.info("Incompleted McastRouteInfo. Abort.");
153 return;
154 }
155 ConnectPoint source = mcastRouteInfo.source().orElse(null);
156 Set<ConnectPoint> sinks = mcastRouteInfo.sinks();
157 IpAddress mcastIp = mcastRouteInfo.route().group();
158
159 sinks.forEach(sink -> {
160 processSinkAddedInternal(source, sink, mcastIp);
161 });
162 }
163
164 /**
165 * Processes the SINK_ADDED event.
166 *
167 * @param event McastEvent with SINK_ADDED type
168 */
169 protected void processSinkAdded(McastEvent event) {
170 log.info("processSinkAdded {}", event);
171 McastRouteInfo mcastRouteInfo = event.subject();
172 if (!mcastRouteInfo.isComplete()) {
173 log.info("Incompleted McastRouteInfo. Abort.");
174 return;
175 }
176 ConnectPoint source = mcastRouteInfo.source().orElse(null);
177 ConnectPoint sink = mcastRouteInfo.sink().orElse(null);
178 IpAddress mcastIp = mcastRouteInfo.route().group();
179
180 processSinkAddedInternal(source, sink, mcastIp);
181 }
182
183 /**
184 * Processes the SINK_REMOVED event.
185 *
186 * @param event McastEvent with SINK_REMOVED type
187 */
188 protected void processSinkRemoved(McastEvent event) {
189 log.info("processSinkRemoved {}", event);
190 McastRouteInfo mcastRouteInfo = event.subject();
191 if (!mcastRouteInfo.isComplete()) {
192 log.info("Incompleted McastRouteInfo. Abort.");
193 return;
194 }
195 ConnectPoint source = mcastRouteInfo.source().orElse(null);
196 ConnectPoint sink = mcastRouteInfo.sink().orElse(null);
197 IpAddress mcastIp = mcastRouteInfo.route().group();
Charles Chanc91c8782016-03-30 17:54:24 -0700198
Charles Chan0932eca2016-06-28 16:50:13 -0700199 // Continue only when this instance is the master of source device
200 if (!srManager.mastershipService.isLocalMaster(source.deviceId())) {
201 log.info("Skip {} due to lack of mastership of the source device {}",
202 mcastIp, source.deviceId());
203 return;
204 }
205
Charles Chanc91c8782016-03-30 17:54:24 -0700206 // When source and sink are on the same device
207 if (source.deviceId().equals(sink.deviceId())) {
208 // Source and sink are on even the same port. There must be something wrong.
209 if (source.port().equals(sink.port())) {
210 log.warn("Sink is on the same port of source. Abort");
211 return;
212 }
Charles Chana8f9dee2016-05-16 18:44:13 -0700213 removePortFromDevice(sink.deviceId(), sink.port(), mcastIp, assignedVlan(source));
Charles Chanc91c8782016-03-30 17:54:24 -0700214 return;
215 }
216
217 // Process the egress device
Charles Chana8f9dee2016-05-16 18:44:13 -0700218 boolean isLast = removePortFromDevice(sink.deviceId(), sink.port(), mcastIp, assignedVlan(null));
Charles Chan72779502016-04-23 17:36:10 -0700219 if (isLast) {
220 mcastRoleStore.remove(new McastStoreKey(mcastIp, sink.deviceId()));
221 }
Charles Chanc91c8782016-03-30 17:54:24 -0700222
223 // If this is the last sink on the device, also update upstream
224 Optional<Path> mcastPath = getPath(source.deviceId(), sink.deviceId(), mcastIp);
225 if (mcastPath.isPresent()) {
226 List<Link> links = Lists.newArrayList(mcastPath.get().links());
227 Collections.reverse(links);
228 for (Link link : links) {
229 if (isLast) {
230 isLast = removePortFromDevice(link.src().deviceId(), link.src().port(),
Charles Chana8f9dee2016-05-16 18:44:13 -0700231 mcastIp,
232 assignedVlan(link.src().deviceId().equals(source.deviceId()) ? source : null));
Charles Chan72779502016-04-23 17:36:10 -0700233 mcastRoleStore.remove(new McastStoreKey(mcastIp, link.src().deviceId()));
Charles Chanc91c8782016-03-30 17:54:24 -0700234 }
235 }
236 }
237 }
238
239 /**
240 * Establishes a path from source to sink for given multicast group.
241 *
242 * @param source connect point of the multicast source
243 * @param sink connection point of the multicast sink
244 * @param mcastIp multicast group IP address
245 */
246 private void processSinkAddedInternal(ConnectPoint source, ConnectPoint sink,
247 IpAddress mcastIp) {
Charles Chan0932eca2016-06-28 16:50:13 -0700248 // Continue only when this instance is the master of source device
249 if (!srManager.mastershipService.isLocalMaster(source.deviceId())) {
250 log.info("Skip {} due to lack of mastership of the source device {}",
251 source.deviceId());
252 return;
253 }
254
Charles Chan72779502016-04-23 17:36:10 -0700255 // Process the ingress device
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000256 addFilterToDevice(source.deviceId(), source.port(), assignedVlan(source), mcastIp);
Charles Chan72779502016-04-23 17:36:10 -0700257
Charles Chanc91c8782016-03-30 17:54:24 -0700258 // When source and sink are on the same device
259 if (source.deviceId().equals(sink.deviceId())) {
260 // Source and sink are on even the same port. There must be something wrong.
261 if (source.port().equals(sink.port())) {
262 log.warn("Sink is on the same port of source. Abort");
263 return;
264 }
Charles Chana8f9dee2016-05-16 18:44:13 -0700265 addPortToDevice(sink.deviceId(), sink.port(), mcastIp, assignedVlan(source));
Charles Chan72779502016-04-23 17:36:10 -0700266 mcastRoleStore.put(new McastStoreKey(mcastIp, sink.deviceId()), McastRole.INGRESS);
Charles Chanc91c8782016-03-30 17:54:24 -0700267 return;
268 }
269
Charles Chanc91c8782016-03-30 17:54:24 -0700270 // Find a path. If present, create/update groups and flows for each hop
271 Optional<Path> mcastPath = getPath(source.deviceId(), sink.deviceId(), mcastIp);
272 if (mcastPath.isPresent()) {
Charles Chan72779502016-04-23 17:36:10 -0700273 List<Link> links = mcastPath.get().links();
274 checkState(links.size() == 2,
275 "Path in leaf-spine topology should always be two hops: ", links);
276
277 links.forEach(link -> {
Charles Chana8f9dee2016-05-16 18:44:13 -0700278 addPortToDevice(link.src().deviceId(), link.src().port(), mcastIp,
279 assignedVlan(link.src().deviceId().equals(source.deviceId()) ? source : null));
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000280 addFilterToDevice(link.dst().deviceId(), link.dst().port(), assignedVlan(null), mcastIp);
Charles Chanc91c8782016-03-30 17:54:24 -0700281 });
Charles Chan72779502016-04-23 17:36:10 -0700282
Charles Chanc91c8782016-03-30 17:54:24 -0700283 // Process the egress device
Charles Chana8f9dee2016-05-16 18:44:13 -0700284 addPortToDevice(sink.deviceId(), sink.port(), mcastIp, assignedVlan(null));
Charles Chan72779502016-04-23 17:36:10 -0700285
286 // Setup mcast roles
287 mcastRoleStore.put(new McastStoreKey(mcastIp, source.deviceId()),
288 McastRole.INGRESS);
289 mcastRoleStore.put(new McastStoreKey(mcastIp, links.get(0).dst().deviceId()),
290 McastRole.TRANSIT);
291 mcastRoleStore.put(new McastStoreKey(mcastIp, sink.deviceId()),
292 McastRole.EGRESS);
293 } else {
294 log.warn("Unable to find a path from {} to {}. Abort sinkAdded",
295 source.deviceId(), sink.deviceId());
Charles Chanc91c8782016-03-30 17:54:24 -0700296 }
297 }
298
299 /**
Charles Chan72779502016-04-23 17:36:10 -0700300 * Processes the LINK_DOWN event.
301 *
302 * @param affectedLink Link that is going down
303 */
304 protected void processLinkDown(Link affectedLink) {
Charles Chan72779502016-04-23 17:36:10 -0700305 getAffectedGroups(affectedLink).forEach(mcastIp -> {
306 // Find out the ingress, transit and egress device of affected group
307 DeviceId ingressDevice = getDevice(mcastIp, McastRole.INGRESS)
308 .stream().findAny().orElse(null);
309 DeviceId transitDevice = getDevice(mcastIp, McastRole.TRANSIT)
310 .stream().findAny().orElse(null);
311 Set<DeviceId> egressDevices = getDevice(mcastIp, McastRole.EGRESS);
Charles Chana8f9dee2016-05-16 18:44:13 -0700312 ConnectPoint source = getSource(mcastIp);
313
314 // Do not proceed if any of these info is missing
315 if (ingressDevice == null || transitDevice == null
316 || egressDevices == null || source == null) {
317 log.warn("Missing ingress {}, transit {}, egress {} devices or source {}",
318 ingressDevice, transitDevice, egressDevices, source);
Charles Chan72779502016-04-23 17:36:10 -0700319 return;
320 }
321
Charles Chan0932eca2016-06-28 16:50:13 -0700322 // Continue only when this instance is the master of source device
323 if (!srManager.mastershipService.isLocalMaster(source.deviceId())) {
324 log.info("Skip {} due to lack of mastership of the source device {}",
325 source.deviceId());
326 return;
327 }
328
Charles Chan72779502016-04-23 17:36:10 -0700329 // Remove entire transit
Charles Chana8f9dee2016-05-16 18:44:13 -0700330 removeGroupFromDevice(transitDevice, mcastIp, assignedVlan(null));
Charles Chan72779502016-04-23 17:36:10 -0700331
332 // Remove transit-facing port on ingress device
333 PortNumber ingressTransitPort = ingressTransitPort(mcastIp);
334 if (ingressTransitPort != null) {
Charles Chana8f9dee2016-05-16 18:44:13 -0700335 removePortFromDevice(ingressDevice, ingressTransitPort, mcastIp, assignedVlan(source));
Charles Chan72779502016-04-23 17:36:10 -0700336 mcastRoleStore.remove(new McastStoreKey(mcastIp, transitDevice));
337 }
338
339 // Construct a new path for each egress device
340 egressDevices.forEach(egressDevice -> {
341 Optional<Path> mcastPath = getPath(ingressDevice, egressDevice, mcastIp);
342 if (mcastPath.isPresent()) {
343 List<Link> links = mcastPath.get().links();
344 links.forEach(link -> {
Charles Chana8f9dee2016-05-16 18:44:13 -0700345 addPortToDevice(link.src().deviceId(), link.src().port(), mcastIp,
346 assignedVlan(link.src().deviceId().equals(source.deviceId()) ? source : null));
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000347 addFilterToDevice(link.dst().deviceId(), link.dst().port(), assignedVlan(null), mcastIp);
Charles Chan72779502016-04-23 17:36:10 -0700348 });
349 // Setup new transit mcast role
350 mcastRoleStore.put(new McastStoreKey(mcastIp,
351 links.get(0).dst().deviceId()), McastRole.TRANSIT);
352 } else {
353 log.warn("Fail to recover egress device {} from link failure {}",
354 egressDevice, affectedLink);
Charles Chana8f9dee2016-05-16 18:44:13 -0700355 removeGroupFromDevice(egressDevice, mcastIp, assignedVlan(null));
Charles Chan72779502016-04-23 17:36:10 -0700356 }
357 });
358 });
359 }
360
361 /**
Charles Chanc91c8782016-03-30 17:54:24 -0700362 * Adds filtering objective for given device and port.
363 *
364 * @param deviceId device ID
365 * @param port ingress port number
366 * @param assignedVlan assigned VLAN ID
367 */
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000368 private void addFilterToDevice(DeviceId deviceId, PortNumber port, VlanId assignedVlan, IpAddress mcastIp) {
Charles Chanc91c8782016-03-30 17:54:24 -0700369 // Do nothing if the port is configured as suppressed
Charles Chan370a65b2016-05-10 17:29:47 -0700370 ConnectPoint connectPoint = new ConnectPoint(deviceId, port);
371 SegmentRoutingAppConfig appConfig = srManager.cfgService
372 .getConfig(srManager.appId, SegmentRoutingAppConfig.class);
373 if (appConfig != null && appConfig.suppressSubnet().contains(connectPoint)) {
374 log.info("Ignore suppressed port {}", connectPoint);
Charles Chanc91c8782016-03-30 17:54:24 -0700375 return;
376 }
377
378 FilteringObjective.Builder filtObjBuilder =
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000379 filterObjBuilder(deviceId, port, assignedVlan, mcastIp);
Charles Chan72779502016-04-23 17:36:10 -0700380 ObjectiveContext context = new DefaultObjectiveContext(
381 (objective) -> log.debug("Successfully add filter on {}/{}, vlan {}",
Charles Chan10b0fb72017-02-02 16:20:42 -0800382 deviceId, port.toLong(), assignedVlan),
Charles Chan72779502016-04-23 17:36:10 -0700383 (objective, error) ->
384 log.warn("Failed to add filter on {}/{}, vlan {}: {}",
Charles Chan10b0fb72017-02-02 16:20:42 -0800385 deviceId, port.toLong(), assignedVlan, error));
Charles Chan72779502016-04-23 17:36:10 -0700386 srManager.flowObjectiveService.filter(deviceId, filtObjBuilder.add(context));
Charles Chanc91c8782016-03-30 17:54:24 -0700387 }
388
389 /**
390 * Adds a port to given multicast group on given device. This involves the
391 * update of L3 multicast group and multicast routing table entry.
392 *
393 * @param deviceId device ID
394 * @param port port to be added
395 * @param mcastIp multicast group
396 * @param assignedVlan assigned VLAN ID
397 */
398 private void addPortToDevice(DeviceId deviceId, PortNumber port,
399 IpAddress mcastIp, VlanId assignedVlan) {
Charles Chan72779502016-04-23 17:36:10 -0700400 McastStoreKey mcastStoreKey = new McastStoreKey(mcastIp, deviceId);
Charles Chanc91c8782016-03-30 17:54:24 -0700401 ImmutableSet.Builder<PortNumber> portBuilder = ImmutableSet.builder();
Charles Chan72779502016-04-23 17:36:10 -0700402 if (!mcastNextObjStore.containsKey(mcastStoreKey)) {
Charles Chanc91c8782016-03-30 17:54:24 -0700403 // First time someone request this mcast group via this device
404 portBuilder.add(port);
405 } else {
406 // This device already serves some subscribers of this mcast group
Charles Chan72779502016-04-23 17:36:10 -0700407 NextObjective nextObj = mcastNextObjStore.get(mcastStoreKey).value();
Charles Chanc91c8782016-03-30 17:54:24 -0700408 // Stop if the port is already in the nextobj
409 Set<PortNumber> existingPorts = getPorts(nextObj.next());
410 if (existingPorts.contains(port)) {
411 log.info("NextObj for {}/{} already exists. Abort", deviceId, port);
412 return;
413 }
414 portBuilder.addAll(existingPorts).add(port).build();
415 }
416 // Create, store and apply the new nextObj and fwdObj
Charles Chan72779502016-04-23 17:36:10 -0700417 ObjectiveContext context = new DefaultObjectiveContext(
418 (objective) -> log.debug("Successfully add {} on {}/{}, vlan {}",
419 mcastIp, deviceId, port.toLong(), assignedVlan),
420 (objective, error) ->
421 log.warn("Failed to add {} on {}/{}, vlan {}: {}",
422 mcastIp, deviceId, port.toLong(), assignedVlan, error));
Charles Chanc91c8782016-03-30 17:54:24 -0700423 NextObjective newNextObj =
424 nextObjBuilder(mcastIp, assignedVlan, portBuilder.build()).add();
425 ForwardingObjective fwdObj =
Charles Chan72779502016-04-23 17:36:10 -0700426 fwdObjBuilder(mcastIp, assignedVlan, newNextObj.id()).add(context);
427 mcastNextObjStore.put(mcastStoreKey, newNextObj);
Charles Chanc91c8782016-03-30 17:54:24 -0700428 srManager.flowObjectiveService.next(deviceId, newNextObj);
429 srManager.flowObjectiveService.forward(deviceId, fwdObj);
Charles Chanc91c8782016-03-30 17:54:24 -0700430 }
431
432 /**
433 * Removes a port from given multicast group on given device.
434 * This involves the update of L3 multicast group and multicast routing
435 * table entry.
436 *
437 * @param deviceId device ID
438 * @param port port to be added
439 * @param mcastIp multicast group
440 * @param assignedVlan assigned VLAN ID
441 * @return true if this is the last sink on this device
442 */
443 private boolean removePortFromDevice(DeviceId deviceId, PortNumber port,
444 IpAddress mcastIp, VlanId assignedVlan) {
Charles Chan72779502016-04-23 17:36:10 -0700445 McastStoreKey mcastStoreKey =
446 new McastStoreKey(mcastIp, deviceId);
Charles Chanc91c8782016-03-30 17:54:24 -0700447 // This device is not serving this multicast group
Charles Chan72779502016-04-23 17:36:10 -0700448 if (!mcastNextObjStore.containsKey(mcastStoreKey)) {
Charles Chanc91c8782016-03-30 17:54:24 -0700449 log.warn("{} is not serving {} on port {}. Abort.", deviceId, mcastIp, port);
450 return false;
451 }
Charles Chan72779502016-04-23 17:36:10 -0700452 NextObjective nextObj = mcastNextObjStore.get(mcastStoreKey).value();
Charles Chanc91c8782016-03-30 17:54:24 -0700453
454 Set<PortNumber> existingPorts = getPorts(nextObj.next());
Charles Chan72779502016-04-23 17:36:10 -0700455 // This port does not serve this multicast group
Charles Chanc91c8782016-03-30 17:54:24 -0700456 if (!existingPorts.contains(port)) {
457 log.warn("{} is not serving {} on port {}. Abort.", deviceId, mcastIp, port);
458 return false;
459 }
460 // Copy and modify the ImmutableSet
461 existingPorts = Sets.newHashSet(existingPorts);
462 existingPorts.remove(port);
463
464 NextObjective newNextObj;
465 ForwardingObjective fwdObj;
466 if (existingPorts.isEmpty()) {
467 // If this is the last sink, remove flows and groups
468 // NOTE: Rely on GroupStore garbage collection rather than explicitly
469 // remove L3MG since there might be other flows/groups refer to
470 // the same L2IG
Charles Chan72779502016-04-23 17:36:10 -0700471 ObjectiveContext context = new DefaultObjectiveContext(
472 (objective) -> log.debug("Successfully remove {} on {}/{}, vlan {}",
473 mcastIp, deviceId, port.toLong(), assignedVlan),
474 (objective, error) ->
475 log.warn("Failed to remove {} on {}/{}, vlan {}: {}",
476 mcastIp, deviceId, port.toLong(), assignedVlan, error));
477 fwdObj = fwdObjBuilder(mcastIp, assignedVlan, nextObj.id()).remove(context);
478 mcastNextObjStore.remove(mcastStoreKey);
Charles Chanc91c8782016-03-30 17:54:24 -0700479 srManager.flowObjectiveService.forward(deviceId, fwdObj);
480 } else {
481 // If this is not the last sink, update flows and groups
Charles Chan72779502016-04-23 17:36:10 -0700482 ObjectiveContext context = new DefaultObjectiveContext(
483 (objective) -> log.debug("Successfully update {} on {}/{}, vlan {}",
484 mcastIp, deviceId, port.toLong(), assignedVlan),
485 (objective, error) ->
486 log.warn("Failed to update {} on {}/{}, vlan {}: {}",
487 mcastIp, deviceId, port.toLong(), assignedVlan, error));
Charles Chanc91c8782016-03-30 17:54:24 -0700488 newNextObj = nextObjBuilder(mcastIp, assignedVlan, existingPorts).add();
Charles Chan82f19972016-05-17 13:13:55 -0700489 fwdObj = fwdObjBuilder(mcastIp, assignedVlan, newNextObj.id()).add(context);
Charles Chan72779502016-04-23 17:36:10 -0700490 mcastNextObjStore.put(mcastStoreKey, newNextObj);
Charles Chanc91c8782016-03-30 17:54:24 -0700491 srManager.flowObjectiveService.next(deviceId, newNextObj);
492 srManager.flowObjectiveService.forward(deviceId, fwdObj);
493 }
Charles Chanc91c8782016-03-30 17:54:24 -0700494 return existingPorts.isEmpty();
495 }
496
Charles Chan72779502016-04-23 17:36:10 -0700497
498 /**
499 * Removes entire group on given device.
500 *
501 * @param deviceId device ID
502 * @param mcastIp multicast group to be removed
503 * @param assignedVlan assigned VLAN ID
504 */
505 private void removeGroupFromDevice(DeviceId deviceId, IpAddress mcastIp,
506 VlanId assignedVlan) {
507 McastStoreKey mcastStoreKey = new McastStoreKey(mcastIp, deviceId);
508 // This device is not serving this multicast group
509 if (!mcastNextObjStore.containsKey(mcastStoreKey)) {
510 log.warn("{} is not serving {}. Abort.", deviceId, mcastIp);
511 return;
512 }
513 NextObjective nextObj = mcastNextObjStore.get(mcastStoreKey).value();
514 // NOTE: Rely on GroupStore garbage collection rather than explicitly
515 // remove L3MG since there might be other flows/groups refer to
516 // the same L2IG
517 ObjectiveContext context = new DefaultObjectiveContext(
518 (objective) -> log.debug("Successfully remove {} on {}, vlan {}",
519 mcastIp, deviceId, assignedVlan),
520 (objective, error) ->
521 log.warn("Failed to remove {} on {}, vlan {}: {}",
522 mcastIp, deviceId, assignedVlan, error));
523 ForwardingObjective fwdObj = fwdObjBuilder(mcastIp, assignedVlan, nextObj.id()).remove(context);
524 srManager.flowObjectiveService.forward(deviceId, fwdObj);
525 mcastNextObjStore.remove(mcastStoreKey);
526 mcastRoleStore.remove(mcastStoreKey);
527 }
528
529 /**
530 * Remove all groups on given device.
531 *
532 * @param deviceId device ID
533 */
534 public void removeDevice(DeviceId deviceId) {
Charles Chanfc115892016-06-17 14:28:07 -0700535 mcastNextObjStore.entrySet().stream()
536 .filter(entry -> entry.getKey().deviceId().equals(deviceId))
537 .forEach(entry -> {
538 ConnectPoint source = getSource(entry.getKey().mcastIp());
539 removeGroupFromDevice(entry.getKey().deviceId(), entry.getKey().mcastIp(),
Charles Chan6961f222018-01-04 14:26:07 -0800540 assignedVlan(source != null && deviceId.equals(source.deviceId()) ? source : null));
Charles Chanfc115892016-06-17 14:28:07 -0700541 mcastNextObjStore.remove(entry.getKey());
542 });
543 log.debug("{} is removed from mcastNextObjStore", deviceId);
Charles Chan72779502016-04-23 17:36:10 -0700544
Charles Chanfc115892016-06-17 14:28:07 -0700545 mcastRoleStore.entrySet().stream()
546 .filter(entry -> entry.getKey().deviceId().equals(deviceId))
547 .forEach(entry -> {
548 mcastRoleStore.remove(entry.getKey());
549 });
550 log.debug("{} is removed from mcastRoleStore", deviceId);
Charles Chan72779502016-04-23 17:36:10 -0700551 }
552
Charles Chanc91c8782016-03-30 17:54:24 -0700553 /**
554 * Creates a next objective builder for multicast.
555 *
556 * @param mcastIp multicast group
557 * @param assignedVlan assigned VLAN ID
558 * @param outPorts set of output port numbers
559 * @return next objective builder
560 */
561 private NextObjective.Builder nextObjBuilder(IpAddress mcastIp,
562 VlanId assignedVlan, Set<PortNumber> outPorts) {
563 int nextId = srManager.flowObjectiveService.allocateNextId();
564
565 TrafficSelector metadata =
566 DefaultTrafficSelector.builder()
567 .matchVlanId(assignedVlan)
568 .matchIPDst(mcastIp.toIpPrefix())
569 .build();
570
571 NextObjective.Builder nextObjBuilder = DefaultNextObjective
572 .builder().withId(nextId)
573 .withType(NextObjective.Type.BROADCAST).fromApp(srManager.appId)
574 .withMeta(metadata);
575
576 outPorts.forEach(port -> {
577 TrafficTreatment.Builder tBuilder = DefaultTrafficTreatment.builder();
578 if (egressVlan().equals(VlanId.NONE)) {
579 tBuilder.popVlan();
580 }
581 tBuilder.setOutput(port);
582 nextObjBuilder.addTreatment(tBuilder.build());
583 });
584
585 return nextObjBuilder;
586 }
587
588 /**
589 * Creates a forwarding objective builder for multicast.
590 *
591 * @param mcastIp multicast group
592 * @param assignedVlan assigned VLAN ID
593 * @param nextId next ID of the L3 multicast group
594 * @return forwarding objective builder
595 */
596 private ForwardingObjective.Builder fwdObjBuilder(IpAddress mcastIp,
597 VlanId assignedVlan, int nextId) {
598 TrafficSelector.Builder sbuilder = DefaultTrafficSelector.builder();
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000599 IpPrefix mcastPrefix = mcastIp.toIpPrefix();
600
601 if (mcastIp.isIp4()) {
602 sbuilder.matchEthType(Ethernet.TYPE_IPV4);
603 sbuilder.matchIPDst(mcastPrefix);
604 } else {
605 sbuilder.matchEthType(Ethernet.TYPE_IPV6);
606 sbuilder.matchIPv6Dst(mcastPrefix);
607 }
608
609
Charles Chanc91c8782016-03-30 17:54:24 -0700610 TrafficSelector.Builder metabuilder = DefaultTrafficSelector.builder();
611 metabuilder.matchVlanId(assignedVlan);
612
613 ForwardingObjective.Builder fwdBuilder = DefaultForwardingObjective.builder();
614 fwdBuilder.withSelector(sbuilder.build())
615 .withMeta(metabuilder.build())
616 .nextStep(nextId)
617 .withFlag(ForwardingObjective.Flag.SPECIFIC)
618 .fromApp(srManager.appId)
619 .withPriority(SegmentRoutingService.DEFAULT_PRIORITY);
620 return fwdBuilder;
621 }
622
623 /**
624 * Creates a filtering objective builder for multicast.
625 *
626 * @param deviceId Device ID
627 * @param ingressPort ingress port of the multicast stream
628 * @param assignedVlan assigned VLAN ID
629 * @return filtering objective builder
630 */
631 private FilteringObjective.Builder filterObjBuilder(DeviceId deviceId, PortNumber ingressPort,
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000632 VlanId assignedVlan, IpAddress mcastIp) {
Charles Chanc91c8782016-03-30 17:54:24 -0700633 FilteringObjective.Builder filtBuilder = DefaultFilteringObjective.builder();
Charles Chan0932eca2016-06-28 16:50:13 -0700634
Julia Fergusonf1d9c342017-08-10 18:15:24 +0000635 if (mcastIp.isIp4()) {
636 filtBuilder.withKey(Criteria.matchInPort(ingressPort))
637 .addCondition(Criteria.matchEthDstMasked(MacAddress.IPV4_MULTICAST,
638 MacAddress.IPV4_MULTICAST_MASK))
639 .addCondition(Criteria.matchVlanId(egressVlan()))
640 .withPriority(SegmentRoutingService.DEFAULT_PRIORITY);
641 } else {
642 filtBuilder.withKey(Criteria.matchInPort(ingressPort))
643 .addCondition(Criteria.matchEthDstMasked(MacAddress.IPV6_MULTICAST,
644 MacAddress.IPV6_MULTICAST_MASK))
645 .addCondition(Criteria.matchVlanId(egressVlan()))
646 .withPriority(SegmentRoutingService.DEFAULT_PRIORITY);
647 }
Charles Chan0932eca2016-06-28 16:50:13 -0700648 TrafficTreatment tt = DefaultTrafficTreatment.builder()
649 .pushVlan().setVlanId(assignedVlan).build();
650 filtBuilder.withMeta(tt);
651
Charles Chanc91c8782016-03-30 17:54:24 -0700652 return filtBuilder.permit().fromApp(srManager.appId);
653 }
654
655 /**
656 * Gets output ports information from treatments.
657 *
658 * @param treatments collection of traffic treatments
659 * @return set of output port numbers
660 */
661 private Set<PortNumber> getPorts(Collection<TrafficTreatment> treatments) {
662 ImmutableSet.Builder<PortNumber> builder = ImmutableSet.builder();
663 treatments.forEach(treatment -> {
664 treatment.allInstructions().stream()
665 .filter(instr -> instr instanceof OutputInstruction)
666 .forEach(instr -> {
667 builder.add(((OutputInstruction) instr).port());
668 });
669 });
670 return builder.build();
671 }
672
673 /**
674 * Gets a path from src to dst.
675 * If a path was allocated before, returns the allocated path.
676 * Otherwise, randomly pick one from available paths.
677 *
678 * @param src source device ID
679 * @param dst destination device ID
680 * @param mcastIp multicast group
681 * @return an optional path from src to dst
682 */
683 private Optional<Path> getPath(DeviceId src, DeviceId dst, IpAddress mcastIp) {
684 List<Path> allPaths = Lists.newArrayList(
685 topologyService.getPaths(topologyService.currentTopology(), src, dst));
Charles Chan72779502016-04-23 17:36:10 -0700686 log.debug("{} path(s) found from {} to {}", allPaths.size(), src, dst);
Charles Chanc91c8782016-03-30 17:54:24 -0700687 if (allPaths.isEmpty()) {
Charles Chanc91c8782016-03-30 17:54:24 -0700688 return Optional.empty();
689 }
690
691 // If one of the available path is used before, use the same path
Charles Chan72779502016-04-23 17:36:10 -0700692 McastStoreKey mcastStoreKey = new McastStoreKey(mcastIp, src);
693 if (mcastNextObjStore.containsKey(mcastStoreKey)) {
694 NextObjective nextObj = mcastNextObjStore.get(mcastStoreKey).value();
Charles Chanc91c8782016-03-30 17:54:24 -0700695 Set<PortNumber> existingPorts = getPorts(nextObj.next());
696 for (Path path : allPaths) {
697 PortNumber srcPort = path.links().get(0).src().port();
698 if (existingPorts.contains(srcPort)) {
699 return Optional.of(path);
700 }
701 }
702 }
703 // Otherwise, randomly pick a path
704 Collections.shuffle(allPaths);
705 return allPaths.stream().findFirst();
706 }
707
708 /**
Charles Chan72779502016-04-23 17:36:10 -0700709 * Gets device(s) of given role in given multicast group.
710 *
711 * @param mcastIp multicast IP
712 * @param role multicast role
713 * @return set of device ID or empty set if not found
714 */
715 private Set<DeviceId> getDevice(IpAddress mcastIp, McastRole role) {
716 return mcastRoleStore.entrySet().stream()
717 .filter(entry -> entry.getKey().mcastIp().equals(mcastIp) &&
718 entry.getValue().value() == role)
719 .map(Map.Entry::getKey).map(McastStoreKey::deviceId)
720 .collect(Collectors.toSet());
721 }
722
723 /**
Charles Chana8f9dee2016-05-16 18:44:13 -0700724 * Gets source connect point of given multicast group.
725 *
726 * @param mcastIp multicast IP
727 * @return source connect point or null if not found
728 */
729 private ConnectPoint getSource(IpAddress mcastIp) {
730 return srManager.multicastRouteService.getRoutes().stream()
731 .filter(mcastRoute -> mcastRoute.group().equals(mcastIp))
732 .map(mcastRoute -> srManager.multicastRouteService.fetchSource(mcastRoute))
733 .findAny().orElse(null);
734 }
735
736 /**
Charles Chan72779502016-04-23 17:36:10 -0700737 * Gets groups which is affected by the link down event.
738 *
739 * @param link link going down
740 * @return a set of multicast IpAddress
741 */
742 private Set<IpAddress> getAffectedGroups(Link link) {
743 DeviceId deviceId = link.src().deviceId();
744 PortNumber port = link.src().port();
745 return mcastNextObjStore.entrySet().stream()
746 .filter(entry -> entry.getKey().deviceId().equals(deviceId) &&
747 getPorts(entry.getValue().value().next()).contains(port))
748 .map(Map.Entry::getKey).map(McastStoreKey::mcastIp)
749 .collect(Collectors.toSet());
750 }
751
752 /**
Charles Chanc91c8782016-03-30 17:54:24 -0700753 * Gets egress VLAN from McastConfig.
754 *
755 * @return egress VLAN or VlanId.NONE if not configured
756 */
757 private VlanId egressVlan() {
758 McastConfig mcastConfig =
759 srManager.cfgService.getConfig(coreAppId, McastConfig.class);
760 return (mcastConfig != null) ? mcastConfig.egressVlan() : VlanId.NONE;
761 }
762
763 /**
764 * Gets assigned VLAN according to the value of egress VLAN.
Charles Chana8f9dee2016-05-16 18:44:13 -0700765 * If connect point is specified, try to reuse the assigned VLAN on the connect point.
Charles Chanc91c8782016-03-30 17:54:24 -0700766 *
Charles Chana8f9dee2016-05-16 18:44:13 -0700767 * @param cp connect point; Can be null if not specified
768 * @return assigned VLAN ID
Charles Chanc91c8782016-03-30 17:54:24 -0700769 */
Charles Chana8f9dee2016-05-16 18:44:13 -0700770 private VlanId assignedVlan(ConnectPoint cp) {
771 // Use the egressVlan if it is tagged
772 if (!egressVlan().equals(VlanId.NONE)) {
773 return egressVlan();
774 }
775 // Reuse unicast VLAN if the port has subnet configured
776 if (cp != null) {
Charles Chanf9759582017-10-20 19:09:16 -0700777 VlanId untaggedVlan = srManager.getInternalVlanId(cp);
Charles Chan10b0fb72017-02-02 16:20:42 -0800778 return (untaggedVlan != null) ? untaggedVlan : INTERNAL_VLAN;
Charles Chana8f9dee2016-05-16 18:44:13 -0700779 }
Charles Chan10b0fb72017-02-02 16:20:42 -0800780 // Use DEFAULT_VLAN if none of the above matches
781 return SegmentRoutingManager.INTERNAL_VLAN;
Charles Chanc91c8782016-03-30 17:54:24 -0700782 }
Charles Chan72779502016-04-23 17:36:10 -0700783
784 /**
785 * Gets the spine-facing port on ingress device of given multicast group.
786 *
787 * @param mcastIp multicast IP
788 * @return spine-facing port on ingress device
789 */
790 private PortNumber ingressTransitPort(IpAddress mcastIp) {
791 DeviceId ingressDevice = getDevice(mcastIp, McastRole.INGRESS)
792 .stream().findAny().orElse(null);
793 if (ingressDevice != null) {
794 NextObjective nextObj = mcastNextObjStore
795 .get(new McastStoreKey(mcastIp, ingressDevice)).value();
796 Set<PortNumber> ports = getPorts(nextObj.next());
797
798 for (PortNumber port : ports) {
799 // Spine-facing port should have no subnet and no xconnect
800 if (srManager.deviceConfiguration != null &&
Pier Ventreb6a7f342016-11-26 21:05:22 -0800801 srManager.deviceConfiguration.getPortSubnets(ingressDevice, port).isEmpty() &&
Charles Chan82f19972016-05-17 13:13:55 -0700802 !srManager.xConnectHandler.hasXConnect(new ConnectPoint(ingressDevice, port))) {
Charles Chan72779502016-04-23 17:36:10 -0700803 return port;
804 }
805 }
806 }
807 return null;
808 }
Jonghwan Hyune5ef7622017-08-25 17:48:36 -0700809
810 /**
811 * Removes filtering objective for given device and port.
812 *
813 * @param deviceId device ID
814 * @param port ingress port number
815 * @param assignedVlan assigned VLAN ID
816 * @param mcastIp multicast IP address
817 */
818 private void removeFilterToDevice(DeviceId deviceId, PortNumber port, VlanId assignedVlan, IpAddress mcastIp) {
819 // Do nothing if the port is configured as suppressed
820 ConnectPoint connectPoint = new ConnectPoint(deviceId, port);
821 SegmentRoutingAppConfig appConfig = srManager.cfgService
822 .getConfig(srManager.appId, SegmentRoutingAppConfig.class);
823 if (appConfig != null && appConfig.suppressSubnet().contains(connectPoint)) {
824 log.info("Ignore suppressed port {}", connectPoint);
825 return;
826 }
827
828 FilteringObjective.Builder filtObjBuilder =
829 filterObjBuilder(deviceId, port, assignedVlan, mcastIp);
830 ObjectiveContext context = new DefaultObjectiveContext(
831 (objective) -> log.debug("Successfully removed filter on {}/{}, vlan {}",
832 deviceId, port.toLong(), assignedVlan),
833 (objective, error) ->
834 log.warn("Failed to remove filter on {}/{}, vlan {}: {}",
835 deviceId, port.toLong(), assignedVlan, error));
836 srManager.flowObjectiveService.filter(deviceId, filtObjBuilder.remove(context));
837 }
838
839 /**
840 * Adds or removes filtering objective for given device and port.
841 *
842 * @param deviceId device ID
843 * @param portNum ingress port number
844 * @param vlanId assigned VLAN ID
845 * @param install true to add, false to remove
846 */
847 protected void updateFilterToDevice(DeviceId deviceId, PortNumber portNum,
848 VlanId vlanId, boolean install) {
849 srManager.multicastRouteService.getRoutes().forEach(mcastRoute -> {
850 ConnectPoint source = srManager.multicastRouteService.fetchSource(mcastRoute);
851 if (source.deviceId().equals(deviceId) && source.port().equals(portNum)) {
852 if (install) {
853 addFilterToDevice(deviceId, portNum, vlanId, mcastRoute.group());
854 } else {
855 removeFilterToDevice(deviceId, portNum, vlanId, mcastRoute.group());
856 }
857 }
858 });
859 }
Charles Chanc91c8782016-03-30 17:54:24 -0700860}