/* * Copyright (c) 2015 Ericsson India Global Services Pvt Ltd. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.vpnservice; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.iana._if.type.rev140508.Tunnel; import java.math.BigInteger; import java.util.Collection; import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.ArrayList; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ExecutionException; import java.util.concurrent.Future; import com.google.common.base.Optional; import com.google.common.collect.ImmutableList; import com.google.common.util.concurrent.Futures; import com.google.common.util.concurrent.FutureCallback; import org.opendaylight.bgpmanager.api.IBgpManager; import org.opendaylight.controller.md.sal.binding.api.DataBroker; import org.opendaylight.controller.md.sal.binding.api.DataChangeListener; import org.opendaylight.fibmanager.api.IFibManager; import org.opendaylight.vpnservice.interfacemgr.interfaces.IInterfaceManager; import org.opendaylight.vpnservice.mdsalutil.FlowEntity; import org.opendaylight.vpnservice.mdsalutil.InstructionInfo; import org.opendaylight.vpnservice.mdsalutil.InstructionType; import org.opendaylight.vpnservice.mdsalutil.MDSALUtil; import org.opendaylight.vpnservice.mdsalutil.MatchFieldType; import org.opendaylight.vpnservice.mdsalutil.MatchInfo; import org.opendaylight.vpnservice.mdsalutil.MetaDataUtil; import org.opendaylight.vpnservice.mdsalutil.NwConstants; import org.opendaylight.vpnservice.mdsalutil.interfaces.IMdsalApiManager; import org.opendaylight.yangtools.concepts.ListenerRegistration; import org.opendaylight.yangtools.yang.binding.DataObject; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier.InstanceIdentifierBuilder; import org.opendaylight.yangtools.yang.common.RpcResult; import org.opendaylight.controller.md.sal.binding.api.ReadOnlyTransaction; import org.opendaylight.controller.md.sal.binding.api.WriteTransaction; import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType; import org.opendaylight.controller.md.sal.common.api.data.AsyncDataBroker.DataChangeScope; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.Interfaces; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.InterfacesState; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.InterfaceKey; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface.OperStatus; import org.opendaylight.yang.gen.v1.urn.opendaylight.l3vpn.rev130911.AdjacencyList; import org.opendaylight.yang.gen.v1.urn.opendaylight.l3vpn.rev130911.adjacency.list.Adjacency; import org.opendaylight.yang.gen.v1.urn.opendaylight.l3vpn.rev130911.adjacency.list.AdjacencyBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.l3vpn.rev130911.VpnInstance1; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.idmanager.rev150403.IdManagerService; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.idmanager.rev150403.AllocateIdOutput; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.idmanager.rev150403.ReleaseIdInput; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.idmanager.rev150403.ReleaseIdInputBuilder; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.VpnInterfaces; import org.opendaylight.yang.gen.v1.urn.opendaylight.l3vpn.rev130911.Adjacencies; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.VpnAfConfig; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.VpnInstances; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.vpn.instances.VpnInstance; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.vpn.instances.VpnInstanceKey; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.vpn.interfaces.VpnInterface; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.vpn.interfaces.VpnInterfaceKey; import org.opendaylight.yang.gen.v1.urn.huawei.params.xml.ns.yang.l3vpn.rev140815.vpn.interfaces.VpnInterfaceBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.l3vpn.rev130911.AdjacenciesBuilder; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class VpnInterfaceManager extends AbstractDataChangeListener implements AutoCloseable { private static final Logger LOG = LoggerFactory.getLogger(VpnInterfaceManager.class); private ListenerRegistration listenerRegistration, interfaceListenerRegistration; private final DataBroker broker; private final IBgpManager bgpManager; private IFibManager fibManager; private IMdsalApiManager mdsalManager; private IInterfaceManager interfaceManager; private IdManagerService idManager; private Map> vpnToDpnsDb; private Map> dpnToInterfaceDb; private InterfaceListener interfaceListener; private static final FutureCallback DEFAULT_CALLBACK = new FutureCallback() { public void onSuccess(Void result) { LOG.debug("Success in Datastore operation"); } public void onFailure(Throwable error) { LOG.error("Error in Datastore operation", error); }; }; /** * Responsible for listening to data change related to VPN Interface * Bind VPN Service on the interface and informs the BGP service * * @param db - dataBroker service reference */ public VpnInterfaceManager(final DataBroker db, final IBgpManager bgpManager) { super(VpnInterface.class); broker = db; this.bgpManager = bgpManager; vpnToDpnsDb = new ConcurrentHashMap<>(); dpnToInterfaceDb = new ConcurrentHashMap<>(); interfaceListener = new InterfaceListener(); registerListener(db); } public void setMdsalManager(IMdsalApiManager mdsalManager) { this.mdsalManager = mdsalManager; } public void setInterfaceManager(IInterfaceManager interfaceManager) { this.interfaceManager = interfaceManager; } public void setFibManager(IFibManager fibManager) { this.fibManager = fibManager; } public void setIdManager(IdManagerService idManager) { this.idManager = idManager; } @Override public void close() throws Exception { if (listenerRegistration != null) { try { listenerRegistration.close(); interfaceListenerRegistration.close(); } catch (final Exception e) { LOG.error("Error when cleaning up DataChangeListener.", e); } listenerRegistration = null; interfaceListenerRegistration = null; } LOG.info("VPN Interface Manager Closed"); } private void registerListener(final DataBroker db) { try { listenerRegistration = db.registerDataChangeListener(LogicalDatastoreType.CONFIGURATION, getWildCardPath(), VpnInterfaceManager.this, DataChangeScope.SUBTREE); interfaceListenerRegistration = db.registerDataChangeListener(LogicalDatastoreType.OPERATIONAL, getInterfaceListenerPath(), interfaceListener, DataChangeScope.SUBTREE); } catch (final Exception e) { LOG.error("VPN Service DataChange listener registration fail!", e); throw new IllegalStateException("VPN Service registration Listener failed.", e); } } private InstanceIdentifier getInterfaceListenerPath() { return InstanceIdentifier.create(InterfacesState.class) .child(org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface.class); } @Override protected void add(final InstanceIdentifier identifier, final VpnInterface vpnInterface) { LOG.trace("key: {} , value: {}", identifier, vpnInterface ); addInterface(identifier, vpnInterface); } private void addInterface(final InstanceIdentifier identifier, final VpnInterface vpnInterface) { final VpnInterfaceKey key = identifier.firstKeyOf(VpnInterface.class, VpnInterfaceKey.class); String interfaceName = key.getName(); InstanceIdentifierBuilder idBuilder = InstanceIdentifier.builder(Interfaces.class).child(Interface.class, new InterfaceKey(interfaceName)); InstanceIdentifier id = idBuilder.build(); Optional port = read(LogicalDatastoreType.CONFIGURATION, id); if (port.isPresent()) { Interface interf = port.get(); bindServiceOnInterface(interf, vpnInterface.getVpnInstanceName()); updateNextHops(identifier, vpnInterface); } } private void updateNextHops(final InstanceIdentifier identifier, VpnInterface intf) { //Read NextHops InstanceIdentifier path = identifier.augmentation(Adjacencies.class); Optional adjacencies = read(LogicalDatastoreType.CONFIGURATION, path); String intfName = intf.getName(); if (adjacencies.isPresent()) { List nextHops = adjacencies.get().getAdjacency(); List value = new ArrayList<>(); //Get the rd of the vpn instance String rd = getRouteDistinguisher(intf.getVpnInstanceName()); BigInteger dpnId = interfaceManager.getDpnForInterface(intfName); String nextHopIp = interfaceManager.getEndpointIpForDpn(dpnId); LOG.trace("NextHops are {}", nextHops); for (Adjacency nextHop : nextHops) { String key = rd + VpnConstants.SEPARATOR + nextHop.getIpAddress(); String prefix = VpnUtil.getIpPrefix(nextHop.getIpAddress()); // long label = getUniqueId(idManager, VpnConstants.VPN_IDPOOL_NAME, VpnUtil // .getNextHopLabelKey((rd == null) ? intf.getVpnInstanceName() : rd, prefix)); long label = VpnUtil.getUniqueId(idManager, VpnConstants.VPN_IDPOOL_NAME, VpnUtil.getNextHopLabelKey(rd, nextHop.getIpAddress())); value.add(new AdjacencyBuilder(nextHop).setLabel(label).build()); } Adjacencies aug = VpnUtil.getVpnInterfaceAugmentation(value); VpnInterface opInterface = VpnUtil.getVpnInterface(intfName, intf.getVpnInstanceName(), aug); InstanceIdentifier interfaceId = VpnUtil.getVpnInterfaceIdentifier(intfName); syncWrite(LogicalDatastoreType.OPERATIONAL, interfaceId, opInterface, DEFAULT_CALLBACK); for (Adjacency nextHop : nextHops) { String key = rd + VpnConstants.SEPARATOR + nextHop.getIpAddress(); String prefix = VpnUtil.getIpPrefix(nextHop.getIpAddress()); // long label = getUniqueId(idManager, VpnConstants.VPN_IDPOOL_NAME, VpnUtil // .getNextHopLabelKey((rd == null) ? intf.getVpnInstanceName() : rd, prefix)); long label = VpnUtil.getUniqueId(idManager, VpnConstants.VPN_IDPOOL_NAME, VpnUtil.getNextHopLabelKey(rd, nextHop.getIpAddress())); updatePrefixToBGP(rd, nextHop, nextHopIp, label); } } } private long getVpnId(String vpnName) { //TODO: This should be a Util function InstanceIdentifier id = InstanceIdentifier.builder(VpnInstances.class) .child(VpnInstance.class, new VpnInstanceKey(vpnName)).augmentation(VpnInstance1.class).build(); Optional vpnInstance = read(LogicalDatastoreType.OPERATIONAL, id); long vpnId = VpnConstants.INVALID_ID; if(vpnInstance.isPresent()) { vpnId = vpnInstance.get().getVpnId(); } return vpnId; } private String getRouteDistinguisher(String vpnName) { InstanceIdentifier id = InstanceIdentifier.builder(VpnInstances.class) .child(VpnInstance.class, new VpnInstanceKey(vpnName)).build(); Optional vpnInstance = read(LogicalDatastoreType.CONFIGURATION, id); String rd = ""; if(vpnInstance.isPresent()) { VpnInstance instance = vpnInstance.get(); VpnAfConfig config = instance.getIpv4Family(); rd = config.getRouteDistinguisher(); } return rd; } private synchronized void updateMappingDbs(long vpnId, BigInteger dpnId, String intfName, String rd) { Collection dpnIds = vpnToDpnsDb.get(vpnId); if(dpnIds == null) { dpnIds = new HashSet<>(); } if(dpnIds.add(dpnId)) { vpnToDpnsDb.put(vpnId, dpnIds); fibManager.populateFibOnNewDpn(dpnId, vpnId, rd); } Collection intfNames = dpnToInterfaceDb.get(dpnId); if(intfNames == null) { intfNames = new ArrayList<>(); } intfNames.add(intfName); dpnToInterfaceDb.put(dpnId, intfNames); } private synchronized void remoteFromMappingDbs(long vpnId, BigInteger dpnId, String inftName, String rd) { Collection intfNames = dpnToInterfaceDb.get(dpnId); if(intfNames == null) { return; } intfNames.remove(inftName); dpnToInterfaceDb.put(dpnId, intfNames); //TODO: Delay 'DPN' removal so that other services can cleanup the entries for this dpn if(intfNames.isEmpty()) { Collection dpnIds = vpnToDpnsDb.get(vpnId); if(dpnIds == null) { return; } dpnIds.remove(dpnId); vpnToDpnsDb.put(vpnId, dpnIds); fibManager.cleanUpDpnForVpn(dpnId, vpnId, rd); } } private void bindServiceOnInterface(Interface intf, String vpnName) { LOG.trace("Bind service on interface {} for VPN: {}", intf, vpnName); long vpnId = getVpnId(vpnName); BigInteger dpId = interfaceManager.getDpnForInterface(intf.getName()); if(dpId.equals(BigInteger.ZERO)) { LOG.warn("DPN for interface {} not found. Bind service on this interface aborted.", intf.getName()); return; } else { String rd = getRouteDistinguisher(vpnName); updateMappingDbs(vpnId, dpId, intf.getName(), rd); } long portNo = interfaceManager.getPortForInterface(intf.getName()); String flowRef = getVpnInterfaceFlowRef(dpId, VpnConstants.LPORT_INGRESS_TABLE, vpnId, portNo); String flowName = intf.getName(); BigInteger COOKIE_VM_INGRESS_TABLE = new BigInteger("8000001", 16); int priority = VpnConstants.DEFAULT_FLOW_PRIORITY; short gotoTableId = VpnConstants.FIB_TABLE; if(intf.getType().equals(Tunnel.class)){ gotoTableId = VpnConstants.LFIB_TABLE; } List mkInstructions = new ArrayList(); mkInstructions.add(new InstructionInfo(InstructionType.write_metadata, new BigInteger[] { BigInteger.valueOf(vpnId), MetaDataUtil.METADATA_MASK_VRFID })); mkInstructions.add(new InstructionInfo(InstructionType.goto_table, new long[] { gotoTableId })); List matches = new ArrayList(); matches.add(new MatchInfo(MatchFieldType.in_port, new BigInteger[] { dpId, BigInteger.valueOf(portNo) })); FlowEntity flowEntity = MDSALUtil.buildFlowEntity(dpId, VpnConstants.LPORT_INGRESS_TABLE, flowRef, priority, flowName, 0, 0, COOKIE_VM_INGRESS_TABLE, matches, mkInstructions); mdsalManager.installFlow(flowEntity); } private String getVpnInterfaceFlowRef(BigInteger dpId, short tableId, long vpnId, long portNo) { return new StringBuilder().append(dpId).append(tableId).append(vpnId).append(portNo).toString(); } private void updatePrefixToBGP(String rd, Adjacency nextHop, String nextHopIp, long label) { try { bgpManager.addPrefix(rd, nextHop.getIpAddress(), nextHopIp, (int)label); } catch(Exception e) { LOG.error("Add prefix failed", e); } } private Optional read(LogicalDatastoreType datastoreType, InstanceIdentifier path) { ReadOnlyTransaction tx = broker.newReadOnlyTransaction(); Optional result = Optional.absent(); try { result = tx.read(datastoreType, path).get(); } catch (Exception e) { throw new RuntimeException(e); } return result; } private InstanceIdentifier getWildCardPath() { return InstanceIdentifier.create(VpnInterfaces.class).child(VpnInterface.class); } @Override protected void remove( InstanceIdentifier identifier, VpnInterface vpnInterface) { LOG.trace("Remove event - key: {}, value: {}" ,identifier, vpnInterface ); final VpnInterfaceKey key = identifier.firstKeyOf(VpnInterface.class, VpnInterfaceKey.class); String interfaceName = key.getName(); InstanceIdentifierBuilder idBuilder = InstanceIdentifier.builder(Interfaces.class).child(Interface.class, new InterfaceKey(interfaceName)); InstanceIdentifier id = idBuilder.build(); Optional port = read(LogicalDatastoreType.CONFIGURATION, id); if (port.isPresent()) { Interface interf = port.get(); removeNextHops(identifier, vpnInterface); unbindServiceOnInterface(interf, vpnInterface.getVpnInstanceName()); //InstanceIdentifier interfaceId = VpnUtil.getVpnInterfaceIdentifier(interfaceName); delete(LogicalDatastoreType.OPERATIONAL, identifier); } else { LOG.warn("No nexthops were available to handle remove event {}", interfaceName); } } private void removeNextHops(final InstanceIdentifier identifier, VpnInterface intf) { //Read NextHops InstanceIdentifier path = identifier.augmentation(Adjacencies.class); Optional adjacencies = read(LogicalDatastoreType.OPERATIONAL, path); String intfName = intf.getName(); String rd = getRouteDistinguisher(intf.getVpnInstanceName()); if (adjacencies.isPresent()) { List nextHops = adjacencies.get().getAdjacency(); if (!nextHops.isEmpty()) { LOG.trace("NextHops are " + nextHops); for (Adjacency nextHop : nextHops) { removePrefixFromBGP(rd, nextHop); } } } } private void delete(LogicalDatastoreType datastoreType, InstanceIdentifier path) { WriteTransaction tx = broker.newWriteOnlyTransaction(); tx.delete(datastoreType, path); Futures.addCallback(tx.submit(), DEFAULT_CALLBACK); } private void unbindServiceOnInterface(Interface intf, String vpnName) { LOG.trace("Unbind service on interface {} for VPN: {}", intf, vpnName); long vpnId = getVpnId(vpnName); BigInteger dpId = interfaceManager.getDpnForInterface(intf); if(dpId.equals(BigInteger.ZERO)) { LOG.warn("DPN for interface {} not found. Unbind service on this interface aborted.", intf.getName()); return; } else { String rd = getRouteDistinguisher(vpnName); remoteFromMappingDbs(vpnId, dpId, intf.getName(), rd); LOG.debug("removed vpn mapping for interface {} from VPN RD {}", intf.getName(), rd); } long portNo = interfaceManager.getPortForInterface(intf); String flowRef = getVpnInterfaceFlowRef(dpId, VpnConstants.LPORT_INGRESS_TABLE, vpnId, portNo); String flowName = intf.getName(); int priority = VpnConstants.DEFAULT_FLOW_PRIORITY; List matches = new ArrayList(); matches.add(new MatchInfo(MatchFieldType.in_port, new BigInteger[] { dpId, BigInteger.valueOf(portNo) })); FlowEntity flowEntity = MDSALUtil.buildFlowEntity(dpId, VpnConstants.LPORT_INGRESS_TABLE, flowRef, priority, flowName, 0, 0, null, matches, null); LOG.debug("Remove ingress flow for port {} in dpn {}", portNo, dpId.intValue()); mdsalManager.removeFlow(flowEntity); } private void removePrefixFromBGP(String rd, Adjacency nextHop) { try { bgpManager.deletePrefix(rd, nextHop.getIpAddress()); } catch(Exception e) { LOG.error("Delete prefix failed", e); } } @Override protected void update(InstanceIdentifier identifier, VpnInterface original, VpnInterface update) { LOG.trace("Update VPN Interface {} , original {}, update {}", identifier, original, update); String vpnName = original.getVpnInstanceName(); boolean vpnNameChanged = false; String rd = getRouteDistinguisher(vpnName); String newRd = rd; if(!vpnName.equals(update.getVpnInstanceName())) { //VPN for this interface got changed. //Remove the interface from old VPN and add it to new VPN String newVpnName = update.getVpnInstanceName(); newRd = getRouteDistinguisher(newVpnName); if(newRd.equals("")) { LOG.warn("VPN Instance {} not found. Update operation aborted", newVpnName); return; } vpnNameChanged = true; LOG.debug("New VPN Name for the interface {} is {}", newVpnName, original.getName()); } BigInteger dpnId = interfaceManager.getDpnForInterface(original.getName()); String nextHopIp = interfaceManager.getEndpointIpForDpn(dpnId); //List oldAdjs = original.getAugmentation(Adjacencies.class).getAdjacency(); List newAdjs = update.getAugmentation(Adjacencies.class).getAdjacency(); if(vpnNameChanged && newAdjs != null && !newAdjs.isEmpty()) { long label = VpnConstants.INVALID_ID; InstanceIdentifier path = identifier.augmentation(Adjacencies.class); Optional adjacencies = read(LogicalDatastoreType.OPERATIONAL, path); if (adjacencies.isPresent()) { List nextHops = adjacencies.get().getAdjacency(); for(Adjacency nextHop : nextHops) { label = nextHop.getLabel(); if(label == VpnConstants.INVALID_ID) { //Generate label using ID Manager String key = newRd + VpnConstants.SEPARATOR + nextHop.getIpAddress(); // label = getUniqueId(key); } removePrefixFromBGP(rd, nextHop); //updatePrefixToBGP(newRd, nextHop, nextHopIp, label); } updateNextHops(identifier, update); asyncUpdate(LogicalDatastoreType.OPERATIONAL, identifier, update, DEFAULT_CALLBACK); } } else { LOG.debug("No Update information is available for VPN Interface to proceed"); } } protected void asyncUpdate(LogicalDatastoreType datastoreType, InstanceIdentifier path, T data, FutureCallback callback) { WriteTransaction tx = broker.newWriteOnlyTransaction(); tx.merge(datastoreType, path, data, true); Futures.addCallback(tx.submit(), callback); } private void asyncWrite(LogicalDatastoreType datastoreType, InstanceIdentifier path, T data, FutureCallback callback) { WriteTransaction tx = broker.newWriteOnlyTransaction(); tx.put(datastoreType, path, data, true); Futures.addCallback(tx.submit(), callback); } private void syncWrite(LogicalDatastoreType datastoreType, InstanceIdentifier path, T data, FutureCallback callback) { WriteTransaction tx = broker.newWriteOnlyTransaction(); tx.put(datastoreType, path, data, true); tx.submit(); } synchronized Collection getDpnsForVpn(long vpnId) { Collection dpnIds = vpnToDpnsDb.get(vpnId); if(dpnIds != null) { return ImmutableList.copyOf(dpnIds); } else { return Collections.emptyList(); } } VpnInterface getVpnInterface(String interfaceName) { Optional optVpnInterfaces = read(LogicalDatastoreType.CONFIGURATION, VpnUtil.getVpnInterfacesIdentifier()); if(optVpnInterfaces.isPresent()) { List interfaces = optVpnInterfaces.get().getVpnInterface(); for(VpnInterface intf : interfaces) { if(intf.getName().equals(interfaceName)) { return intf; } } } return null; } private Interface getInterface(String interfaceName) { Optional optInterface = read(LogicalDatastoreType.CONFIGURATION, VpnUtil.getInterfaceIdentifier(interfaceName)); if(optInterface.isPresent()) { return optInterface.get(); } return null; } private String getTunnelInterfaceFlowRef(BigInteger dpnId, short tableId, String ifName) { return new StringBuilder().append(dpnId).append(tableId).append(ifName).toString(); } protected void makeTunnelIngressFlow(BigInteger dpnId, String ifName, int addOrRemoveFlow) { /* long portNo = 0; String flowName = ifName; String flowRef = getTunnelInterfaceFlowRef(dpnId, VpnConstants.LPORT_INGRESS_TABLE, ifName); List matches = new ArrayList(); List mkInstructions = new ArrayList(); if (NwConstants.ADD_FLOW == addOrRemoveFlow) { portNo = interfaceManager.getPortForInterface(ifName); matches.add(new MatchInfo(MatchFieldType.in_port, new BigInteger[] { dpnId, BigInteger.valueOf(portNo) })); mkInstructions.add(new InstructionInfo(InstructionType.goto_table, new long[] {VpnConstants.LFIB_TABLE})); } BigInteger COOKIE_VM_INGRESS_TABLE = new BigInteger("8000001", 16); FlowEntity flowEntity = MDSALUtil.buildFlowEntity(dpnId, VpnConstants.LPORT_INGRESS_TABLE, flowRef, VpnConstants.DEFAULT_FLOW_PRIORITY, flowName, 0, 0, COOKIE_VM_INGRESS_TABLE, matches, mkInstructions); if (NwConstants.ADD_FLOW == addOrRemoveFlow) { mdsalManager.installFlow(flowEntity); } else { mdsalManager.removeFlow(flowEntity); }*/ } private class InterfaceListener extends AbstractDataChangeListener { public InterfaceListener() { super(org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface.class); } @Override protected void remove(InstanceIdentifier identifier, org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface del) { LOG.trace("Operational Interface remove event - {}", del); } @Override protected void update(InstanceIdentifier identifier, org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface original, org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface update) { LOG.trace("Operation Interface update event - Old: {}, New: {}", original, update); String interfaceName = update.getName(); Interface intf = getInterface(interfaceName); if (intf != null && intf.getType().equals(Tunnel.class)) { BigInteger dpnId = interfaceManager.getDpnForInterface(interfaceName); if(update.getOperStatus().equals(OperStatus.Up)) { //Create ingress to LFIB LOG.debug("Installing Ingress for tunnel interface {}", interfaceName); makeTunnelIngressFlow(dpnId, interfaceName, NwConstants.ADD_FLOW); } else if(update.getOperStatus().equals(OperStatus.Down)) { LOG.debug("Removing Ingress flow for tunnel interface {}", interfaceName); makeTunnelIngressFlow(dpnId, interfaceName, NwConstants.DEL_FLOW); } } else { VpnInterface vpnInterface = getVpnInterface(interfaceName); if(vpnInterface != null) { if(update.getOperStatus().equals(OperStatus.Up)) { LOG.debug("Installing VPN related rules for interface {}", interfaceName); addInterface(VpnUtil.getVpnInterfaceIdentifier(vpnInterface.getName()), vpnInterface); } else if(update.getOperStatus().equals(OperStatus.Down)) { LOG.debug("Removing VPN related rules for interface {}", interfaceName); VpnInterfaceManager.this.remove(VpnUtil.getVpnInterfaceIdentifier(vpnInterface.getName()), vpnInterface); } } else { LOG.debug("No VPN Interface associated with interface {} to handle Update Operation", interfaceName); } } } @Override protected void add(InstanceIdentifier identifier, org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface add) { LOG.trace("Operational Interface add event - {}", add); String interfaceName = add.getName(); Interface intf = getInterface(interfaceName); if (intf != null && intf.getType().equals(Tunnel.class)) { BigInteger dpnId = interfaceManager.getDpnForInterface(interfaceName); if(add.getOperStatus().equals(OperStatus.Up)) { //Create ingress to LFIB LOG.debug("Installing Ingress for tunnel interface {}", interfaceName); makeTunnelIngressFlow(dpnId, interfaceName, NwConstants.ADD_FLOW); } } else { VpnInterface vpnInterface = getVpnInterface(interfaceName); if(vpnInterface != null) { if(add.getOperStatus().equals(OperStatus.Up)) { LOG.debug("Installing VPN related rules for interface {}", interfaceName); addInterface(VpnUtil.getVpnInterfaceIdentifier(vpnInterface.getName()), vpnInterface); } } else { LOG.debug("No VPN Interface associated with interface {} to handle add Operation", interfaceName); } } } } }