Line data Source code
1 : /*
2 : * Copyright (c) 2017 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include "bgp/mvpn/mvpn_table.h"
6 :
7 : #include <utility>
8 : #include <boost/foreach.hpp>
9 :
10 : #include "base/task_annotations.h"
11 : #include "bgp/ermvpn/ermvpn_table.h"
12 : #include "bgp/extended-community/source_as.h"
13 : #include "bgp/ipeer.h"
14 : #include "bgp/bgp_factory.h"
15 : #include "bgp/bgp_log.h"
16 : #include "bgp/bgp_multicast.h"
17 : #include "bgp/bgp_mvpn.h"
18 : #include "bgp/bgp_server.h"
19 : #include "bgp/bgp_update.h"
20 : #include "bgp/inet/inet_table.h"
21 : #include "bgp/origin-vn/origin_vn.h"
22 : #include "bgp/routing-instance/path_resolver.h"
23 : #include "bgp/routing-instance/routing_instance.h"
24 : #include "bgp/routing-instance/routing_instance_analytics_types.h"
25 : #include "bgp/routing-instance/routing_instance_log.h"
26 :
27 : using std::unique_ptr;
28 : using std::pair;
29 : using std::string;
30 : using std::set;
31 :
32 1769302 : size_t MvpnTable::HashFunction(const MvpnPrefix &prefix) const {
33 2324020 : if ((prefix.type() == MvpnPrefix::IntraASPMSIADRoute) ||
34 554718 : (prefix.type() == MvpnPrefix::LeafADRoute)) {
35 1269254 : uint32_t data = prefix.originator().to_ulong();
36 1269252 : return boost::hash_value(data);
37 : }
38 500048 : if (prefix.type() == MvpnPrefix::InterASPMSIADRoute) {
39 10 : uint32_t data = prefix.asn();
40 10 : return boost::hash_value(data);
41 : }
42 500038 : return boost::hash_value(prefix.group().to_ulong());
43 : }
44 :
45 51130 : MvpnTable::MvpnTable(DB *db, const string &name)
46 51130 : : BgpTable(db, name), manager_(NULL) {
47 51131 : }
48 :
49 42904 : PathResolver *MvpnTable::CreatePathResolver() {
50 42904 : if (routing_instance()->IsMasterRoutingInstance())
51 0 : return NULL;
52 42904 : PathResolver * path_resolver = new PathResolver(this);
53 42904 : path_resolver->set_nexthop_longest_match(true);
54 42904 : return path_resolver;
55 : }
56 :
57 83518 : unique_ptr<DBEntry> MvpnTable::AllocEntry(
58 : const DBRequestKey *key) const {
59 83518 : const RequestKey *pfxkey = static_cast<const RequestKey *>(key);
60 83518 : return unique_ptr<DBEntry> (new MvpnRoute(pfxkey->prefix));
61 : }
62 :
63 49 : unique_ptr<DBEntry> MvpnTable::AllocEntryStr(
64 : const string &key_str) const {
65 49 : MvpnPrefix prefix = MvpnPrefix::FromString(key_str);
66 98 : return unique_ptr<DBEntry> (new MvpnRoute(prefix));
67 49 : }
68 :
69 1769302 : size_t MvpnTable::Hash(const DBEntry *entry) const {
70 1769302 : const MvpnRoute *rt_entry = static_cast<const MvpnRoute *>(entry);
71 1769302 : const MvpnPrefix &mvpnprefix = rt_entry->GetPrefix();
72 1769302 : size_t value = MvpnTable::HashFunction(mvpnprefix);
73 1769300 : return value % kPartitionCount;
74 : }
75 :
76 40301 : size_t MvpnTable::Hash(const DBRequestKey *key) const {
77 40301 : const RequestKey *rkey = static_cast<const RequestKey *>(key);
78 40301 : Ip4Prefix prefix(rkey->prefix.group(), 32);
79 40301 : size_t value = InetTable::HashFunction(prefix);
80 40301 : return value % kPartitionCount;
81 : }
82 :
83 40133 : BgpRoute *MvpnTable::TableFind(DBTablePartition *rtp,
84 : const DBRequestKey *prefix) {
85 40133 : const RequestKey *pfxkey = static_cast<const RequestKey *>(prefix);
86 40133 : MvpnRoute rt_key(pfxkey->prefix);
87 80266 : return static_cast<BgpRoute *>(rtp->Find(&rt_key));
88 40133 : }
89 :
90 51129 : DBTableBase *MvpnTable::CreateTable(DB *db, const string &name) {
91 51129 : MvpnTable *table = new MvpnTable(db, name);
92 51131 : table->Init();
93 51131 : return table;
94 : }
95 :
96 59304 : void MvpnTable::CreateManager() {
97 59304 : if (manager_)
98 12599 : return;
99 :
100 : // Don't create the MvpnManager if ProjectManager is not present.
101 46705 : MvpnProjectManager *pm = GetProjectManager();
102 46705 : if (!pm)
103 4864 : return;
104 41841 : manager_ = BgpStaticObjectFactory::Create<MvpnManager>(this, pm->table());
105 41841 : manager_->Initialize();
106 :
107 : // Notify all routes in the table for further evaluation.
108 41841 : NotifyAllEntries();
109 :
110 : // TODO(Ananth): Should we also notify routes in the bgp.mvpn.0 table ?
111 : }
112 :
113 41841 : void MvpnTable::DestroyManager() {
114 41841 : if (!manager_)
115 0 : return;
116 41841 : if (IsDeleted())
117 22563 : DeleteMvpnManager();
118 41841 : manager_->Terminate();
119 41841 : delete manager_;
120 41841 : manager_ = NULL;
121 : }
122 :
123 51131 : void MvpnTable::CreateMvpnManagers() {
124 51131 : if (!server()->mvpn_ipv4_enable())
125 38542 : return;
126 22563 : RoutingInstance *rtinstance = routing_instance();
127 22563 : std::scoped_lock lock(rtinstance->manager()->mutex());
128 :
129 : // Don't create the MvpnManager for the VPN table.
130 45126 : if (!rtinstance->IsMasterRoutingInstance() &&
131 22563 : !rtinstance->mvpn_project_manager_network().empty()) {
132 : pair<MvpnProjectManagerNetworks::iterator, bool> ret =
133 45126 : rtinstance->manager()->mvpn_project_managers().insert(make_pair(
134 45126 : rtinstance->mvpn_project_manager_network(), set<string>()));
135 22563 : ret.first->second.insert(rtinstance->name());
136 :
137 : // Initialize MVPN Manager.
138 22563 : CreateManager();
139 : }
140 :
141 : MvpnProjectManagerNetworks::iterator iter =
142 45126 : rtinstance->manager()->mvpn_project_managers().find(
143 22563 : rtinstance->name());
144 22563 : if (iter == rtinstance->manager()->mvpn_project_managers().end())
145 9974 : return;
146 :
147 86071 : BOOST_FOREACH(const string &mvpn_network, iter->second) {
148 : RoutingInstance *rti =
149 36741 : rtinstance->manager()->GetRoutingInstance(mvpn_network);
150 36741 : if (!rti || rti->deleted())
151 0 : continue;
152 : MvpnTable *table =
153 36741 : dynamic_cast<MvpnTable *>(rti->GetTable(Address::MVPN));
154 36741 : if (!table || table->IsDeleted())
155 0 : continue;
156 36741 : table->CreateManager();
157 : }
158 22563 : }
159 :
160 22563 : void MvpnTable::DeleteMvpnManager() {
161 22563 : if (!server()->mvpn_ipv4_enable())
162 0 : return;
163 22563 : if (routing_instance()->mvpn_project_manager_network().empty())
164 0 : return;
165 22563 : std::scoped_lock lock(routing_instance()->manager()->mutex());
166 : MvpnProjectManagerNetworks::iterator iter =
167 45126 : routing_instance()->manager()->mvpn_project_managers().find(
168 22563 : routing_instance()->mvpn_project_manager_network());
169 22563 : if (iter != routing_instance()->manager()->mvpn_project_managers().end()) {
170 22563 : iter->second.erase(routing_instance()->name());
171 22563 : if (iter->second.empty())
172 6163 : routing_instance()->manager()->mvpn_project_managers().erase(iter);
173 : }
174 22563 : }
175 :
176 : // Call the const version to avoid code duplication.
177 97467 : MvpnProjectManager *MvpnTable::GetProjectManager() {
178 : return const_cast<MvpnProjectManager *>(
179 97467 : static_cast<const MvpnTable *>(this)->GetProjectManager());
180 : }
181 :
182 : // Get MvpnProjectManager object for this Mvpn. Each MVPN network is associated
183 : // with a parent project maanger network via configuration. MvpnProjectManager
184 : // is retrieved from this parent network RoutingInstance's ErmVpnTable.
185 97588 : const MvpnProjectManager *MvpnTable::GetProjectManager() const {
186 97588 : std::string pm_network = routing_instance()->mvpn_project_manager_network();
187 97588 : if (pm_network.empty())
188 0 : return NULL;
189 : const RoutingInstance *rtinstance =
190 97588 : routing_instance()->manager()->GetRoutingInstance(pm_network);
191 97588 : if (!rtinstance)
192 4864 : return NULL;
193 92724 : const ErmVpnTable *table = dynamic_cast<const ErmVpnTable *>(
194 92724 : rtinstance->GetTable(Address::ERMVPN));
195 92724 : if (!table)
196 0 : return NULL;
197 92724 : return table->mvpn_project_manager();
198 97588 : }
199 :
200 64859 : bool MvpnTable::IsProjectManagerUsable() const {
201 64859 : std::string pm_network = routing_instance()->mvpn_project_manager_network();
202 64859 : if (pm_network.empty())
203 0 : return false;
204 : const RoutingInstance *rtinstance =
205 64859 : routing_instance()->manager()->GetRoutingInstance(pm_network);
206 64859 : if (!rtinstance || rtinstance->deleted())
207 18247 : return false;
208 46612 : const ErmVpnTable *table = dynamic_cast<const ErmVpnTable *>(
209 46612 : rtinstance->GetTable(Address::ERMVPN));
210 46612 : if (!table || table->IsDeleted())
211 0 : return false;
212 :
213 93224 : if (!table->mvpn_project_manager() ||
214 46612 : table->mvpn_project_manager()->deleter()->IsDeleted()) {
215 0 : return false;
216 : }
217 46612 : return true;
218 64859 : }
219 :
220 : // Return the MvpnProjectManagerPartition for this route using the same DB
221 : // partition index as of the route.
222 121 : const MvpnProjectManagerPartition *MvpnTable::GetProjectManagerPartition(
223 : BgpRoute *route) const {
224 121 : const MvpnProjectManager *manager = GetProjectManager();
225 121 : if (!manager)
226 0 : return NULL;
227 121 : int part_id = route->get_table_partition()->index();
228 121 : return manager->GetPartition(part_id);
229 : }
230 :
231 : // Override virtual method to retrive target table for MVPN routes. For now,
232 : // only Type-4 LeafAD routes require special treatment, as they always come
233 : // with the same route target <router-id>:0. Hence, if normal rtf selection
234 : // mode is used, every table with MVPN enalbled would have to be notified for
235 : // replication. Instead, find the table based on the correspondong S-PMSI route.
236 : // This route can be retrieved from the MVPN state of the <S-G> maintained in
237 : // the MvpnProjectManagerPartition object.
238 196633 : void MvpnTable::UpdateSecondaryTablesForReplication(BgpRoute *rt,
239 : TableSet *secondary_tables) {
240 196633 : MvpnRoute *mvpn_rt = dynamic_cast<MvpnRoute *>(rt);
241 196633 : assert(mvpn_rt);
242 :
243 : // Special table lookup is required only for the Type4 LeafAD routes.
244 196633 : if (mvpn_rt->GetPrefix().type() != MvpnPrefix::LeafADRoute)
245 196512 : return;
246 :
247 : // Find Type-3 S-PMSI route from the Type-4 prefix route.
248 3776 : MvpnPrefix spmsi_prefix;
249 3776 : spmsi_prefix.SetSPMSIPrefixFromLeafADPrefix(mvpn_rt->GetPrefix());
250 3776 : const MvpnRoute *spmsi_rt = FindRoute(spmsi_prefix);
251 3776 : if (!spmsi_rt || !spmsi_rt->IsUsable())
252 0 : return;
253 3776 : if (!spmsi_rt->BestPath()->IsReplicated())
254 1 : return;
255 :
256 3775 : const BgpTable *table = dynamic_cast<const BgpSecondaryPath *>(
257 7550 : spmsi_rt->BestPath())->src_table();
258 3775 : const MvpnTable *mvpn_table = dynamic_cast<const MvpnTable *>(table);
259 3775 : if (!mvpn_table || mvpn_table->IsMaster() || !mvpn_table->manager())
260 3654 : return;
261 121 : mvpn_table->manager()->UpdateSecondaryTablesForReplication(
262 : mvpn_rt, secondary_tables);
263 3776 : }
264 :
265 : // Find or create the route.
266 82309 : MvpnRoute *MvpnTable::FindRoute(const MvpnPrefix &prefix) {
267 82309 : MvpnRoute rt_key(prefix);
268 : DBTablePartition *rtp = static_cast<DBTablePartition *>(
269 82309 : GetTablePartition(&rt_key));
270 164618 : return dynamic_cast<MvpnRoute *>(rtp->Find(&rt_key));
271 82309 : }
272 :
273 24087 : const MvpnRoute *MvpnTable::FindRoute(const MvpnPrefix &prefix) const {
274 24087 : MvpnRoute rt_key(prefix);
275 : const DBTablePartition *rtp = static_cast<const DBTablePartition *>(
276 24087 : GetTablePartition(&rt_key));
277 48174 : return dynamic_cast<const MvpnRoute *>(rtp->Find(&rt_key));
278 24087 : }
279 :
280 : // Find or create the route.
281 50006 : MvpnRoute *MvpnTable::LocateRoute(const MvpnPrefix &prefix) {
282 50006 : MvpnRoute rt_key(prefix);
283 : DBTablePartition *rtp = static_cast<DBTablePartition *>(
284 50006 : GetTablePartition(&rt_key));
285 50006 : MvpnRoute *dest_route = dynamic_cast<MvpnRoute *>(rtp->Find(&rt_key));
286 50006 : if (dest_route == NULL) {
287 48078 : dest_route = new MvpnRoute(prefix);
288 48078 : rtp->Add(dest_route);
289 : } else {
290 1928 : dest_route->ClearDelete();
291 : }
292 50006 : return dest_route;
293 50006 : }
294 :
295 3049 : MvpnPrefix MvpnTable::CreateType4LeafADRoutePrefix(const MvpnRoute *type3_rt) {
296 3049 : assert(type3_rt->GetPrefix().type() == MvpnPrefix::SPMSIADRoute);
297 3049 : const Ip4Address originator_ip(server()->bgp_identifier());
298 3049 : MvpnPrefix prefix(MvpnPrefix::LeafADRoute, originator_ip);
299 3049 : prefix.SetLeafADPrefixFromSPMSIPrefix(type3_rt->GetPrefix());
300 6098 : return prefix;
301 0 : }
302 :
303 3048 : MvpnRoute *MvpnTable::LocateType4LeafADRoute(const MvpnRoute *type3_spmsi_rt) {
304 3048 : MvpnPrefix prefix = CreateType4LeafADRoutePrefix(type3_spmsi_rt);
305 6096 : return LocateRoute(prefix);
306 3048 : }
307 :
308 4506 : MvpnPrefix MvpnTable::CreateType3SPMSIRoutePrefix(const MvpnRoute *type7_rt) {
309 4506 : assert(type7_rt->GetPrefix().type() == MvpnPrefix::SourceTreeJoinRoute);
310 4506 : const RouteDistinguisher rd = type7_rt->GetPrefix().route_distinguisher();
311 4506 : Ip4Address source = type7_rt->GetPrefix().source();
312 4506 : Ip4Address group = type7_rt->GetPrefix().group();
313 4506 : const Ip4Address originator_ip(server()->bgp_identifier());
314 : MvpnPrefix prefix(MvpnPrefix::SPMSIADRoute, rd, originator_ip,
315 4506 : group, source);
316 9012 : return prefix;
317 : }
318 :
319 17794 : MvpnPrefix MvpnTable::CreateLocalType7Prefix(MvpnRoute *rt) const {
320 17794 : const RouteDistinguisher rd = RouteDistinguisher::kZeroRd;
321 17794 : Ip4Address source = rt->GetPrefix().source();
322 17794 : Ip4Address group = rt->GetPrefix().group();
323 : MvpnPrefix prefix(MvpnPrefix::SourceTreeJoinRoute, rd,
324 17794 : 0, group, source);
325 35588 : return prefix;
326 : }
327 :
328 6293 : MvpnPrefix MvpnTable::CreateType7SourceTreeJoinRoutePrefix(
329 : MvpnRoute *rt) const {
330 : // get the source-rd from attributes as we store type-5 route with zero-rd
331 6293 : const BgpAttr *attr = rt->BestPath()->GetAttr();
332 6293 : assert(attr);
333 6293 : assert(!attr->source_rd().IsZero());
334 6293 : const RouteDistinguisher rd = attr->source_rd();
335 6293 : Ip4Address source = rt->GetPrefix().source();
336 6293 : Ip4Address group = rt->GetPrefix().group();
337 : MvpnPrefix prefix(MvpnPrefix::SourceTreeJoinRoute, rd,
338 6293 : server()->autonomous_system(), group, source);
339 12586 : return prefix;
340 : }
341 :
342 4505 : MvpnRoute *MvpnTable::LocateType3SPMSIRoute(const MvpnRoute *type7_rt) {
343 4505 : MvpnPrefix prefix = CreateType3SPMSIRoutePrefix(type7_rt);
344 9010 : return LocateRoute(prefix);
345 4505 : }
346 :
347 1 : MvpnPrefix MvpnTable::CreateType2ADRoutePrefix() {
348 1 : const RouteDistinguisher *rd = routing_instance()->GetRD();
349 : MvpnPrefix prefix(MvpnPrefix::InterASPMSIADRoute, *rd,
350 1 : server()->autonomous_system());
351 1 : return prefix;
352 : }
353 :
354 112755 : MvpnPrefix MvpnTable::CreateType1ADRoutePrefix(
355 : const Ip4Address &originator_ip) {
356 112755 : const RouteDistinguisher rd(originator_ip.to_ulong(),
357 112755 : routing_instance()->index());
358 112755 : MvpnPrefix prefix(MvpnPrefix::IntraASPMSIADRoute, rd, originator_ip);
359 225510 : return prefix;
360 : }
361 :
362 42454 : MvpnPrefix MvpnTable::CreateType1ADRoutePrefix() {
363 84908 : return CreateType1ADRoutePrefix(Ip4Address(server()->bgp_identifier()));
364 : }
365 :
366 42453 : MvpnRoute *MvpnTable::LocateType1ADRoute() {
367 42453 : MvpnPrefix prefix = CreateType1ADRoutePrefix();
368 84906 : return LocateRoute(prefix);
369 42453 : }
370 :
371 70301 : MvpnRoute *MvpnTable::FindType1ADRoute(const Ip4Address &originator_ip) {
372 70301 : MvpnPrefix prefix = CreateType1ADRoutePrefix(originator_ip);
373 140602 : return FindRoute(prefix);
374 70301 : }
375 :
376 69131 : MvpnRoute *MvpnTable::FindType1ADRoute() {
377 69131 : Ip4Address originator_ip(server()->bgp_identifier());
378 69131 : return FindType1ADRoute(Ip4Address(server()->bgp_identifier()));
379 : }
380 :
381 24087 : const MvpnRoute *MvpnTable::FindType7SourceTreeJoinRoute(MvpnRoute *rt) const {
382 24087 : MvpnPrefix prefix;
383 24087 : if (rt->GetPrefix().type() == MvpnPrefix::SourceActiveADRoute)
384 6293 : prefix = CreateType7SourceTreeJoinRoutePrefix(rt);
385 24087 : if (rt->GetPrefix().type() == MvpnPrefix::SPMSIADRoute) {
386 17794 : prefix = CreateLocalType7Prefix(rt);
387 : }
388 48174 : return FindRoute(prefix);
389 24087 : }
390 :
391 272618 : BgpRoute *MvpnTable::RouteReplicate(BgpServer *server, BgpTable *stable,
392 : BgpRoute *rt, const BgpPath *src_path, ExtCommunityPtr community) {
393 272618 : MvpnTable *src_table = dynamic_cast<MvpnTable *>(stable);
394 272618 : assert(src_table);
395 272618 : MvpnRoute *src_rt = dynamic_cast<MvpnRoute *>(rt);
396 272618 : assert(src_rt);
397 :
398 272618 : if (!server->mvpn_ipv4_enable()) {
399 572 : return ReplicatePath(server, src_rt->GetPrefix(), src_table, src_rt,
400 286 : src_path, community);
401 : }
402 :
403 : // Replicate Type7 C-Join route.
404 272332 : if (src_rt->GetPrefix().type() == MvpnPrefix::SourceTreeJoinRoute) {
405 100448 : return ReplicateType7SourceTreeJoin(server, src_table, src_rt,
406 50224 : src_path, community);
407 : }
408 :
409 222108 : if (!IsMaster()) {
410 : // For type-4 paths, only replicate if there is a type-3 primary path
411 : // present in the table.
412 88710 : if (src_rt->GetPrefix().type() == MvpnPrefix::LeafADRoute) {
413 1999 : MvpnProjectManager *pm = GetProjectManager();
414 1999 : if (!pm)
415 1878 : return NULL;
416 1999 : MvpnStatePtr mvpn_state = pm->GetState(src_rt);
417 2120 : if (!mvpn_state || !mvpn_state->spmsi_rt() ||
418 121 : !mvpn_state->spmsi_rt()->IsUsable()) {
419 1878 : return NULL;
420 : }
421 121 : if (mvpn_state->spmsi_rt()->table() != this)
422 0 : return NULL;
423 1999 : }
424 : }
425 :
426 : // Replicate all other types.
427 440460 : return ReplicatePath(server, src_rt->GetPrefix(), src_table, src_rt,
428 220230 : src_path, community);
429 : }
430 :
431 50224 : BgpRoute *MvpnTable::ReplicateType7SourceTreeJoin(BgpServer *server,
432 : MvpnTable *src_table, MvpnRoute *src_rt, const BgpPath *src_path,
433 : ExtCommunityPtr ext_community) {
434 :
435 : // Only replicate if route has a target that matches this table's auto
436 : // created route target (vit).
437 50224 : if (!IsMaster()) {
438 19209 : RouteTarget vit(Ip4Address(server->bgp_identifier()),
439 38418 : routing_instance()->index());
440 19209 : bool vit_found = false;
441 56261 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &comm,
442 : ext_community->communities()) {
443 20305 : if (ExtCommunity::is_route_target(comm)) {
444 19546 : RouteTarget rtarget(comm);
445 19546 : if (rtarget == vit) {
446 3558 : vit_found = true;
447 3558 : break;
448 : }
449 : }
450 : }
451 :
452 19209 : if (!vit_found) {
453 15651 : MVPN_RT_LOG(src_rt, "Route was not replicated as rt-import "
454 : "extended-community was not found");
455 15651 : return NULL;
456 : }
457 : }
458 :
459 : // If replicating from Master table, no special checks are required.
460 34573 : if (src_table->IsMaster()) {
461 7072 : return ReplicatePath(server, src_rt->GetPrefix(), src_table, src_rt,
462 3536 : src_path, ext_community);
463 : }
464 :
465 : // This is the case when routes are replicated either to Master or to other
466 : // vrf.mvpn.0 as identified the route targets. In either case, basic idea
467 : // is to target the replicated path directly to vrf where sender resides.
468 : //
469 : // Route-target of the target vrf is derived from the Vrf Import Target of
470 : // the route the source resolves to. Resolver code would have already
471 : // computed this and encoded inside source-rd. Also source-as to encode in
472 : // the RD is also encoded as part of the SourceAS extended community.
473 31037 : const BgpAttr *attr = src_path->GetAttr();
474 31037 : if (!attr)
475 0 : return NULL;
476 :
477 : // Do not resplicate if the source is not resolvable.
478 31037 : if (attr->source_rd().IsZero()) {
479 30881 : MVPN_RT_LOG(src_rt, "Route was not replicated as source_rd is zero");
480 30881 : return NULL;
481 : }
482 :
483 : // Find source-as extended-community. If not present, do not replicate
484 156 : bool source_as_found = false;
485 156 : SourceAs source_as;
486 312 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &value,
487 : attr->ext_community()->communities()) {
488 156 : if (ExtCommunity::is_source_as(value)) {
489 156 : source_as_found = true;
490 156 : source_as = SourceAs(value);
491 156 : break;
492 : }
493 : }
494 :
495 156 : if (!source_as_found) {
496 0 : MVPN_RT_LOG(src_rt, "Route was not replicated as source_as is zero");
497 0 : return NULL;
498 : }
499 :
500 : // No need to send SourceAS with this mvpn route. This is only sent along
501 : // with the unicast routes.
502 : ext_community =
503 156 : server->extcomm_db()->RemoveSourceASAndLocate(ext_community.get());
504 :
505 : // Replicate path using source route's<C-S,G>, source_rd and asn as encoded
506 : // in the source-as attribute.
507 : MvpnPrefix prefix(MvpnPrefix::SourceTreeJoinRoute, attr->source_rd(),
508 156 : source_as.GetAsn(), src_rt->GetPrefix().group(),
509 312 : src_rt->GetPrefix().source());
510 :
511 : // Replicate the path with the computed prefix and attributes.
512 312 : return ReplicatePath(server, prefix, src_table, src_rt, src_path,
513 156 : ext_community);
514 156 : }
515 :
516 224208 : BgpRoute *MvpnTable::ReplicatePath(BgpServer *server, const MvpnPrefix &mprefix,
517 : MvpnTable *src_table, MvpnRoute *src_rt, const BgpPath *src_path,
518 : ExtCommunityPtr comm) {
519 224208 : MvpnRoute rt_key(mprefix);
520 :
521 : // Find or create the route.
522 : DBTablePartition *rtp =
523 224208 : static_cast<DBTablePartition *>(GetTablePartition(&rt_key));
524 224208 : BgpRoute *dest_route = static_cast<BgpRoute *>(rtp->Find(&rt_key));
525 224208 : if (dest_route == NULL) {
526 77734 : dest_route = new MvpnRoute(mprefix);
527 77734 : rtp->Add(dest_route);
528 : } else {
529 146474 : dest_route->ClearDelete();
530 : }
531 :
532 : BgpAttrPtr new_attr =
533 : server->attr_db()->ReplaceExtCommunityAndLocate(src_path->GetAttr(),
534 224208 : comm.get());
535 : // Replace Nexthop with controller address, MX rejects the route if
536 : // nexthop is same as neighbor address, needed for Type-7 routes
537 672624 : new_attr = server->attr_db()->ReplaceNexthopAndLocate(new_attr.get(),
538 448416 : Ip4Address(server->bgp_identifier()));
539 : // Need to strip off route targets other than sender-ip:0
540 224208 : if (src_rt->GetPrefix().type() == MvpnPrefix::LeafADRoute) {
541 3794 : ExtCommunity::ExtCommunityList rtarget;
542 3794 : Ip4Address ip = src_rt->GetPrefix().GetType3OriginatorFromType4Route();
543 3794 : RouteTarget leaf_ad_rtarget(ip, 0);
544 14971 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &value,
545 : comm->communities()) {
546 7485 : if (ExtCommunity::is_route_target(value)) {
547 7484 : if (leaf_ad_rtarget == RouteTarget(value)) {
548 3793 : rtarget.push_back(value);
549 3793 : break;
550 : }
551 : }
552 : }
553 :
554 3794 : if (rtarget.size() == 1) {
555 : ExtCommunityPtr ext_community = server->extcomm_db()->
556 3793 : ReplaceRTargetAndLocate(new_attr->ext_community(), rtarget);
557 7586 : new_attr = server->attr_db()->ReplaceExtCommunityAndLocate(
558 3793 : src_path->GetAttr(), ext_community.get());
559 3793 : } else {
560 1 : MVPN_RT_LOG(src_rt,
561 : "Could not find <originator>:0 route-target community");
562 : }
563 3794 : }
564 :
565 : // Check whether peer already has a path.
566 224208 : BgpPath *dest_path = dest_route->FindSecondaryPath(src_rt,
567 : src_path->GetSource(), src_path->GetPeer(),
568 : src_path->GetPathId());
569 224208 : if (dest_path != NULL) {
570 265801 : if (new_attr != dest_path->GetOriginalAttr() ||
571 132564 : src_path->GetFlags() != dest_path->GetFlags()) {
572 673 : bool success = dest_route->RemoveSecondaryPath(src_rt,
573 : src_path->GetSource(), src_path->GetPeer(),
574 : src_path->GetPathId());
575 673 : assert(success);
576 : } else {
577 132564 : return dest_route;
578 : }
579 : }
580 :
581 : // Create replicated path and insert it on the route.
582 : BgpSecondaryPath *replicated_path =
583 91644 : new BgpSecondaryPath(src_path->GetPeer(), src_path->GetPathId(),
584 91644 : src_path->GetSource(), new_attr,
585 91644 : src_path->GetFlags(), src_path->GetLabel());
586 91644 : replicated_path->SetReplicateInfo(src_table, src_rt);
587 91644 : dest_route->InsertPath(replicated_path);
588 91644 : rtp->Notify(dest_route);
589 91644 : MVPN_RT_LOG(src_rt, "Route was successfully replicated");
590 91644 : return dest_route;
591 224208 : }
592 :
593 11859 : bool MvpnTable::Export(RibOut *ribout, Route *route,
594 : const RibPeerSet &peerset, UpdateInfoSList &uinfo_slist) {
595 11859 : MvpnRoute *mvpn_route = dynamic_cast<MvpnRoute *>(route);
596 :
597 11859 : if (ribout->IsEncodingXmpp()) {
598 7848 : UpdateInfo *uinfo = GetMvpnUpdateInfo(ribout, mvpn_route, peerset);
599 7848 : if (!uinfo)
600 7583 : return false;
601 265 : uinfo_slist->push_front(*uinfo);
602 265 : return true;
603 : }
604 4011 : BgpRoute *bgp_route = static_cast<BgpRoute *> (route);
605 :
606 4011 : UpdateInfo *uinfo = GetUpdateInfo(ribout, bgp_route, peerset);
607 4011 : if (!uinfo) {
608 2892 : MVPN_RT_LOG(mvpn_route, "Route was exported as update_info could not "
609 : "be computed");
610 2892 : return false;
611 : }
612 1119 : uinfo_slist->push_front(*uinfo);
613 1119 : return true;
614 : }
615 :
616 429 : void MvpnTable::GetPeerSet(RibOut *ribout, MvpnRoute *route,
617 : const RibPeerSet &peerset, RibPeerSet *new_peerset) {
618 429 : RibOut::PeerIterator iter(ribout, peerset);
619 1694 : while (iter.HasNext()) {
620 1265 : int current_index = iter.index();
621 1265 : IPeer *peer = dynamic_cast<IPeer *>(iter.Next());
622 1265 : assert(peer);
623 2530 : for (Route::PathList::const_iterator it = route->GetPathList().begin();
624 4578 : it != route->GetPathList().end(); ++it) {
625 1461 : const BgpPath *path = static_cast<const BgpPath *>(it.operator->());
626 1461 : if (path->IsFeasible() && peer == path->GetPeer()) {
627 437 : new_peerset->set(current_index);
628 437 : break;
629 : }
630 : }
631 : }
632 429 : }
633 :
634 7848 : UpdateInfo *MvpnTable::GetMvpnUpdateInfo(RibOut *ribout, MvpnRoute *route,
635 : const RibPeerSet &peerset) {
636 15179 : if ((route->GetPrefix().type() != MvpnPrefix::SourceActiveADRoute) &&
637 7331 : (route->GetPrefix().type() != MvpnPrefix::SourceTreeJoinRoute))
638 6936 : return NULL;
639 912 : if (!route->IsUsable())
640 183 : return NULL;
641 :
642 729 : if (route->BestPath()->IsReplicated())
643 300 : return NULL;
644 :
645 429 : MvpnProjectManager *pm = GetProjectManager();
646 429 : if (!pm) {
647 0 : MVPN_RT_LOG(route, "Route was exported as ProjectManager was "
648 : "not found");
649 0 : return NULL;
650 : }
651 :
652 429 : RibPeerSet new_peerset;
653 429 : GetPeerSet(ribout, route, peerset, &new_peerset);
654 :
655 429 : if (new_peerset.empty())
656 8 : return NULL;
657 :
658 421 : UpdateInfo *uinfo = pm->GetUpdateInfo(route);
659 421 : if (uinfo)
660 265 : uinfo->target = new_peerset;
661 421 : return uinfo;
662 429 : }
663 :
664 1382241 : bool MvpnTable::IsMaster() const {
665 1382241 : return routing_instance()->IsMasterRoutingInstance();
666 : }
667 :
668 159 : static void RegisterFactory() {
669 159 : DB::RegisterFactory("mvpn.0", &MvpnTable::CreateTable);
670 159 : }
671 :
672 : MODULE_INITIALIZER(RegisterFactory);
|