Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include "bgp/routing-instance/routepath_replicator.h"
6 :
7 : #include <boost/foreach.hpp>
8 :
9 : #include <utility>
10 :
11 : #include "base/set_util.h"
12 : #include "base/task_annotations.h"
13 : #include "base/task_trigger.h"
14 : #include "bgp/bgp_config.h"
15 : #include "bgp/bgp_log.h"
16 : #include "bgp/bgp_route.h"
17 : #include "bgp/bgp_server.h"
18 : #include "bgp/origin-vn/origin_vn.h"
19 : #include "bgp/routing-instance/path_resolver.h"
20 : #include "bgp/routing-instance/routing_instance.h"
21 : #include "bgp/routing-instance/rtarget_group_mgr.h"
22 : #include "bgp/routing-instance/routing_instance_analytics_types.h"
23 :
24 : using std::ostringstream;
25 : using std::make_pair;
26 : using std::pair;
27 : using std::string;
28 : using std::vector;
29 :
30 : //
31 : // RoutePathReplication trace macro. Optionally logs the server name as well for
32 : // easier debugging in multi server unit tests
33 : //
34 : #define RPR_TRACE(obj, ...) \
35 : do { \
36 : if (LoggingDisabled()) break; \
37 : bgp_log_test::LogServerName(server()); \
38 : Rpr##obj##Log::Send("RoutingInstance", \
39 : SandeshLevel::SYS_DEBUG, __FILE__, __LINE__, __VA_ARGS__); \
40 : Rpr##obj::TraceMsg(trace_buf_, __FILE__, __LINE__, __VA_ARGS__); \
41 : } while (false)
42 :
43 : #define RPR_TRACE_ONLY(obj, ...) \
44 : do { \
45 : if (LoggingDisabled()) break; \
46 : bgp_log_test::LogServerName(server()); \
47 : Rpr##obj::TraceMsg(trace_buf_, __FILE__, __LINE__, __VA_ARGS__); \
48 : } while (false)
49 :
50 : class TableState::DeleteActor : public LifetimeActor {
51 : public:
52 254217 : explicit DeleteActor(TableState *ts)
53 254217 : : LifetimeActor(ts->replicator()->server()->lifetime_manager()),
54 254217 : ts_(ts) {
55 254219 : }
56 :
57 254302 : virtual bool MayDelete() const {
58 254302 : return ts_->MayDelete();
59 : }
60 :
61 254220 : virtual void Shutdown() {
62 254220 : }
63 :
64 254220 : virtual void Destroy() {
65 254220 : ts_->replicator()->DeleteTableState(ts_->table());
66 254220 : }
67 :
68 : private:
69 : TableState *ts_;
70 : };
71 :
72 254215 : TableState::TableState(RoutePathReplicator *replicator, BgpTable *table)
73 254215 : : replicator_(replicator),
74 254215 : table_(table),
75 254215 : listener_id_(DBTableBase::kInvalidId),
76 254215 : deleter_(new DeleteActor(this)),
77 254218 : table_delete_ref_(this, table->deleter()) {
78 254213 : assert(table->deleter() != NULL);
79 254213 : }
80 :
81 254220 : TableState::~TableState() {
82 254220 : if (walk_ref() != NULL) {
83 22602 : table()->ReleaseWalker(walk_ref());
84 : }
85 254220 : }
86 :
87 254220 : void TableState::ManagedDelete() {
88 254220 : deleter()->Delete();
89 254220 : }
90 :
91 0 : bool TableState::deleted() const {
92 0 : return deleter()->IsDeleted();
93 : }
94 :
95 899686 : LifetimeActor *TableState::deleter() {
96 899686 : return deleter_.get();
97 : }
98 :
99 0 : const LifetimeActor *TableState::deleter() const {
100 0 : return deleter_.get();
101 : }
102 :
103 254302 : bool TableState::MayDelete() const {
104 531226 : if (list_.empty() && !route_count() &&
105 276924 : ((walk_ref() == NULL) || !walk_ref()->walk_is_active()))
106 254220 : return true;
107 82 : return false;
108 : }
109 :
110 626380 : void TableState::RetryDelete() {
111 626380 : if (!deleter()->IsDeleted())
112 607227 : return;
113 19092 : deleter()->RetryDelete();
114 : }
115 :
116 668124 : void TableState::AddGroup(RtGroup *group) {
117 668124 : list_.insert(group);
118 668318 : }
119 :
120 668344 : void TableState::RemoveGroup(RtGroup *group) {
121 668344 : list_.erase(group);
122 668345 : }
123 :
124 429496 : const RtGroup *TableState::FindGroup(RtGroup *group) const {
125 429496 : GroupList::const_iterator it = list_.find(group);
126 429451 : return (it != list_.end() ? *it : NULL);
127 : }
128 :
129 254313 : uint32_t TableState::route_count() const {
130 254313 : return table_->GetDBStateCount(listener_id());
131 : }
132 :
133 1030938 : RtReplicated::RtReplicated(RoutePathReplicator *replicator)
134 1030938 : : replicator_(replicator) {
135 1030859 : }
136 :
137 517485 : void RtReplicated::AddRouteInfo(BgpTable *table, BgpRoute *rt,
138 : ReplicatedRtPathList::const_iterator it) {
139 517485 : pair<ReplicatedRtPathList::iterator, bool> result;
140 517477 : result = replicate_list_.insert(*it);
141 517519 : assert(result.second);
142 517519 : }
143 :
144 516753 : void RtReplicated::DeleteRouteInfo(BgpTable *table, BgpRoute *rt,
145 : ReplicatedRtPathList::const_iterator it) {
146 516753 : replicator_->DeleteSecondaryPath(table, rt, *it);
147 517175 : replicate_list_.erase(it);
148 517344 : }
149 :
150 : //
151 : // Return the list of secondary table names for the given primary path.
152 : // We go through all SecondaryRouteInfos and skip the ones that don't
153 : // match the primary path.
154 : //
155 283 : vector<string> RtReplicated::GetTableNameList(const BgpPath *path) const {
156 283 : vector<string> table_list;
157 1373 : BOOST_FOREACH(const SecondaryRouteInfo &rinfo, replicate_list_) {
158 545 : if (rinfo.peer_ != path->GetPeer())
159 0 : continue;
160 545 : if (rinfo.path_id_ != path->GetPathId())
161 16 : continue;
162 529 : if (rinfo.src_ != path->GetSource())
163 0 : continue;
164 529 : table_list.push_back(rinfo.table_->name());
165 : }
166 283 : return table_list;
167 0 : }
168 :
169 48715 : RoutePathReplicator::RoutePathReplicator(BgpServer *server,
170 48715 : Address::Family family)
171 48715 : : server_(server),
172 48715 : family_(family),
173 48715 : vpn_table_(NULL),
174 48715 : trace_buf_(SandeshTraceBufferCreate("RoutePathReplicator", 500)) {
175 48715 : }
176 :
177 97430 : RoutePathReplicator::~RoutePathReplicator() {
178 48715 : assert(table_state_list_.empty());
179 97430 : }
180 :
181 41140 : void RoutePathReplicator::Initialize() {
182 41140 : assert(!vpn_table_);
183 41140 : RoutingInstanceMgr *mgr = server_->routing_instance_mgr();
184 41140 : assert(mgr);
185 41140 : RoutingInstance *master = mgr->GetDefaultRoutingInstance();
186 41140 : assert(master);
187 41140 : vpn_table_ = master->GetTable(family_);
188 41140 : assert(vpn_table_);
189 41140 : assert(AddTableState(vpn_table_));
190 41140 : }
191 :
192 709453 : TableState *RoutePathReplicator::AddTableState(BgpTable *table,
193 : RtGroup *group) {
194 709453 : assert(table->IsVpnTable() || group);
195 :
196 709438 : TableStateList::iterator loc = table_state_list_.find(table);
197 709144 : if (loc == table_state_list_.end()) {
198 254209 : TableState *ts = new TableState(this, table);
199 254213 : DBTableBase::ListenerId id = table->Register(
200 : boost::bind(&RoutePathReplicator::RouteListener, this, ts, _1, _2),
201 : "RoutePathReplicator");
202 254194 : ts->set_listener_id(id);
203 254191 : if (group)
204 213051 : ts->AddGroup(group);
205 254214 : table_state_list_.insert(make_pair(table, ts));
206 254212 : RPR_TRACE(RegTable, table->name());
207 254210 : return ts;
208 : } else {
209 455082 : assert(group);
210 455082 : TableState *ts = loc->second;
211 455085 : ts->AddGroup(group);
212 455256 : return ts;
213 : }
214 : }
215 :
216 668345 : void RoutePathReplicator::RemoveTableState(BgpTable *table, RtGroup *group) {
217 668345 : TableState *ts = FindTableState(table);
218 668344 : assert(ts);
219 668344 : ts->RemoveGroup(group);
220 668345 : }
221 :
222 254220 : void RoutePathReplicator::DeleteTableState(BgpTable *table) {
223 254220 : TableState *ts = FindTableState(table);
224 254220 : assert(ts);
225 254220 : RPR_TRACE(UnregTable, table->name());
226 254220 : table->Unregister(ts->listener_id());
227 254220 : table_state_list_.erase(table);
228 254220 : delete ts;
229 254220 : }
230 :
231 1858341 : TableState *RoutePathReplicator::FindTableState(BgpTable *table) {
232 1858341 : TableStateList::iterator loc = table_state_list_.find(table);
233 1858260 : return (loc != table_state_list_.end() ? loc->second : NULL);
234 : }
235 :
236 5644 : const TableState *RoutePathReplicator::FindTableState(
237 : const BgpTable *table) const {
238 : TableStateList::const_iterator loc =
239 5644 : table_state_list_.find(const_cast<BgpTable *>(table));
240 5637 : return (loc != table_state_list_.end() ? loc->second : NULL);
241 : }
242 :
243 : void
244 51325 : RoutePathReplicator::RequestWalk(BgpTable *table) {
245 51325 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper");
246 51325 : TableState *ts = FindTableState(table);
247 51325 : assert(ts);
248 51325 : if (!ts->walk_ref()) {
249 : DBTable::DBTableWalkRef walk_ref = table->AllocWalker(
250 : boost::bind(&RoutePathReplicator::RouteListener, this, ts, _1, _2),
251 45204 : boost::bind(&RoutePathReplicator::BulkReplicationDone, this, _2));
252 22602 : table->WalkTable(walk_ref);
253 22602 : ts->set_walk_ref(walk_ref);
254 22602 : } else {
255 28723 : table->WalkAgain(ts->walk_ref());
256 : }
257 51325 : }
258 :
259 : void
260 24561 : RoutePathReplicator::BulkReplicationDone(DBTableBase *dbtable) {
261 24561 : CHECK_CONCURRENCY("db::Walker");
262 24561 : BgpTable *table = static_cast<BgpTable *>(dbtable);
263 24561 : RPR_TRACE(WalkDone, table->name());
264 24561 : TableState *ts = FindTableState(table);
265 24561 : ts->RetryDelete();
266 24561 : }
267 :
268 429939 : void RoutePathReplicator::JoinVpnTable(RtGroup *group) {
269 429939 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper");
270 429968 : TableState *vpn_ts = FindTableState(vpn_table_);
271 429862 : if (!vpn_ts || vpn_ts->FindGroup(group))
272 370 : return;
273 429437 : RPR_TRACE(TableJoin, vpn_table_->name(), group->rt().ToString(), true);
274 429574 : group->AddImportTable(family(), vpn_table_);
275 429723 : RPR_TRACE(TableJoin, vpn_table_->name(), group->rt().ToString(), false);
276 429720 : group->AddExportTable(family(), vpn_table_);
277 429726 : AddTableState(vpn_table_, group);
278 : }
279 :
280 429920 : void RoutePathReplicator::LeaveVpnTable(RtGroup *group) {
281 429920 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper");
282 429919 : TableState *vpn_ts = FindTableState(vpn_table_);
283 429918 : if (!vpn_ts)
284 185 : return;
285 429733 : RPR_TRACE(TableLeave, vpn_table_->name(), group->rt().ToString(), true);
286 429734 : group->RemoveImportTable(family(), vpn_table_);
287 429734 : RPR_TRACE(TableLeave, vpn_table_->name(), group->rt().ToString(), false);
288 429734 : group->RemoveExportTable(family(), vpn_table_);
289 429735 : RemoveTableState(vpn_table_, group);
290 : }
291 :
292 : //
293 : // Add a given BgpTable to RtGroup of given RouteTarget.
294 : // It will create a new RtGroup if none exists.
295 : // In case of export RouteTarget, create TableState if it doesn't exist.
296 : //
297 927971 : void RoutePathReplicator::Join(BgpTable *table, const RouteTarget &rt,
298 : bool import) {
299 927971 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper");
300 :
301 927798 : std::scoped_lock lock(mutex_);
302 928426 : RPR_TRACE(TableJoin, table->name(), rt.ToString(), import);
303 :
304 928438 : bool first = false;
305 928438 : RtGroup *group = server()->rtarget_group_mgr()->LocateRtGroup(rt);
306 928559 : if (import) {
307 689950 : first = group->AddImportTable(family(), table);
308 689948 : if (group->HasDepRoutes())
309 100 : server()->rtarget_group_mgr()->NotifyRtGroup(rt);
310 689872 : if (family_ == Address::INETVPN)
311 137991 : server_->NotifyAllStaticRoutes();
312 1338586 : BOOST_FOREACH(BgpTable *sec_table, group->GetExportTables(family())) {
313 324349 : if (sec_table->IsVpnTable() || sec_table->empty())
314 323173 : continue;
315 1180 : RequestWalk(sec_table);
316 : }
317 : } else {
318 238609 : first = group->AddExportTable(family(), table);
319 238609 : AddTableState(table, group);
320 238600 : if (!table->empty()) {
321 58 : RequestWalk(table);
322 : }
323 : }
324 :
325 : // Join the vpn table when group is created.
326 928171 : if (first)
327 429940 : JoinVpnTable(group);
328 928306 : }
329 :
330 : //
331 : // Remove a BgpTable from RtGroup of given RouteTarget.
332 : // If the last group is going away, the RtGroup will be removed
333 : // In case of export RouteTarget, trigger remove of TableState appropriate.
334 : //
335 928563 : void RoutePathReplicator::Leave(BgpTable *table, const RouteTarget &rt,
336 : bool import) {
337 928563 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper");
338 :
339 928562 : std::scoped_lock lock(mutex_);
340 928564 : RtGroup *group = server()->rtarget_group_mgr()->GetRtGroup(rt);
341 928565 : assert(group);
342 928565 : RPR_TRACE(TableLeave, table->name(), rt.ToString(), import);
343 :
344 928565 : if (import) {
345 689955 : group->RemoveImportTable(family(), table);
346 689951 : if (group->HasDepRoutes())
347 132315 : server()->rtarget_group_mgr()->NotifyRtGroup(rt);
348 689948 : if (family_ == Address::INETVPN)
349 137991 : server_->NotifyAllStaticRoutes();
350 2674185 : BOOST_FOREACH(BgpTable *sec_table, group->GetExportTables(family())) {
351 992117 : if (sec_table->IsVpnTable() || sec_table->empty())
352 965207 : continue;
353 26911 : RequestWalk(sec_table);
354 : }
355 : } else {
356 238610 : group->RemoveExportTable(family(), table);
357 238610 : RemoveTableState(table, group);
358 238610 : if (!table->empty()) {
359 23162 : RequestWalk(table);
360 : }
361 : }
362 :
363 : // Leave the vpn table when the last VRF has left the group.
364 928559 : if (!group->HasVrfTables(family())) {
365 429920 : LeaveVpnTable(group);
366 429920 : server()->rtarget_group_mgr()->RemoveRtGroup(rt);
367 : }
368 928559 : }
369 :
370 1418388 : void RoutePathReplicator::DBStateSync(BgpTable *table, TableState *ts,
371 : BgpRoute *rt, RtReplicated *dbstate,
372 : const RtReplicated::ReplicatedRtPathList *future) {
373 1418388 : set_synchronize(dbstate->GetMutableList(), future,
374 : boost::bind(&RtReplicated::AddRouteInfo, dbstate, table, rt, _1),
375 : boost::bind(&RtReplicated::DeleteRouteInfo, dbstate, table, rt, _1));
376 :
377 1418162 : if (dbstate->GetList().empty()) {
378 1030956 : rt->ClearState(table, ts->listener_id());
379 1031443 : delete dbstate;
380 1031182 : if (table->GetDBStateCount(ts->listener_id()) == 0)
381 601836 : ts->RetryDelete();
382 : }
383 1418527 : }
384 :
385 223995 : static ExtCommunityPtr UpdateOriginVn(BgpServer *server,
386 : const ExtCommunity *ext_community,
387 : int vn_index) {
388 223995 : as_t asn = server->autonomous_system();
389 223994 : ExtCommunityPtr extcomm_ptr;
390 223994 : if (vn_index) {
391 53851 : if (asn > AS2_MAX && vn_index > 0xffff) {
392 0 : OriginVn origin_vn(asn, AS_TRANS);
393 0 : extcomm_ptr = server->extcomm_db()->ReplaceOriginVnAndLocate(
394 0 : ext_community, origin_vn.GetExtCommunity());
395 0 : OriginVn origin_vn2(AS_TRANS, vn_index);
396 0 : extcomm_ptr = server->extcomm_db()->AppendAndLocate(
397 0 : extcomm_ptr.get(), origin_vn2.GetExtCommunity());
398 0 : } else {
399 53851 : OriginVn origin_vn(asn, vn_index);
400 161563 : extcomm_ptr = server->extcomm_db()->ReplaceOriginVnAndLocate(
401 107711 : ext_community, origin_vn.GetExtCommunity());
402 53858 : return extcomm_ptr;
403 : }
404 : }
405 170143 : extcomm_ptr = server->extcomm_db()->RemoveOriginVnAndLocate(ext_community);
406 170179 : return extcomm_ptr;
407 0 : }
408 :
409 : //
410 : // Update the ExtCommunity with the RouteTargets from the export list
411 : // and the OriginVn. The OriginVn is derived from the RouteTargets in
412 : // vpn routes.
413 : //
414 516422 : static ExtCommunityPtr UpdateExtCommunity(BgpServer *server,
415 : const RoutingInstance *rtinstance, const ExtCommunity *ext_community,
416 : const ExtCommunity::ExtCommunityList &export_list) {
417 : // Add RouteTargets exported by the instance for a non-master instance.
418 516422 : ExtCommunityPtr extcomm_ptr;
419 516422 : if (!rtinstance->IsMasterRoutingInstance()) {
420 : extcomm_ptr =
421 322731 : server->extcomm_db()->AppendAndLocate(ext_community, export_list);
422 322812 : return extcomm_ptr;
423 : }
424 :
425 : // Bail if we have a vpn route without extended communities.
426 193693 : if (!ext_community)
427 1053 : return ExtCommunityPtr(NULL);
428 :
429 : // Nothing to do if we already have the OriginVn community with our AS
430 : // or with a vn index from the global range.
431 3242407 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &comm,
432 : ext_community->communities()) {
433 1541513 : if (!ExtCommunity::is_origin_vn(comm))
434 1524961 : continue;
435 18512 : OriginVn origin_vn(comm);
436 37024 : if (!origin_vn.IsGlobal() &&
437 18512 : origin_vn.as_number() != server->autonomous_system()) {
438 1999 : continue;
439 : }
440 16513 : return ExtCommunityPtr(ext_community);
441 : }
442 :
443 : // Add the OriginVn if we have a valid vn index.
444 : int vn_index =
445 176156 : server->routing_instance_mgr()->GetVnIndexByExtCommunity(ext_community);
446 176297 : return UpdateOriginVn(server, ext_community, vn_index);
447 516714 : }
448 :
449 : //
450 : // Concurrency: Called in the context of the DB partition task.
451 : //
452 : // This function handles
453 : // 1. Table Notification for path replication
454 : // 2. Table walk for import/export of new targets
455 : //
456 : // Replicate a path (clone the BgpPath) to secondary BgpTables based on the
457 : // export targets of the primary BgpTable.
458 : // If primary table is a VRF table attach it's export targets to replicated
459 : // path in the VPN table.
460 : //
461 1976310 : bool RoutePathReplicator::RouteListener(TableState *ts,
462 : DBTablePartBase *root, DBEntryBase *entry) {
463 1976310 : CHECK_CONCURRENCY("db::DBTable");
464 :
465 1975650 : BgpTable *table = static_cast<BgpTable *>(root->parent());
466 1975615 : BgpRoute *rt = static_cast<BgpRoute *>(entry);
467 1975615 : const RoutingInstance *rtinstance = table->routing_instance();
468 :
469 1975602 : DBTableBase::ListenerId id = ts->listener_id();
470 1975586 : assert(id != DBTableBase::kInvalidId);
471 :
472 : // Get the DBState.
473 : RtReplicated *dbstate =
474 1975586 : static_cast<RtReplicated *>(rt->GetState(table, id));
475 1977674 : RtReplicated::ReplicatedRtPathList replicated_path_list;
476 :
477 : //Flag to track if any change happenned to route in last 30 minutes
478 1976946 : bool route_unchanged = false;
479 : //List of tables that needs replication of route
480 1976946 : RtGroup::RtGroupMemberList addedTables;
481 : //List of tables that doesn't need replication of route
482 1976181 : RtGroup::RtGroupMemberList deletedTables;
483 : //List of tables having the latest changes in route
484 1975704 : RtGroup::RtGroupMemberList previousTables;
485 : //Threshold used for optimised replication
486 1975455 : uint64_t optimizationThresholdTime = 30ULL * 60 * 1000000;
487 1975455 : uint64_t start = UTCTimestampUsec();
488 :
489 : //Optimize only if last route change occurred 30 mins before.
490 : //We update the route_unchanged flag to true
491 1977651 : if ( (start - rt->last_update_at() > optimizationThresholdTime) &&
492 : dbstate != NULL)
493 : {
494 139 : RPR_TRACE(RouteListener, "Route change occurred before threshold \
495 : time, optimising replication");
496 139 : route_unchanged = true;
497 : }
498 :
499 : //
500 : // Cleanup if the route is not usable.
501 : // If route aggregation is enabled, contributing route/more specific route
502 : // for a aggregate route will NOT be replicated to destination table
503 : //
504 2440187 : if (!rt->IsUsable() || (table->IsRouteAggregationSupported() &&
505 462773 : !rtinstance->deleted() &&
506 459928 : table->IsContributingRoute(rt))) {
507 717215 : if (!dbstate) {
508 558921 : return true;
509 : }
510 158294 : DBStateSync(table, ts, rt, dbstate, &replicated_path_list);
511 158426 : return true;
512 : }
513 :
514 : // Create and set new DBState on the route. This will get cleaned up via
515 : // via the call to DBStateSync if we don't need to replicate the route to
516 : // any tables.
517 1259562 : if (dbstate == NULL) {
518 1030935 : dbstate = new RtReplicated(this);
519 1030858 : rt->SetState(table, id, dbstate);
520 : }
521 :
522 : // Get the export route target list from the routing instance.
523 1259998 : ExtCommunity::ExtCommunityList export_list;
524 1259863 : if (!rtinstance->IsMasterRoutingInstance()) {
525 5802887 : BOOST_FOREACH(RouteTarget rtarget, rtinstance->GetExportList()) {
526 2494060 : export_list.push_back(rtarget.GetExtCommunity());
527 : }
528 : }
529 :
530 : // Updating previousTables with the list of tables having the information
531 : // about the route.
532 1259623 : if (route_unchanged){
533 139 : replicated_path_list = dbstate->GetList();
534 435 : BOOST_FOREACH (RtReplicated::SecondaryRouteInfo path,
535 : replicated_path_list){
536 148 : previousTables.insert(path.table_);
537 : }
538 : }
539 :
540 : // Replicate all feasible and non-replicated paths.
541 1259623 : for (Route::PathList::iterator it = rt->GetPathList().begin();
542 6016399 : it != rt->GetPathList().end(); ++it) {
543 1754523 : BgpPath *path = static_cast<BgpPath *>(it.operator->());
544 :
545 : // Skip if the source peer is down.
546 2580130 : if (!path->IsStale() && !path->IsLlgrStale() && path->GetPeer() &&
547 825314 : !path->GetPeer()->IsReady())
548 1262950 : continue;
549 :
550 : // No need to replicate aliased or replicated paths.
551 1667964 : if (path->IsReplicated() || path->IsAliased())
552 1145274 : continue;
553 :
554 : // Do not replicate non-ecmp paths.
555 522672 : if (rt->BestPath()->PathCompare(*path, true))
556 6228 : break;
557 :
558 : // Do not replicate if nexthop is not hitched by ermvpn tree.
559 516429 : if (path->CheckErmVpn())
560 0 : break;
561 :
562 516428 : const BgpAttr *attr = path->GetAttr();
563 516425 : const ExtCommunity *ext_community = attr->ext_community();
564 :
565 : ExtCommunityPtr extcomm_ptr =
566 516421 : UpdateExtCommunity(server(), rtinstance, ext_community, export_list);
567 516708 : ext_community = extcomm_ptr.get();
568 516713 : if (!ext_community)
569 1053 : continue;
570 :
571 : // Go through all extended communities.
572 : //
573 : // Get the vn_index from the OriginVn extended community.
574 : // For each RouteTarget extended community, get the list of tables
575 : // to which we need to replicate the path.
576 515660 : int vn_index = 0;
577 515660 : RtGroup::RtGroupMemberList secondary_tables;
578 5822691 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &comm,
579 : ext_community->communities()) {
580 2653490 : if (ExtCommunity::is_origin_vn(comm)) {
581 79514 : OriginVn origin_vn(comm);
582 79514 : vn_index = origin_vn.vn_index();
583 2573904 : } else if (ExtCommunity::is_route_target(comm)) {
584 : RtGroup *group =
585 2117576 : server()->rtarget_group_mgr()->GetRtGroup(comm);
586 2117954 : if (!group)
587 574684 : continue;
588 : RtGroup::RtGroupMemberList import_list =
589 2116154 : group->GetImportTables(family());
590 2115868 : if (import_list.empty())
591 572884 : continue;
592 1542995 : secondary_tables.insert(import_list.begin(), import_list.end());
593 2115701 : }
594 : }
595 :
596 : // Update with family specific secondary tables.
597 515634 : table->UpdateSecondaryTablesForReplication(rt, &secondary_tables);
598 :
599 : // Skip if we don't need to replicate the path to any tables.
600 515633 : if (secondary_tables.empty())
601 29858 : continue;
602 :
603 : // Add OriginVn when replicating self-originated routes from a VRF.
604 406497 : if (!vn_index && !rtinstance->IsMasterRoutingInstance() &&
605 892269 : path->IsVrfOriginated() && rtinstance->virtual_network_index()) {
606 47329 : vn_index = rtinstance->virtual_network_index();
607 47328 : extcomm_ptr = UpdateOriginVn(server_, extcomm_ptr.get(), vn_index);
608 : }
609 :
610 485777 : if (route_unchanged){
611 141 : addedTables = RtGroup::RtGroupMemberList();
612 : // Finding the difference between sets, secondary_tables and
613 : // previousTables
614 141 : std::set_difference(secondary_tables.begin(), secondary_tables.end(),
615 : previousTables.begin(), previousTables.end(),
616 : std::inserter(addedTables, addedTables.end()));
617 141 : std::set_difference(previousTables.begin(), previousTables.end(),
618 : secondary_tables.begin(), secondary_tables.end(),
619 : std::inserter(deletedTables, deletedTables.end()));
620 461 : BOOST_FOREACH (RtReplicated::SecondaryRouteInfo path,
621 : dbstate->GetList()){
622 160 : if (deletedTables.find(path.table_)!=deletedTables.end()) {
623 1 : replicated_path_list.erase(path);
624 : }
625 : }
626 : //Updating the secondary tables with the difference.
627 141 : secondary_tables = addedTables;
628 :
629 : }
630 :
631 :
632 : // Replicate path to all destination tables.
633 3721494 : BOOST_FOREACH(BgpTable *dest, secondary_tables) {
634 : // Skip if destination is same as source table.
635 1617783 : if (dest == table)
636 668637 : continue;
637 :
638 1132131 : const RoutingInstance *dest_rtinstance = dest->routing_instance();
639 1132129 : ExtCommunityPtr new_extcomm_ptr = extcomm_ptr;
640 :
641 : // If the origin vn is unresolved, see if route has a RouteTarget
642 : // that's in the set of export RouteTargets for the dest instance.
643 : // If so, we set the origin vn for the replicated route to be the
644 : // vn for the dest instance.
645 1132857 : if (!vn_index && dest_rtinstance->virtual_network_index() &&
646 468 : dest_rtinstance->HasExportTarget(ext_community)) {
647 368 : int dest_vn_index = dest_rtinstance->virtual_network_index();
648 736 : new_extcomm_ptr = UpdateOriginVn(server_, extcomm_ptr.get(),
649 368 : dest_vn_index);
650 : }
651 :
652 : // Replicate the route to the destination table. The destination
653 : // table may decide to not replicate based on it's own policy e.g.
654 : // multicast routes are never leaked across routing-instances.
655 1132380 : BgpRoute *replicated_rt = dest->RouteReplicate(
656 : server_, table, rt, path, new_extcomm_ptr);
657 1132377 : if (!replicated_rt)
658 182993 : continue;
659 :
660 : // Add information about the secondary path to the replicated path
661 : // list.
662 949370 : RtReplicated::SecondaryRouteInfo rtinfo(dest, path->GetPeer(),
663 1898753 : path->GetPathId(), path->GetSource(), replicated_rt);
664 949362 : pair<RtReplicated::ReplicatedRtPathList::iterator, bool> result;
665 949343 : result = replicated_path_list.insert(rtinfo);
666 : // Assert if the insertion to replication path list fails
667 949371 : if (!route_unchanged)
668 949370 : assert(result.second);
669 949371 : RPR_TRACE_ONLY(Replicate, table->name(), rt->ToString(),
670 : path->ToString(),
671 : BgpPath::PathIdString(path->GetPathId()),
672 : dest->name(), replicated_rt->ToString());
673 1132357 : }
674 546572 : }
675 :
676 : // Update the DBState to reflect the new list of secondary paths. The
677 : // DBState will get cleared if the list is empty.
678 1259636 : DBStateSync(table, ts, rt, dbstate, &replicated_path_list);
679 1260129 : return true;
680 1977476 : }
681 173 : const RtReplicated *RoutePathReplicator::GetReplicationState(
682 : BgpTable *table, BgpRoute *rt) const {
683 173 : const TableState *ts = FindTableState(table);
684 172 : if (!ts)
685 0 : return NULL;
686 : RtReplicated *dbstate =
687 172 : static_cast<RtReplicated *>(rt->GetState(table, ts->listener_id()));
688 173 : return dbstate;
689 : }
690 :
691 : //
692 : // Return the list of secondary table names for the given primary path.
693 : //
694 5471 : vector<string> RoutePathReplicator::GetReplicatedTableNameList(
695 : const BgpTable *table, const BgpRoute *rt, const BgpPath *path) const {
696 5471 : const TableState *ts = FindTableState(table);
697 5463 : if (!ts)
698 5159 : return vector<string>();
699 : const RtReplicated *dbstate = static_cast<const RtReplicated *>(
700 304 : rt->GetState(table, ts->listener_id()));
701 304 : if (!dbstate)
702 21 : return vector<string>();
703 283 : return dbstate->GetTableNameList(path);
704 : }
705 :
706 516781 : void RoutePathReplicator::DeleteSecondaryPath(BgpTable *table, BgpRoute *rt,
707 : const RtReplicated::SecondaryRouteInfo &rtinfo) {
708 516781 : BgpRoute *rt_secondary = rtinfo.rt_;
709 516781 : BgpTable *secondary_table = rtinfo.table_;
710 516781 : const IPeer *peer = rtinfo.peer_;
711 516781 : uint32_t path_id = rtinfo.path_id_;
712 516781 : BgpPath::PathSource src = rtinfo.src_;
713 :
714 : DBTablePartBase *partition =
715 516781 : secondary_table->GetTablePartition(rt_secondary);
716 516590 : BgpPath *secondary_path = rt_secondary->FindSecondaryPath(rt, src,
717 : peer, path_id);
718 516913 : if (secondary_path && secondary_path->NeedsResolution()) {
719 0 : secondary_table->path_resolver()->StopPathResolution(partition->index(),
720 : secondary_path);
721 : }
722 516880 : assert(rt_secondary->RemoveSecondaryPath(rt, src, peer, path_id));
723 517328 : if (rt_secondary->count() == 0) {
724 493280 : RPR_TRACE_ONLY(Flush, secondary_table->name(), rt_secondary->ToString(),
725 : peer ? peer->ToString() : "Nil",
726 : BgpPath::PathIdString(path_id), table->name(),
727 : rt->ToString(), "Delete");
728 493436 : partition->Delete(rt_secondary);
729 : } else {
730 23833 : partition->Notify(rt_secondary);
731 23848 : RPR_TRACE_ONLY(Flush, secondary_table->name(), rt_secondary->ToString(),
732 : peer ? peer->ToString() : "Nil",
733 : BgpPath::PathIdString(path_id), table->name(),
734 : rt->ToString(), "Path update");
735 : }
736 517009 : }
737 :
738 0 : string RtReplicated::SecondaryRouteInfo::ToString() const {
739 0 : ostringstream out;
740 0 : out << table_->name() << "(" << table_ << ")" << ":" <<
741 0 : peer_->ToString() << "(" << peer_ << ")" << ":" <<
742 0 : rt_->ToString() << "(" << rt_ << ")";
743 0 : return out.str();
744 0 : }
|