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 42907 : explicit DeleteActor(McastTreeManager *tree_manager)
30 42907 : : LifetimeActor(tree_manager->table_->routing_instance()->server()->
31 : lifetime_manager()),
32 42907 : tree_manager_(tree_manager) {
33 42907 : }
34 85814 : virtual ~DeleteActor() {
35 85814 : }
36 :
37 42907 : virtual bool MayDelete() const {
38 42907 : return tree_manager_->MayDelete();
39 : }
40 :
41 42907 : virtual void Shutdown() {
42 42907 : tree_manager_->Shutdown();
43 42907 : }
44 :
45 42907 : virtual void Destroy() {
46 42907 : tree_manager_->table_->DestroyTreeManager();
47 42907 : }
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 11473 : McastForwarder::McastForwarder(McastSGEntry *sg_entry, ErmVpnRoute *route)
64 11473 : : sg_entry_(sg_entry),
65 11473 : route_(route),
66 11473 : global_tree_route_(NULL),
67 11473 : label_(0),
68 11473 : address_(0),
69 11473 : rd_(route->GetPrefix().route_distinguisher()),
70 22946 : router_id_(route->GetPrefix().router_id()) {
71 11473 : const BgpPath *path = route->BestPath();
72 11473 : const BgpAttr *attr = path->GetAttr();
73 :
74 11473 : if (route_->GetPrefix().type() == ErmVpnPrefix::NativeRoute) {
75 6150 : level_ = McastTreeManager::LevelNative;
76 6150 : address_ = attr->nexthop().to_v4();
77 6150 : label_block_ = attr->label_block();
78 : } else {
79 5323 : level_ = McastTreeManager::LevelLocal;
80 5323 : const EdgeDiscovery::Edge *edge = attr->edge_discovery()->edge_list[0];
81 5323 : address_ = edge->address;
82 5323 : label_block_ = edge->label_block;
83 : }
84 :
85 11473 : if (path->GetAttr()->ext_community())
86 6416 : encap_ = path->GetAttr()->ext_community()->GetTunnelEncap();
87 11473 : }
88 :
89 : //
90 : // Destructor for McastForwarder. Flushes forward and reverse links to and
91 : // from other McastForwarders.
92 : //
93 13395 : McastForwarder::~McastForwarder() {
94 11473 : DeleteGlobalTreeRoute();
95 11473 : FlushLinks();
96 11473 : ReleaseLabel();
97 13395 : }
98 :
99 : //
100 : // Update the McastForwarder based on information in the ErmVpnRoute.
101 : // Return true if something changed.
102 : //
103 9551 : bool McastForwarder::Update(ErmVpnRoute *route) {
104 9551 : McastForwarder forwarder(sg_entry_, route);
105 :
106 9551 : bool changed = false;
107 9551 : if (label_block_ != forwarder.label_block_) {
108 1158 : ReleaseLabel();
109 1158 : label_block_ = forwarder.label_block_;
110 1158 : changed = true;
111 : }
112 9551 : if (address_ != forwarder.address_) {
113 526 : address_ = forwarder.address_;
114 526 : changed = true;
115 : }
116 9551 : if (encap_ != forwarder.encap_) {
117 13 : encap_ = forwarder.encap_;
118 13 : changed = true;
119 : }
120 :
121 9551 : return changed;
122 9551 : }
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 5938 : McastForwarder *McastForwarder::FindLink(McastForwarder *forwarder) {
139 5938 : for (McastForwarderList::iterator it = tree_links_.begin();
140 10435 : it != tree_links_.end(); ++it) {
141 4497 : if (*it == forwarder) return forwarder;
142 : }
143 5938 : return NULL;
144 : }
145 :
146 : //
147 : // Add a link to the given McastForwarder.
148 : //
149 5938 : void McastForwarder::AddLink(McastForwarder *forwarder) {
150 5938 : assert(!FindLink(forwarder));
151 5938 : tree_links_.push_back(forwarder);
152 5938 : }
153 :
154 : //
155 : // Remove a link to the given McastForwarder.
156 : //
157 2969 : void McastForwarder::RemoveLink(McastForwarder *forwarder) {
158 2969 : for (McastForwarderList::iterator it = tree_links_.begin();
159 3158 : it != tree_links_.end(); ++it) {
160 3158 : if (*it == forwarder) {
161 2969 : tree_links_.erase(it);
162 2969 : 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 18403 : void McastForwarder::FlushLinks() {
172 18403 : for (McastForwarderList::iterator it = tree_links_.begin();
173 21372 : it != tree_links_.end(); ++it) {
174 2969 : (*it)->RemoveLink(this);
175 : }
176 18403 : tree_links_.clear();
177 18403 : }
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 4897 : void McastForwarder::AllocateLabel() {
186 4897 : label_ = label_block_->AllocateLabel();
187 4897 : }
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 19561 : void McastForwarder::ReleaseLabel() {
194 19561 : if (label_ != 0) {
195 4887 : label_block_->ReleaseLabel(label_);
196 4887 : label_ = 0;
197 : }
198 19561 : }
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 3553 : void McastForwarder::AddGlobalTreeRoute() {
206 3553 : assert(level_ == McastTreeManager::LevelLocal);
207 3553 : assert(!global_tree_route_);
208 :
209 : // Bail if there's no label allocated.
210 3553 : if (label_ == 0)
211 2160 : return;
212 :
213 : // Bail if we can't build a source RD.
214 1519 : if (sg_entry_->GetSourceRd().IsZero())
215 126 : return;
216 :
217 : // Construct the prefix and route key.
218 1393 : BgpTable *table = static_cast<BgpTable *>(route_->get_table());
219 : ErmVpnPrefix prefix(ErmVpnPrefix::GlobalTreeRoute,
220 1393 : RouteDistinguisher::kZeroRd, router_id_,
221 1393 : sg_entry_->group(), sg_entry_->source());
222 1393 : ErmVpnRoute rt_key(prefix);
223 :
224 : // Find or create the route.
225 1393 : McastManagerPartition *partition = sg_entry_->partition();
226 : DBTablePartition *tbl_partition =
227 1393 : static_cast<DBTablePartition *>(partition->GetTablePartition());
228 : ErmVpnRoute *route =
229 1393 : static_cast<ErmVpnRoute *>(tbl_partition->Find(&rt_key));
230 1393 : if (!route) {
231 411 : route = new ErmVpnRoute(prefix);
232 411 : tbl_partition->Add(route);
233 : } else {
234 982 : route->ClearDelete();
235 : }
236 :
237 : // Build the attributes. Need to go through the tree links to build the
238 : // EdgeForwardingSpec.
239 1393 : BgpServer *server = table->routing_instance()->server();
240 1393 : BgpAttrSpec attr_spec;
241 1393 : BgpAttrNextHop nexthop(server->bgp_identifier());
242 1393 : attr_spec.push_back(&nexthop);
243 1393 : BgpAttrSourceRd source_rd(sg_entry_->GetSourceRd());
244 1393 : attr_spec.push_back(&source_rd);
245 1393 : EdgeForwardingSpec efspec;
246 1393 : for (McastForwarderList::const_iterator it = tree_links_.begin();
247 2377 : it != tree_links_.end(); ++it) {
248 984 : EdgeForwardingSpec::Edge *edge = new EdgeForwardingSpec::Edge;
249 984 : edge->SetInboundIp4Address(address_);
250 984 : edge->inbound_label = label_;
251 984 : edge->SetOutboundIp4Address((*it)->address());
252 984 : edge->outbound_label = (*it)->label();
253 984 : efspec.edge_list.push_back(edge);
254 : }
255 1393 : attr_spec.push_back(&efspec);
256 : // Add tunnel encaps for remote nodes
257 1393 : ExtCommunitySpec ext;
258 1393 : ext.AddTunnelEncaps(encap_);
259 1393 : if (!ext.communities.empty())
260 179 : attr_spec.push_back(&ext);
261 1393 : BgpAttrPtr attr = server->attr_db()->Locate(attr_spec);
262 :
263 : // Add a path with source BgpPath::Local.
264 1393 : BgpPath *path = new BgpPath(0, BgpPath::Local, attr);
265 1393 : route->InsertPath(path);
266 1393 : tbl_partition->Notify(route);
267 1393 : global_tree_route_ = route;
268 1393 : }
269 :
270 : //
271 : // Delete the GlobalTreeRoute for this McastForwarder.
272 : //
273 15026 : void McastForwarder::DeleteGlobalTreeRoute() {
274 15026 : if (!global_tree_route_)
275 13633 : return;
276 :
277 1393 : McastManagerPartition *partition = sg_entry_->partition();
278 : DBTablePartition *tbl_partition =
279 1393 : static_cast<DBTablePartition *>(partition->GetTablePartition());
280 1393 : global_tree_route_->RemovePath(BgpPath::Local);
281 :
282 1393 : if (!global_tree_route_->HasPaths()) {
283 1329 : tbl_partition->Delete(global_tree_route_);
284 : } else {
285 64 : tbl_partition->Notify(global_tree_route_);
286 : }
287 1393 : 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 4177 : void McastForwarder::AddLocalOListElems(BgpOListSpec *olist_spec) {
295 4177 : assert(level_ == McastTreeManager::LevelNative);
296 :
297 4177 : for (McastForwarderList::const_iterator it = tree_links_.begin();
298 7767 : it != tree_links_.end(); ++it) {
299 3590 : BgpOListElem elem((*it)->address(), (*it)->label(), (*it)->encap());
300 3590 : olist_spec->elements.push_back(elem);
301 3590 : }
302 4177 : }
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 4177 : void McastForwarder::AddGlobalOListElems(BgpOListSpec *olist_spec) {
309 4177 : assert(level_ == McastTreeManager::LevelNative);
310 :
311 : // Bail if this is not the forest node for the Local tree.
312 4177 : if (!sg_entry_->IsForestNode(this))
313 2448 : return;
314 :
315 2214 : const ErmVpnRoute *route = sg_entry_->tree_result_route();
316 2214 : if (!route)
317 477 : return;
318 :
319 1737 : const BgpPath *path = route->BestPath();
320 1737 : if (!path)
321 8 : return;
322 1729 : const BgpAttr *attr = path->GetAttr();
323 1729 : vector<string> encaps;
324 1729 : if (attr && attr->ext_community())
325 577 : encaps = attr->ext_community()->GetTunnelEncap();
326 :
327 : // Go through each forwarding edge and add it to the list.
328 1729 : const EdgeForwarding *eforwarding = path->GetAttr()->edge_forwarding();
329 1729 : for (EdgeForwarding::EdgeList::const_iterator it =
330 4630 : eforwarding->edge_list.begin(); it != eforwarding->edge_list.end();
331 1172 : ++it) {
332 1172 : const EdgeForwarding::Edge *edge = *it;
333 1172 : if (edge->inbound_address == address_) {
334 999 : BgpOListElem elem(edge->outbound_address, edge->outbound_label,
335 999 : encaps);
336 999 : olist_spec->elements.push_back(elem);
337 999 : }
338 : }
339 1729 : }
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 4177 : UpdateInfo *McastForwarder::GetUpdateInfo(ErmVpnTable *table) {
352 4177 : CHECK_CONCURRENCY("db::DBTable");
353 :
354 4177 : assert(level_ == McastTreeManager::LevelNative);
355 :
356 4177 : BgpOListSpec olist_spec(BgpAttribute::OList);
357 4177 : AddLocalOListElems(&olist_spec);
358 4177 : AddGlobalOListElems(&olist_spec);
359 :
360 : // Bail if there is no label allocated.
361 4177 : if (label_ == 0)
362 643 : return NULL;
363 :
364 3534 : BgpAttrSpec attr_spec;
365 3534 : attr_spec.push_back(&olist_spec);
366 3534 : BgpAttrPtr attr = table->server()->attr_db()->Locate(attr_spec);
367 :
368 3534 : UpdateInfo *uinfo = new UpdateInfo;
369 3534 : uinfo->roattr = RibOutAttr(table, route_, attr.get(), label_, true, true);
370 5724 : if (route_ && sg_entry_->IsForestNode(this) &&
371 2190 : sg_entry_->IsTreeBuilder(McastTreeManager::LevelLocal)) {
372 1287 : table->GetMvpnSourceAddress(route_, uinfo->roattr.source_address());
373 : }
374 3534 : return uinfo;
375 4177 : }
376 :
377 : //
378 : // Constructor for McastSGEntry.
379 : //
380 25880 : McastSGEntry::McastSGEntry(McastManagerPartition *partition,
381 25880 : Ip4Address group, Ip4Address source)
382 25880 : : partition_(partition),
383 25880 : group_(group),
384 25880 : source_(source),
385 25880 : forest_node_(NULL),
386 25880 : local_tree_route_(NULL),
387 25880 : tree_result_route_(NULL),
388 51760 : on_work_queue_(false) {
389 77640 : for (int level = McastTreeManager::LevelFirst;
390 77640 : level < McastTreeManager::LevelCount; ++level) {
391 51760 : ForwarderSet *forwarders = new ForwarderSet;
392 51760 : forwarder_sets_.push_back(forwarders);
393 51760 : update_needed_.push_back(false);
394 : }
395 25880 : }
396 :
397 : //
398 : // Destructor for McastSGEntry.
399 : //
400 26284 : McastSGEntry::~McastSGEntry() {
401 25880 : STLDeleteValues(&forwarder_sets_);
402 26284 : }
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 1210 : void McastSGEntry::ChangeForwarder(McastForwarder *forwarder) {
430 1210 : uint8_t level = forwarder->level();
431 1210 : update_needed_[level] = true;
432 1210 : partition_->EnqueueSGEntry(this);
433 1210 : }
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 416 : 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 3850 : const RouteDistinguisher &McastSGEntry::GetSourceRd() const {
456 3850 : if (!forest_node_)
457 126 : return RouteDistinguisher::kZeroRd;
458 3724 : 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 1286 : void McastSGEntry::AddLocalTreeRoute() {
475 1286 : assert(!forest_node_);
476 1286 : 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 1286 : uint8_t level = McastTreeManager::LevelNative;
482 1286 : ForwarderSet *forwarders = forwarder_sets_[level];
483 1286 : for (ForwarderSet::reverse_iterator rit = forwarders->rbegin();
484 1293 : rit != forwarders->rend(); ++rit) {
485 945 : McastForwarder *forwarder = *rit;
486 945 : if (forwarder->label()) {
487 938 : forest_node_ = forwarder;
488 938 : break;
489 : }
490 : }
491 :
492 : // Bail if we couldn't designate a forest node.
493 1286 : if (!forest_node_)
494 348 : return;
495 :
496 : // Construct the prefix and route key.
497 938 : BgpServer *server = partition_->server();
498 938 : Ip4Address router_id(server->bgp_identifier());
499 : ErmVpnPrefix prefix(ErmVpnPrefix::LocalTreeRoute,
500 938 : RouteDistinguisher::kZeroRd, router_id, group_, source_);
501 938 : ErmVpnRoute rt_key(prefix);
502 :
503 : // Find or create the route.
504 : DBTablePartition *tbl_partition =
505 938 : static_cast<DBTablePartition *>(partition_->GetTablePartition());
506 : ErmVpnRoute *route =
507 938 : static_cast<ErmVpnRoute *>(tbl_partition->Find(&rt_key));
508 938 : if (!route) {
509 348 : route = new ErmVpnRoute(prefix);
510 348 : tbl_partition->Add(route);
511 : } else {
512 590 : route->ClearDelete();
513 : }
514 :
515 : // Build the attributes.
516 938 : BgpAttrSpec attr_spec;
517 938 : BgpAttrNextHop nexthop(server->bgp_identifier());
518 938 : attr_spec.push_back(&nexthop);
519 938 : BgpAttrSourceRd source_rd(GetSourceRd());
520 938 : attr_spec.push_back(&source_rd);
521 938 : EdgeDiscoverySpec edspec;
522 3752 : for (int idx = 1; idx <= McastTreeManager::kDegree - 1; ++idx) {
523 2814 : EdgeDiscoverySpec::Edge *edge = new EdgeDiscoverySpec::Edge;
524 2814 : edge->SetIp4Address(forest_node_->address());
525 2814 : edge->SetLabels(forest_node_->label(), forest_node_->label());
526 2814 : edspec.edge_list.push_back(edge);
527 : }
528 938 : attr_spec.push_back(&edspec);
529 : // Add tunnel encaps for remote nodes
530 938 : ExtCommunitySpec ext;
531 938 : ext.AddTunnelEncaps(forest_node_->encap());
532 938 : if (!ext.communities.empty())
533 184 : attr_spec.push_back(&ext);
534 938 : BgpAttrPtr attr = server->attr_db()->Locate(attr_spec);
535 :
536 : // Add a path with source BgpPath::Local.
537 938 : BgpPath *path = new BgpPath(0, BgpPath::Local, attr);
538 938 : route->InsertPath(path);
539 938 : tbl_partition->Notify(route);
540 938 : local_tree_route_ = route;
541 938 : }
542 :
543 : //
544 : // Delete the LocalTreeRoute for this McastSGEntry.
545 : //
546 1286 : void McastSGEntry::DeleteLocalTreeRoute() {
547 1286 : if (!local_tree_route_)
548 348 : return;
549 :
550 938 : forest_node_ = NULL;
551 : DBTablePartition *tbl_partition =
552 938 : static_cast<DBTablePartition *>(partition_->GetTablePartition());
553 938 : local_tree_route_->RemovePath(BgpPath::Local);
554 938 : if (!local_tree_route_->HasPaths()) {
555 938 : tbl_partition->Delete(local_tree_route_);
556 : } else {
557 0 : tbl_partition->Notify(local_tree_route_);
558 : }
559 938 : local_tree_route_ = NULL;
560 : }
561 :
562 : //
563 : // Update the LocalTreeRoute for this McastSGEntry if RouterId has changed.
564 : //
565 722 : void McastSGEntry::UpdateLocalTreeRoute() {
566 722 : if (!local_tree_route_)
567 722 : return;
568 :
569 : // Bail if the RouterId hasn't changed.
570 605 : const BgpServer *server = partition_->server();
571 605 : Ip4Address router_id = local_tree_route_->GetPrefix().router_id();
572 605 : if (router_id.to_ulong() == server->bgp_identifier())
573 605 : 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 3622 : void McastSGEntry::UpdateRoutes(uint8_t level) {
584 3622 : if (level == McastTreeManager::LevelNative) {
585 1286 : DeleteLocalTreeRoute();
586 1286 : AddLocalTreeRoute();
587 : } else {
588 2336 : ForwarderSet *forwarders = forwarder_sets_[level];
589 2336 : for (ForwarderSet::iterator it = forwarders->begin();
590 5889 : it != forwarders->end(); ++it) {
591 3553 : (*it)->DeleteGlobalTreeRoute();
592 3553 : (*it)->AddGlobalTreeRoute();
593 : }
594 : }
595 3622 : }
596 :
597 3401 : ErmVpnRoute *McastSGEntry::GetGlobalTreeRootRoute() const {
598 3401 : if (!IsTreeBuilder(McastTreeManager::LevelLocal))
599 1662 : return NULL;
600 1739 : ForwarderSet *forwarders = forwarder_sets_[McastTreeManager::LevelLocal];
601 1739 : assert(!forwarders->empty());
602 1739 : ForwarderSet::const_iterator it = forwarders->begin();
603 1739 : return (*it)->global_tree_route();
604 : }
605 :
606 : //
607 : // Implement tree builder election.
608 : //
609 9217 : bool McastSGEntry::IsTreeBuilder(uint8_t level) const {
610 9217 : if (level == McastTreeManager::LevelNative)
611 1288 : return true;
612 :
613 7929 : const ForwarderSet *forwarders = forwarder_sets_[level];
614 7929 : ForwarderSet::const_iterator it = forwarders->begin();
615 7929 : if (it == forwarders->end())
616 677 : return false;
617 :
618 7252 : Ip4Address router_id(partition_->server()->bgp_identifier());
619 7252 : if ((*it)->router_id() != router_id)
620 3243 : return false;
621 :
622 4009 : 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 6954 : void McastSGEntry::UpdateTree(uint8_t level) {
636 6954 : CHECK_CONCURRENCY("db::DBTable");
637 :
638 6954 : if (!update_needed_[level])
639 4687 : return;
640 3622 : update_needed_[level] = false;
641 :
642 : int degree;
643 3622 : if (level == McastTreeManager::LevelNative) {
644 1286 : degree = McastTreeManager::kDegree;
645 : } else {
646 2336 : 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 3622 : ForwarderSet *forwarders = forwarder_sets_[level];
653 3622 : for (ForwarderSet::iterator it = forwarders->begin();
654 10552 : it != forwarders->end(); ++it) {
655 6930 : (*it)->FlushLinks();
656 6930 : (*it)->ReleaseLabel();
657 6930 : partition_->GetTablePartition()->Notify((*it)->route());
658 : }
659 :
660 : // Bail if we're not the tree builder.
661 3622 : if (!IsTreeBuilder(level)) {
662 1355 : UpdateRoutes(level);
663 1355 : 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 2267 : McastForwarderList vec;
670 2267 : vec.reserve(forwarders->size());
671 2267 : for (ForwarderSet::iterator it = forwarders->begin();
672 7164 : it != forwarders->end(); ++it) {
673 4897 : McastForwarder *forwarder = *it;
674 4897 : forwarder->AllocateLabel();
675 4897 : if (!forwarder->label())
676 10 : continue;
677 4887 : 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 7154 : for (McastForwarderList::iterator it = vec.begin(); it != vec.end(); ++it) {
684 4887 : int idx = it - vec.begin();
685 4887 : if (idx == 0)
686 1918 : continue;
687 :
688 2969 : int parent_idx = (idx - 1) / degree;
689 2969 : McastForwarderList::iterator parent_it = vec.begin() + parent_idx;
690 2969 : assert(parent_it != vec.end());
691 2969 : McastForwarder *forwarder = *it;
692 2969 : McastForwarder *parent_forwarder = *parent_it;
693 2969 : forwarder->AddLink(parent_forwarder);
694 2969 : parent_forwarder->AddLink(forwarder);
695 : }
696 :
697 : // Update [Local|Global]TreeRoutes.
698 2267 : UpdateRoutes(level);
699 2267 : }
700 :
701 : //
702 : // Update distribution trees for both levels.
703 : //
704 3477 : void McastSGEntry::UpdateTree() {
705 10431 : for (uint8_t level = McastTreeManager::LevelFirst;
706 10431 : level < McastTreeManager::LevelCount; ++level) {
707 6954 : UpdateTree(level);
708 : }
709 3477 : }
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 2101 : void McastSGEntry::NotifyForestNode() {
717 2101 : if (!forest_node_)
718 424 : return;
719 1677 : partition_->GetTablePartition()->Notify(forest_node_->route());
720 : }
721 :
722 177 : bool McastSGEntry::GetForestNodePMSI(uint32_t *label, Ip4Address *address,
723 : vector<string> *tunnel_encap) const {
724 177 : if (!forest_node_)
725 0 : return false;
726 177 : *label = forest_node_->label();
727 177 : *address = forest_node_->address();
728 177 : *tunnel_encap = forest_node_->encap();
729 177 : return true;
730 : }
731 :
732 7711 : bool McastSGEntry::IsForestNode(McastForwarder *forwarder) {
733 7711 : return (forwarder == forest_node_);
734 : }
735 :
736 3477 : bool McastSGEntry::empty() const {
737 3477 : if (local_tree_route_ || tree_result_route_)
738 2782 : return false;
739 695 : if (!forwarder_sets_[McastTreeManager::LevelNative]->empty())
740 3 : 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 42873 : McastManagerPartition::McastManagerPartition(McastTreeManager *tree_manager,
750 42873 : size_t part_id)
751 42873 : : tree_manager_(tree_manager),
752 42873 : part_id_(part_id),
753 42874 : update_count_(0),
754 85747 : work_queue_(TaskScheduler::GetInstance()->GetTaskId("db::DBTable"),
755 42873 : part_id_,
756 42873 : boost::bind(&McastManagerPartition::ProcessSGEntry, this, _1)) {
757 42874 : }
758 :
759 : //
760 : // Destructor for McastManagerPartition.
761 : //
762 42874 : McastManagerPartition::~McastManagerPartition() {
763 42874 : work_queue_.Shutdown();
764 42874 : }
765 :
766 : // Find the McastSGEntry for the given group and source.
767 21898 : McastSGEntry *McastManagerPartition::FindSGEntry(
768 : const Ip4Address &group, const Ip4Address &source) {
769 : return const_cast<McastSGEntry *>(
770 21898 : static_cast<const McastManagerPartition *>(this)->FindSGEntry(group,
771 21898 : source));
772 : }
773 :
774 : //
775 : // Find the McastSGEntry for the given group and source.
776 : //
777 25476 : const McastSGEntry *McastManagerPartition::FindSGEntry(
778 : const Ip4Address &group, const Ip4Address &source) const {
779 : McastSGEntry temp_sg_entry(const_cast<McastManagerPartition *>(this),
780 25476 : group, source);
781 25476 : SGList::const_iterator it = sg_list_.find(&temp_sg_entry);
782 50952 : return (it != sg_list_.end() ? *it : NULL);
783 25476 : }
784 :
785 : //
786 : // Find or create the McastSGEntry for the given group and source.
787 : //
788 2359 : McastSGEntry *McastManagerPartition::LocateSGEntry(
789 : Ip4Address group, Ip4Address source) {
790 2359 : McastSGEntry *sg_entry = FindSGEntry(group, source);
791 2359 : if (!sg_entry) {
792 404 : sg_entry = new McastSGEntry(this, group, source);
793 404 : sg_list_.insert(sg_entry);
794 : }
795 2359 : return sg_entry;
796 : }
797 :
798 3401 : ErmVpnRoute *McastManagerPartition::GetGlobalTreeRootRoute(
799 : const Ip4Address &source, const Ip4Address &group) const {
800 3401 : const McastSGEntry *sg = FindSGEntry(group, source);
801 3401 : return sg ? sg->GetGlobalTreeRootRoute() : NULL;
802 : }
803 :
804 7983 : void McastManagerPartition::NotifyForestNode(
805 : const Ip4Address &source, const Ip4Address &group) {
806 7983 : McastSGEntry *sg = FindSGEntry(group, source);
807 7983 : if (sg)
808 310 : sg->NotifyForestNode();
809 7983 : }
810 :
811 177 : bool McastManagerPartition::GetForestNodePMSI(ErmVpnRoute *rt, uint32_t *label,
812 : Ip4Address *address, vector<string> *encap) const {
813 177 : const McastSGEntry *sg = FindSGEntry(rt->GetPrefix().group(),
814 177 : rt->GetPrefix().source());
815 177 : 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 5491 : void McastManagerPartition::EnqueueSGEntry(McastSGEntry *sg_entry) {
822 5491 : if (sg_entry->on_work_queue())
823 2014 : return;
824 3477 : work_queue_.Enqueue(sg_entry);
825 3477 : 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 3477 : bool McastManagerPartition::ProcessSGEntry(McastSGEntry *sg_entry) {
833 3477 : CHECK_CONCURRENCY("db::DBTable");
834 :
835 3477 : sg_entry->clear_on_work_queue();
836 3477 : sg_entry->UpdateTree();
837 3477 : update_count_++;
838 :
839 3477 : if (sg_entry->empty()) {
840 404 : sg_list_.erase(sg_entry);
841 404 : delete sg_entry;
842 : }
843 :
844 3477 : if (sg_list_.empty())
845 289 : tree_manager_->RetryDelete();
846 :
847 3477 : return true;
848 : }
849 :
850 : //
851 : // Get the DBTablePartBase for the ErmVpnTable for our partition id.
852 : //
853 13269 : DBTablePartBase *McastManagerPartition::GetTablePartition() {
854 13269 : 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 8795 : BgpServer *McastManagerPartition::server() {
862 8795 : 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 42907 : deleter_.reset(new DeleteActor(this));
877 42907 : }
878 :
879 : //
880 : // Destructor for McastTreeManager.
881 : //
882 66863 : McastTreeManager::~McastTreeManager() {
883 66863 : }
884 :
885 : //
886 : // Initialize the McastTreeManager. We allocate the McastManagerPartitions
887 : // and register a DBListener for the ErmVpnTable.
888 : //
889 42873 : void McastTreeManager::Initialize() {
890 42873 : AllocPartitions();
891 42873 : listener_id_ = table_->Register(
892 : boost::bind(&McastTreeManager::RouteListener, this, _1, _2),
893 : "McastTreeManager");
894 42874 : }
895 :
896 : //
897 : // Terminate the McastTreeManager. We free the McastManagerPartitions
898 : // and unregister from the ErmVpnTable.
899 : //
900 42874 : void McastTreeManager::Terminate() {
901 42874 : table_->Unregister(listener_id_);
902 42874 : FreePartitions();
903 42874 : }
904 :
905 : //
906 : // Allocate the McastManagerPartitions.
907 : //
908 42874 : void McastTreeManager::AllocPartitions() {
909 85747 : for (int part_id = 0; part_id < table_->PartitionCount(); part_id++) {
910 42874 : partitions_.push_back(new McastManagerPartition(this, part_id));
911 : }
912 42872 : }
913 :
914 : //
915 : // Free the McastManagerPartitions.
916 : //
917 42874 : void McastTreeManager::FreePartitions() {
918 85748 : for (size_t part_id = 0; part_id < partitions_.size(); part_id++) {
919 42874 : delete partitions_[part_id];
920 : }
921 42874 : partitions_.clear();
922 42874 : }
923 :
924 7985 : McastManagerPartition *McastTreeManager::GetPartition(int part_id) {
925 7985 : return partitions_[part_id];
926 : }
927 :
928 3578 : const McastManagerPartition *McastTreeManager::GetPartition(int part_id) const {
929 3578 : return partitions_[part_id];
930 : }
931 :
932 : //
933 : // Get the DBTablePartBase for the ErmVpnTable for given partition id.
934 : //
935 13269 : DBTablePartBase *McastTreeManager::GetTablePartition(size_t part_id) {
936 13269 : 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 3207 : UpdateInfo *McastTreeManager::GetUpdateInfo(ErmVpnRoute *route) {
944 3207 : CHECK_CONCURRENCY("db::DBTable");
945 :
946 3207 : DBState *dbstate = route->GetState(table_, listener_id_);
947 3207 : McastForwarder *forwarder = dynamic_cast<McastForwarder *>(dbstate);
948 :
949 3207 : if (!forwarder)
950 0 : return NULL;
951 :
952 3207 : 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 13416 : void McastTreeManager::TreeNodeListener(McastManagerPartition *partition,
964 : ErmVpnRoute *route) {
965 13416 : CHECK_CONCURRENCY("db::DBTable");
966 :
967 13416 : DBState *dbstate = route->GetState(table_, listener_id_);
968 13416 : 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 722 : sg_entry->UpdateLocalTreeRoute();
988 : } else {
989 11473 : McastSGEntry *sg_entry = partition->FindSGEntry(
990 11473 : route->GetPrefix().group(), route->GetPrefix().source());
991 11473 : assert(sg_entry);
992 11473 : McastForwarder *forwarder = dynamic_cast<McastForwarder *>(dbstate);
993 11473 : assert(forwarder);
994 :
995 11473 : 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 9551 : } else if (forwarder->Update(route)) {
1001 : // Trigger update of the distribution tree.
1002 1210 : 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 9587 : void McastTreeManager::TreeResultListener(McastManagerPartition *partition,
1013 : ErmVpnRoute *route) {
1014 9587 : CHECK_CONCURRENCY("db::DBTable");
1015 :
1016 9587 : DBState *dbstate = route->GetState(table_, listener_id_);
1017 9587 : if (!dbstate) {
1018 : // We have no previous DBState for this route.
1019 : // Bail if the route is not valid.
1020 8233 : if (!route->IsValid())
1021 6000 : return;
1022 :
1023 : // Ignore GlobalTreeRoute if it's not applicable to this control-node.
1024 2233 : BgpServer *server = table_->routing_instance()->server();
1025 2233 : if (route->GetPrefix().router_id().to_ulong() !=
1026 2233 : server->bgp_identifier())
1027 1796 : return;
1028 :
1029 437 : McastSGEntry *sg_entry = partition->LocateSGEntry(
1030 437 : route->GetPrefix().group(), route->GetPrefix().source());
1031 437 : route->SetState(table_, listener_id_, sg_entry);
1032 437 : sg_entry->set_tree_result_route(route);
1033 437 : sg_entry->NotifyForestNode();
1034 : } else {
1035 1354 : McastSGEntry *sg_entry = dynamic_cast<McastSGEntry *>(dbstate);
1036 1354 : assert(sg_entry);
1037 :
1038 1354 : if (!route->IsValid()) {
1039 437 : sg_entry->clear_tree_result_route();
1040 437 : route->ClearState(table_, listener_id_);
1041 437 : partition->EnqueueSGEntry(sg_entry);
1042 : }
1043 1354 : 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 23003 : void McastTreeManager::RouteListener(
1053 : DBTablePartBase *tpart, DBEntryBase *db_entry) {
1054 23003 : CHECK_CONCURRENCY("db::DBTable");
1055 :
1056 23003 : McastManagerPartition *partition = partitions_[tpart->index()];
1057 23003 : ErmVpnRoute *route = dynamic_cast<ErmVpnRoute *>(db_entry);
1058 23003 : if (route->GetPrefix().type() == ErmVpnPrefix::GlobalTreeRoute) {
1059 9587 : TreeResultListener(partition, route);
1060 : } else {
1061 13416 : TreeNodeListener(partition, route);
1062 : }
1063 23003 : }
1064 :
1065 :
1066 : //
1067 : // Check if the McastTreeManager can be deleted. This can happen only if all
1068 : // the McastManagerPartitions are empty.
1069 : //
1070 42907 : bool McastTreeManager::MayDelete() const {
1071 42907 : CHECK_CONCURRENCY("bgp::Config");
1072 :
1073 42907 : for (PartitionList::const_iterator it = partitions_.begin();
1074 85781 : it != partitions_.end(); ++it) {
1075 42874 : if (!(*it)->empty())
1076 0 : return false;
1077 : }
1078 :
1079 42907 : return true;
1080 : }
1081 :
1082 : //
1083 : // Initiate shutdown for the McastTreeManager.
1084 : //
1085 42907 : void McastTreeManager::Shutdown() {
1086 42907 : CHECK_CONCURRENCY("bgp::Config");
1087 42907 : }
1088 :
1089 : //
1090 : // Trigger deletion of the McastTreeManager and propagate the delete to any
1091 : // dependents.
1092 : //
1093 42907 : void McastTreeManager::ManagedDelete() {
1094 42907 : deleter_->Delete();
1095 42907 : }
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 6654 : LifetimeActor *McastTreeManager::deleter() {
1110 6654 : 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 3401 : ErmVpnRoute *McastTreeManager::GetGlobalTreeRootRoute(
1129 : const Ip4Address &source, const Ip4Address &group) const {
1130 3401 : const McastManagerPartition *partition = GetPartition(table_->Hash(group));
1131 3401 : return partition->GetGlobalTreeRootRoute(source, group);
1132 : }
1133 :
1134 7983 : void McastTreeManager::NotifyForestNode(int part_id, const Ip4Address &source,
1135 : const Ip4Address &group) {
1136 7983 : McastManagerPartition *partition = GetPartition(part_id);
1137 7983 : partition->NotifyForestNode(source, group);
1138 7983 : }
1139 :
1140 246 : bool McastTreeManager::GetForestNodePMSI(ErmVpnRoute *rt, uint32_t *label,
1141 : Ip4Address *address, vector<string> *encap) const {
1142 246 : if (!rt || !rt->IsUsable())
1143 69 : return false;
1144 : const McastManagerPartition *partition =
1145 177 : GetPartition(rt->get_table_partition()->index());
1146 177 : return partition->GetForestNodePMSI(rt, label, address, encap);
1147 : }
|