Bump upstreams for Silicon
[netvirt.git] / vpnmanager / impl / src / main / java / org / opendaylight / netvirt / vpnmanager / populator / impl / L3vpnPopulator.java
1 /*
2  * Copyright (c) 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 package org.opendaylight.netvirt.vpnmanager.populator.impl;
9
10 import static java.util.Objects.requireNonNull;
11 import static org.opendaylight.infrautils.utils.concurrent.LoggingFutures.addErrorLogging;
12 import static org.opendaylight.mdsal.binding.util.Datastore.OPERATIONAL;
13
14 import java.util.Collections;
15 import java.util.List;
16 import java.util.Optional;
17 import java.util.concurrent.ExecutionException;
18 import java.util.concurrent.locks.ReentrantLock;
19 import org.eclipse.jdt.annotation.Nullable;
20 import org.opendaylight.genius.datastoreutils.SingleTransactionDataBroker;
21 import org.opendaylight.genius.utils.JvmGlobalLocks;
22 import org.opendaylight.mdsal.binding.api.DataBroker;
23 import org.opendaylight.mdsal.binding.util.Datastore.Configuration;
24 import org.opendaylight.mdsal.binding.util.ManagedNewTransactionRunner;
25 import org.opendaylight.mdsal.binding.util.ManagedNewTransactionRunnerImpl;
26 import org.opendaylight.mdsal.binding.util.TypedWriteTransaction;
27 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
28 import org.opendaylight.netvirt.bgpmanager.api.IBgpManager;
29 import org.opendaylight.netvirt.fibmanager.api.FibHelper;
30 import org.opendaylight.netvirt.fibmanager.api.IFibManager;
31 import org.opendaylight.netvirt.fibmanager.api.RouteOrigin;
32 import org.opendaylight.netvirt.vpnmanager.VpnUtil;
33 import org.opendaylight.netvirt.vpnmanager.populator.input.L3vpnInput;
34 import org.opendaylight.netvirt.vpnmanager.populator.intfc.VpnPopulator;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.FibEntries;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.LabelRouteMap;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.SubnetRoute;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.SubnetRouteBuilder;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.VrfEntryBase;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.fibentries.VrfTables;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.fibentries.VrfTablesBuilder;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.fibentries.VrfTablesKey;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.label.route.map.LabelRouteInfo;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.label.route.map.LabelRouteInfoBuilder;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.label.route.map.LabelRouteInfoKey;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.vrfentries.VrfEntry;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.vrfentries.VrfEntryKey;
48 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.instance.op.data.VpnInstanceOpDataEntry;
49 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.neutronvpn.l3vpn.rev200204.adjacency.list.Adjacency;
50 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.neutronvpn.l3vpn.rev200204.adjacency.list.AdjacencyBuilder;
51 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
52 import org.opendaylight.yangtools.yang.common.Uint32;
53 import org.opendaylight.yangtools.yang.common.Uint64;
54 import org.slf4j.Logger;
55 import org.slf4j.LoggerFactory;
56
57 public abstract class L3vpnPopulator implements VpnPopulator {
58     private static final Logger LOG = LoggerFactory.getLogger(L3vpnPopulator.class);
59
60     protected final IBgpManager bgpManager;
61     protected final IFibManager fibManager;
62     protected final DataBroker broker;
63     protected final ManagedNewTransactionRunner txRunner;
64     protected final VpnUtil vpnUtil;
65
66     protected L3vpnPopulator(DataBroker dataBroker, IBgpManager bgpManager, IFibManager fibManager,
67                              VpnUtil vpnUtil) {
68         this.bgpManager = bgpManager;
69         this.fibManager = fibManager;
70         this.broker = dataBroker;
71         this.txRunner = new ManagedNewTransactionRunnerImpl(dataBroker);
72         this.vpnUtil = vpnUtil;
73     }
74
75     @Override
76     public void populateFib(L3vpnInput input, TypedWriteTransaction<Configuration> writeCfgTxn) {
77
78     }
79
80     public void addSubnetRouteFibEntry(L3vpnInput input) {
81         String rd = input.getRd();
82         String vpnName = input.getVpnName();
83         String prefix = input.getSubnetIp();
84         String nextHop = input.getNextHopIp();
85         Uint32 label = Uint32.valueOf(input.getLabel());
86         Uint32 l3vni = Uint32.valueOf(input.getL3vni());
87         Uint32 elantag = Uint32.valueOf(input.getElanTag());
88         String networkName = input.getNetworkName();
89         String gwMacAddress = input.getGatewayMac();
90         SubnetRoute route = new SubnetRouteBuilder().setElantag(elantag).build();
91         RouteOrigin origin = RouteOrigin.CONNECTED; // Only case when a route is considered as directly connected
92         VrfEntry vrfEntry = FibHelper.getVrfEntryBuilder(prefix, label, nextHop, origin, networkName)
93                 .addAugmentation(route).setL3vni(l3vni).setGatewayMacAddress(gwMacAddress).build();
94         LOG.debug("Created vrfEntry for {} nexthop {} label {} and elantag {}", prefix, nextHop, label, elantag);
95         InstanceIdentifier<VrfEntry> vrfEntryId =
96                 InstanceIdentifier.builder(FibEntries.class)
97                         .child(VrfTables.class, new VrfTablesKey(rd))
98                         .child(VrfEntry.class, new VrfEntryKey(prefix)).build();
99         Optional<VrfEntry> entry = Optional.empty();
100         try {
101             entry = SingleTransactionDataBroker.syncReadOptional(broker,
102                     LogicalDatastoreType.CONFIGURATION, vrfEntryId);
103         } catch (ExecutionException | InterruptedException e) {
104             LOG.error("addSubnetRouteFibEntry: Exception while reading vrfEntry for the prefix {} rd {}", prefix,
105                     rd, e);
106         }
107         if (!entry.isPresent()) {
108             List<VrfEntry> vrfEntryList = Collections.singletonList(vrfEntry);
109
110             InstanceIdentifier.InstanceIdentifierBuilder<VrfTables> idBuilder =
111                     InstanceIdentifier.builder(FibEntries.class).child(VrfTables.class, new VrfTablesKey(rd));
112             InstanceIdentifier<VrfTables> vrfTableId = idBuilder.build();
113
114             VrfTables vrfTableNew = new VrfTablesBuilder().setRouteDistinguisher(rd).setVrfEntry(vrfEntryList).build();
115             vpnUtil.syncUpdate(LogicalDatastoreType.CONFIGURATION, vrfTableId, vrfTableNew);
116             LOG.info("SUBNETROUTE: addSubnetRouteFibEntryToDS: Added vrfEntry for {} nexthop {} label {} rd {}"
117                     + " vpnName {}", prefix, nextHop, label, rd, vpnName);
118         } else { // Found in MDSAL database
119             vpnUtil.syncWrite(LogicalDatastoreType.CONFIGURATION, vrfEntryId, vrfEntry);
120             LOG.info("SUBNETROUTE: addSubnetRouteFibEntryToDS: Updated vrfEntry for {} nexthop {} label {} rd {}"
121                     + " vpnName {}", prefix, nextHop, label, rd, vpnName);
122         }
123
124         //Will be handled appropriately with the iRT patch for EVPN
125         if (input.getEncapType().equals(VrfEntryBase.EncapType.Mplsgre)) {
126             List<VpnInstanceOpDataEntry> vpnsToImportRoute = vpnUtil.getVpnsImportingMyRoute(vpnName);
127             if (vpnsToImportRoute.size() > 0) {
128                 VrfEntry importingVrfEntry = FibHelper.getVrfEntryBuilder(prefix, label, nextHop,
129                         RouteOrigin.SELF_IMPORTED, rd).addAugmentation(route).build();
130                 List<VrfEntry> importingVrfEntryList = Collections.singletonList(importingVrfEntry);
131                 for (VpnInstanceOpDataEntry vpnInstance : vpnsToImportRoute) {
132                     String importingRd = vpnInstance.getVrfId();
133                     InstanceIdentifier<VrfTables> importingVrfTableId =
134                             InstanceIdentifier.builder(FibEntries.class).child(VrfTables.class,
135                                     new VrfTablesKey(importingRd)).build();
136                     VrfTables importingVrfTable = new VrfTablesBuilder().setRouteDistinguisher(importingRd)
137                             .setVrfEntry(importingVrfEntryList).build();
138                     vpnUtil.syncUpdate(LogicalDatastoreType.CONFIGURATION, importingVrfTableId, importingVrfTable);
139                     LOG.info("SUBNETROUTE: addSubnetRouteFibEntryToDS: Exported route rd {} prefix {} nexthop {}"
140                                     + " label {} to vpn {} importingRd {}", rd, prefix, nextHop, label,
141                             vpnInstance.getVpnInstanceName(), importingRd);
142                 }
143             }
144         }
145         LOG.info("SUBNETROUTE: addSubnetRouteFibEntryToDS: Created vrfEntry for {} nexthop {} label {} and elantag {}"
146                 + "rd {} vpnName {}", prefix, nextHop, label, elantag, rd, vpnName);
147     }
148
149     public void addToLabelMapper(Uint32 label, Uint64 dpnId, String prefix, List<String> nextHopIpList, Uint32 vpnId,
150                                  @Nullable String vpnInterfaceName, @Nullable Uint32 elanTag,
151                                  boolean isSubnetRoute, String rd) {
152         final String labelStr = requireNonNull(label, "addToLabelMapper: label cannot be null or empty!").toString();
153         requireNonNull(prefix, "addToLabelMapper: prefix cannot be null or empty!");
154         requireNonNull(vpnId, "addToLabelMapper: vpnId cannot be null or empty!");
155         requireNonNull(rd, "addToLabelMapper: rd cannot be null or empty!");
156         if (!isSubnetRoute) {
157             // NextHop must be present for non-subnetroute entries
158             requireNonNull(nextHopIpList, "addToLabelMapper: nextHopIp cannot be null or empty!");
159         }
160
161         // FIXME: separate this out somehow?
162         final ReentrantLock lock = JvmGlobalLocks.getLockForString(labelStr);
163         lock.lock();
164         try {
165             addErrorLogging(txRunner.callWithNewWriteOnlyTransactionAndSubmit(OPERATIONAL, tx -> {
166                 LOG.info("addToLabelMapper: label {} dpn {} prefix {} nexthoplist {} vpnid {} vpnIntfcName {} rd {}"
167                         + " elanTag {}", labelStr, dpnId, prefix, nextHopIpList, vpnId, vpnInterfaceName, rd, elanTag);
168                 if (dpnId != null) {
169                     LabelRouteInfoBuilder lriBuilder = new LabelRouteInfoBuilder();
170                     lriBuilder.setLabel(label).setDpnId(dpnId).setPrefix(prefix).setNextHopIpList(nextHopIpList)
171                             .setParentVpnid(vpnId).setIsSubnetRoute(isSubnetRoute);
172                     if (elanTag != null) {
173                         lriBuilder.setElanTag(elanTag);
174                     } else {
175                         LOG.warn("addToLabelMapper: elanTag is null for label {} prefix {} rd {} vpnId {}",
176                                 labelStr, prefix, rd, vpnId);
177                     }
178                     if (vpnInterfaceName != null) {
179                         lriBuilder.setVpnInterfaceName(vpnInterfaceName);
180                     } else {
181                         LOG.warn("addToLabelMapper: vpn interface is null for label {} prefix {} rd {} vpnId {}",
182                                 labelStr, prefix, rd, vpnId);
183                     }
184                     lriBuilder.setParentVpnRd(rd);
185                     VpnInstanceOpDataEntry vpnInstanceOpDataEntry = vpnUtil.getVpnInstanceOpData(rd);
186                     if (vpnInstanceOpDataEntry != null) {
187                         List<String> vpnInstanceNames = Collections
188                                 .singletonList(vpnInstanceOpDataEntry.getVpnInstanceName());
189                         lriBuilder.setVpnInstanceList(vpnInstanceNames);
190                     }
191                     LabelRouteInfo lri = lriBuilder.build();
192                     InstanceIdentifier<LabelRouteInfo> lriIid = InstanceIdentifier.builder(LabelRouteMap.class)
193                             .child(LabelRouteInfo.class, new LabelRouteInfoKey(label)).build();
194                     tx.mergeParentStructureMerge(lriIid, lri);
195                     LOG.info("addToLabelMapper: Added label route info to label {} prefix {} nextHopList {} vpnId {}"
196                                     + " interface {} rd {} elantag {}", labelStr, prefix, nextHopIpList, vpnId,
197                             vpnInterfaceName, rd, elanTag);
198                 } else {
199                     LOG.warn("addToLabelMapper: Can't add entry to label map for label {} prefix {} nextHopList {}"
200                                     + " vpnId {} interface {} rd {} elantag {}, dpnId is null",
201                             labelStr, prefix, nextHopIpList,
202                             vpnId, vpnInterfaceName, rd, elanTag);
203                 }
204             }), LOG, "addToLabelMapper");
205         } finally {
206             lock.unlock();
207         }
208     }
209
210     @Override
211     public Adjacency createOperationalAdjacency(L3vpnInput input) {
212         return new AdjacencyBuilder().build();
213     }
214
215     @SuppressWarnings("checkstyle:IllegalCatch")
216     protected void addPrefixToBGP(String rd, String primaryRd, @Nullable String macAddress, String prefix,
217                                   String nextHopIp, VrfEntry.EncapType encapType, Uint32 label, Uint32 l3vni,
218                                   String gatewayMac, RouteOrigin origin,
219                                   TypedWriteTransaction<Configuration> writeConfigTxn) {
220         try {
221             List<String> nextHopList = Collections.singletonList(nextHopIp);
222             LOG.info("ADD: addPrefixToBGP: Adding Fib entry rd {} prefix {} nextHop {} label {} gwMac {}", rd, prefix,
223                     nextHopList, label, gatewayMac);
224             fibManager.addOrUpdateFibEntry(primaryRd, macAddress, prefix, nextHopList,
225                     encapType, label, l3vni, gatewayMac, null /*parentVpnRd*/, origin, writeConfigTxn);
226             LOG.info("ADD: addPrefixToBGP: Added Fib entry rd {} prefix {} nextHop {} label {} gwMac {}", rd, prefix,
227                     nextHopList, label, gatewayMac);
228             // Advertise the prefix to BGP only if nexthop ip is available
229             if (!nextHopList.isEmpty()) {
230                 bgpManager.advertisePrefix(rd, macAddress, prefix, nextHopList, encapType, label,
231                         l3vni, Uint32.ZERO /*l2vni*/, gatewayMac);
232             } else {
233                 LOG.error("addPrefixToBGP: NextHopList is null/empty. Hence rd {} prefix {} nextHop {} label {}"
234                         + " gwMac {} is not advertised to BGP", rd, prefix, nextHopList, label, gatewayMac);
235             }
236         } catch (Exception e) {
237             LOG.error("addPrefixToBGP: Add prefix {} with rd {} nextHop {} label {} gwMac {} failed", prefix, rd,
238                     nextHopIp, label, gatewayMac, e);
239         }
240     }
241 }