Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include "bgp/bgp_multicast.h"
6 :
7 : #include <boost/bind/bind.hpp>
8 : #include <boost/foreach.hpp>
9 :
10 : #include "base/string_util.h"
11 : #include "base/task_annotations.h"
12 : #include "bgp/bgp_log.h"
13 : #include "bgp/bgp_mvpn.h"
14 : #include "bgp/bgp_server.h"
15 : #include "bgp/bgp_update.h"
16 : #include "bgp/ermvpn/ermvpn_table.h"
17 : #include "bgp/mvpn/mvpn_route.h"
18 : #include "bgp/routing-instance/routing_instance.h"
19 : #include "bgp/routing-instance/routing_instance_analytics_types.h"
20 : #include "bgp/routing-instance/routing_instance_log.h"
21 : #include "bgp/tunnel_encap/tunnel_encap.h"
22 :
23 : using std::string;
24 : using std::vector;
25 : using namespace boost::placeholders;
26 :
27 : class McastTreeManager::DeleteActor : public LifetimeActor {
28 : public:
29 42902 : explicit DeleteActor(McastTreeManager *tree_manager)
30 42902 : : LifetimeActor(tree_manager->table_->routing_instance()->server()->
31 : lifetime_manager()),
32 42902 : tree_manager_(tree_manager) {
33 42904 : }
34 85808 : virtual ~DeleteActor() {
35 85808 : }
36 :
37 42904 : virtual bool MayDelete() const {
38 42904 : return tree_manager_->MayDelete();
39 : }
40 :
41 42904 : virtual void Shutdown() {
42 42904 : tree_manager_->Shutdown();
43 42904 : }
44 :
45 42904 : virtual void Destroy() {
46 42904 : tree_manager_->table_->DestroyTreeManager();
47 42904 : }
48 :
49 : private:
50 : McastTreeManager *tree_manager_;
51 : };
52 :
53 : //
54 : // Constructor for McastForwarder. The level is determined by the route type.
55 : // We get the address of the forwarder and the label_block from the attributes
56 : // of the active path. The LabelBLockPtr needs to be copied so that we can
57 : // release the label when processing a delete notification - we won't have the
58 : // path at that point.
59 : //
60 : // The RD will be zero for BGP learnt routes and the RouterId will be zero for
61 : // XMPP learnt routes.
62 : //
63 10713 : McastForwarder::McastForwarder(McastSGEntry *sg_entry, ErmVpnRoute *route)
64 10713 : : sg_entry_(sg_entry),
65 10713 : route_(route),
66 10713 : global_tree_route_(NULL),
67 10713 : label_(0),
68 10713 : address_(0),
69 10713 : rd_(route->GetPrefix().route_distinguisher()),
70 21426 : router_id_(route->GetPrefix().router_id()) {
71 10713 : const BgpPath *path = route->BestPath();
72 10713 : const BgpAttr *attr = path->GetAttr();
73 :
74 10713 : if (route_->GetPrefix().type() == ErmVpnPrefix::NativeRoute) {
75 5506 : level_ = McastTreeManager::LevelNative;
76 5506 : address_ = attr->nexthop().to_v4();
77 5506 : label_block_ = attr->label_block();
78 : } else {
79 5207 : level_ = McastTreeManager::LevelLocal;
80 5207 : const EdgeDiscovery::Edge *edge = attr->edge_discovery()->edge_list[0];
81 5207 : address_ = edge->address;
82 5207 : label_block_ = edge->label_block;
83 : }
84 :
85 10713 : if (path->GetAttr()->ext_community())
86 5525 : encap_ = path->GetAttr()->ext_community()->GetTunnelEncap();
87 10713 : }
88 :
89 : //
90 : // Destructor for McastForwarder. Flushes forward and reverse links to and
91 : // from other McastForwarders.
92 : //
93 12635 : McastForwarder::~McastForwarder() {
94 10713 : DeleteGlobalTreeRoute();
95 10713 : FlushLinks();
96 10713 : ReleaseLabel();
97 12635 : }
98 :
99 : //
100 : // Update the McastForwarder based on information in the ErmVpnRoute.
101 : // Return true if something changed.
102 : //
103 8791 : bool McastForwarder::Update(ErmVpnRoute *route) {
104 8791 : McastForwarder forwarder(sg_entry_, route);
105 :
106 8791 : bool changed = false;
107 8791 : if (label_block_ != forwarder.label_block_) {
108 1128 : ReleaseLabel();
109 1128 : label_block_ = forwarder.label_block_;
110 1128 : changed = true;
111 : }
112 8791 : if (address_ != forwarder.address_) {
113 500 : address_ = forwarder.address_;
114 500 : changed = true;
115 : }
116 8791 : if (encap_ != forwarder.encap_) {
117 11 : encap_ = forwarder.encap_;
118 11 : changed = true;
119 : }
120 :
121 8791 : return changed;
122 8791 : }
123 :
124 : //
125 : // Printable string for McastForwarder.
126 : //
127 2792 : std::string McastForwarder::ToString() const {
128 2792 : if (level_ == McastTreeManager::LevelNative) {
129 5584 : return rd_.ToString() + " -> " + integerToString(label_);
130 : } else {
131 0 : return router_id_.to_string() + " -> " + integerToString(label_);
132 : }
133 : }
134 :
135 : //
136 : // Find a link to the given McastForwarder.
137 : //
138 4712 : McastForwarder *McastForwarder::FindLink(McastForwarder *forwarder) {
139 4712 : for (McastForwarderList::iterator it = tree_links_.begin();
140 7966 : it != tree_links_.end(); ++it) {
141 3254 : if (*it == forwarder) return forwarder;
142 : }
143 4712 : return NULL;
144 : }
145 :
146 : //
147 : // Add a link to the given McastForwarder.
148 : //
149 4712 : void McastForwarder::AddLink(McastForwarder *forwarder) {
150 4712 : assert(!FindLink(forwarder));
151 4712 : tree_links_.push_back(forwarder);
152 4712 : }
153 :
154 : //
155 : // Remove a link to the given McastForwarder.
156 : //
157 2356 : void McastForwarder::RemoveLink(McastForwarder *forwarder) {
158 2356 : for (McastForwarderList::iterator it = tree_links_.begin();
159 2494 : it != tree_links_.end(); ++it) {
160 2494 : if (*it == forwarder) {
161 2356 : tree_links_.erase(it);
162 2356 : return;
163 : }
164 : }
165 : }
166 :
167 : //
168 : // Flush all links from this McastForwarder. Takes care of removing the
169 : // reverse links as well.
170 : //
171 16938 : void McastForwarder::FlushLinks() {
172 16938 : for (McastForwarderList::iterator it = tree_links_.begin();
173 19294 : it != tree_links_.end(); ++it) {
174 2356 : (*it)->RemoveLink(this);
175 : }
176 16938 : tree_links_.clear();
177 16938 : }
178 :
179 : //
180 : // Allocate a label for this McastForwarder. The label gets allocated from
181 : // the LabelBlock corresponding to the label range advertised by the peer.
182 : // This is used when updating the distribution tree for the McastSGEntry to
183 : // this McastForwarder belongs.
184 : //
185 4208 : void McastForwarder::AllocateLabel() {
186 4208 : label_ = label_block_->AllocateLabel();
187 4208 : }
188 :
189 : //
190 : // Release the label, if any, for this McastForwarder. This is required when
191 : // updating the distribution tree for the McastSGEntry to which we belong.
192 : //
193 18066 : void McastForwarder::ReleaseLabel() {
194 18066 : if (label_ != 0) {
195 4197 : label_block_->ReleaseLabel(label_);
196 4197 : label_ = 0;
197 : }
198 18066 : }
199 :
200 : //
201 : // Add the GlobalTreeRoute for this McastForwarder. The GlobalTreeRoute is
202 : // used by the tree builder to tell the associated control-node about the
203 : // forwarding edges for Native McastForwarders attached it.
204 : //
205 3486 : void McastForwarder::AddGlobalTreeRoute() {
206 3486 : assert(level_ == McastTreeManager::LevelLocal);
207 3486 : assert(!global_tree_route_);
208 :
209 : // Bail if there's no label allocated.
210 3486 : if (label_ == 0)
211 2119 : return;
212 :
213 : // Bail if we can't build a source RD.
214 1469 : if (sg_entry_->GetSourceRd().IsZero())
215 102 : return;
216 :
217 : // Construct the prefix and route key.
218 1367 : BgpTable *table = static_cast<BgpTable *>(route_->get_table());
219 : ErmVpnPrefix prefix(ErmVpnPrefix::GlobalTreeRoute,
220 1367 : RouteDistinguisher::kZeroRd, router_id_,
221 1367 : sg_entry_->group(), sg_entry_->source());
222 1367 : ErmVpnRoute rt_key(prefix);
223 :
224 : // Find or create the route.
225 1367 : McastManagerPartition *partition = sg_entry_->partition();
226 : DBTablePartition *tbl_partition =
227 1367 : static_cast<DBTablePartition *>(partition->GetTablePartition());
228 : ErmVpnRoute *route =
229 1367 : static_cast<ErmVpnRoute *>(tbl_partition->Find(&rt_key));
230 1367 : if (!route) {
231 401 : route = new ErmVpnRoute(prefix);
232 401 : tbl_partition->Add(route);
233 : } else {
234 966 : route->ClearDelete();
235 : }
236 :
237 : // Build the attributes. Need to go through the tree links to build the
238 : // EdgeForwardingSpec.
239 1367 : BgpServer *server = table->routing_instance()->server();
240 1367 : BgpAttrSpec attr_spec;
241 1367 : BgpAttrNextHop nexthop(server->bgp_identifier());
242 1367 : attr_spec.push_back(&nexthop);
243 1367 : BgpAttrSourceRd source_rd(sg_entry_->GetSourceRd());
244 1367 : attr_spec.push_back(&source_rd);
245 1367 : EdgeForwardingSpec efspec;
246 1367 : for (McastForwarderList::const_iterator it = tree_links_.begin();
247 2353 : it != tree_links_.end(); ++it) {
248 986 : EdgeForwardingSpec::Edge *edge = new EdgeForwardingSpec::Edge;
249 986 : edge->SetInboundIp4Address(address_);
250 986 : edge->inbound_label = label_;
251 986 : edge->SetOutboundIp4Address((*it)->address());
252 986 : edge->outbound_label = (*it)->label();
253 986 : efspec.edge_list.push_back(edge);
254 : }
255 1367 : attr_spec.push_back(&efspec);
256 : // Add tunnel encaps for remote nodes
257 1367 : ExtCommunitySpec ext;
258 1367 : ext.AddTunnelEncaps(encap_);
259 1367 : if (!ext.communities.empty())
260 115 : attr_spec.push_back(&ext);
261 1367 : BgpAttrPtr attr = server->attr_db()->Locate(attr_spec);
262 :
263 : // Add a path with source BgpPath::Local.
264 1367 : BgpPath *path = new BgpPath(0, BgpPath::Local, attr);
265 1367 : route->InsertPath(path);
266 1367 : tbl_partition->Notify(route);
267 1367 : global_tree_route_ = route;
268 1367 : }
269 :
270 : //
271 : // Delete the GlobalTreeRoute for this McastForwarder.
272 : //
273 14199 : void McastForwarder::DeleteGlobalTreeRoute() {
274 14199 : if (!global_tree_route_)
275 12832 : return;
276 :
277 1367 : McastManagerPartition *partition = sg_entry_->partition();
278 : DBTablePartition *tbl_partition =
279 1367 : static_cast<DBTablePartition *>(partition->GetTablePartition());
280 1367 : global_tree_route_->RemovePath(BgpPath::Local);
281 :
282 1367 : if (!global_tree_route_->HasPaths()) {
283 1324 : tbl_partition->Delete(global_tree_route_);
284 : } else {
285 43 : tbl_partition->Notify(global_tree_route_);
286 : }
287 1367 : global_tree_route_ = NULL;
288 : }
289 :
290 : //
291 : // Append list of BgpOListElems from the Local tree to the BgpOListSpec. The
292 : // list is built based on the tree links in this McastForwarder.
293 : //
294 4248 : void McastForwarder::AddLocalOListElems(BgpOListSpec *olist_spec) {
295 4248 : assert(level_ == McastTreeManager::LevelNative);
296 :
297 4248 : for (McastForwarderList::const_iterator it = tree_links_.begin();
298 7908 : it != tree_links_.end(); ++it) {
299 3660 : BgpOListElem elem((*it)->address(), (*it)->label(), (*it)->encap());
300 3660 : olist_spec->elements.push_back(elem);
301 3660 : }
302 4248 : }
303 :
304 : //
305 : // Append list of BgpOListElems from the Global tree to the BgpOListSpec. The
306 : // list is built based on EdgeForwarding attribute in the GlobalTreeRoute.
307 : //
308 4248 : void McastForwarder::AddGlobalOListElems(BgpOListSpec *olist_spec) {
309 4248 : assert(level_ == McastTreeManager::LevelNative);
310 :
311 : // Bail if this is not the forest node for the Local tree.
312 4248 : if (!sg_entry_->IsForestNode(this))
313 2487 : return;
314 :
315 2261 : const ErmVpnRoute *route = sg_entry_->tree_result_route();
316 2261 : if (!route)
317 494 : return;
318 :
319 1767 : const BgpPath *path = route->BestPath();
320 1767 : if (!path)
321 6 : return;
322 1761 : const BgpAttr *attr = path->GetAttr();
323 1761 : vector<string> encaps;
324 1761 : if (attr && attr->ext_community())
325 578 : encaps = attr->ext_community()->GetTunnelEncap();
326 :
327 : // Go through each forwarding edge and add it to the list.
328 1761 : const EdgeForwarding *eforwarding = path->GetAttr()->edge_forwarding();
329 1761 : for (EdgeForwarding::EdgeList::const_iterator it =
330 4698 : eforwarding->edge_list.begin(); it != eforwarding->edge_list.end();
331 1176 : ++it) {
332 1176 : const EdgeForwarding::Edge *edge = *it;
333 1176 : if (edge->inbound_address == address_) {
334 1008 : BgpOListElem elem(edge->outbound_address, edge->outbound_label,
335 1008 : encaps);
336 1008 : olist_spec->elements.push_back(elem);
337 1008 : }
338 : }
339 1761 : }
340 :
341 : //
342 : // Construct an UpdateInfo with the RibOutAttr that needs to be advertised to
343 : // the IPeer for the ErmVpnRoute associated with this McastForwarder. This is
344 : // used as Export method of the ErmVpnTable. It is expected that the caller
345 : // fills in the target RibPeerSet in the UpdateInfo.
346 : //
347 : // The main functionality here is to transform the McastForwarderList for the
348 : // distribution tree and the EdgeForwarding attribute from the GlobalTreeRoute
349 : // into a BgpOList.
350 : //
351 4248 : UpdateInfo *McastForwarder::GetUpdateInfo(ErmVpnTable *table) {
352 4248 : CHECK_CONCURRENCY("db::DBTable");
353 :
354 4248 : assert(level_ == McastTreeManager::LevelNative);
355 :
356 4248 : BgpOListSpec olist_spec(BgpAttribute::OList);
357 4248 : AddLocalOListElems(&olist_spec);
358 4248 : AddGlobalOListElems(&olist_spec);
359 :
360 : // Bail if there is no label allocated.
361 4248 : if (label_ == 0)
362 647 : return NULL;
363 :
364 3601 : BgpAttrSpec attr_spec;
365 3601 : attr_spec.push_back(&olist_spec);
366 3601 : BgpAttrPtr attr = table->server()->attr_db()->Locate(attr_spec);
367 :
368 3601 : UpdateInfo *uinfo = new UpdateInfo;
369 3601 : uinfo->roattr = RibOutAttr(table, route_, attr.get(), label_, true, true);
370 5838 : if (route_ && sg_entry_->IsForestNode(this) &&
371 2237 : sg_entry_->IsTreeBuilder(McastTreeManager::LevelLocal)) {
372 1334 : table->GetMvpnSourceAddress(route_, uinfo->roattr.source_address());
373 : }
374 3601 : return uinfo;
375 4248 : }
376 :
377 : //
378 : // Constructor for McastSGEntry.
379 : //
380 25116 : McastSGEntry::McastSGEntry(McastManagerPartition *partition,
381 25116 : Ip4Address group, Ip4Address source)
382 25116 : : partition_(partition),
383 25116 : group_(group),
384 25116 : source_(source),
385 25116 : forest_node_(NULL),
386 25116 : local_tree_route_(NULL),
387 25116 : tree_result_route_(NULL),
388 50232 : on_work_queue_(false) {
389 75348 : for (int level = McastTreeManager::LevelFirst;
390 75348 : level < McastTreeManager::LevelCount; ++level) {
391 50232 : ForwarderSet *forwarders = new ForwarderSet;
392 50232 : forwarder_sets_.push_back(forwarders);
393 50232 : update_needed_.push_back(false);
394 : }
395 25116 : }
396 :
397 : //
398 : // Destructor for McastSGEntry.
399 : //
400 25520 : McastSGEntry::~McastSGEntry() {
401 25116 : STLDeleteValues(&forwarder_sets_);
402 25520 : }
403 :
404 : //
405 : // Printable string for McastSGEntry.
406 : //
407 56 : std::string McastSGEntry::ToString() const {
408 112 : return group_.to_string() + "," + source_.to_string();
409 : }
410 :
411 : //
412 : // Add the given McastForwarder under this McastSGEntry and trigger update
413 : // of the distribution tree.
414 : //
415 1922 : void McastSGEntry::AddForwarder(McastForwarder *forwarder) {
416 1922 : uint8_t level = forwarder->level();
417 1922 : forwarder_sets_[level]->insert(forwarder);
418 1922 : update_needed_[level] = true;
419 1922 : partition_->EnqueueSGEntry(this);
420 1922 : }
421 :
422 : //
423 : // Handle change for the given McastForwarder under this McastSGEntry. Trigger
424 : // update of the distribution tree.
425 : //
426 : // Note that this method only handles the change = the caller determines that
427 : // there has been a change.
428 : //
429 1183 : void McastSGEntry::ChangeForwarder(McastForwarder *forwarder) {
430 1183 : uint8_t level = forwarder->level();
431 1183 : update_needed_[level] = true;
432 1183 : partition_->EnqueueSGEntry(this);
433 1183 : }
434 :
435 : //
436 : // Delete the given McastForwarder from this McastSGEntry and trigger update
437 : // of the distribution tree.
438 : //
439 1922 : void McastSGEntry::DeleteForwarder(McastForwarder *forwarder) {
440 1922 : if (forwarder == forest_node_)
441 411 : forest_node_ = NULL;
442 1922 : uint8_t level = forwarder->level();
443 1922 : forwarder_sets_[level]->erase(forwarder);
444 1922 : update_needed_[level] = true;
445 1922 : partition_->EnqueueSGEntry(this);
446 1922 : }
447 :
448 : //
449 : // Get the SourceRD to be used when adding [Local|Global]TreeRoutes. This
450 : // SourceRD gets used as the RD when the ErmVpnRoute is replicated from the
451 : // VRF table to the VPN table.
452 : //
453 : // We simply use the RD for the forest node.
454 : //
455 3731 : const RouteDistinguisher &McastSGEntry::GetSourceRd() const {
456 3731 : if (!forest_node_)
457 102 : return RouteDistinguisher::kZeroRd;
458 3629 : return forest_node_->route()->GetPrefix().route_distinguisher();
459 : }
460 :
461 : //
462 : // Add the LocalTreeRoute for this McastSGEntry. This route advertises a set
463 : // of candidate edges from McastForwarders attached to this control-node that
464 : // can be used by the tree builder to build the higher level tree. We simply
465 : // advertise edges McastTreeManager::kDegree - 1 edges from the forest node.
466 : //
467 : // We advertise kDegree-1 candidate edges via the EdgeDiscovery attribute. All
468 : // the edges are for the forest node for the tree of native McastForwarders.
469 : // The label block for each edge in the EdgeDiscovery attribute is of size 1 -
470 : // this is label that has been allocated for the forest node. Using a single
471 : // label is acceptable because the tree builder algorithm does not change the
472 : // relative order of nodes in the tree.
473 : //
474 1244 : void McastSGEntry::AddLocalTreeRoute() {
475 1244 : assert(!forest_node_);
476 1244 : assert(!local_tree_route_);
477 :
478 : // Select last usable leaf in the distribution tree as the forest node.
479 : // A leaf is considered usable if it has a valid label i.e. it has not
480 : // run out of labels.
481 1244 : uint8_t level = McastTreeManager::LevelNative;
482 1244 : ForwarderSet *forwarders = forwarder_sets_[level];
483 1244 : for (ForwarderSet::reverse_iterator rit = forwarders->rbegin();
484 1253 : rit != forwarders->rend(); ++rit) {
485 904 : McastForwarder *forwarder = *rit;
486 904 : if (forwarder->label()) {
487 895 : forest_node_ = forwarder;
488 895 : break;
489 : }
490 : }
491 :
492 : // Bail if we couldn't designate a forest node.
493 1244 : if (!forest_node_)
494 349 : return;
495 :
496 : // Construct the prefix and route key.
497 895 : BgpServer *server = partition_->server();
498 895 : Ip4Address router_id(server->bgp_identifier());
499 : ErmVpnPrefix prefix(ErmVpnPrefix::LocalTreeRoute,
500 895 : RouteDistinguisher::kZeroRd, router_id, group_, source_);
501 895 : ErmVpnRoute rt_key(prefix);
502 :
503 : // Find or create the route.
504 : DBTablePartition *tbl_partition =
505 895 : static_cast<DBTablePartition *>(partition_->GetTablePartition());
506 : ErmVpnRoute *route =
507 895 : static_cast<ErmVpnRoute *>(tbl_partition->Find(&rt_key));
508 895 : if (!route) {
509 347 : route = new ErmVpnRoute(prefix);
510 347 : tbl_partition->Add(route);
511 : } else {
512 548 : route->ClearDelete();
513 : }
514 :
515 : // Build the attributes.
516 895 : BgpAttrSpec attr_spec;
517 895 : BgpAttrNextHop nexthop(server->bgp_identifier());
518 895 : attr_spec.push_back(&nexthop);
519 895 : BgpAttrSourceRd source_rd(GetSourceRd());
520 895 : attr_spec.push_back(&source_rd);
521 895 : EdgeDiscoverySpec edspec;
522 3580 : for (int idx = 1; idx <= McastTreeManager::kDegree - 1; ++idx) {
523 2685 : EdgeDiscoverySpec::Edge *edge = new EdgeDiscoverySpec::Edge;
524 2685 : edge->SetIp4Address(forest_node_->address());
525 2685 : edge->SetLabels(forest_node_->label(), forest_node_->label());
526 2685 : edspec.edge_list.push_back(edge);
527 : }
528 895 : attr_spec.push_back(&edspec);
529 : // Add tunnel encaps for remote nodes
530 895 : ExtCommunitySpec ext;
531 895 : ext.AddTunnelEncaps(forest_node_->encap());
532 895 : if (!ext.communities.empty())
533 120 : attr_spec.push_back(&ext);
534 895 : BgpAttrPtr attr = server->attr_db()->Locate(attr_spec);
535 :
536 : // Add a path with source BgpPath::Local.
537 895 : BgpPath *path = new BgpPath(0, BgpPath::Local, attr);
538 895 : route->InsertPath(path);
539 895 : tbl_partition->Notify(route);
540 895 : local_tree_route_ = route;
541 895 : }
542 :
543 : //
544 : // Delete the LocalTreeRoute for this McastSGEntry.
545 : //
546 1244 : void McastSGEntry::DeleteLocalTreeRoute() {
547 1244 : if (!local_tree_route_)
548 349 : return;
549 :
550 895 : forest_node_ = NULL;
551 : DBTablePartition *tbl_partition =
552 895 : static_cast<DBTablePartition *>(partition_->GetTablePartition());
553 895 : local_tree_route_->RemovePath(BgpPath::Local);
554 895 : if (!local_tree_route_->HasPaths()) {
555 895 : tbl_partition->Delete(local_tree_route_);
556 : } else {
557 0 : tbl_partition->Notify(local_tree_route_);
558 : }
559 895 : local_tree_route_ = NULL;
560 : }
561 :
562 : //
563 : // Update the LocalTreeRoute for this McastSGEntry if RouterId has changed.
564 : //
565 721 : void McastSGEntry::UpdateLocalTreeRoute() {
566 721 : if (!local_tree_route_)
567 721 : return;
568 :
569 : // Bail if the RouterId hasn't changed.
570 613 : const BgpServer *server = partition_->server();
571 613 : Ip4Address router_id = local_tree_route_->GetPrefix().router_id();
572 613 : if (router_id.to_ulong() == server->bgp_identifier())
573 613 : return;
574 :
575 : // Add and delete the route.
576 0 : DeleteLocalTreeRoute();
577 0 : AddLocalTreeRoute();
578 : }
579 :
580 : //
581 : // Update relevant [Local|Global]TreeRoutes for the McastSGEntry.
582 : //
583 3536 : void McastSGEntry::UpdateRoutes(uint8_t level) {
584 3536 : if (level == McastTreeManager::LevelNative) {
585 1244 : DeleteLocalTreeRoute();
586 1244 : AddLocalTreeRoute();
587 : } else {
588 2292 : ForwarderSet *forwarders = forwarder_sets_[level];
589 2292 : for (ForwarderSet::iterator it = forwarders->begin();
590 5778 : it != forwarders->end(); ++it) {
591 3486 : (*it)->DeleteGlobalTreeRoute();
592 3486 : (*it)->AddGlobalTreeRoute();
593 : }
594 : }
595 3536 : }
596 :
597 3395 : ErmVpnRoute *McastSGEntry::GetGlobalTreeRootRoute() const {
598 3395 : if (!IsTreeBuilder(McastTreeManager::LevelLocal))
599 1673 : return NULL;
600 1722 : ForwarderSet *forwarders = forwarder_sets_[McastTreeManager::LevelLocal];
601 1722 : assert(!forwarders->empty());
602 1722 : ForwarderSet::const_iterator it = forwarders->begin();
603 1722 : return (*it)->global_tree_route();
604 : }
605 :
606 : //
607 : // Implement tree builder election.
608 : //
609 9172 : bool McastSGEntry::IsTreeBuilder(uint8_t level) const {
610 9172 : if (level == McastTreeManager::LevelNative)
611 1246 : return true;
612 :
613 7926 : const ForwarderSet *forwarders = forwarder_sets_[level];
614 7926 : ForwarderSet::const_iterator it = forwarders->begin();
615 7926 : if (it == forwarders->end())
616 680 : return false;
617 :
618 7246 : Ip4Address router_id(partition_->server()->bgp_identifier());
619 7246 : if ((*it)->router_id() != router_id)
620 3242 : return false;
621 :
622 4004 : return true;
623 : }
624 :
625 : //
626 : //
627 : // Update specified distribution tree for the McastSGEntry. We traverse all
628 : // McastForwarders in sorted order and arrange them in breadth first fashion
629 : // in a k-ary tree. Building the tree in this manner guarantees that we get
630 : // the same tree for a given set of forwarders, independent of the order in
631 : // in which they joined. This predictability is deemed to be more important
632 : // than other criteria such as minimizing disruption of traffic, minimizing
633 : // the cost/weight of the tree etc.
634 : //
635 6818 : void McastSGEntry::UpdateTree(uint8_t level) {
636 6818 : CHECK_CONCURRENCY("db::DBTable");
637 :
638 6818 : if (!update_needed_[level])
639 4628 : return;
640 3536 : update_needed_[level] = false;
641 :
642 : int degree;
643 3536 : if (level == McastTreeManager::LevelNative) {
644 1244 : degree = McastTreeManager::kDegree;
645 : } else {
646 2292 : degree = McastTreeManager::kDegree - 1;
647 : }
648 :
649 : // First get rid of the previous distribution tree and enqueue all the
650 : // associated ErmVpnRoutes for notification. Note that DBListeners will
651 : // not get invoked until after this routine is done.
652 3536 : ForwarderSet *forwarders = forwarder_sets_[level];
653 3536 : for (ForwarderSet::iterator it = forwarders->begin();
654 9761 : it != forwarders->end(); ++it) {
655 6225 : (*it)->FlushLinks();
656 6225 : (*it)->ReleaseLabel();
657 6225 : partition_->GetTablePartition()->Notify((*it)->route());
658 : }
659 :
660 : // Bail if we're not the tree builder.
661 3536 : if (!IsTreeBuilder(level)) {
662 1346 : UpdateRoutes(level);
663 1346 : return;
664 : }
665 :
666 : // Create a vector of pointers to the McastForwarders in sorted order.
667 : // We do this because std::set doesn't support random access iterators.
668 : // Skip if we can't allocate a label for the McastForwarder.
669 2190 : McastForwarderList vec;
670 2190 : vec.reserve(forwarders->size());
671 2190 : for (ForwarderSet::iterator it = forwarders->begin();
672 6398 : it != forwarders->end(); ++it) {
673 4208 : McastForwarder *forwarder = *it;
674 4208 : forwarder->AllocateLabel();
675 4208 : if (!forwarder->label())
676 11 : continue;
677 4197 : vec.push_back(forwarder);
678 : }
679 :
680 : // Go through each McastForwarder in the vector and link it to it's parent
681 : // McastForwarder in the k-ary tree. We also add a link from the parent to
682 : // the entry in question.
683 6387 : for (McastForwarderList::iterator it = vec.begin(); it != vec.end(); ++it) {
684 4197 : int idx = it - vec.begin();
685 4197 : if (idx == 0)
686 1841 : continue;
687 :
688 2356 : int parent_idx = (idx - 1) / degree;
689 2356 : McastForwarderList::iterator parent_it = vec.begin() + parent_idx;
690 2356 : assert(parent_it != vec.end());
691 2356 : McastForwarder *forwarder = *it;
692 2356 : McastForwarder *parent_forwarder = *parent_it;
693 2356 : forwarder->AddLink(parent_forwarder);
694 2356 : parent_forwarder->AddLink(forwarder);
695 : }
696 :
697 : // Update [Local|Global]TreeRoutes.
698 2190 : UpdateRoutes(level);
699 2190 : }
700 :
701 : //
702 : // Update distribution trees for both levels.
703 : //
704 3409 : void McastSGEntry::UpdateTree() {
705 10227 : for (uint8_t level = McastTreeManager::LevelFirst;
706 10227 : level < McastTreeManager::LevelCount; ++level) {
707 6818 : UpdateTree(level);
708 : }
709 3409 : }
710 :
711 : //
712 : // Trigger notification of the ErmVpnRoute associated with the McastForwarder
713 : // that is the forest node. This is used to trigger a rebuild of the BgpOlist
714 : // when the GlobalTreeRoute is updated.
715 : //
716 2091 : void McastSGEntry::NotifyForestNode() {
717 2091 : if (!forest_node_)
718 442 : return;
719 1649 : partition_->GetTablePartition()->Notify(forest_node_->route());
720 : }
721 :
722 174 : bool McastSGEntry::GetForestNodePMSI(uint32_t *label, Ip4Address *address,
723 : vector<string> *tunnel_encap) const {
724 174 : if (!forest_node_)
725 0 : return false;
726 174 : *label = forest_node_->label();
727 174 : *address = forest_node_->address();
728 174 : *tunnel_encap = forest_node_->encap();
729 174 : return true;
730 : }
731 :
732 7849 : bool McastSGEntry::IsForestNode(McastForwarder *forwarder) {
733 7849 : return (forwarder == forest_node_);
734 : }
735 :
736 3409 : bool McastSGEntry::empty() const {
737 3409 : if (local_tree_route_ || tree_result_route_)
738 2712 : return false;
739 697 : if (!forwarder_sets_[McastTreeManager::LevelNative]->empty())
740 5 : return false;
741 692 : if (!forwarder_sets_[McastTreeManager::LevelLocal]->empty())
742 288 : return false;
743 404 : return true;
744 : }
745 :
746 : //
747 : // Constructor for McastManagerPartition.
748 : //
749 42871 : McastManagerPartition::McastManagerPartition(McastTreeManager *tree_manager,
750 42871 : size_t part_id)
751 42871 : : tree_manager_(tree_manager),
752 42871 : part_id_(part_id),
753 42871 : update_count_(0),
754 85742 : work_queue_(TaskScheduler::GetInstance()->GetTaskId("db::DBTable"),
755 42871 : part_id_,
756 42871 : boost::bind(&McastManagerPartition::ProcessSGEntry, this, _1)) {
757 42871 : }
758 :
759 : //
760 : // Destructor for McastManagerPartition.
761 : //
762 42871 : McastManagerPartition::~McastManagerPartition() {
763 42871 : work_queue_.Shutdown();
764 42871 : }
765 :
766 : // Find the McastSGEntry for the given group and source.
767 21143 : McastSGEntry *McastManagerPartition::FindSGEntry(
768 : const Ip4Address &group, const Ip4Address &source) {
769 : return const_cast<McastSGEntry *>(
770 21143 : static_cast<const McastManagerPartition *>(this)->FindSGEntry(group,
771 21143 : source));
772 : }
773 :
774 : //
775 : // Find the McastSGEntry for the given group and source.
776 : //
777 24712 : const McastSGEntry *McastManagerPartition::FindSGEntry(
778 : const Ip4Address &group, const Ip4Address &source) const {
779 : McastSGEntry temp_sg_entry(const_cast<McastManagerPartition *>(this),
780 24712 : group, source);
781 24712 : SGList::const_iterator it = sg_list_.find(&temp_sg_entry);
782 49424 : return (it != sg_list_.end() ? *it : NULL);
783 24712 : }
784 :
785 : //
786 : // Find or create the McastSGEntry for the given group and source.
787 : //
788 2367 : McastSGEntry *McastManagerPartition::LocateSGEntry(
789 : Ip4Address group, Ip4Address source) {
790 2367 : McastSGEntry *sg_entry = FindSGEntry(group, source);
791 2367 : if (!sg_entry) {
792 404 : sg_entry = new McastSGEntry(this, group, source);
793 404 : sg_list_.insert(sg_entry);
794 : }
795 2367 : return sg_entry;
796 : }
797 :
798 3395 : ErmVpnRoute *McastManagerPartition::GetGlobalTreeRootRoute(
799 : const Ip4Address &source, const Ip4Address &group) const {
800 3395 : const McastSGEntry *sg = FindSGEntry(group, source);
801 3395 : return sg ? sg->GetGlobalTreeRootRoute() : NULL;
802 : }
803 :
804 7980 : void McastManagerPartition::NotifyForestNode(
805 : const Ip4Address &source, const Ip4Address &group) {
806 7980 : McastSGEntry *sg = FindSGEntry(group, source);
807 7980 : if (sg)
808 308 : sg->NotifyForestNode();
809 7980 : }
810 :
811 174 : bool McastManagerPartition::GetForestNodePMSI(ErmVpnRoute *rt, uint32_t *label,
812 : Ip4Address *address, vector<string> *encap) const {
813 174 : const McastSGEntry *sg = FindSGEntry(rt->GetPrefix().group(),
814 174 : rt->GetPrefix().source());
815 174 : return sg ? sg->GetForestNodePMSI(label, address, encap) : false;
816 : }
817 :
818 : //
819 : // Enqueue the given McastSGEntry on the WorkQueue if it's not already on it.
820 : //
821 5472 : void McastManagerPartition::EnqueueSGEntry(McastSGEntry *sg_entry) {
822 5472 : if (sg_entry->on_work_queue())
823 2063 : return;
824 3409 : work_queue_.Enqueue(sg_entry);
825 3409 : sg_entry->set_on_work_queue();
826 : }
827 :
828 : //
829 : // Callback for the WorkQueue. Updates distribution trees for the McastSGEntry.
830 : // Also gets rid of the McastSGEntry if it is eligible to be deleted.
831 : //
832 3409 : bool McastManagerPartition::ProcessSGEntry(McastSGEntry *sg_entry) {
833 3409 : CHECK_CONCURRENCY("db::DBTable");
834 :
835 3409 : sg_entry->clear_on_work_queue();
836 3409 : sg_entry->UpdateTree();
837 3409 : update_count_++;
838 :
839 3409 : if (sg_entry->empty()) {
840 404 : sg_list_.erase(sg_entry);
841 404 : delete sg_entry;
842 : }
843 :
844 3409 : if (sg_list_.empty())
845 289 : tree_manager_->RetryDelete();
846 :
847 3409 : return true;
848 : }
849 :
850 : //
851 : // Get the DBTablePartBase for the ErmVpnTable for our partition id.
852 : //
853 12398 : DBTablePartBase *McastManagerPartition::GetTablePartition() {
854 12398 : return tree_manager_->GetTablePartition(part_id_);
855 : }
856 :
857 0 : const RoutingInstance *McastManagerPartition::routing_instance() const {
858 0 : return tree_manager_->table()->routing_instance();
859 : }
860 :
861 8754 : BgpServer *McastManagerPartition::server() {
862 8754 : return tree_manager_->table()->server();
863 : }
864 :
865 0 : const BgpServer *McastManagerPartition::server() const {
866 0 : return tree_manager_->table()->server();
867 : }
868 :
869 : //
870 : // Constructor for McastTreeManager.
871 : //
872 42904 : McastTreeManager::McastTreeManager(ErmVpnTable *table)
873 42904 : : table_(table),
874 42904 : listener_id_(DBTable::kInvalidId),
875 42904 : table_delete_ref_(this, table->deleter()) {
876 42904 : deleter_.reset(new DeleteActor(this));
877 42904 : }
878 :
879 : //
880 : // Destructor for McastTreeManager.
881 : //
882 66857 : McastTreeManager::~McastTreeManager() {
883 66857 : }
884 :
885 : //
886 : // Initialize the McastTreeManager. We allocate the McastManagerPartitions
887 : // and register a DBListener for the ErmVpnTable.
888 : //
889 42871 : void McastTreeManager::Initialize() {
890 42871 : AllocPartitions();
891 42871 : listener_id_ = table_->Register(
892 : boost::bind(&McastTreeManager::RouteListener, this, _1, _2),
893 : "McastTreeManager");
894 42871 : }
895 :
896 : //
897 : // Terminate the McastTreeManager. We free the McastManagerPartitions
898 : // and unregister from the ErmVpnTable.
899 : //
900 42871 : void McastTreeManager::Terminate() {
901 42871 : table_->Unregister(listener_id_);
902 42871 : FreePartitions();
903 42871 : }
904 :
905 : //
906 : // Allocate the McastManagerPartitions.
907 : //
908 42871 : void McastTreeManager::AllocPartitions() {
909 85742 : for (int part_id = 0; part_id < table_->PartitionCount(); part_id++) {
910 42871 : partitions_.push_back(new McastManagerPartition(this, part_id));
911 : }
912 42871 : }
913 :
914 : //
915 : // Free the McastManagerPartitions.
916 : //
917 42871 : void McastTreeManager::FreePartitions() {
918 85742 : for (size_t part_id = 0; part_id < partitions_.size(); part_id++) {
919 42871 : delete partitions_[part_id];
920 : }
921 42871 : partitions_.clear();
922 42871 : }
923 :
924 7982 : McastManagerPartition *McastTreeManager::GetPartition(int part_id) {
925 7982 : return partitions_[part_id];
926 : }
927 :
928 3569 : const McastManagerPartition *McastTreeManager::GetPartition(int part_id) const {
929 3569 : return partitions_[part_id];
930 : }
931 :
932 : //
933 : // Get the DBTablePartBase for the ErmVpnTable for given partition id.
934 : //
935 12398 : DBTablePartBase *McastTreeManager::GetTablePartition(size_t part_id) {
936 12398 : return table_->GetTablePartition(part_id);
937 : }
938 :
939 : //
940 : // Construct export state for the given ErmVpnRoute. Note that the route
941 : // only needs to be exported to the IPeer from which it was learnt.
942 : //
943 3278 : UpdateInfo *McastTreeManager::GetUpdateInfo(ErmVpnRoute *route) {
944 3278 : CHECK_CONCURRENCY("db::DBTable");
945 :
946 3278 : DBState *dbstate = route->GetState(table_, listener_id_);
947 3278 : McastForwarder *forwarder = dynamic_cast<McastForwarder *>(dbstate);
948 :
949 3278 : if (!forwarder)
950 0 : return NULL;
951 :
952 3278 : return forwarder->GetUpdateInfo(table_);
953 : }
954 :
955 : //
956 : // DBListener callback handler for Native and Local routes in the ErmVpnTable.
957 : // It creates, updates or deletes the associated McastForwarder as appropriate.
958 : //
959 : // Creates a McastSGEntry if one doesn't already exist. However, McastSGEntrys
960 : // don't get deleted from here. They only get deleted from WorkQueue callback
961 : // routine i.e. McastManagerPartition::ProcessSGEntry.
962 : //
963 12656 : void McastTreeManager::TreeNodeListener(McastManagerPartition *partition,
964 : ErmVpnRoute *route) {
965 12656 : CHECK_CONCURRENCY("db::DBTable");
966 :
967 12656 : DBState *dbstate = route->GetState(table_, listener_id_);
968 12656 : if (!dbstate) {
969 : // We have no previous DBState for this route.
970 : // Bail if the route is not valid.
971 1943 : if (!route->IsValid())
972 21 : return;
973 :
974 : // Create a new McastForwarder and associate it with the route.
975 1922 : McastSGEntry *sg_entry = partition->LocateSGEntry(
976 1922 : route->GetPrefix().group(), route->GetPrefix().source());
977 1922 : McastForwarder *forwarder = new McastForwarder(sg_entry, route);
978 1922 : sg_entry->AddForwarder(forwarder);
979 1922 : route->SetState(table_, listener_id_, forwarder);
980 :
981 : // Update local tree route if our RouterId has changed. Ideally,
982 : // we should trigger an update of all local trees routes when we
983 : // detect a change in RouterId. Instead, we currently check and
984 : // update the local route when we detect a new local route from
985 : // another node.
986 1922 : if (route->GetPrefix().type() == ErmVpnPrefix::LocalTreeRoute)
987 721 : sg_entry->UpdateLocalTreeRoute();
988 : } else {
989 10713 : McastSGEntry *sg_entry = partition->FindSGEntry(
990 10713 : route->GetPrefix().group(), route->GetPrefix().source());
991 10713 : assert(sg_entry);
992 10713 : McastForwarder *forwarder = dynamic_cast<McastForwarder *>(dbstate);
993 10713 : assert(forwarder);
994 :
995 10713 : if (!route->IsValid()) {
996 : // Delete the McastForwarder associated with the route.
997 1922 : route->ClearState(table_, listener_id_);
998 1922 : sg_entry->DeleteForwarder(forwarder);
999 1922 : delete forwarder;
1000 8791 : } else if (forwarder->Update(route)) {
1001 : // Trigger update of the distribution tree.
1002 1183 : sg_entry->ChangeForwarder(forwarder);
1003 : }
1004 : }
1005 : }
1006 :
1007 : //
1008 : // DBListener callback handler for GlobalTreeRoutes in the ErmVpnTable. It
1009 : // updates the tree_result_route_ and triggers re-evaluation of the forest
1010 : // node McastForwarder's BgpOlist.
1011 : //
1012 9584 : void McastTreeManager::TreeResultListener(McastManagerPartition *partition,
1013 : ErmVpnRoute *route) {
1014 9584 : CHECK_CONCURRENCY("db::DBTable");
1015 :
1016 9584 : DBState *dbstate = route->GetState(table_, listener_id_);
1017 9584 : if (!dbstate) {
1018 : // We have no previous DBState for this route.
1019 : // Bail if the route is not valid.
1020 8246 : if (!route->IsValid())
1021 5989 : return;
1022 :
1023 : // Ignore GlobalTreeRoute if it's not applicable to this control-node.
1024 2257 : BgpServer *server = table_->routing_instance()->server();
1025 2257 : if (route->GetPrefix().router_id().to_ulong() !=
1026 2257 : server->bgp_identifier())
1027 1812 : return;
1028 :
1029 445 : McastSGEntry *sg_entry = partition->LocateSGEntry(
1030 445 : route->GetPrefix().group(), route->GetPrefix().source());
1031 445 : route->SetState(table_, listener_id_, sg_entry);
1032 445 : sg_entry->set_tree_result_route(route);
1033 445 : sg_entry->NotifyForestNode();
1034 : } else {
1035 1338 : McastSGEntry *sg_entry = dynamic_cast<McastSGEntry *>(dbstate);
1036 1338 : assert(sg_entry);
1037 :
1038 1338 : if (!route->IsValid()) {
1039 445 : sg_entry->clear_tree_result_route();
1040 445 : route->ClearState(table_, listener_id_);
1041 445 : partition->EnqueueSGEntry(sg_entry);
1042 : }
1043 1338 : sg_entry->NotifyForestNode();
1044 : }
1045 : }
1046 :
1047 : //
1048 : // DBListener callback handler for the ErmVpnTable. GlobalTreeRoutes provide
1049 : // result information and hence are handled differently than Native and Local
1050 : // routes, which result in update of a McastForwarder.
1051 : //
1052 22240 : void McastTreeManager::RouteListener(
1053 : DBTablePartBase *tpart, DBEntryBase *db_entry) {
1054 22240 : CHECK_CONCURRENCY("db::DBTable");
1055 :
1056 22240 : McastManagerPartition *partition = partitions_[tpart->index()];
1057 22240 : ErmVpnRoute *route = dynamic_cast<ErmVpnRoute *>(db_entry);
1058 22240 : if (route->GetPrefix().type() == ErmVpnPrefix::GlobalTreeRoute) {
1059 9584 : TreeResultListener(partition, route);
1060 : } else {
1061 12656 : TreeNodeListener(partition, route);
1062 : }
1063 22240 : }
1064 :
1065 :
1066 : //
1067 : // Check if the McastTreeManager can be deleted. This can happen only if all
1068 : // the McastManagerPartitions are empty.
1069 : //
1070 42904 : bool McastTreeManager::MayDelete() const {
1071 42904 : CHECK_CONCURRENCY("bgp::Config");
1072 :
1073 42904 : for (PartitionList::const_iterator it = partitions_.begin();
1074 85775 : it != partitions_.end(); ++it) {
1075 42871 : if (!(*it)->empty())
1076 0 : return false;
1077 : }
1078 :
1079 42904 : return true;
1080 : }
1081 :
1082 : //
1083 : // Initiate shutdown for the McastTreeManager.
1084 : //
1085 42904 : void McastTreeManager::Shutdown() {
1086 42904 : CHECK_CONCURRENCY("bgp::Config");
1087 42904 : }
1088 :
1089 : //
1090 : // Trigger deletion of the McastTreeManager and propagate the delete to any
1091 : // dependents.
1092 : //
1093 42904 : void McastTreeManager::ManagedDelete() {
1094 42904 : deleter_->Delete();
1095 42904 : }
1096 :
1097 : //
1098 : // Attempt to enqueue a delete for the McastTreeManager.
1099 : //
1100 289 : void McastTreeManager::RetryDelete() {
1101 289 : if (!deleter()->IsDeleted())
1102 289 : return;
1103 0 : deleter()->RetryDelete();
1104 : }
1105 :
1106 : //
1107 : // Return the LifetimeActor for the McastTreeManager.
1108 : //
1109 6785 : LifetimeActor *McastTreeManager::deleter() {
1110 6785 : return deleter_.get();
1111 : }
1112 :
1113 : //
1114 : // Return the LifetimeActor for the McastTreeManager.
1115 : // Const version.
1116 : //
1117 190 : const LifetimeActor *McastTreeManager::deleter() const {
1118 190 : return deleter_.get();
1119 : }
1120 :
1121 : //
1122 : // Return true if the McastTreeManager is deleted.
1123 : //
1124 190 : bool McastTreeManager::deleted() const {
1125 190 : return deleter_->IsDeleted();
1126 : }
1127 :
1128 3395 : ErmVpnRoute *McastTreeManager::GetGlobalTreeRootRoute(
1129 : const Ip4Address &source, const Ip4Address &group) const {
1130 3395 : const McastManagerPartition *partition = GetPartition(table_->Hash(group));
1131 3395 : return partition->GetGlobalTreeRootRoute(source, group);
1132 : }
1133 :
1134 7980 : void McastTreeManager::NotifyForestNode(int part_id, const Ip4Address &source,
1135 : const Ip4Address &group) {
1136 7980 : McastManagerPartition *partition = GetPartition(part_id);
1137 7980 : partition->NotifyForestNode(source, group);
1138 7980 : }
1139 :
1140 242 : bool McastTreeManager::GetForestNodePMSI(ErmVpnRoute *rt, uint32_t *label,
1141 : Ip4Address *address, vector<string> *encap) const {
1142 242 : if (!rt || !rt->IsUsable())
1143 68 : return false;
1144 : const McastManagerPartition *partition =
1145 174 : GetPartition(rt->get_table_partition()->index());
1146 174 : return partition->GetForestNodePMSI(rt, label, address, encap);
1147 : }
|