Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 : #include "bgp/routing-instance/rtarget_group_mgr.h"
5 :
6 : #include <boost/foreach.hpp>
7 :
8 : #include <utility>
9 :
10 : #include "base/map_util.h"
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_peer.h"
16 : #include "bgp/bgp_ribout.h"
17 : #include "bgp/bgp_server.h"
18 : #include "bgp/bgp_table.h"
19 : #include "bgp/routing-instance/routing_instance.h"
20 : #include "bgp/rtarget/rtarget_route.h"
21 :
22 : using std::pair;
23 :
24 498800 : void VpnRouteState::AddRouteTarget(RTargetGroupMgr *mgr, int part_id,
25 : BgpRoute *rt, RTargetList::const_iterator it) {
26 498800 : pair<RTargetList::iterator, bool> result;
27 498802 : result = list_.insert(*it);
28 498928 : assert(result.second);
29 498928 : RtGroup *rtgroup = mgr->LocateRtGroup(*it);
30 498972 : rtgroup->AddDepRoute(part_id, rt);
31 498964 : }
32 :
33 498705 : void VpnRouteState::DeleteRouteTarget(RTargetGroupMgr *mgr, int part_id,
34 : BgpRoute *rt, RTargetList::const_iterator it) {
35 498705 : RtGroup *rtgroup = mgr->GetRtGroup(*it);
36 498969 : rtgroup->RemoveDepRoute(part_id, rt);
37 498907 : mgr->RemoveRtGroup(*it);
38 498951 : list_.erase(it);
39 498928 : }
40 :
41 27078 : void RTargetState::AddInterestedPeer(RTargetGroupMgr *mgr, RtGroup *rtgroup,
42 : RTargetRoute *rt, RtGroup::InterestedPeerList::const_iterator it) {
43 27078 : pair<RtGroup::InterestedPeerList::iterator, bool> result;
44 27078 : result = list_.insert(*it);
45 27078 : assert(result.second);
46 27078 : rtgroup->AddInterestedPeer(it->first, rt);
47 27078 : mgr->NotifyRtGroupUnlocked(rtgroup->rt());
48 27078 : }
49 :
50 27078 : void RTargetState::DeleteInterestedPeer(RTargetGroupMgr *mgr, RtGroup *rtgroup,
51 : RTargetRoute *rt, RtGroup::InterestedPeerList::iterator it) {
52 27078 : rtgroup->RemoveInterestedPeer(it->first, rt);
53 27078 : mgr->NotifyRtGroupUnlocked(rtgroup->rt());
54 27078 : list_.erase(it);
55 27078 : }
56 :
57 9743 : RTargetGroupMgr::RTargetGroupMgr(BgpServer *server) : server_(server),
58 19486 : rtarget_route_trigger_(new TaskTrigger(
59 : boost::bind(&RTargetGroupMgr::ProcessRTargetRouteList, this),
60 19486 : TaskScheduler::GetInstance()->GetTaskId("bgp::RTFilter"), 0)),
61 19486 : remove_rtgroup_trigger_(new TaskTrigger(
62 : boost::bind(&RTargetGroupMgr::ProcessRtGroupList, this),
63 19486 : TaskScheduler::GetInstance()->GetTaskId("bgp::RTFilter"), 0)),
64 9743 : rtarget_trigger_lists_(DB::PartitionCount()),
65 29229 : master_instance_delete_ref_(this, NULL) {
66 48190 : for (int i = 0; i < DB::PartitionCount(); i++) {
67 115341 : rtarget_dep_triggers_.push_back(boost::shared_ptr<TaskTrigger>(new
68 : TaskTrigger(boost::bind(&RTargetGroupMgr::ProcessRouteTargetList,
69 : this, i),
70 76894 : TaskScheduler::GetInstance()->GetTaskId("db::DBTable"), i)));
71 : }
72 9743 : }
73 :
74 102490 : void RTargetGroupMgr::RTargetPeerSync(BgpTable *table, RTargetRoute *rt,
75 : DBTableBase::ListenerId id, RTargetState *dbstate,
76 : const RtGroup::InterestedPeerList *future) {
77 102490 : CHECK_CONCURRENCY("bgp::RTFilter");
78 :
79 102490 : RouteTarget rtarget = rt->GetPrefix().rtarget();
80 102490 : RtGroup *rtgroup = LocateRtGroup(rtarget);
81 102490 : assert(rtgroup);
82 :
83 102490 : map_synchronize(dbstate->GetMutableList(), future,
84 : boost::bind(&RTargetState::AddInterestedPeer, dbstate, this, rtgroup,
85 : rt, _1),
86 : boost::bind(&RTargetState::DeleteInterestedPeer, dbstate, this, rtgroup,
87 : rt, _1));
88 :
89 102490 : if (dbstate->GetList()->empty()) {
90 64490 : rt->ClearState(table, id);
91 64490 : delete dbstate;
92 64490 : RemoveRtGroup(rtarget);
93 : }
94 102490 : }
95 :
96 102490 : void RTargetGroupMgr::BuildRTargetDistributionGraph(BgpTable *table,
97 : RTargetRoute *rt, DBTableBase::ListenerId id) {
98 102490 : CHECK_CONCURRENCY("bgp::RTFilter");
99 :
100 : RTargetState *dbstate =
101 102490 : static_cast<RTargetState *>(rt->GetState(table, id));
102 :
103 102490 : RtGroup::InterestedPeerList peer_list;
104 :
105 179178 : if (rt->IsDeleted() || !rt->BestPath() ||
106 76688 : !rt->BestPath()->IsFeasible()) {
107 25802 : RTargetPeerSync(table, rt, id, dbstate, &peer_list);
108 25802 : return;
109 : }
110 :
111 76688 : const BgpPath *best_ebgp_path = NULL;
112 223082 : for (Route::PathList::iterator it = rt->GetPathList().begin();
113 446164 : it != rt->GetPathList().end(); it++) {
114 146394 : BgpPath *path = static_cast<BgpPath *>(it.operator->());
115 146394 : if (!path->IsFeasible())
116 0 : break;
117 146394 : if (!path->GetPeer() || path->GetPeer()->IsXmppPeer())
118 107129 : continue;
119 :
120 39333 : const BgpPeer *peer = static_cast<const BgpPeer *>(path->GetPeer());
121 39333 : if (peer->PeerType() == BgpProto::EBGP) {
122 17316 : if (!best_ebgp_path) {
123 17139 : best_ebgp_path = path;
124 177 : } else if (!best_ebgp_path->PathSameNeighborAs(*path)) {
125 68 : continue;
126 : }
127 : }
128 :
129 : std::pair<RtGroup::InterestedPeerList::iterator, bool> ret =
130 39265 : peer_list.insert(std::pair<const BgpPeer *,
131 78530 : RtGroup::RTargetRouteList>(peer, RtGroup::RTargetRouteList()));
132 39265 : assert(ret.second);
133 39265 : ret.first->second.insert(rt);
134 : }
135 :
136 76688 : RTargetPeerSync(table, rt, id, dbstate, &peer_list);
137 102490 : }
138 :
139 159466 : bool RTargetGroupMgr::ProcessRouteTargetList(int part_id) {
140 159466 : CHECK_CONCURRENCY("db::DBTable");
141 :
142 752033 : BOOST_FOREACH(const RouteTarget &rtarget, rtarget_trigger_lists_[part_id]) {
143 295642 : RtGroup *rtgroup = GetRtGroup(rtarget);
144 298030 : if (!rtgroup)
145 27794 : continue;
146 270236 : rtgroup->NotifyDepRoutes(part_id);
147 : }
148 :
149 159766 : rtarget_trigger_lists_[part_id].clear();
150 159803 : return true;
151 : }
152 :
153 186531 : void RTargetGroupMgr::AddRouteTargetToLists(const RouteTarget &rtarget) {
154 915795 : for (int idx = 0; idx < DB::PartitionCount(); ++idx) {
155 729264 : rtarget_trigger_lists_[idx].insert(rtarget);
156 729264 : rtarget_dep_triggers_[idx]->Set();
157 : }
158 186531 : }
159 :
160 8 : void RTargetGroupMgr::DisableRouteTargetProcessing() {
161 40 : for (int idx = 0; idx < DB::PartitionCount(); ++idx) {
162 32 : rtarget_dep_triggers_[idx]->set_disable();
163 : }
164 8 : }
165 :
166 8 : void RTargetGroupMgr::EnableRouteTargetProcessing() {
167 40 : for (int idx = 0; idx < DB::PartitionCount(); ++idx) {
168 32 : rtarget_dep_triggers_[idx]->set_enable();
169 : }
170 8 : }
171 :
172 22 : bool RTargetGroupMgr::IsRouteTargetOnList(const RouteTarget &rtarget) const {
173 66 : for (int idx = 0; idx < DB::PartitionCount(); ++idx) {
174 55 : if (rtarget_trigger_lists_[idx].find(rtarget) !=
175 110 : rtarget_trigger_lists_[idx].end()) {
176 11 : return true;
177 : }
178 : }
179 11 : return false;
180 : }
181 :
182 40179 : bool RTargetGroupMgr::ProcessRTargetRouteList() {
183 40179 : CHECK_CONCURRENCY("bgp::RTFilter");
184 :
185 40179 : RoutingInstanceMgr *mgr = server()->routing_instance_mgr();
186 40179 : RoutingInstance *master = mgr->GetDefaultRoutingInstance();
187 40179 : BgpTable *table = master->GetTable(Address::RTARGET);
188 :
189 : // Get the Listener id
190 40179 : DBTableBase::ListenerId id = GetListenerId(table);
191 :
192 40179 : for (RTargetRouteTriggerList::iterator it = rtarget_route_list_.begin();
193 142669 : it != rtarget_route_list_.end(); it++) {
194 102490 : BuildRTargetDistributionGraph(table, *it, id);
195 : }
196 :
197 40179 : rtarget_route_list_.clear();
198 40179 : return true;
199 : }
200 :
201 8 : void RTargetGroupMgr::DisableRTargetRouteProcessing() {
202 8 : rtarget_route_trigger_->set_disable();
203 8 : }
204 :
205 8 : void RTargetGroupMgr::EnableRTargetRouteProcessing() {
206 8 : rtarget_route_trigger_->set_enable();
207 8 : }
208 :
209 19 : bool RTargetGroupMgr::IsRTargetRouteOnList(RTargetRoute *rt) const {
210 19 : return rtarget_route_list_.find(rt) != rtarget_route_list_.end();
211 : }
212 :
213 6510 : void RTargetGroupMgr::Initialize() {
214 6510 : assert(table_state_.empty());
215 6510 : RoutingInstanceMgr *mgr = server()->routing_instance_mgr();
216 6510 : RoutingInstance *master = mgr->GetDefaultRoutingInstance();
217 6510 : assert(master);
218 :
219 6510 : master_instance_delete_ref_.Reset(master->deleter());
220 :
221 6510 : RoutingInstance::RouteTableList const table_list = master->GetTables();
222 : DBTableBase::ListenerId id;
223 6510 : RtGroupMgrTableState *ts = NULL;
224 6510 : for (RoutingInstance::RouteTableList::const_iterator it =
225 71610 : table_list.begin(); it != table_list.end(); ++it) {
226 58590 : if (!it->second->IsVpnTable()) continue;
227 :
228 32550 : BgpTable *vpntable = it->second;
229 32550 : id = vpntable->Register(
230 : boost::bind(&RTargetGroupMgr::VpnRouteNotify, this, _1, _2),
231 : "RTargetGroupMgr");
232 32550 : ts = new RtGroupMgrTableState(vpntable, id);
233 32550 : table_state_.insert(std::make_pair(vpntable, ts));
234 : }
235 :
236 6510 : BgpTable *rttable = master->GetTable(Address::RTARGET);
237 6510 : id = rttable->Register(
238 : boost::bind(&RTargetGroupMgr::RTargetRouteNotify, this, _1, _2),
239 : "RTargetGroupMgr");
240 6510 : ts = new RtGroupMgrTableState(rttable, id);
241 6510 : table_state_.insert(std::make_pair(rttable, ts));
242 6510 : }
243 :
244 6510 : void RTargetGroupMgr::ManagedDelete() {
245 6510 : if (rtgroup_map_.empty()) remove_rtgroup_trigger_->Set();
246 6510 : }
247 :
248 : void
249 584638 : RTargetGroupMgr::RTargetDepSync(DBTablePartBase *root, BgpRoute *rt,
250 : DBTableBase::ListenerId id,
251 : VpnRouteState *dbstate,
252 : const VpnRouteState::RTargetList *future) {
253 584638 : CHECK_CONCURRENCY("db::DBTable");
254 :
255 584490 : BgpTable *table = static_cast<BgpTable *>(root->parent());
256 584488 : if (!dbstate) {
257 184604 : dbstate = new VpnRouteState();
258 184603 : rt->SetState(table, id, dbstate);
259 : }
260 :
261 584501 : int part_id = root->index();
262 584496 : set_synchronize(dbstate->GetMutableList(), future,
263 : boost::bind(
264 : &VpnRouteState::AddRouteTarget, dbstate, this, part_id, rt, _1),
265 : boost::bind(
266 : &VpnRouteState::DeleteRouteTarget, dbstate, this, part_id, rt, _1));
267 :
268 584529 : if (dbstate->GetList()->empty()) {
269 184620 : rt->ClearState(root->parent(), id);
270 184622 : delete dbstate;
271 : }
272 584553 : }
273 :
274 743124 : DBTableBase::ListenerId RTargetGroupMgr::GetListenerId(BgpTable *table) {
275 743124 : RtGroupMgrTableStateList::iterator loc = table_state_.find(table);
276 743029 : assert(loc != table_state_.end());
277 743002 : RtGroupMgrTableState *ts = loc->second;
278 743005 : DBTableBase::ListenerId id = ts->GetListenerId();
279 742981 : assert(id != DBTableBase::kInvalidId);
280 742981 : return id;
281 : }
282 :
283 584658 : bool RTargetGroupMgr::VpnRouteNotify(DBTablePartBase *root,
284 : DBEntryBase *entry) {
285 584658 : CHECK_CONCURRENCY("db::DBTable");
286 :
287 584554 : BgpTable *table = static_cast<BgpTable *>(root->parent());
288 584551 : BgpRoute *rt = static_cast<BgpRoute *>(entry);
289 : // Get the Listener id
290 584551 : DBTableBase::ListenerId id = GetListenerId(table);
291 :
292 : // Get the dbstate
293 : VpnRouteState *dbstate =
294 584434 : static_cast<VpnRouteState *>(rt->GetState(table, id));
295 :
296 584874 : VpnRouteState::RTargetList list;
297 :
298 984781 : if (entry->IsDeleted() || !rt->BestPath() ||
299 400028 : !rt->BestPath()->IsFeasible()) {
300 184707 : if (!dbstate)
301 122 : return true;
302 184585 : RTargetDepSync(root, rt, id, dbstate, &list);
303 184601 : return true;
304 : }
305 :
306 400013 : const BgpPath *path = rt->BestPath();
307 399971 : const BgpAttr *attr = path->GetAttr();
308 399974 : const ExtCommunity *ext_community = attr->ext_community();
309 :
310 399971 : if (ext_community) {
311 : // Gather all Route Target
312 7414720 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &comm,
313 : ext_community->communities()) {
314 3507032 : if (ExtCommunity::is_route_target(comm)) {
315 2852462 : list.insert(RouteTarget(comm));
316 : }
317 : }
318 : }
319 :
320 400057 : RTargetDepSync(root, rt, id, dbstate, &list);
321 399951 : return true;
322 584674 : }
323 :
324 118420 : bool RTargetGroupMgr::RTargetRouteNotify(DBTablePartBase *root,
325 : DBEntryBase *entry) {
326 118420 : CHECK_CONCURRENCY("db::DBTable");
327 :
328 118420 : BgpTable *table = static_cast<BgpTable *>(root->parent());
329 118420 : RTargetRoute *rt = static_cast<RTargetRoute *>(entry);
330 : // Get the Listener id
331 118420 : DBTableBase::ListenerId id = GetListenerId(table);
332 :
333 : // Get the dbstate
334 : RTargetState *dbstate =
335 118420 : static_cast<RTargetState *>(rt->GetState(table, id));
336 :
337 118420 : if (!dbstate) {
338 78832 : if (rt->IsDeleted()) return true;
339 64490 : dbstate = new RTargetState();
340 64490 : rt->SetState(table, id, dbstate);
341 : }
342 104078 : if (rtarget_route_list_.empty())
343 40179 : rtarget_route_trigger_->Set();
344 104078 : rtarget_route_list_.insert(rt);
345 104078 : return true;
346 : }
347 :
348 18648 : RTargetGroupMgr::~RTargetGroupMgr() {
349 9743 : assert(rtgroup_map_.empty());
350 18648 : }
351 :
352 : // Search a RtGroup
353 6403216 : RtGroup *RTargetGroupMgr::GetRtGroup(const RouteTarget &rt) {
354 6403216 : std::scoped_lock lock(mutex_);
355 6407617 : RtGroupMap::iterator loc = rtgroup_map_.find(rt);
356 6406791 : if (loc != rtgroup_map_.end()) {
357 6061103 : return loc->second;
358 : }
359 345592 : return NULL;
360 6406619 : }
361 :
362 : // Search a RtGroup
363 4364635 : RtGroup *RTargetGroupMgr::GetRtGroup(const ExtCommunity::ExtCommunityValue
364 : &community) {
365 4364635 : RouteTarget rt(community);
366 8730685 : return GetRtGroup(rt);
367 : }
368 :
369 1529653 : RtGroup *RTargetGroupMgr::LocateRtGroup(const RouteTarget &rt) {
370 1529653 : std::scoped_lock lock(mutex_);
371 1529995 : RtGroupMap::iterator loc = rtgroup_map_.find(rt);
372 1529971 : RtGroup *group = (loc != rtgroup_map_.end()) ? loc->second : NULL;
373 1529962 : if (group == NULL) {
374 108895 : group = new RtGroup(rt);
375 108895 : rtgroup_map_.insert(rt, group);
376 : }
377 1530001 : return group;
378 1529962 : }
379 :
380 186531 : void RTargetGroupMgr::NotifyRtGroupUnlocked(const RouteTarget &rt) {
381 186531 : CHECK_CONCURRENCY("bgp::RTFilter", "bgp::Config", "bgp::ConfigHelper");
382 :
383 186531 : AddRouteTargetToLists(rt);
384 186531 : if (!rt.IsNull())
385 186515 : return;
386 :
387 16 : for (RtGroupMgrTableStateList::iterator it = table_state_.begin();
388 112 : it != table_state_.end(); ++it) {
389 96 : BgpTable *table = it->first;
390 96 : if (!table->IsVpnTable())
391 16 : continue;
392 80 : table->NotifyAllEntries();
393 : }
394 : }
395 :
396 132375 : void RTargetGroupMgr::NotifyRtGroup(const RouteTarget &rt) {
397 132375 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper");
398 132375 : std::scoped_lock lock(mutex_);
399 132375 : NotifyRtGroupUnlocked(rt);
400 132375 : }
401 :
402 993287 : void RTargetGroupMgr::RemoveRtGroup(const RouteTarget &rt) {
403 993287 : std::scoped_lock lock(mutex_);
404 993352 : RtGroupMap::iterator loc = rtgroup_map_.find(rt);
405 993323 : RtGroup *rtgroup = (loc != rtgroup_map_.end()) ? loc->second : NULL;
406 993307 : assert(rtgroup);
407 :
408 993307 : rtgroup_remove_list_.insert(rtgroup);
409 993372 : remove_rtgroup_trigger_->Set();
410 993367 : }
411 :
412 315791 : void RTargetGroupMgr::GetRibOutInterestedPeers(RibOut *ribout,
413 : const ExtCommunity *ext_community,
414 : const RibPeerSet &peerset, RibPeerSet *new_peerset) {
415 315791 : RtGroupInterestedPeerSet peer_set;
416 315796 : RtGroup *null_rtgroup = GetRtGroup(RouteTarget::null_rtarget);
417 316067 : if (null_rtgroup) peer_set = null_rtgroup->GetInterestedPeers();
418 6076777 : BOOST_FOREACH(const ExtCommunity::ExtCommunityValue &comm,
419 : ext_community->communities()) {
420 2880225 : if (ExtCommunity::is_route_target(comm)) {
421 2231179 : RtGroup *rtgroup = GetRtGroup(comm);
422 2231586 : if (!rtgroup) continue;
423 2231586 : peer_set |= rtgroup->GetInterestedPeers();
424 : }
425 : }
426 316058 : RibOut::PeerIterator iter(ribout, peerset);
427 635815 : while (iter.HasNext()) {
428 319750 : int current_index = iter.index();
429 319770 : IPeerUpdate *peer = iter.Next();
430 319759 : BgpPeer *tmp = dynamic_cast<BgpPeer *>(peer);
431 319759 : assert(tmp);
432 319759 : if (tmp->IsFamilyNegotiated(Address::RTARGET)) {
433 311755 : if (!peer_set.test(tmp->GetIndex())) {
434 20559 : new_peerset->reset(current_index);
435 : }
436 : }
437 : }
438 316036 : }
439 :
440 11015 : void RTargetGroupMgr::UnregisterTables() {
441 11015 : CHECK_CONCURRENCY("bgp::RTFilter");
442 :
443 11015 : if (rtgroup_map_.empty()) {
444 11015 : RoutingInstanceMgr *mgr = server()->routing_instance_mgr();
445 11015 : RoutingInstance *master = mgr->GetDefaultRoutingInstance();
446 11015 : if (master && master->deleted()) {
447 7963 : for (RtGroupMgrTableStateList::iterator it =
448 54986 : table_state_.begin(), itnext; it != table_state_.end();
449 39060 : it = itnext) {
450 39060 : itnext = it;
451 39060 : itnext++;
452 39060 : RtGroupMgrTableState *ts = it->second;
453 39060 : DBTableBase::ListenerId id = ts->GetListenerId();
454 39060 : BgpTable *bgptable = it->first;
455 39060 : bgptable->Unregister(id);
456 39060 : table_state_.erase(it);
457 39060 : delete ts;
458 : }
459 7963 : master_instance_delete_ref_.Reset(NULL);
460 : }
461 : }
462 11015 : }
463 :
464 140711 : bool RTargetGroupMgr::ProcessRtGroupList() {
465 140711 : CHECK_CONCURRENCY("bgp::RTFilter");
466 791985 : BOOST_FOREACH(RtGroup *rtgroup, rtgroup_remove_list_) {
467 325637 : if (!rtgroup->MayDelete())
468 216742 : continue;
469 108895 : RouteTarget rt = rtgroup->rt();
470 108895 : rtgroup_map_.erase(rt);
471 : }
472 140711 : rtgroup_remove_list_.clear();
473 :
474 140711 : if (rtgroup_map_.empty()) UnregisterTables();
475 :
476 140711 : return true;
477 : }
478 :
479 3 : void RTargetGroupMgr::DisableRtGroupProcessing() {
480 3 : remove_rtgroup_trigger_->set_disable();
481 3 : }
482 :
483 3 : void RTargetGroupMgr::EnableRtGroupProcessing() {
484 3 : remove_rtgroup_trigger_->set_enable();
485 3 : }
486 :
487 12 : bool RTargetGroupMgr::IsRtGroupOnList(RtGroup *rtgroup) const {
488 12 : return rtgroup_remove_list_.find(rtgroup) != rtgroup_remove_list_.end();
489 : }
490 :
491 39060 : RtGroupMgrTableState::RtGroupMgrTableState(BgpTable *table,
492 39060 : DBTableBase::ListenerId id)
493 39060 : : id_(id), table_delete_ref_(this, table->deleter()) {
494 39060 : assert(table->deleter() != NULL);
495 39060 : }
496 :
497 39060 : RtGroupMgrTableState::~RtGroupMgrTableState() {
498 39060 : }
499 :
500 34494 : void RtGroupMgrTableState::ManagedDelete() {
501 34494 : }
|