MRI version bumpup for Aluminium
[netvirt.git] / elanmanager / impl / src / main / java / org / opendaylight / netvirt / elan / internal / ElanServiceProvider.java
1 /*
2  * Copyright (c) 2016, 2017 Ericsson India Global Services Pvt Ltd. and others.  All rights reserved.
3  *
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
7  */
8
9 package org.opendaylight.netvirt.elan.internal;
10
11 import static java.util.Collections.emptyMap;
12
13 import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
14 import java.util.ArrayList;
15 import java.util.Collection;
16 import java.util.Collections;
17 import java.util.HashMap;
18 import java.util.HashSet;
19 import java.util.List;
20 import java.util.Map;
21 import java.util.Objects;
22 import java.util.Optional;
23 import java.util.Set;
24 import java.util.concurrent.Future;
25 import java.util.function.BiFunction;
26 import javax.inject.Inject;
27 import javax.inject.Singleton;
28 import org.eclipse.jdt.annotation.NonNull;
29 import org.eclipse.jdt.annotation.Nullable;
30 import org.opendaylight.genius.infra.Datastore;
31 import org.opendaylight.genius.infra.ManagedNewTransactionRunner;
32 import org.opendaylight.genius.infra.ManagedNewTransactionRunnerImpl;
33 import org.opendaylight.genius.interfacemanager.exceptions.InterfaceAlreadyExistsException;
34 import org.opendaylight.genius.interfacemanager.globals.IfmConstants;
35 import org.opendaylight.genius.interfacemanager.interfaces.IInterfaceManager;
36 import org.opendaylight.genius.mdsalutil.MDSALUtil;
37 import org.opendaylight.genius.mdsalutil.NwConstants;
38 import org.opendaylight.genius.mdsalutil.interfaces.IMdsalApiManager;
39 import org.opendaylight.genius.utils.ServiceIndex;
40 import org.opendaylight.genius.utils.hwvtep.HwvtepSouthboundConstants;
41 import org.opendaylight.infrautils.inject.AbstractLifecycle;
42 import org.opendaylight.infrautils.utils.concurrent.LoggingFutures;
43 import org.opendaylight.mdsal.binding.api.DataBroker;
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;
96
97 @Singleton
98 public class ElanServiceProvider extends AbstractLifecycle implements IElanService {
99
100     private static final Logger LOG = LoggerFactory.getLogger(ElanServiceProvider.class);
101
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;
113
114     private final EntityOwnershipCandidateRegistration candidateRegistration;
115
116     @Inject
117     public ElanServiceProvider(IdManagerService idManager, IInterfaceManager interfaceManager,
118                                ElanBridgeManager bridgeMgr,
119                                DataBroker dataBroker,
120                                ElanUtils elanUtils,
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;
134
135         candidateRegistration = registerCandidate(entityOwnershipService);
136     }
137
138     @Nullable
139     private static EntityOwnershipCandidateRegistration registerCandidate(
140             EntityOwnershipService entityOwnershipService) {
141         try {
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");
146             return null;
147         }
148     }
149
150     @Override
151     @SuppressWarnings("checkstyle:IllegalCatch")
152     protected void start() throws Exception {
153         LOG.info("Starting ElanServiceProvider");
154         setIsL2BeforeL3();
155         createIdPool();
156     }
157
158     @Override
159     protected void stop() {
160         if (candidateRegistration != null) {
161             candidateRegistration.close();
162         }
163
164         LOG.info("ElanServiceProvider stopped");
165     }
166
167     @Override
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();
172     }
173
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");
180         }
181     }
182
183     @Override
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);
190                 return true;
191             } else {
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);
199             }
200         } else {
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);
207         }
208         return isSuccess;
209     }
210
211     @Override
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);
218                 return true;
219             } else {
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.class, 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);
229             }
230         } else {
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.class, etreeInstance).build();
236             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
237                     ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName), elanInstance);
238             LOG.debug("Creating the new Etree Instance {}", elanInstance);
239         }
240         return isSuccess;
241     }
242
243     @Override
244     @Nullable
245     public EtreeInterface getEtreeInterfaceByElanInterfaceName(String elanInterface) {
246         return elanInterfaceCache.getEtreeInterface(elanInterface).orElse(null);
247     }
248
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)) {
254             isEqual = true;
255         }
256         return isEqual;
257     }
258
259     @Override
260     public void updateElanInstance(String elanInstanceName, long newMacTimout, String newDescription) {
261         createElanInstance(elanInstanceName, newMacTimout, newDescription);
262     }
263
264     @Override
265     public boolean deleteEtreeInstance(String etreeInstanceName) {
266         return deleteElanInstance(etreeInstanceName);
267     }
268
269     @Override
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);
274             return false;
275         }
276         LOG.debug("Deletion of the existing Elan Instance {}", existingElanInstance);
277         ElanUtils.delete(broker, LogicalDatastoreType.CONFIGURATION,
278                 ElanHelper.getElanInstanceConfigurationDataPath(elanInstanceName));
279         return true;
280     }
281
282     @Override
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.class, etreeInterface).build();
294             } else {
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.class, etreeInterface).build();
301             }
302             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
303                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
304             LOG.debug("Creating the new Etree Interface {}", elanInterface);
305         }
306     }
307
308     @Override
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);
320             }
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);
325         }
326     }
327
328     @Override
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()) {
333             return;
334         }
335
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);
343     }
344
345     @Override
346     public void deleteEtreeInterface(String interfaceName) {
347         deleteElanInterface(interfaceName);
348         LOG.debug("deleting the Etree Interface {}", interfaceName);
349     }
350
351     @Override
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);
358         }
359     }
360
361     @Override
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);
371         }
372     }
373
374     @Override
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,
380                     macAddress);
381             MDSALUtil.syncDelete(broker, LogicalDatastoreType.CONFIGURATION, staticMacEntriesIdentifier);
382         }
383     }
384
385     @Override
386     public Collection<MacEntry> getElanMacTable(String elanInstanceName) {
387         Elan elanInfo = ElanUtils.getElanByName(broker, elanInstanceName);
388         List<MacEntry> macAddress = new ArrayList<>();
389         if (elanInfo == null) {
390             return macAddress;
391         }
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.getMacEntry().values());
399                 }
400             }
401         }
402         return macAddress;
403     }
404
405     @Override
406     public void flushMACTable(String elanInstanceName) {
407         Elan elanInfo = ElanUtils.getElanByName(broker, elanInstanceName);
408         if (elanInfo == null) {
409             return;
410         }
411         List<String> elanInterfaces = elanInfo.getElanInterfaces();
412         if (elanInterfaces == null || elanInterfaces.isEmpty()) {
413             return;
414         }
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.getMacEntry();
419                 for (MacEntry macEntry : macEntries.values()) {
420                     deleteStaticMacAddress(elanInterface, macEntry.getMacAddress().getValue());
421                 }
422             }
423         }
424
425     }
426
427     @Override
428     @Nullable
429     public ElanInstance getElanInstance(String elanName) {
430         return elanInstanceCache.get(elanName).orElse(null);
431     }
432
433     @Override
434     public List<ElanInstance> getElanInstances() {
435         InstanceIdentifier<ElanInstances> elanInstancesIdentifier = InstanceIdentifier.builder(ElanInstances.class)
436                 .build();
437         return new ArrayList<>(ElanUtils.read(broker, LogicalDatastoreType.CONFIGURATION, elanInstancesIdentifier).map(
438                 ElanInstances::getElanInstance).orElse(emptyMap()).values());
439     }
440
441     @Override
442     @NonNull
443     public List<String> getElanInterfaces(String elanInstanceName) {
444         List<String> elanInterfaces = new ArrayList<>();
445         InstanceIdentifier<ElanInterfaces> elanInterfacesIdentifier = InstanceIdentifier.builder(ElanInterfaces.class)
446                 .build();
447         Optional<ElanInterfaces> elanInterfacesOptional = ElanUtils.read(broker, LogicalDatastoreType.CONFIGURATION,
448                 elanInterfacesIdentifier);
449         if (!elanInterfacesOptional.isPresent()) {
450             return elanInterfaces;
451         }
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());
456             }
457         }
458         return elanInterfaces;
459     }
460
461     @Override
462     public void createExternalElanNetworks(Node node) {
463         handleExternalElanNetworks(node, true, (elanInstance, interfaceName) -> {
464             createExternalElanNetwork(elanInstance, interfaceName);
465             return null;
466         });
467     }
468
469     @Override
470     public void createExternalElanNetwork(ElanInstance elanInstance) {
471         handleExternalElanNetwork(elanInstance, false, (elanInstance1, interfaceName) -> {
472             createExternalElanNetwork(elanInstance1, interfaceName);
473             return null;
474         });
475     }
476
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(),
483                     dpId);
484             createExternalElanNetwork(elanInstance, providerIntfName);
485         }
486     }
487
488     private void createExternalElanNetwork(ElanInstance elanInstance, String interfaceName) {
489         if (interfaceName == null) {
490             LOG.trace("No physical interface is attached to {}", elanInstance.getPhysicalNetworkName());
491             return;
492         }
493
494         String elanInterfaceName = createIetfInterfaces(elanInstance, interfaceName);
495         addElanInterface(elanInstance.getElanInstanceName(), elanInterfaceName, null, null);
496     }
497
498     @Override
499     public void updateExternalElanNetwork(ElanInstance elanInstance) {
500         handleExternalElanNetwork(elanInstance, true, (elanInstance1, interfaceName) -> {
501             createExternalElanNetwork(elanInstance1, interfaceName);
502             return null;
503         });
504     }
505
506     @Override
507     public void deleteExternalElanNetworks(Node node) {
508         handleExternalElanNetworks(node, false, (elanInstance, interfaceName) -> {
509             deleteExternalElanNetwork(elanInstance, interfaceName);
510             return null;
511         });
512     }
513
514     @Override
515     public void deleteExternalElanNetwork(ElanInstance elanInstance) {
516         handleExternalElanNetwork(elanInstance, false, (elanInstance1, interfaceName) -> {
517             deleteExternalElanNetwork(elanInstance1, interfaceName);
518             return null;
519         });
520     }
521
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);
531         } else {
532             LOG.debug("vlan prv intf {} not found in interfacemgr config DS", intfName);
533         }
534     }
535
536     private void deleteExternalElanNetwork(ElanInstance elanInstance, String interfaceName) {
537         if (interfaceName == null) {
538             LOG.trace("No physial interface is attached to {}", elanInstance.getPhysicalNetworkName());
539             return;
540         }
541
542         String elanInstanceName = elanInstance.getElanInstanceName();
543         for (String elanInterface : getExternalElanInterfaces(elanInstanceName)) {
544             if (elanInterface.startsWith(interfaceName)) {
545                 if (ElanUtils.isVlan(elanInstance)) {
546                     deleteIetfInterface(elanInterface);
547                 }
548                 String trunkInterfaceName = getTrunkInterfaceName(interfaceName);
549                 if (shouldDeleteTrunk(trunkInterfaceName, elanInterface)) {
550                     deleteIetfInterface(trunkInterfaceName);
551                 }
552                 deleteElanInterface(elanInterface);
553             }
554         }
555     }
556
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);
562             return true;
563         }
564
565         LOG.debug("Trunk interface {} has {} VLAN member interfaces left", trunkInterfaceName, childInterfaces.size());
566         return false;
567     }
568
569     @Override
570     public void updateExternalElanNetworks(Node origNode, Node updatedNode) {
571         if (!bridgeMgr.isIntegrationBridge(updatedNode)) {
572             return;
573         }
574
575         List<ElanInstance> elanInstances = getElanInstances();
576         if (elanInstances == null || elanInstances.isEmpty()) {
577             LOG.trace("No ELAN instances found");
578             return;
579         }
580
581         LOG.trace("updateExternalElanNetworks, orig bridge {} . updated bridge {}", origNode, updatedNode);
582
583         Map<String, String> origProviderMappping = getMapFromOtherConfig(origNode,
584                 ElanBridgeManager.PROVIDER_MAPPINGS_KEY);
585         Map<String, String> updatedProviderMappping = getMapFromOtherConfig(updatedNode,
586                 ElanBridgeManager.PROVIDER_MAPPINGS_KEY);
587
588         boolean hasDatapathIdOnOrigNode = bridgeMgr.hasDatapathID(origNode);
589         boolean hasDatapathIdOnUpdatedNode = bridgeMgr.hasDatapathID(updatedNode);
590         Uint64 origDpnID = bridgeMgr.getDatapathId(origNode);
591
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);
598                 /**
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
602                  */
603                 if (hasDatapathIdOnOrigNode && !elanInstance.isExternal()
604                         && ElanUtils.isVlan(elanInstance)) {
605                     String externalIntf = getExternalElanInterface(elanInstance.getElanInstanceName(),
606                             origDpnID);
607                     if (externalIntf == null) {
608                         createExternalElanNw = false;
609                     }
610                 }
611                 if (hasPortNameRemoved(origPortName, updatedPortName)) {
612                     deleteExternalElanNetwork(elanInstance,
613                             bridgeMgr.getProviderInterfaceName(origNode, physicalNetworkName));
614                 }
615
616                 if (createExternalElanNw && (hasPortNameUpdated(origPortName, updatedPortName)
617                         || hasDatapathIdAdded(hasDatapathIdOnOrigNode, hasDatapathIdOnUpdatedNode))) {
618                     createExternalElanNetwork(elanInstance,
619                             bridgeMgr.getProviderInterfaceName(updatedNode, physicalNetworkName));
620                 }
621             }
622         }
623     }
624
625     private static boolean hasDatapathIdAdded(boolean hasDatapathIdOnOrigNode, boolean hasDatapathIdOnUpdatedNode) {
626         return !hasDatapathIdOnOrigNode && hasDatapathIdOnUpdatedNode;
627     }
628
629     private static boolean hasPortNameUpdated(String origPortName, String updatedPortName) {
630         return updatedPortName != null && !updatedPortName.equals(origPortName);
631     }
632
633     private static boolean hasPortNameRemoved(String origPortName, String updatedPortName) {
634         return origPortName != null && !origPortName.equals(updatedPortName);
635     }
636
637     private Map<String, String> getMapFromOtherConfig(Node node, String otherConfigColumn) {
638         return bridgeMgr.getOpenvswitchOtherConfigMap(node, otherConfigColumn);
639     }
640
641     @Override
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();
647         }
648
649         Set<String> externalElanInterfaces = new HashSet<>();
650         for (String elanInterface : elanInterfaces) {
651             if (interfaceManager.isExternalInterface(elanInterface)) {
652                 externalElanInterfaces.add(elanInterface);
653             }
654         }
655
656         return externalElanInterfaces;
657     }
658
659     @Override
660     public String getExternalElanInterface(String elanInstanceName, Uint64 dpnId) {
661         return elanUtils.getExternalElanInterface(elanInstanceName, dpnId);
662     }
663
664     @Override
665     public boolean isExternalInterface(String interfaceName) {
666         return interfaceManager.isExternalInterface(interfaceName);
667     }
668
669     @Override
670     @Nullable
671     public ElanInterface getElanInterfaceByElanInterfaceName(String interfaceName) {
672         return elanInterfaceCache.get(interfaceName).orElse(null);
673     }
674
675     @Override
676     public void handleKnownL3DmacAddress(String macAddress, String elanInstanceName, int addOrRemove) {
677         if (addOrRemove == NwConstants.ADD_FLOW) {
678             addKnownL3DmacAddress(macAddress, elanInstanceName);
679         } else {
680             removeKnownL3DmacAddress(macAddress, elanInstanceName);
681         }
682     }
683
684     @Override
685     public void addKnownL3DmacAddress(String macAddress, String elanInstanceName) {
686         if (!isL2BeforeL3) {
687             LOG.trace("ELAN service is after L3VPN in the Netvirt pipeline skip known L3DMAC flows installation");
688             return;
689         }
690         ElanInstance elanInstance = elanInstanceCache.get(elanInstanceName).orElse(null);
691         if (elanInstance == null) {
692             LOG.warn("Null elan instance {}", elanInstanceName);
693             return;
694         }
695
696         List<Uint64> dpnsIdsForElanInstance = elanUtils.getParticipatingDpnsInElanInstance(elanInstanceName);
697         if (dpnsIdsForElanInstance.isEmpty()) {
698             LOG.warn("No DPNs for elan instance {}", elanInstance);
699             return;
700         }
701
702         elanUtils.addDmacRedirectToDispatcherFlows(elanInstance.getElanTag(), elanInstanceName, macAddress,
703                 dpnsIdsForElanInstance);
704     }
705
706     @Override
707     public void removeKnownL3DmacAddress(String macAddress, String elanInstanceName) {
708         if (!isL2BeforeL3) {
709             LOG.trace("ELAN service is after L3VPN in the Netvirt pipeline skip known L3DMAC flows installation");
710             return;
711         }
712         ElanInstance elanInstance = elanInstanceCache.get(elanInstanceName).orElse(null);
713         if (elanInstance == null) {
714             LOG.warn("Null elan instance {}", elanInstanceName);
715             return;
716         }
717
718         List<Uint64> dpnsIdsForElanInstance = elanUtils.getParticipatingDpnsInElanInstance(elanInstanceName);
719         if (dpnsIdsForElanInstance.isEmpty()) {
720             LOG.warn("No DPNs for elan instance {}", elanInstance);
721             return;
722         }
723
724         elanUtils.removeDmacRedirectToDispatcherFlows(elanInstance.getElanTag(),
725             macAddress, dpnsIdsForElanInstance);
726     }
727
728     /**
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
734      * trunk interface.
735      *
736      * @param elanInstance
737      *            ELAN instance
738      * @param parentRef
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
743      */
744     private String createIetfInterfaces(ElanInstance elanInstance, String parentRef) {
745         String interfaceName = null;
746
747         try {
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);
754             }
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);
762             }
763         } catch (InterfaceAlreadyExistsException e) {
764             LOG.trace("Interface {} was already created", interfaceName);
765         }
766
767         return interfaceName;
768     }
769
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);
776     }
777
778     private void handleExternalElanNetworks(Node node, boolean skipIntVlanNw,
779                                             BiFunction<ElanInstance, String, Void> function) {
780         if (!bridgeMgr.isIntegrationBridge(node)) {
781             return;
782         }
783
784         List<ElanInstance> elanInstances = getElanInstances();
785         if (elanInstances == null || elanInstances.isEmpty()) {
786             LOG.trace("No ELAN instances found");
787             return;
788         }
789
790         for (ElanInstance elanInstance : elanInstances) {
791             if (skipIntVlanNw && !elanInstance.isExternal() && ElanUtils.isVlan(elanInstance)) {
792                 continue;
793             }
794             String interfaceName = bridgeMgr.getProviderInterfaceName(node, elanInstance.getPhysicalNetworkName());
795             if (interfaceName != null) {
796                 function.apply(elanInstance, interfaceName);
797             }
798         }
799     }
800
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());
809             return;
810         }
811         if (elanInstance.getPhysicalNetworkName() == null) {
812             LOG.trace("No physical network attached to {}", elanInstanceName);
813             return;
814         }
815
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());
820             return;
821         }
822
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()) {
829                         continue;
830                     }
831                 }
832                 String interfaceName = bridgeMgr.getProviderInterfaceName(node, elanInstance.getPhysicalNetworkName());
833                 function.apply(elanInstance, interfaceName);
834             }
835         }
836     }
837
838     private static String getTrunkInterfaceName(String parentRef) {
839         return parentRef + IfmConstants.OF_URI_SEPARATOR + "trunk";
840     }
841
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");
849             isL2BeforeL3 = true;
850         } else {
851             LOG.info("ELAN service is set after L3VPN service in the Netvirt pipeline");
852             isL2BeforeL3 = false;
853         }
854     }
855
856     @Override
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());
863
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);
872             return;
873         }
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);
883         }
884         Flow flowEntity =
885             MDSALUtil.buildFlowNew(NwConstants.ARP_RESPONDER_TABLE, flowId, NwConstants.DEFAULT_ARP_FLOW_PRIORITY,
886                 flowId, 0, 0,
887                 ArpResponderUtil.generateCookie(lportTag, ipAddress),
888                 ArpResponderUtil.getMatchCriteria(lportTag, elanInstance, ipAddress),
889                     arpResponderInputInstructionsMap);
890         LoggingFutures.addErrorLogging(txRunner.callWithNewWriteOnlyTransactionAndSubmit(Datastore.CONFIGURATION,
891             tx -> mdsalManager.addFlow(tx, dpnId, flowEntity)), LOG, "Error adding flow {}", flowEntity);
892         LOG.info("Installed the ARP Responder flow for Interface {}", ingressInterfaceName);
893     }
894
895     @Override
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();
900
901         LOG.trace("Installing the ExternalTunnel ARP responder flow on DPN {} for ElanInstance {} with MAC {} & IP {}",
902                 dpnId, elanInstanceName, macAddress, ipAddress);
903
904         ElanInstance elanInstance = elanInstanceCache.get(elanInstanceName).orElse(null);
905         if (elanInstance == null) {
906             LOG.warn("Null elan instance {}", elanInstanceName);
907             return;
908         }
909
910         int lportTag = arpResponderInput.getLportTag();
911         String flowId = ArpResponderUtil.getFlowId(lportTag, ipAddress);
912
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);
921         }
922
923         Flow flowEntity =
924             MDSALUtil.buildFlowNew(NwConstants.ARP_RESPONDER_TABLE, flowId, NwConstants.DEFAULT_ARP_FLOW_PRIORITY,
925                 flowId, 0, 0,
926                 ArpResponderUtil.generateCookie(lportTag, ipAddress),
927                 ArpResponderUtil.getMatchCriteria(lportTag, elanInstance, ipAddress),
928                     arpResponderInputInstructionsMap);
929         LoggingFutures.addErrorLogging(txRunner.callWithNewWriteOnlyTransactionAndSubmit(Datastore.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);
932     }
933
934     @Override
935     public void removeArpResponderFlow(ArpResponderInput arpResponderInput) {
936         elanUtils.removeArpResponderFlow(Uint64.valueOf(arpResponderInput.getDpId()),
937                 arpResponderInput.getInterfaceName(),
938                 arpResponderInput.getSpa(), arpResponderInput.getLportTag());
939     }
940
941     /**
942      * Uses the IdManager to retrieve a brand new ElanTag.
943      *
944      * @param idKey
945      *            the id key
946      * @return the integer
947      */
948     @Override
949     public Uint32 retrieveNewElanTag(String idKey) {
950         return ElanUtils.retrieveNewElanTag(idManager, idKey);
951     }
952
953     @Override
954     public InstanceIdentifier<DpnInterfaces> getElanDpnInterfaceOperationalDataPath(
955                                                                 String elanInstanceName, Uint64 dpnId) {
956         return ElanUtils.getElanDpnInterfaceOperationalDataPath(elanInstanceName, dpnId);
957     }
958
959     @Override
960     public DpnInterfaces getElanInterfaceInfoByElanDpn(String elanInstanceName, Uint64 dpId) {
961         return elanUtils.getElanInterfaceInfoByElanDpn(elanInstanceName, dpId);
962     }
963
964 }