2 * Copyright (c) 2016, 2017 Ericsson India Global Services Pvt Ltd. and others. All rights reserved.
4 * This program and the accompanying materials are made available under the
5 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6 * and is available at http://www.eclipse.org/legal/epl-v10.html
9 package org.opendaylight.netvirt.elan.internal;
11 import static java.util.Collections.emptyMap;
12 import static org.opendaylight.mdsal.binding.util.Datastore.CONFIGURATION;
14 import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
15 import java.util.ArrayList;
16 import java.util.Collection;
17 import java.util.Collections;
18 import java.util.HashMap;
19 import java.util.HashSet;
20 import java.util.List;
22 import java.util.Objects;
23 import java.util.Optional;
25 import java.util.concurrent.Future;
26 import java.util.function.BiFunction;
27 import javax.inject.Inject;
28 import javax.inject.Singleton;
29 import org.eclipse.jdt.annotation.NonNull;
30 import org.eclipse.jdt.annotation.Nullable;
31 import org.opendaylight.genius.interfacemanager.exceptions.InterfaceAlreadyExistsException;
32 import org.opendaylight.genius.interfacemanager.globals.IfmConstants;
33 import org.opendaylight.genius.interfacemanager.interfaces.IInterfaceManager;
34 import org.opendaylight.genius.mdsalutil.MDSALUtil;
35 import org.opendaylight.genius.mdsalutil.NwConstants;
36 import org.opendaylight.genius.mdsalutil.interfaces.IMdsalApiManager;
37 import org.opendaylight.genius.utils.ServiceIndex;
38 import org.opendaylight.genius.utils.hwvtep.HwvtepSouthboundConstants;
39 import org.opendaylight.infrautils.inject.AbstractLifecycle;
40 import org.opendaylight.infrautils.utils.concurrent.LoggingFutures;
41 import org.opendaylight.mdsal.binding.api.DataBroker;
42 import org.opendaylight.mdsal.binding.util.ManagedNewTransactionRunner;
43 import org.opendaylight.mdsal.binding.util.ManagedNewTransactionRunnerImpl;
44 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
45 import org.opendaylight.mdsal.eos.binding.api.Entity;
46 import org.opendaylight.mdsal.eos.binding.api.EntityOwnershipCandidateRegistration;
47 import org.opendaylight.mdsal.eos.binding.api.EntityOwnershipService;
48 import org.opendaylight.mdsal.eos.common.api.CandidateAlreadyRegisteredException;
49 import org.opendaylight.netvirt.elan.arp.responder.ArpResponderInput;
50 import org.opendaylight.netvirt.elan.arp.responder.ArpResponderUtil;
51 import org.opendaylight.netvirt.elan.cache.ElanInstanceCache;
52 import org.opendaylight.netvirt.elan.cache.ElanInterfaceCache;
53 import org.opendaylight.netvirt.elan.utils.ElanConstants;
54 import org.opendaylight.netvirt.elan.utils.ElanUtils;
55 import org.opendaylight.netvirt.elanmanager.api.ElanHelper;
56 import org.opendaylight.netvirt.elanmanager.api.IElanService;
57 import org.opendaylight.ovsdb.utils.southbound.utils.SouthboundUtils;
58 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.Interfaces;
59 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface;
60 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.InterfaceKey;
61 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.yang.types.rev130715.PhysAddress;
62 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.inventory.rev130819.tables.table.Flow;
63 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.CreateIdPoolInput;
64 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.CreateIdPoolInputBuilder;
65 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.CreateIdPoolOutput;
66 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.IdManagerService;
67 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.rev160406.IfL2vlan;
68 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInstance;
69 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInstanceBuilder;
70 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInterface;
71 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInterface.EtreeInterfaceType;
72 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInterfaceBuilder;
73 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.ElanInstances;
74 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.ElanInterfaces;
75 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan._interface.forwarding.entries.ElanInterfaceMac;
76 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.dpn.interfaces.elan.dpn.interfaces.list.DpnInterfaces;
77 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.instances.ElanInstance;
78 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.instances.ElanInstanceBuilder;
79 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.instances.ElanInstanceKey;
80 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.ElanInterface;
81 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.ElanInterfaceBuilder;
82 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.ElanInterfaceKey;
83 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.elan._interface.StaticMacEntries;
84 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.elan._interface.StaticMacEntriesBuilder;
85 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.state.Elan;
86 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.forwarding.entries.MacEntry;
87 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.forwarding.entries.MacEntryKey;
88 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Node;
89 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.NodeKey;
90 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
91 import org.opendaylight.yangtools.yang.common.RpcResult;
92 import org.opendaylight.yangtools.yang.common.Uint32;
93 import org.opendaylight.yangtools.yang.common.Uint64;
94 import org.slf4j.Logger;
95 import org.slf4j.LoggerFactory;
98 public class ElanServiceProvider extends AbstractLifecycle implements IElanService {
100 private static final Logger LOG = LoggerFactory.getLogger(ElanServiceProvider.class);
102 private final IdManagerService idManager;
103 private final IInterfaceManager interfaceManager;
104 private final ElanBridgeManager bridgeMgr;
105 private final DataBroker broker;
106 private final ManagedNewTransactionRunner txRunner;
107 private final ElanUtils elanUtils;
108 private final SouthboundUtils southboundUtils;
109 private final IMdsalApiManager mdsalManager;
110 private final ElanInstanceCache elanInstanceCache;
111 private final ElanInterfaceCache elanInterfaceCache;
112 private boolean isL2BeforeL3;
114 private final EntityOwnershipCandidateRegistration candidateRegistration;
117 public ElanServiceProvider(IdManagerService idManager, IInterfaceManager interfaceManager,
118 ElanBridgeManager bridgeMgr,
119 DataBroker dataBroker,
121 EntityOwnershipService entityOwnershipService,
122 SouthboundUtils southboundUtils, ElanInstanceCache elanInstanceCache,
123 ElanInterfaceCache elanInterfaceCache, IMdsalApiManager mdsalManager) {
124 this.idManager = idManager;
125 this.interfaceManager = interfaceManager;
126 this.bridgeMgr = bridgeMgr;
127 this.broker = dataBroker;
128 this.txRunner = new ManagedNewTransactionRunnerImpl(dataBroker);
129 this.elanUtils = elanUtils;
130 this.southboundUtils = southboundUtils;
131 this.elanInstanceCache = elanInstanceCache;
132 this.elanInterfaceCache = elanInterfaceCache;
133 this.mdsalManager = mdsalManager;
135 candidateRegistration = registerCandidate(entityOwnershipService);
139 private static EntityOwnershipCandidateRegistration registerCandidate(
140 EntityOwnershipService entityOwnershipService) {
142 return entityOwnershipService.registerCandidate(
143 new Entity(HwvtepSouthboundConstants.ELAN_ENTITY_TYPE, HwvtepSouthboundConstants.ELAN_ENTITY_TYPE));
144 } catch (CandidateAlreadyRegisteredException e) {
145 LOG.error("failed to register the entity");
151 @SuppressWarnings("checkstyle:IllegalCatch")
152 protected void start() throws Exception {
153 LOG.info("Starting ElanServiceProvider");
159 protected void stop() {
160 if (candidateRegistration != null) {
161 candidateRegistration.close();
164 LOG.info("ElanServiceProvider stopped");
168 // Confusing with isOpenstackVniSemanticsEnforced but this is an interface method so can't change it.
169 @SuppressFBWarnings("NM_CONFUSING")
170 public Boolean isOpenStackVniSemanticsEnforced() {
171 return elanUtils.isOpenstackVniSemanticsEnforced();
174 private void createIdPool() throws Exception {
175 CreateIdPoolInput createPool = new CreateIdPoolInputBuilder().setPoolName(ElanConstants.ELAN_ID_POOL_NAME)
176 .setLow(ElanConstants.ELAN_ID_LOW_VALUE).setHigh(ElanConstants.ELAN_ID_HIGH_VALUE).build();
177 Future<RpcResult<CreateIdPoolOutput>> result = idManager.createIdPool(createPool);
178 if (result != null && result.get().isSuccessful()) {
179 LOG.debug("ELAN Id Pool is created successfully");
184 public boolean createElanInstance(String elanInstanceName, long macTimeout, String description) {
185 Optional<ElanInstance> existingElanInstance = elanInstanceCache.get(elanInstanceName);
186 boolean isSuccess = true;
187 if (existingElanInstance.isPresent()) {
188 if (compareWithExistingElanInstance(existingElanInstance.get(), macTimeout, description)) {
189 LOG.debug("Elan Instance is already present in the Operational DS {}", existingElanInstance);
192 ElanInstance updateElanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
193 .setDescription(description).setMacTimeout(macTimeout)
194 .withKey(new ElanInstanceKey(elanInstanceName)).build();
195 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
196 ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName), updateElanInstance);
197 LOG.debug("Updating the Elan Instance {} with MAC TIME-OUT {} and Description {}",
198 updateElanInstance, macTimeout, description);
201 ElanInstance elanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
202 .setMacTimeout(macTimeout).setDescription(description)
203 .withKey(new ElanInstanceKey(elanInstanceName)).build();
204 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
205 ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName), elanInstance);
206 LOG.debug("Creating the new Elan Instance {}", elanInstance);
212 public boolean createEtreeInstance(String elanInstanceName, long macTimeout, String description) {
213 Optional<ElanInstance> existingElanInstance = elanInstanceCache.get(elanInstanceName);
214 boolean isSuccess = true;
215 if (existingElanInstance.isPresent()) {
216 if (compareWithExistingElanInstance(existingElanInstance.get(), macTimeout, description)) {
217 LOG.warn("Etree Instance is already present in the Operational DS {}", existingElanInstance);
220 EtreeInstance etreeInstance = new EtreeInstanceBuilder().build();
221 ElanInstance updateElanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
222 .setDescription(description).setMacTimeout(macTimeout)
223 .withKey(new ElanInstanceKey(elanInstanceName))
224 .addAugmentation(etreeInstance).build();
225 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
226 ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName), updateElanInstance);
227 LOG.debug("Updating the Etree Instance {} with MAC TIME-OUT {} and Description {} ",
228 updateElanInstance, macTimeout, description);
231 EtreeInstance etreeInstance = new EtreeInstanceBuilder().build();
232 ElanInstance elanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
233 .setMacTimeout(macTimeout).setDescription(description)
234 .withKey(new ElanInstanceKey(elanInstanceName))
235 .addAugmentation(etreeInstance).build();
236 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
237 ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName), elanInstance);
238 LOG.debug("Creating the new Etree Instance {}", elanInstance);
245 public EtreeInterface getEtreeInterfaceByElanInterfaceName(String elanInterface) {
246 return elanInterfaceCache.getEtreeInterface(elanInterface).orElse(null);
249 public static boolean compareWithExistingElanInstance(ElanInstance existingElanInstance, long macTimeOut,
250 String description) {
251 boolean isEqual = false;
252 if (existingElanInstance.getMacTimeout().longValue() == macTimeOut
253 && Objects.equals(existingElanInstance.getDescription(), description)) {
260 public void updateElanInstance(String elanInstanceName, long newMacTimout, String newDescription) {
261 createElanInstance(elanInstanceName, newMacTimout, newDescription);
265 public boolean deleteEtreeInstance(String etreeInstanceName) {
266 return deleteElanInstance(etreeInstanceName);
270 public boolean deleteElanInstance(String elanInstanceName) {
271 Optional<ElanInstance> existingElanInstance = elanInstanceCache.get(elanInstanceName);
272 if (!existingElanInstance.isPresent()) {
273 LOG.debug("Elan Instance is not present for {}", elanInstanceName);
276 LOG.debug("Deletion of the existing Elan Instance {}", existingElanInstance);
277 ElanUtils.delete(broker, LogicalDatastoreType.CONFIGURATION,
278 ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName));
283 public void addEtreeInterface(String etreeInstanceName, String interfaceName, EtreeInterfaceType interfaceType,
284 List<String> staticMacAddresses, String description) {
285 Optional<ElanInstance> existingElanInstance = elanInstanceCache.get(etreeInstanceName);
286 if (existingElanInstance.isPresent()
287 && existingElanInstance.get().augmentation(EtreeInstance.class) != null) {
288 EtreeInterface etreeInterface = new EtreeInterfaceBuilder().setEtreeInterfaceType(interfaceType).build();
289 ElanInterface elanInterface;
290 if (staticMacAddresses == null) {
291 elanInterface = new ElanInterfaceBuilder().setElanInstanceName(etreeInstanceName)
292 .setDescription(description).setName(interfaceName).withKey(new ElanInterfaceKey(interfaceName))
293 .addAugmentation(etreeInterface).build();
295 List<StaticMacEntries> staticMacEntries = ElanUtils.getStaticMacEntries(staticMacAddresses);
296 elanInterface = new ElanInterfaceBuilder().setElanInstanceName(etreeInstanceName)
297 .setDescription(description).setName(interfaceName)
298 .setStaticMacEntries(staticMacEntries)
299 .withKey(new ElanInterfaceKey(interfaceName))
300 .addAugmentation(etreeInterface).build();
302 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
303 ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
304 LOG.debug("Creating the new Etree Interface {}", elanInterface);
309 public void addElanInterface(String elanInstanceName, String interfaceName,
310 @Nullable List<String> staticMacAddresses, @Nullable String description) {
311 Optional<ElanInstance> existingElanInstance = elanInstanceCache.get(elanInstanceName);
312 if (existingElanInstance.isPresent()) {
313 ElanInterfaceBuilder elanInterfaceBuilder = new ElanInterfaceBuilder()
314 .setElanInstanceName(elanInstanceName)
315 .setDescription(description).setName(interfaceName)
316 .withKey(new ElanInterfaceKey(interfaceName));
317 if (staticMacAddresses != null) {
318 List<StaticMacEntries> staticMacEntries = ElanUtils.getStaticMacEntries(staticMacAddresses);
319 elanInterfaceBuilder.setStaticMacEntries(staticMacEntries);
321 ElanInterface elanInterface = elanInterfaceBuilder.build();
322 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
323 ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
324 LOG.debug("Created the new ELan Interface {}", elanInterface);
329 public void updateElanInterface(String elanInstanceName, String interfaceName,
330 List<String> updatedStaticMacAddresses, String newDescription) {
331 Optional<ElanInterface> existingElanInterface = elanInterfaceCache.get(interfaceName);
332 if (!existingElanInterface.isPresent()) {
336 List<StaticMacEntries> updatedStaticMacEntries = ElanUtils.getStaticMacEntries(updatedStaticMacAddresses);
337 LOG.debug("updating the ElanInterface with new Mac Entries {}", updatedStaticMacAddresses);
338 ElanInterface elanInterface = new ElanInterfaceBuilder().setElanInstanceName(elanInstanceName)
339 .setName(interfaceName).setDescription(newDescription).setStaticMacEntries(updatedStaticMacEntries)
340 .withKey(new ElanInterfaceKey(interfaceName)).build();
341 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
342 ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
346 public void deleteEtreeInterface(String interfaceName) {
347 deleteElanInterface(interfaceName);
348 LOG.debug("deleting the Etree Interface {}", interfaceName);
352 public void deleteElanInterface(String interfaceName) {
353 Optional<ElanInterface> existingElanInterface = elanInterfaceCache.get(interfaceName);
354 if (existingElanInterface.isPresent()) {
355 ElanUtils.delete(broker, LogicalDatastoreType.CONFIGURATION,
356 ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName));
357 LOG.debug("deleting the Elan Interface {}", existingElanInterface);
362 public void addStaticMacAddress(String interfaceName, String macAddress) {
363 Optional<ElanInterface> existingElanInterface = elanInterfaceCache.get(interfaceName);
364 if (existingElanInterface.isPresent()) {
365 StaticMacEntriesBuilder staticMacEntriesBuilder = new StaticMacEntriesBuilder();
366 StaticMacEntries staticMacEntry = staticMacEntriesBuilder.setMacAddress(
367 new PhysAddress(macAddress)).build();
368 InstanceIdentifier<StaticMacEntries> staticMacEntriesIdentifier =
369 ElanUtils.getStaticMacEntriesCfgDataPathIdentifier(interfaceName, macAddress);
370 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION, staticMacEntriesIdentifier, staticMacEntry);
375 public void deleteStaticMacAddress(String interfaceName, String macAddress) {
376 Optional<ElanInterface> existingElanInterface = elanInterfaceCache.get(interfaceName);
377 if (existingElanInterface.isPresent()) {
378 InstanceIdentifier<StaticMacEntries> staticMacEntriesIdentifier =
379 ElanUtils.getStaticMacEntriesCfgDataPathIdentifier(interfaceName,
381 MDSALUtil.syncDelete(broker, LogicalDatastoreType.CONFIGURATION, staticMacEntriesIdentifier);
386 public Collection<MacEntry> getElanMacTable(String elanInstanceName) {
387 Elan elanInfo = ElanUtils.getElanByName(broker, elanInstanceName);
388 List<MacEntry> macAddress = new ArrayList<>();
389 if (elanInfo == null) {
392 List<String> elanInterfaces = elanInfo.getElanInterfaces();
393 if (elanInterfaces != null && elanInterfaces.size() > 0) {
394 for (String elanInterface : elanInterfaces) {
395 ElanInterfaceMac elanInterfaceMac = elanUtils.getElanInterfaceMacByInterfaceName(elanInterface);
396 if (elanInterfaceMac != null && elanInterfaceMac.getMacEntry() != null
397 && elanInterfaceMac.getMacEntry().size() > 0) {
398 macAddress.addAll(elanInterfaceMac.nonnullMacEntry().values());
406 public void flushMACTable(String elanInstanceName) {
407 Elan elanInfo = ElanUtils.getElanByName(broker, elanInstanceName);
408 if (elanInfo == null) {
411 List<String> elanInterfaces = elanInfo.getElanInterfaces();
412 if (elanInterfaces == null || elanInterfaces.isEmpty()) {
415 for (String elanInterface : elanInterfaces) {
416 ElanInterfaceMac elanInterfaceMac = elanUtils.getElanInterfaceMacByInterfaceName(elanInterface);
417 if (elanInterfaceMac.getMacEntry() != null && elanInterfaceMac.getMacEntry().size() > 0) {
418 Map<MacEntryKey, MacEntry> macEntries = elanInterfaceMac.nonnullMacEntry();
419 for (MacEntry macEntry : macEntries.values()) {
420 deleteStaticMacAddress(elanInterface, macEntry.getMacAddress().getValue());
429 public ElanInstance getElanInstance(String elanName) {
430 return elanInstanceCache.get(elanName).orElse(null);
434 public List<ElanInstance> getElanInstances() {
435 InstanceIdentifier<ElanInstances> elanInstancesIdentifier = InstanceIdentifier.builder(ElanInstances.class)
437 return new ArrayList<>(ElanUtils.read(broker, LogicalDatastoreType.CONFIGURATION, elanInstancesIdentifier).map(
438 ElanInstances::nonnullElanInstance).orElse(emptyMap()).values());
443 public List<String> getElanInterfaces(String elanInstanceName) {
444 List<String> elanInterfaces = new ArrayList<>();
445 InstanceIdentifier<ElanInterfaces> elanInterfacesIdentifier = InstanceIdentifier.builder(ElanInterfaces.class)
447 Optional<ElanInterfaces> elanInterfacesOptional = ElanUtils.read(broker, LogicalDatastoreType.CONFIGURATION,
448 elanInterfacesIdentifier);
449 if (!elanInterfacesOptional.isPresent()) {
450 return elanInterfaces;
452 Map<ElanInterfaceKey, ElanInterface> elanInterfaceList = elanInterfacesOptional.get().nonnullElanInterface();
453 for (ElanInterface elanInterface : elanInterfaceList.values()) {
454 if (Objects.equals(elanInterface.getElanInstanceName(), elanInstanceName)) {
455 elanInterfaces.add(elanInterface.getName());
458 return elanInterfaces;
462 public void createExternalElanNetworks(Node node) {
463 handleExternalElanNetworks(node, true, (elanInstance, interfaceName) -> {
464 createExternalElanNetwork(elanInstance, interfaceName);
470 public void createExternalElanNetwork(ElanInstance elanInstance) {
471 handleExternalElanNetwork(elanInstance, false, (elanInstance1, interfaceName) -> {
472 createExternalElanNetwork(elanInstance1, interfaceName);
477 protected void createExternalElanNetwork(ElanInstance elanInstance, Uint64 dpId) {
478 String providerIntfName = bridgeMgr.getProviderInterfaceName(dpId, elanInstance.getPhysicalNetworkName());
479 String intfName = providerIntfName + IfmConstants.OF_URI_SEPARATOR + elanInstance.getSegmentationId();
480 Interface memberIntf = interfaceManager.getInterfaceInfoFromConfigDataStore(intfName);
481 if (memberIntf == null) {
482 LOG.debug("creating vlan prv intf in elan {}, dpn {}", elanInstance.getElanInstanceName(),
484 createExternalElanNetwork(elanInstance, providerIntfName);
488 private void createExternalElanNetwork(ElanInstance elanInstance, String interfaceName) {
489 if (interfaceName == null) {
490 LOG.trace("No physical interface is attached to {}", elanInstance.getPhysicalNetworkName());
494 String elanInterfaceName = createIetfInterfaces(elanInstance, interfaceName);
495 addElanInterface(elanInstance.getElanInstanceName(), elanInterfaceName, null, null);
499 public void updateExternalElanNetwork(ElanInstance elanInstance) {
500 handleExternalElanNetwork(elanInstance, true, (elanInstance1, interfaceName) -> {
501 createExternalElanNetwork(elanInstance1, interfaceName);
507 public void deleteExternalElanNetworks(Node node) {
508 handleExternalElanNetworks(node, false, (elanInstance, interfaceName) -> {
509 deleteExternalElanNetwork(elanInstance, interfaceName);
515 public void deleteExternalElanNetwork(ElanInstance elanInstance) {
516 handleExternalElanNetwork(elanInstance, false, (elanInstance1, interfaceName) -> {
517 deleteExternalElanNetwork(elanInstance1, interfaceName);
522 protected void deleteExternalElanNetwork(ElanInstance elanInstance, Uint64 dpnId) {
523 String providerIntfName = bridgeMgr.getProviderInterfaceName(dpnId, elanInstance.getPhysicalNetworkName());
524 String intfName = providerIntfName + IfmConstants.OF_URI_SEPARATOR + elanInstance.getSegmentationId();
525 Interface memberIntf = interfaceManager.getInterfaceInfoFromConfigDataStore(intfName);
526 if (memberIntf != null) {
527 deleteElanInterface(intfName);
528 deleteIetfInterface(intfName);
529 LOG.debug("delete vlan prv intf {} in elan {}, dpID {}", intfName,
530 elanInstance.getElanInstanceName(), dpnId);
532 LOG.debug("vlan prv intf {} not found in interfacemgr config DS", intfName);
536 private void deleteExternalElanNetwork(ElanInstance elanInstance, String interfaceName) {
537 if (interfaceName == null) {
538 LOG.trace("No physial interface is attached to {}", elanInstance.getPhysicalNetworkName());
542 String elanInstanceName = elanInstance.getElanInstanceName();
543 for (String elanInterface : getExternalElanInterfaces(elanInstanceName)) {
544 if (elanInterface.startsWith(interfaceName)) {
545 if (ElanUtils.isVlan(elanInstance)) {
546 deleteIetfInterface(elanInterface);
548 String trunkInterfaceName = getTrunkInterfaceName(interfaceName);
549 if (shouldDeleteTrunk(trunkInterfaceName, elanInterface)) {
550 deleteIetfInterface(trunkInterfaceName);
552 deleteElanInterface(elanInterface);
557 private boolean shouldDeleteTrunk(String trunkInterfaceName, String elanInterfaceName) {
558 List<Interface> childInterfaces = interfaceManager.getChildInterfaces(trunkInterfaceName);
559 if (childInterfaces == null || childInterfaces.isEmpty()
560 || childInterfaces.size() == 1 && elanInterfaceName.equals(childInterfaces.get(0).getName())) {
561 LOG.debug("No more VLAN member interfaces left for trunk {}", trunkInterfaceName);
565 LOG.debug("Trunk interface {} has {} VLAN member interfaces left", trunkInterfaceName, childInterfaces.size());
570 public void updateExternalElanNetworks(Node origNode, Node updatedNode) {
571 if (!bridgeMgr.isIntegrationBridge(updatedNode)) {
575 List<ElanInstance> elanInstances = getElanInstances();
576 if (elanInstances == null || elanInstances.isEmpty()) {
577 LOG.trace("No ELAN instances found");
581 LOG.trace("updateExternalElanNetworks, orig bridge {} . updated bridge {}", origNode, updatedNode);
583 Map<String, String> origProviderMappping = getMapFromOtherConfig(origNode,
584 ElanBridgeManager.PROVIDER_MAPPINGS_KEY);
585 Map<String, String> updatedProviderMappping = getMapFromOtherConfig(updatedNode,
586 ElanBridgeManager.PROVIDER_MAPPINGS_KEY);
588 boolean hasDatapathIdOnOrigNode = bridgeMgr.hasDatapathID(origNode);
589 boolean hasDatapathIdOnUpdatedNode = bridgeMgr.hasDatapathID(updatedNode);
590 Uint64 origDpnID = bridgeMgr.getDatapathId(origNode);
592 for (ElanInstance elanInstance : elanInstances) {
593 String physicalNetworkName = elanInstance.getPhysicalNetworkName();
594 boolean createExternalElanNw = true;
595 if (physicalNetworkName != null) {
596 String origPortName = origProviderMappping.get(physicalNetworkName);
597 String updatedPortName = updatedProviderMappping.get(physicalNetworkName);
599 * for internal vlan network, vlan provider interface creation should be
600 * triggered only if there is existing vlan provider intf indicating presence
601 * of VM ports on the DPN
603 if (hasDatapathIdOnOrigNode && !elanInstance.isExternal()
604 && ElanUtils.isVlan(elanInstance)) {
605 String externalIntf = getExternalElanInterface(elanInstance.getElanInstanceName(),
607 if (externalIntf == null) {
608 createExternalElanNw = false;
611 if (hasPortNameRemoved(origPortName, updatedPortName)) {
612 deleteExternalElanNetwork(elanInstance,
613 bridgeMgr.getProviderInterfaceName(origNode, physicalNetworkName));
616 if (createExternalElanNw && (hasPortNameUpdated(origPortName, updatedPortName)
617 || hasDatapathIdAdded(hasDatapathIdOnOrigNode, hasDatapathIdOnUpdatedNode))) {
618 createExternalElanNetwork(elanInstance,
619 bridgeMgr.getProviderInterfaceName(updatedNode, physicalNetworkName));
625 private static boolean hasDatapathIdAdded(boolean hasDatapathIdOnOrigNode, boolean hasDatapathIdOnUpdatedNode) {
626 return !hasDatapathIdOnOrigNode && hasDatapathIdOnUpdatedNode;
629 private static boolean hasPortNameUpdated(String origPortName, String updatedPortName) {
630 return updatedPortName != null && !updatedPortName.equals(origPortName);
633 private static boolean hasPortNameRemoved(String origPortName, String updatedPortName) {
634 return origPortName != null && !origPortName.equals(updatedPortName);
637 private Map<String, String> getMapFromOtherConfig(Node node, String otherConfigColumn) {
638 return bridgeMgr.getOpenvswitchOtherConfigMap(node, otherConfigColumn);
642 public Collection<String> getExternalElanInterfaces(String elanInstanceName) {
643 List<String> elanInterfaces = getElanInterfaces(elanInstanceName);
644 if (elanInterfaces.isEmpty()) {
645 LOG.trace("No ELAN interfaces defined for {}", elanInstanceName);
646 return Collections.emptySet();
649 Set<String> externalElanInterfaces = new HashSet<>();
650 for (String elanInterface : elanInterfaces) {
651 if (interfaceManager.isExternalInterface(elanInterface)) {
652 externalElanInterfaces.add(elanInterface);
656 return externalElanInterfaces;
660 public String getExternalElanInterface(String elanInstanceName, Uint64 dpnId) {
661 return elanUtils.getExternalElanInterface(elanInstanceName, dpnId);
665 public boolean isExternalInterface(String interfaceName) {
666 return interfaceManager.isExternalInterface(interfaceName);
671 public ElanInterface getElanInterfaceByElanInterfaceName(String interfaceName) {
672 return elanInterfaceCache.get(interfaceName).orElse(null);
676 public void handleKnownL3DmacAddress(String macAddress, String elanInstanceName, int addOrRemove) {
677 if (addOrRemove == NwConstants.ADD_FLOW) {
678 addKnownL3DmacAddress(macAddress, elanInstanceName);
680 removeKnownL3DmacAddress(macAddress, elanInstanceName);
685 public void addKnownL3DmacAddress(String macAddress, String elanInstanceName) {
687 LOG.trace("ELAN service is after L3VPN in the Netvirt pipeline skip known L3DMAC flows installation");
690 ElanInstance elanInstance = elanInstanceCache.get(elanInstanceName).orElse(null);
691 if (elanInstance == null) {
692 LOG.warn("Null elan instance {}", elanInstanceName);
696 List<Uint64> dpnsIdsForElanInstance = elanUtils.getParticipatingDpnsInElanInstance(elanInstanceName);
697 if (dpnsIdsForElanInstance.isEmpty()) {
698 LOG.warn("No DPNs for elan instance {}", elanInstance);
702 elanUtils.addDmacRedirectToDispatcherFlows(elanInstance.getElanTag(), elanInstanceName, macAddress,
703 dpnsIdsForElanInstance);
707 public void removeKnownL3DmacAddress(String macAddress, String elanInstanceName) {
709 LOG.trace("ELAN service is after L3VPN in the Netvirt pipeline skip known L3DMAC flows installation");
712 ElanInstance elanInstance = elanInstanceCache.get(elanInstanceName).orElse(null);
713 if (elanInstance == null) {
714 LOG.warn("Null elan instance {}", elanInstanceName);
718 List<Uint64> dpnsIdsForElanInstance = elanUtils.getParticipatingDpnsInElanInstance(elanInstanceName);
719 if (dpnsIdsForElanInstance.isEmpty()) {
720 LOG.warn("No DPNs for elan instance {}", elanInstance);
724 elanUtils.removeDmacRedirectToDispatcherFlows(elanInstance.getElanTag(),
725 macAddress, dpnsIdsForElanInstance);
729 * Create ietf-interfaces based on the ELAN segment type.<br>
730 * For segment type flat - create transparent interface pointing to the
731 * patch-port attached to the physnet port.<br>
732 * For segment type vlan - create trunk interface pointing to the patch-port
733 * attached to the physnet port + trunk-member interface pointing to the
736 * @param elanInstance
739 * parent interface name
740 * @return the name of the interface to be added to the ELAN instance i.e.
741 * trunk-member name for vlan network and transparent for flat
742 * network or null otherwise
744 private String createIetfInterfaces(ElanInstance elanInstance, String parentRef) {
745 String interfaceName = null;
748 String trunkName = getTrunkInterfaceName(parentRef);
749 // trunk interface may have been created by other vlan network
750 Interface trunkInterface = interfaceManager.getInterfaceInfoFromConfigDataStore(trunkName);
751 if (trunkInterface == null) {
752 interfaceManager.createVLANInterface(trunkName, parentRef, null, null,
753 IfL2vlan.L2vlanMode.Trunk, true);
755 if (ElanUtils.isFlat(elanInstance)) {
756 interfaceName = trunkName;
757 } else if (ElanUtils.isVlan(elanInstance)) {
758 Long segmentationId = elanInstance.getSegmentationId().toJava();
759 interfaceName = parentRef + IfmConstants.OF_URI_SEPARATOR + segmentationId;
760 interfaceManager.createVLANInterface(interfaceName, trunkName, segmentationId.intValue(), null,
761 IfL2vlan.L2vlanMode.TrunkMember, true);
763 } catch (InterfaceAlreadyExistsException e) {
764 LOG.trace("Interface {} was already created", interfaceName);
767 return interfaceName;
770 private void deleteIetfInterface(String interfaceName) {
771 InterfaceKey interfaceKey = new InterfaceKey(interfaceName);
772 InstanceIdentifier<Interface> interfaceInstanceIdentifier = InstanceIdentifier.builder(Interfaces.class)
773 .child(Interface.class, interfaceKey).build();
774 MDSALUtil.syncDelete(broker, LogicalDatastoreType.CONFIGURATION, interfaceInstanceIdentifier);
775 LOG.debug("Deleting IETF interface {}", interfaceName);
778 private void handleExternalElanNetworks(Node node, boolean skipIntVlanNw,
779 BiFunction<ElanInstance, String, Void> function) {
780 if (!bridgeMgr.isIntegrationBridge(node)) {
784 List<ElanInstance> elanInstances = getElanInstances();
785 if (elanInstances == null || elanInstances.isEmpty()) {
786 LOG.trace("No ELAN instances found");
790 for (ElanInstance elanInstance : elanInstances) {
791 if (skipIntVlanNw && !elanInstance.isExternal() && ElanUtils.isVlan(elanInstance)) {
794 String interfaceName = bridgeMgr.getProviderInterfaceName(node, elanInstance.getPhysicalNetworkName());
795 if (interfaceName != null) {
796 function.apply(elanInstance, interfaceName);
801 private void handleExternalElanNetwork(ElanInstance elanInstance, boolean update,
802 BiFunction<ElanInstance, String, Void> function) {
803 String elanInstanceName = elanInstance.getElanInstanceName();
804 boolean isFlatOrVlanNetwork = (ElanUtils.isFlat(elanInstance) || ElanUtils.isVlan(elanInstance));
805 if (!isFlatOrVlanNetwork) {
806 LOG.error("Network is not of type FLAT/VLAN."
807 + "Ignoring Elan-interface creation for given ProviderInterface {}",
808 elanInstance.getPhysicalNetworkName());
811 if (elanInstance.getPhysicalNetworkName() == null) {
812 LOG.trace("No physical network attached to {}", elanInstanceName);
816 Map<NodeKey, Node> nodes = southboundUtils.getOvsdbNodes();
817 if (nodes == null || nodes.isEmpty()) {
818 LOG.trace("No OVS nodes found while creating external network for ELAN {}",
819 elanInstance.getElanInstanceName());
823 for (Node node : nodes.values()) {
824 if (bridgeMgr.isIntegrationBridge(node)) {
825 if (update && !elanInstance.isExternal()) {
826 DpnInterfaces dpnInterfaces = elanUtils.getElanInterfaceInfoByElanDpn(elanInstanceName,
827 bridgeMgr.getDatapathId(node));
828 if (dpnInterfaces == null || dpnInterfaces.getInterfaces().isEmpty()) {
832 String interfaceName = bridgeMgr.getProviderInterfaceName(node, elanInstance.getPhysicalNetworkName());
833 function.apply(elanInstance, interfaceName);
838 private static String getTrunkInterfaceName(String parentRef) {
839 return parentRef + IfmConstants.OF_URI_SEPARATOR + "trunk";
842 private void setIsL2BeforeL3() {
843 short elanServiceRealIndex = ServiceIndex.getIndex(NwConstants.ELAN_SERVICE_NAME,
844 NwConstants.ELAN_SERVICE_INDEX);
845 short l3vpnServiceRealIndex = ServiceIndex.getIndex(NwConstants.L3VPN_SERVICE_NAME,
846 NwConstants.L3VPN_SERVICE_INDEX);
847 if (elanServiceRealIndex < l3vpnServiceRealIndex) {
848 LOG.info("ELAN service is set before L3VPN service in the Netvirt pipeline");
851 LOG.info("ELAN service is set after L3VPN service in the Netvirt pipeline");
852 isL2BeforeL3 = false;
857 public void addArpResponderFlow(ArpResponderInput arpResponderInput) {
858 String ingressInterfaceName = arpResponderInput.getInterfaceName();
859 String macAddress = arpResponderInput.getSha();
860 String ipAddress = arpResponderInput.getSpa();
861 int lportTag = arpResponderInput.getLportTag();
862 Uint64 dpnId = Uint64.valueOf(arpResponderInput.getDpId());
864 LOG.info("Installing the ARP responder flow on DPN {} for Interface {} with MAC {} & IP {}", dpnId,
865 ingressInterfaceName, macAddress, ipAddress);
866 Optional<ElanInterface> elanIface = elanInterfaceCache.get(ingressInterfaceName);
867 ElanInstance elanInstance = elanIface.isPresent()
868 ? elanInstanceCache.get(elanIface.get().getElanInstanceName()).orElse(null) : null;
869 if (elanInstance == null) {
870 LOG.debug("addArpResponderFlow: elanInstance is null, Failed to install arp responder flow for dpnId {}"
871 + "for Interface {} with MAC {} & IP {}", dpnId, ingressInterfaceName, macAddress, ipAddress);
874 String flowId = ArpResponderUtil.getFlowId(lportTag, ipAddress);
875 Map<org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.InstructionKey,
876 org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.Instruction>
877 arpResponderInputInstructionsMap = new HashMap<>();
878 int instructionKey = 0;
879 for (org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.Instruction
880 instructionObj : arpResponderInput.getInstructions()) {
881 arpResponderInputInstructionsMap.put(new org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types
882 .rev131026.instruction.list.InstructionKey(++instructionKey), instructionObj);
885 MDSALUtil.buildFlowNew(NwConstants.ARP_RESPONDER_TABLE, flowId, NwConstants.DEFAULT_ARP_FLOW_PRIORITY,
887 ArpResponderUtil.generateCookie(lportTag, ipAddress),
888 ArpResponderUtil.getMatchCriteria(lportTag, elanInstance, ipAddress),
889 arpResponderInputInstructionsMap);
890 LoggingFutures.addErrorLogging(txRunner.callWithNewWriteOnlyTransactionAndSubmit(CONFIGURATION,
891 tx -> mdsalManager.addFlow(tx, dpnId, flowEntity)), LOG, "Error adding flow {}", flowEntity);
892 LOG.info("Installed the ARP Responder flow for Interface {}", ingressInterfaceName);
896 public void addExternalTunnelArpResponderFlow(ArpResponderInput arpResponderInput, String elanInstanceName) {
897 Uint64 dpnId = Uint64.valueOf(arpResponderInput.getDpId());
898 String ipAddress = arpResponderInput.getSpa();
899 String macAddress = arpResponderInput.getSha();
901 LOG.trace("Installing the ExternalTunnel ARP responder flow on DPN {} for ElanInstance {} with MAC {} & IP {}",
902 dpnId, elanInstanceName, macAddress, ipAddress);
904 ElanInstance elanInstance = elanInstanceCache.get(elanInstanceName).orElse(null);
905 if (elanInstance == null) {
906 LOG.warn("Null elan instance {}", elanInstanceName);
910 int lportTag = arpResponderInput.getLportTag();
911 String flowId = ArpResponderUtil.getFlowId(lportTag, ipAddress);
913 Map<org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.InstructionKey,
914 org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.Instruction>
915 arpResponderInputInstructionsMap = new HashMap<>();
916 int instructionKey = 0;
917 for (org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.Instruction
918 instructionObj : arpResponderInput.getInstructions()) {
919 arpResponderInputInstructionsMap.put(new org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types
920 .rev131026.instruction.list.InstructionKey(++instructionKey), instructionObj);
924 MDSALUtil.buildFlowNew(NwConstants.ARP_RESPONDER_TABLE, flowId, NwConstants.DEFAULT_ARP_FLOW_PRIORITY,
926 ArpResponderUtil.generateCookie(lportTag, ipAddress),
927 ArpResponderUtil.getMatchCriteria(lportTag, elanInstance, ipAddress),
928 arpResponderInputInstructionsMap);
929 LoggingFutures.addErrorLogging(txRunner.callWithNewWriteOnlyTransactionAndSubmit(CONFIGURATION,
930 tx -> mdsalManager.addFlow(tx, dpnId, flowEntity)), LOG, "Error adding flow {}", flowEntity);
931 LOG.trace("Installed the ExternalTunnel ARP Responder flow for ElanInstance {}", elanInstanceName);
935 public void removeArpResponderFlow(ArpResponderInput arpResponderInput) {
936 elanUtils.removeArpResponderFlow(Uint64.valueOf(arpResponderInput.getDpId()),
937 arpResponderInput.getInterfaceName(),
938 arpResponderInput.getSpa(), arpResponderInput.getLportTag());
942 * Uses the IdManager to retrieve a brand new ElanTag.
946 * @return the integer
949 public Uint32 retrieveNewElanTag(String idKey) {
950 return ElanUtils.retrieveNewElanTag(idManager, idKey);
954 public InstanceIdentifier<DpnInterfaces> getElanDpnInterfaceOperationalDataPath(
955 String elanInstanceName, Uint64 dpnId) {
956 return ElanUtils.getElanDpnInterfaceOperationalDataPath(elanInstanceName, dpnId);
960 public DpnInterfaces getElanInterfaceInfoByElanDpn(String elanInstanceName, Uint64 dpId) {
961 return elanUtils.getElanInterfaceInfoByElanDpn(elanInstanceName, dpId);