Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include "bgp/bgp_attr.h"
6 :
7 : #include <algorithm>
8 : #include <string>
9 :
10 : #include "bgp/bgp_server.h"
11 : #include "bgp/extended-community/esi_label.h"
12 : #include "bgp/extended-community/etree.h"
13 : #include "bgp/extended-community/mac_mobility.h"
14 : #include "bgp/extended-community/router_mac.h"
15 : #include "net/bgp_af.h"
16 :
17 : using std::sort;
18 : using std::vector;
19 : using std::string;
20 :
21 9985 : int BgpAttrOrigin::CompareTo(const BgpAttribute &rhs_attr) const {
22 9985 : int ret = BgpAttribute::CompareTo(rhs_attr);
23 9985 : if (ret != 0) return ret;
24 9985 : KEY_COMPARE(origin, static_cast<const BgpAttrOrigin &>(rhs_attr).origin);
25 9985 : return 0;
26 : }
27 :
28 184290 : void BgpAttrOrigin::ToCanonical(BgpAttr *attr) {
29 184290 : attr->set_origin(static_cast<BgpAttrOrigin::OriginType>(origin));
30 184292 : }
31 :
32 111197 : string BgpAttrOrigin::ToString() const {
33 : char repr[80];
34 111197 : snprintf(repr, sizeof(repr), "ORIGIN <code: %d, flags: %02x> : %02x",
35 111197 : code, flags, origin);
36 111197 : return string(repr);
37 : }
38 :
39 9981 : int BgpAttrNextHop::CompareTo(const BgpAttribute &rhs_attr) const {
40 9981 : int ret = BgpAttribute::CompareTo(rhs_attr);
41 9981 : if (ret != 0) return ret;
42 9981 : KEY_COMPARE(nexthop,
43 : static_cast<const BgpAttrNextHop &>(rhs_attr).nexthop);
44 9981 : return 0;
45 : }
46 :
47 191419 : void BgpAttrNextHop::ToCanonical(BgpAttr *attr) {
48 191419 : if (v6_nexthop.is_unspecified()) {
49 188887 : attr->set_nexthop(Ip4Address(nexthop));
50 : } else {
51 2532 : attr->set_nexthop(v6_nexthop);
52 : }
53 191419 : }
54 :
55 3207 : string BgpAttrNextHop::ToString() const {
56 : char repr[80];
57 3207 : snprintf(repr, sizeof(repr), "NEXTHOP <code: %d, flags: %02x> : %04x",
58 3207 : code, flags, nexthop);
59 3207 : return string(repr);
60 : }
61 :
62 1988 : int BgpAttrMultiExitDisc::CompareTo(const BgpAttribute &rhs_attr) const {
63 1988 : int ret = BgpAttribute::CompareTo(rhs_attr);
64 1988 : if (ret != 0) return ret;
65 1988 : KEY_COMPARE(med, static_cast<const BgpAttrMultiExitDisc &>(rhs_attr).med);
66 1986 : return 0;
67 : }
68 76368 : void BgpAttrMultiExitDisc::ToCanonical(BgpAttr *attr) {
69 76368 : attr->set_med(med);
70 76368 : }
71 :
72 51529 : string BgpAttrMultiExitDisc::ToString() const {
73 : char repr[80];
74 51529 : snprintf(repr, sizeof(repr), "MED <code: %d, flags: %02x> : %d",
75 51529 : code, flags, med);
76 51529 : return string(repr);
77 : }
78 :
79 1994 : int BgpAttrLocalPref::CompareTo(const BgpAttribute &rhs_attr) const {
80 1994 : int ret = BgpAttribute::CompareTo(rhs_attr);
81 1994 : if (ret != 0) return ret;
82 1994 : KEY_COMPARE(local_pref,
83 : static_cast<const BgpAttrLocalPref &>(rhs_attr).local_pref);
84 1994 : return 0;
85 : }
86 92452 : void BgpAttrLocalPref::ToCanonical(BgpAttr *attr) {
87 92452 : attr->set_local_pref(local_pref);
88 92452 : }
89 :
90 24222 : string BgpAttrLocalPref::ToString() const {
91 : char repr[80];
92 24222 : snprintf(repr, sizeof(repr), "LOCAL_PREF <code: %d, flags: %02x> : %d",
93 24222 : code, flags, local_pref);
94 24222 : return string(repr);
95 : }
96 :
97 9 : void BgpAttrAtomicAggregate::ToCanonical(BgpAttr *attr) {
98 9 : attr->set_atomic_aggregate(true);
99 9 : }
100 :
101 1 : string BgpAttrAtomicAggregate::ToString() const {
102 : char repr[80];
103 1 : snprintf(repr, sizeof(repr), "ATOMIC_AGGR <code: %d, flags: %02x>",
104 1 : code, flags);
105 1 : return string(repr);
106 : }
107 :
108 3 : int BgpAttr4ByteAggregator::CompareTo(const BgpAttribute &rhs_attr) const {
109 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
110 3 : if (ret != 0) return ret;
111 3 : KEY_COMPARE(as_num,
112 : static_cast<const BgpAttr4ByteAggregator &>(rhs_attr).as_num);
113 3 : KEY_COMPARE(address,
114 : static_cast<const BgpAttr4ByteAggregator &>(rhs_attr).address);
115 3 : return 0;
116 : }
117 1 : void BgpAttr4ByteAggregator::ToCanonical(BgpAttr *attr) {
118 1 : attr->set_aggregator(as_num, Ip4Address(address));
119 1 : }
120 :
121 0 : string BgpAttr4ByteAggregator::ToString() const {
122 : char repr[80];
123 0 : snprintf(repr, sizeof(repr),
124 : "Aggregator <code: %d, flags: %02x> : %d:%08x",
125 0 : code, flags, as_num, address);
126 0 : return string(repr);
127 : }
128 :
129 1973 : int BgpAttrAggregator::CompareTo(const BgpAttribute &rhs_attr) const {
130 1973 : int ret = BgpAttribute::CompareTo(rhs_attr);
131 1973 : if (ret != 0) return ret;
132 1973 : KEY_COMPARE(as_num,
133 : static_cast<const BgpAttrAggregator &>(rhs_attr).as_num);
134 1971 : KEY_COMPARE(address,
135 : static_cast<const BgpAttrAggregator &>(rhs_attr).address);
136 1969 : return 0;
137 : }
138 8 : void BgpAttrAggregator::ToCanonical(BgpAttr *attr) {
139 8 : attr->set_aggregator(as_num, Ip4Address(address));
140 8 : }
141 :
142 1 : string BgpAttrAggregator::ToString() const {
143 : char repr[80];
144 1 : snprintf(repr, sizeof(repr),
145 : "Aggregator <code: %d, flags: %02x> : %d:%08x",
146 1 : code, flags, as_num, address);
147 1 : return string(repr);
148 : }
149 :
150 0 : int BgpAttrAs4Aggregator::CompareTo(const BgpAttribute &rhs_attr) const {
151 0 : int ret = BgpAttribute::CompareTo(rhs_attr);
152 0 : if (ret != 0) return ret;
153 0 : KEY_COMPARE(as_num,
154 : static_cast<const BgpAttrAs4Aggregator &>(rhs_attr).as_num);
155 0 : KEY_COMPARE(address,
156 : static_cast<const BgpAttrAs4Aggregator &>(rhs_attr).address);
157 0 : return 0;
158 : }
159 0 : void BgpAttrAs4Aggregator::ToCanonical(BgpAttr *attr) {
160 0 : attr->set_as4_aggregator(as_num, Ip4Address(address));
161 0 : }
162 :
163 0 : string BgpAttrAs4Aggregator::ToString() const {
164 : char repr[80];
165 0 : snprintf(repr, sizeof(repr),
166 : "Aggregator <code: %d, flags: %02x> : %d:%08x",
167 0 : code, flags, as_num, address);
168 0 : return string(repr);
169 : }
170 :
171 3 : int BgpAttrOriginatorId::CompareTo(const BgpAttribute &rhs_attr) const {
172 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
173 3 : if (ret != 0) return ret;
174 3 : KEY_COMPARE(originator_id,
175 : static_cast<const BgpAttrOriginatorId &>(rhs_attr).originator_id);
176 1 : return 0;
177 : }
178 :
179 18 : void BgpAttrOriginatorId::ToCanonical(BgpAttr *attr) {
180 18 : attr->set_originator_id(Ip4Address(originator_id));
181 18 : }
182 :
183 3 : string BgpAttrOriginatorId::ToString() const {
184 : char repr[80];
185 3 : snprintf(repr, sizeof(repr), "OriginatorId <code: %d, flags: 0x%02x> : %s",
186 6 : code, flags, Ip4Address(originator_id).to_string().c_str());
187 3 : return string(repr);
188 : }
189 :
190 9 : ClusterListSpec::ClusterListSpec(uint32_t cluster_id,
191 9 : const ClusterListSpec *rhs)
192 9 : : BgpAttribute(BgpAttribute::ClusterList, kFlags) {
193 9 : cluster_list.push_back(cluster_id);
194 9 : if (rhs) {
195 1 : cluster_list.insert(cluster_list.end(), rhs->cluster_list.begin(),
196 : rhs->cluster_list.end());
197 : }
198 9 : }
199 :
200 46 : int ClusterListSpec::CompareTo(const BgpAttribute &rhs_attr) const {
201 46 : int ret = BgpAttribute::CompareTo(rhs_attr);
202 46 : if (ret != 0) return ret;
203 46 : KEY_COMPARE(cluster_list,
204 : static_cast<const ClusterListSpec &>(rhs_attr).cluster_list);
205 38 : return 0;
206 : }
207 :
208 8 : void ClusterListSpec::ToCanonical(BgpAttr *attr) {
209 8 : attr->set_cluster_list(this);
210 8 : }
211 :
212 3 : string ClusterListSpec::ToString() const {
213 3 : std::stringstream repr;
214 3 : repr << "CLUSTER_LIST <code: " << int(code);
215 3 : repr << ", flags: 0x" << std::hex << int(flags) << "> :";
216 3 : for (vector<uint32_t>::const_iterator iter = cluster_list.begin();
217 9 : iter != cluster_list.end(); ++iter) {
218 6 : repr << " " << Ip4Address(*iter).to_string();
219 : }
220 6 : return repr.str();
221 3 : }
222 :
223 14 : bool ClusterListSpec::ClusterListLoop(uint32_t cluster_id) const {
224 14 : for (vector<uint32_t>::const_iterator iter = cluster_list.begin();
225 56 : iter != cluster_list.end(); ++iter) {
226 47 : if (*iter == cluster_id)
227 5 : return true;
228 : }
229 9 : return false;
230 : }
231 :
232 22 : ClusterList::ClusterList(ClusterListDB *cluster_list_db,
233 22 : const ClusterListSpec &spec)
234 22 : : cluster_list_db_(cluster_list_db),
235 22 : spec_(spec) {
236 22 : refcount_ = 0;
237 22 : }
238 :
239 18 : void ClusterList::Remove() {
240 18 : cluster_list_db_->Delete(this);
241 18 : }
242 :
243 9742 : ClusterListDB::ClusterListDB(BgpServer *server) {
244 9742 : }
245 :
246 1990 : int BgpMpNlri::CompareTo(const BgpAttribute &rhs_attr) const {
247 1990 : int ret = BgpAttribute::CompareTo(rhs_attr);
248 1990 : if (ret != 0) return ret;
249 1990 : const BgpMpNlri &rhs = static_cast<const BgpMpNlri &>(rhs_attr);
250 1990 : KEY_COMPARE(afi, rhs.afi);
251 1990 : KEY_COMPARE(safi, rhs.safi);
252 1990 : KEY_COMPARE(nexthop, rhs.nexthop);
253 1990 : KEY_COMPARE(nlri.size(), rhs.nlri.size());
254 :
255 498340 : for (size_t i = 0; i < nlri.size(); i++) {
256 496350 : KEY_COMPARE(nlri[i]->type, rhs.nlri[i]->type);
257 496350 : KEY_COMPARE(nlri[i]->prefixlen, rhs.nlri[i]->prefixlen);
258 496350 : KEY_COMPARE(nlri[i]->prefix, rhs.nlri[i]->prefix);
259 : }
260 1990 : return 0;
261 : }
262 :
263 158195 : void BgpMpNlri::ToCanonical(BgpAttr *attr) {
264 158195 : }
265 :
266 507383 : size_t BgpMpNlri::EncodeLength() const {
267 507383 : size_t sz = 2 /* safi */ + 1 /* afi */ +
268 : 1 /* NlriNextHopLength */ +
269 : 1 /* Reserved */;
270 507383 : sz += nexthop.size();
271 507382 : for (vector<BgpProtoPrefix*>::const_iterator iter = nlri.begin();
272 2858661 : iter != nlri.end(); ++iter) {
273 2351300 : size_t bytes = 0;
274 2351300 : if (afi == BgpAf::L2Vpn &&
275 39698 : (safi == BgpAf::EVpn || safi == BgpAf::ErmVpn)) {
276 39698 : bytes = (*iter)->prefixlen;
277 : } else {
278 2311602 : bytes = ((*iter)->prefixlen + 7) / 8;
279 : }
280 2351297 : sz += 1 + bytes;
281 : }
282 507340 : return sz;
283 : }
284 :
285 25402 : PmsiTunnelSpec::PmsiTunnelSpec()
286 : : BgpAttribute(PmsiTunnel, kFlags),
287 25402 : tunnel_flags(0), tunnel_type(0), label(0) {
288 25401 : }
289 :
290 3007 : PmsiTunnelSpec::PmsiTunnelSpec(const BgpAttribute &rhs)
291 3007 : : BgpAttribute(rhs), tunnel_flags(0), tunnel_type(0), label(0) {
292 3007 : }
293 :
294 479946 : int PmsiTunnelSpec::CompareTo(const BgpAttribute &rhs_attr) const {
295 479946 : int ret = BgpAttribute::CompareTo(rhs_attr);
296 479946 : if (ret != 0) return ret;
297 479946 : const PmsiTunnelSpec &rhs = static_cast<const PmsiTunnelSpec &>(rhs_attr);
298 479946 : KEY_COMPARE(tunnel_flags, rhs.tunnel_flags);
299 468323 : KEY_COMPARE(tunnel_type, rhs.tunnel_type);
300 468316 : KEY_COMPARE(label, rhs.label);
301 444400 : KEY_COMPARE(identifier, rhs.identifier);
302 441705 : return 0;
303 : }
304 :
305 8133 : void PmsiTunnelSpec::ToCanonical(BgpAttr *attr) {
306 8133 : attr->set_pmsi_tunnel(this);
307 8133 : }
308 :
309 835 : string PmsiTunnelSpec::ToString() const {
310 835 : std::ostringstream oss;
311 835 : oss << "PmsiTunnel <code: " << int(code);
312 835 : oss << ", flags: 0x" << std::hex << int(flags) << std::dec << ">";
313 835 : oss << " Tunnel Flags: 0x" << std::hex << int(tunnel_flags) << std::dec;
314 835 : oss << " Tunnel Type: " << int(tunnel_type);
315 835 : oss << " Label: 0x" << std::hex << int(label) << std::dec;
316 835 : oss << " (" << int(label) << ")";
317 835 : oss << " Identifier: " << GetIdentifier().to_string();
318 :
319 1670 : return oss.str();
320 835 : }
321 :
322 2 : uint32_t PmsiTunnelSpec::GetLabel(const ExtCommunity *ext) const {
323 2 : assert(ext);
324 2 : bool is_vni = ext->ContainsTunnelEncapVxlan();
325 2 : return (is_vni ? label : (label >> 4));
326 : }
327 :
328 7748 : void PmsiTunnelSpec::SetLabel(uint32_t in_label, const ExtCommunity *ext) {
329 7748 : assert(ext);
330 7748 : bool is_vni = ext->ContainsTunnelEncapVxlan();
331 7747 : label = (is_vni ? in_label : (in_label << 4 | 0x01));
332 7747 : }
333 :
334 18763 : Ip4Address PmsiTunnelSpec::GetIdentifier() const {
335 18763 : if (identifier.size() < 4)
336 9828 : return Ip4Address();
337 8935 : return Ip4Address(get_value(&identifier[0], 4));
338 : }
339 :
340 11690 : void PmsiTunnelSpec::SetIdentifier(Ip4Address in_identifier) {
341 11690 : identifier.resize(4, 0);
342 11691 : const Ip4Address::bytes_type &bytes = in_identifier.to_bytes();
343 11691 : std::copy(bytes.begin(), bytes.begin() + 4, identifier.begin());
344 11691 : }
345 :
346 9 : string PmsiTunnelSpec::GetTunnelTypeString() const {
347 9 : switch (tunnel_type) {
348 1 : case RsvpP2mpLsp:
349 1 : return "RsvpP2mpLsp";
350 1 : case LdpP2mpLsp:
351 1 : return "LdpP2mpLsp";
352 1 : case PimSsmTree:
353 1 : return "PimSsmTree";
354 1 : case PimSmTree:
355 1 : return "PimSmTree";
356 1 : case BidirPimTree:
357 1 : return "BidirPimTree";
358 1 : case IngressReplication:
359 1 : return "IngressReplication";
360 1 : case MldpMp2mpLsp:
361 1 : return "MldpMp2mpLsp";
362 1 : case AssistedReplicationContrail:
363 1 : return "AssistedReplication";
364 1 : default:
365 1 : break;
366 : }
367 2 : std::ostringstream oss;
368 1 : oss << "Unknown(" << int(tunnel_type) << ")";
369 1 : return oss.str();
370 : }
371 :
372 4 : string PmsiTunnelSpec::GetTunnelArTypeString() const {
373 4 : switch (tunnel_flags & AssistedReplicationType) {
374 1 : case RegularNVE:
375 1 : return "RegularNVE";
376 1 : case ARReplicator:
377 1 : return "ARReplicator";
378 1 : case ARLeaf:
379 1 : return "ARLeaf";
380 1 : default:
381 1 : break;
382 : }
383 1 : return "Unknown";
384 : }
385 :
386 4 : vector<string> PmsiTunnelSpec::GetTunnelFlagsStrings() const {
387 4 : vector<string> flags;
388 4 : if (tunnel_flags & LeafInfoRequired) {
389 2 : flags.push_back("LeafInfoRequired");
390 : }
391 4 : if (tunnel_flags & EdgeReplicationSupported) {
392 2 : flags.push_back("EdgeReplicationSupported");
393 : }
394 4 : if (flags.empty()) {
395 1 : flags.push_back("None");
396 : }
397 4 : return flags;
398 0 : }
399 :
400 17926 : PmsiTunnel::PmsiTunnel(PmsiTunnelDB *pmsi_tunnel_db,
401 17926 : const PmsiTunnelSpec &pmsi_spec)
402 17926 : : pmsi_tunnel_db_(pmsi_tunnel_db),
403 17926 : pmsi_spec_(pmsi_spec) {
404 17916 : refcount_ = 0;
405 17926 : tunnel_flags_ = pmsi_spec_.tunnel_flags;
406 17926 : tunnel_type_ = pmsi_spec_.tunnel_type;
407 17926 : label_ = pmsi_spec_.label;
408 17926 : identifier_ = pmsi_spec_.GetIdentifier();
409 17925 : }
410 :
411 4965 : void PmsiTunnel::Remove() {
412 4965 : pmsi_tunnel_db_->Delete(this);
413 4965 : }
414 :
415 2595 : uint32_t PmsiTunnel::GetLabel(const ExtCommunity *ext) const {
416 2595 : bool is_vni = false;
417 2595 : if (ext)
418 2583 : is_vni = ext->ContainsTunnelEncapVxlan();
419 2595 : return (is_vni ? label_ : label_ >> 4);
420 : }
421 :
422 9742 : PmsiTunnelDB::PmsiTunnelDB(BgpServer *server) {
423 9742 : }
424 :
425 5938 : EdgeDiscoverySpec::EdgeDiscoverySpec()
426 5938 : : BgpAttribute(McastEdgeDiscovery, kFlags) {
427 5937 : }
428 :
429 3812 : EdgeDiscoverySpec::EdgeDiscoverySpec(const BgpAttribute &rhs)
430 3812 : : BgpAttribute(rhs) {
431 3812 : }
432 :
433 4184 : EdgeDiscoverySpec::EdgeDiscoverySpec(const EdgeDiscoverySpec &rhs)
434 4184 : : BgpAttribute(BgpAttribute::McastEdgeDiscovery, kFlags) {
435 13654 : for (size_t i = 0; i < rhs.edge_list.size(); i++) {
436 9470 : Edge *edge = new Edge;
437 9470 : *edge = *rhs.edge_list[i];
438 9470 : edge_list.push_back(edge);
439 : }
440 4184 : }
441 :
442 22764 : EdgeDiscoverySpec::~EdgeDiscoverySpec() {
443 13914 : STLDeleteValues(&edge_list);
444 22733 : }
445 :
446 9975 : Ip4Address EdgeDiscoverySpec::Edge::GetIp4Address() const {
447 9975 : return Ip4Address(get_value(&address[0], 4));
448 : }
449 :
450 130538 : void EdgeDiscoverySpec::Edge::SetIp4Address(Ip4Address addr) {
451 130538 : address.resize(4, 0);
452 130538 : const Ip4Address::bytes_type &addr_bytes = addr.to_bytes();
453 130538 : std::copy(addr_bytes.begin(), addr_bytes.begin() + 4, address.begin());
454 130538 : }
455 :
456 9975 : void EdgeDiscoverySpec::Edge::GetLabels(
457 : uint32_t *first_label, uint32_t *last_label) const {
458 9975 : *first_label = labels[0];
459 9975 : *last_label = labels[1];
460 9975 : }
461 :
462 130538 : void EdgeDiscoverySpec::Edge::SetLabels(
463 : uint32_t first_label, uint32_t last_label) {
464 130538 : labels.push_back(first_label);
465 130538 : labels.push_back(last_label);
466 130538 : }
467 :
468 : struct EdgeDiscoverySpecEdgeCompare {
469 63710 : int operator()(const EdgeDiscoverySpec::Edge *lhs,
470 : const EdgeDiscoverySpec::Edge *rhs) const {
471 63710 : KEY_COMPARE(lhs->address, rhs->address);
472 63710 : KEY_COMPARE(lhs->labels, rhs->labels);
473 63710 : return 0;
474 : }
475 : };
476 :
477 2003 : int EdgeDiscoverySpec::CompareTo(const BgpAttribute &rhs_attr) const {
478 2003 : int ret = BgpAttribute::CompareTo(rhs_attr);
479 2003 : if (ret != 0) return ret;
480 2003 : const EdgeDiscoverySpec &rhs =
481 : static_cast<const EdgeDiscoverySpec &>(rhs_attr);
482 2003 : ret = STLSortedCompare(edge_list.begin(), edge_list.end(),
483 : rhs.edge_list.begin(), rhs.edge_list.end(),
484 : EdgeDiscoverySpecEdgeCompare());
485 2003 : return ret;
486 : }
487 :
488 2106 : void EdgeDiscoverySpec::ToCanonical(BgpAttr *attr) {
489 2106 : attr->set_edge_discovery(this);
490 2106 : }
491 :
492 1205 : string EdgeDiscoverySpec::ToString() const {
493 1205 : std::ostringstream oss;
494 1205 : oss << "EdgeDiscovery <code: " << int(code);
495 1205 : oss << ", flags: 0x" << std::hex << int(flags) << std::dec << ">";
496 1205 : int idx = 0;
497 1205 : for (EdgeList::const_iterator it = edge_list.begin();
498 4816 : it != edge_list.end(); ++it, ++idx) {
499 3611 : const Edge *edge = *it;
500 : uint32_t first_label, last_label;
501 3611 : edge->GetLabels(&first_label, &last_label);
502 3611 : oss << " Edge[" << idx << "] = (" << edge->GetIp4Address() << ", ";
503 3611 : oss << first_label << "-" << last_label << ")";
504 : }
505 :
506 2410 : return oss.str();
507 1205 : }
508 :
509 13918 : size_t EdgeDiscoverySpec::EncodeLength() const {
510 13918 : size_t sz = 0;
511 13918 : for (EdgeList::const_iterator iter = edge_list.begin();
512 276639 : iter != edge_list.end(); ++iter) {
513 262721 : sz += 2; /* AddressLen + LabelLen */
514 262721 : sz += (*iter)->address.size();
515 262721 : sz += (*iter)->labels.size() * sizeof(uint32_t);
516 : }
517 13918 : return sz;
518 : }
519 :
520 6336 : EdgeDiscovery::Edge::Edge(const EdgeDiscoverySpec::Edge *spec_edge) {
521 6336 : address = spec_edge->GetIp4Address();
522 : uint32_t first_label, last_label;
523 6336 : spec_edge->GetLabels(&first_label, &last_label);
524 6336 : label_block = new LabelBlock(first_label, last_label);
525 6336 : }
526 :
527 54709 : bool EdgeDiscovery::Edge::operator<(const Edge &rhs) const {
528 54709 : BOOL_KEY_COMPARE(address, rhs.address);
529 47436 : BOOL_KEY_COMPARE(label_block->first(), rhs.label_block->first());
530 42188 : BOOL_KEY_COMPARE(label_block->last(), rhs.label_block->last());
531 42188 : return false;
532 : }
533 :
534 3141 : EdgeDiscovery::EdgeDiscovery(EdgeDiscoveryDB *edge_discovery_db,
535 3141 : const EdgeDiscoverySpec &edspec)
536 3139 : : edge_discovery_db_(edge_discovery_db),
537 3141 : edspec_(edspec) {
538 3139 : refcount_ = 0;
539 3139 : for (EdgeDiscoverySpec::EdgeList::const_iterator it =
540 9474 : edspec_.edge_list.begin(); it != edspec_.edge_list.end(); ++it) {
541 6336 : Edge *edge = new Edge(*it);
542 6336 : edge_list.push_back(edge);
543 : }
544 3139 : sort(edge_list.begin(), edge_list.end(), EdgeDiscovery::EdgeCompare());
545 3139 : }
546 :
547 5258 : EdgeDiscovery::~EdgeDiscovery() {
548 3141 : STLDeleteValues(&edge_list);
549 5258 : }
550 :
551 : struct EdgeDiscoveryEdgeCompare {
552 21639 : int operator()(const EdgeDiscovery::Edge *lhs,
553 : const EdgeDiscovery::Edge *rhs) const {
554 21639 : KEY_COMPARE(*lhs, *rhs);
555 12686 : return 0;
556 : }
557 : };
558 :
559 975780 : int EdgeDiscovery::CompareTo(const EdgeDiscovery &rhs) const {
560 975780 : int result = STLSortedCompare(edge_list.begin(), edge_list.end(),
561 : rhs.edge_list.begin(), rhs.edge_list.end(),
562 : EdgeDiscoveryEdgeCompare());
563 975780 : return result;
564 : }
565 :
566 1844 : void EdgeDiscovery::Remove() {
567 1844 : edge_discovery_db_->Delete(this);
568 1844 : }
569 :
570 9742 : EdgeDiscoveryDB::EdgeDiscoveryDB(BgpServer *server) {
571 9742 : }
572 :
573 6400 : EdgeForwardingSpec::EdgeForwardingSpec()
574 6400 : : BgpAttribute(McastEdgeForwarding, kFlags) {
575 6400 : }
576 :
577 4564 : EdgeForwardingSpec::EdgeForwardingSpec(const BgpAttribute &rhs)
578 4564 : : BgpAttribute(rhs) {
579 4564 : }
580 :
581 6140 : EdgeForwardingSpec::EdgeForwardingSpec(const EdgeForwardingSpec &rhs)
582 6140 : : BgpAttribute(BgpAttribute::McastEdgeForwarding, kFlags) {
583 11220 : for (size_t i = 0; i < rhs.edge_list.size(); i++) {
584 5083 : Edge *edge = new Edge;
585 5083 : *edge = *rhs.edge_list[i];
586 5083 : edge_list.push_back(edge);
587 : }
588 6137 : }
589 :
590 27356 : EdgeForwardingSpec:: ~EdgeForwardingSpec() {
591 17089 : STLDeleteValues(&edge_list);
592 27333 : }
593 :
594 5436 : Ip4Address EdgeForwardingSpec::Edge::GetInboundIp4Address() const {
595 5436 : return Ip4Address(get_value(&inbound_address[0], 4));
596 : }
597 :
598 5436 : Ip4Address EdgeForwardingSpec::Edge::GetOutboundIp4Address() const {
599 5436 : return Ip4Address(get_value(&outbound_address[0], 4));
600 : }
601 :
602 126181 : void EdgeForwardingSpec::Edge::SetInboundIp4Address(Ip4Address addr) {
603 126181 : inbound_address.resize(4, 0);
604 126181 : const Ip4Address::bytes_type &addr_bytes = addr.to_bytes();
605 126181 : std::copy(addr_bytes.begin(), addr_bytes.begin() + 4,
606 : inbound_address.begin());
607 126181 : }
608 :
609 126181 : void EdgeForwardingSpec::Edge::SetOutboundIp4Address(Ip4Address addr) {
610 126181 : outbound_address.resize(4, 0);
611 126181 : const Ip4Address::bytes_type &addr_bytes = addr.to_bytes();
612 126181 : std::copy(addr_bytes.begin(), addr_bytes.begin() + 4,
613 : outbound_address.begin());
614 126181 : }
615 :
616 : struct EdgeForwardingSpecEdgeCompare {
617 59919 : int operator()(const EdgeForwardingSpec::Edge *lhs,
618 : const EdgeForwardingSpec::Edge *rhs) const {
619 59919 : KEY_COMPARE(lhs->inbound_address, rhs->inbound_address);
620 59919 : KEY_COMPARE(lhs->outbound_address, rhs->outbound_address);
621 59919 : KEY_COMPARE(lhs->inbound_label, rhs->inbound_label);
622 59919 : KEY_COMPARE(lhs->outbound_label, rhs->outbound_label);
623 59919 : return 0;
624 : }
625 : };
626 :
627 1898 : int EdgeForwardingSpec::CompareTo(const BgpAttribute &rhs_attr) const {
628 1898 : int ret = BgpAttribute::CompareTo(rhs_attr);
629 1898 : if (ret != 0) return ret;
630 1898 : const EdgeForwardingSpec &rhs =
631 : static_cast<const EdgeForwardingSpec &>(rhs_attr);
632 1898 : ret = STLSortedCompare(edge_list.begin(), edge_list.end(),
633 : rhs.edge_list.begin(), rhs.edge_list.end(),
634 : EdgeForwardingSpecEdgeCompare());
635 1898 : return ret;
636 : }
637 :
638 3386 : void EdgeForwardingSpec::ToCanonical(BgpAttr *attr) {
639 3386 : attr->set_edge_forwarding(this);
640 3386 : }
641 :
642 2013 : string EdgeForwardingSpec::ToString() const {
643 2013 : std::ostringstream oss;
644 2013 : oss << "EdgeForwarding <code: " << int(code);
645 2013 : oss << ", flags: 0x" << std::hex << int(flags) << std::dec << ">";
646 2013 : int idx = 0;
647 2013 : for (EdgeList::const_iterator it = edge_list.begin();
648 4221 : it != edge_list.end(); ++it, ++idx) {
649 2208 : const Edge *edge = *it;
650 2208 : oss << " Edge[" << idx << "] = (";
651 2208 : oss << "InAddress=" << edge->GetInboundIp4Address() << ", ";
652 2208 : oss << "InLabel=" << edge->inbound_label << ", ";
653 2208 : oss << "OutAddress=" << edge->GetOutboundIp4Address() << ", ";
654 2208 : oss << "OutLabel=" << edge->outbound_label << ")";
655 : }
656 :
657 4026 : return oss.str();
658 2013 : }
659 :
660 15998 : size_t EdgeForwardingSpec::EncodeLength() const {
661 15998 : size_t sz = 0;
662 15998 : for (EdgeList::const_iterator iter = edge_list.begin();
663 271035 : iter != edge_list.end(); ++iter) {
664 255037 : sz += 1 /* address len */ + 8 /* 2 labels */;
665 255037 : sz += (*iter)->inbound_address.size();
666 255037 : sz += (*iter)->outbound_address.size();
667 : }
668 :
669 15998 : return sz;
670 : }
671 :
672 3224 : EdgeForwarding::Edge::Edge(const EdgeForwardingSpec::Edge *spec_edge) {
673 3224 : inbound_address = spec_edge->GetInboundIp4Address();
674 3224 : outbound_address = spec_edge->GetOutboundIp4Address();
675 3224 : inbound_label = spec_edge->inbound_label;
676 3224 : outbound_label = spec_edge->outbound_label;
677 3224 : }
678 :
679 32576 : bool EdgeForwarding::Edge::operator<(const Edge &rhs) const {
680 32576 : BOOL_KEY_COMPARE(inbound_address, rhs.inbound_address);
681 22679 : BOOL_KEY_COMPARE(outbound_address, rhs.outbound_address);
682 18055 : BOOL_KEY_COMPARE(inbound_label, rhs.inbound_label);
683 15897 : BOOL_KEY_COMPARE(outbound_label, rhs.outbound_label);
684 13573 : return false;
685 : }
686 :
687 4421 : EdgeForwarding::EdgeForwarding(EdgeForwardingDB *edge_forwarding_db,
688 4421 : const EdgeForwardingSpec &efspec)
689 4420 : : edge_forwarding_db_(edge_forwarding_db),
690 4421 : efspec_(efspec) {
691 4417 : refcount_ = 0;
692 4415 : for (EdgeForwardingSpec::EdgeList::const_iterator it =
693 7640 : efspec_.edge_list.begin(); it != efspec_.edge_list.end(); ++it) {
694 3224 : Edge *edge = new Edge(*it);
695 3224 : edge_list.push_back(edge);
696 : }
697 4413 : sort(edge_list.begin(), edge_list.end(), EdgeForwarding::EdgeCompare());
698 4412 : }
699 :
700 7819 : EdgeForwarding::~EdgeForwarding() {
701 4421 : STLDeleteValues(&edge_list);
702 7819 : }
703 :
704 : struct EdgeForwardingEdgeCompare {
705 18584 : int operator()(const EdgeForwarding::Edge *lhs,
706 : const EdgeForwarding::Edge *rhs) const {
707 18584 : KEY_COMPARE(*lhs, *rhs);
708 6787 : return 0;
709 : }
710 : };
711 :
712 592168 : int EdgeForwarding::CompareTo(const EdgeForwarding &rhs) const {
713 592168 : int result = STLSortedCompare(edge_list.begin(), edge_list.end(),
714 : rhs.edge_list.begin(), rhs.edge_list.end(),
715 : EdgeForwardingEdgeCompare());
716 592167 : return result;
717 : }
718 :
719 2331 : void EdgeForwarding::Remove() {
720 2331 : edge_forwarding_db_->Delete(this);
721 2331 : }
722 :
723 9742 : EdgeForwardingDB::EdgeForwardingDB(BgpServer *server) {
724 9742 : }
725 :
726 199600 : bool BgpOListElem::operator<(const BgpOListElem &rhs) const {
727 199600 : BOOL_KEY_COMPARE(address, rhs.address);
728 131666 : BOOL_KEY_COMPARE(label, rhs.label);
729 127637 : BOOL_KEY_COMPARE(encap, rhs.encap);
730 127622 : return false;
731 : }
732 :
733 3 : int BgpOListSpec::CompareTo(const BgpAttribute &rhs_attr) const {
734 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
735 3 : if (ret != 0) return ret;
736 3 : return 0;
737 : }
738 :
739 3573 : void BgpOListSpec::ToCanonical(BgpAttr *attr) {
740 3573 : if (subcode == BgpAttribute::OList) {
741 3573 : attr->set_olist(this);
742 0 : } else if (subcode == BgpAttribute::LeafOList) {
743 0 : attr->set_leaf_olist(this);
744 : } else {
745 0 : assert(false);
746 : }
747 3573 : }
748 :
749 1 : string BgpOListSpec::ToString() const {
750 1 : std::ostringstream oss;
751 1 : oss << "OList <subcode: " << int(subcode) << ">";
752 1 : int idx = 0;
753 1 : for (Elements::const_iterator it = elements.begin();
754 3 : it != elements.end(); ++it, ++idx) {
755 2 : BgpOListElem elem(*it);
756 2 : oss << "OList[" << idx << "] = (";
757 2 : oss << "address: " << elem.address << ", ";
758 2 : oss << "label: " << elem.label << ", ";
759 2 : oss << "encap-list:";
760 2 : int eid = 0;
761 2 : for (vector<string>::const_iterator e_it = elem.encap.begin();
762 6 : e_it != elem.encap.end(); ++e_it, ++eid) {
763 4 : oss << " " << *e_it;
764 : }
765 2 : oss << ")";
766 2 : }
767 2 : return (oss.str());
768 1 : }
769 :
770 14006 : BgpOList::BgpOList(BgpOListDB *olist_db, const BgpOListSpec &olist_spec)
771 14006 : : olist_db_(olist_db),
772 14006 : olist_spec_(olist_spec) {
773 14003 : refcount_ = 0;
774 14006 : for (BgpOListSpec::Elements::const_iterator it =
775 37096 : olist_spec_.elements.begin(); it != olist_spec_.elements.end(); ++it) {
776 23090 : BgpOListElem *elem = new BgpOListElem(*it);
777 23090 : sort(elem->encap.begin(), elem->encap.end());
778 23090 : elements_.push_back(elem);
779 : }
780 14005 : sort(elements_.begin(), elements_.end(), BgpOListElemCompare());
781 14005 : }
782 :
783 26988 : BgpOList::~BgpOList() {
784 14006 : STLDeleteValues(&elements_);
785 26988 : }
786 :
787 : struct BgpOListElementCompare {
788 78334 : int operator()(const BgpOListElem *lhs, const BgpOListElem *rhs) const {
789 78334 : KEY_COMPARE(*lhs, *rhs);
790 63811 : return 0;
791 : }
792 : };
793 :
794 627341 : int BgpOList::CompareTo(const BgpOList &rhs) const {
795 627341 : KEY_COMPARE(olist().subcode, rhs.olist().subcode);
796 1224910 : int result = STLSortedCompare(elements().begin(), elements().end(),
797 1224910 : rhs.elements().begin(), rhs.elements().end(),
798 : BgpOListElementCompare());
799 612455 : return result;
800 : }
801 :
802 9672 : void BgpOList::Remove() {
803 9672 : olist_db_->Delete(this);
804 9672 : }
805 :
806 9742 : BgpOListDB::BgpOListDB(BgpServer *server) {
807 9742 : }
808 :
809 0 : int BgpAttrLabelBlock::CompareTo(const BgpAttribute &rhs_attr) const {
810 0 : int ret = BgpAttribute::CompareTo(rhs_attr);
811 0 : if (ret != 0) return ret;
812 0 : KEY_COMPARE(label_block.get(),
813 : static_cast<const BgpAttrLabelBlock &>(rhs_attr).label_block.get());
814 0 : return 0;
815 : }
816 :
817 1033 : void BgpAttrLabelBlock::ToCanonical(BgpAttr *attr) {
818 1033 : attr->set_label_block(label_block);
819 1033 : }
820 :
821 0 : string BgpAttrLabelBlock::ToString() const {
822 : char repr[80];
823 0 : snprintf(repr, sizeof(repr), "LabelBlock <subcode: %d> : %d-%d",
824 0 : subcode, label_block->first(), label_block->last());
825 0 : return string(repr);
826 : }
827 :
828 3 : int BgpAttrSourceRd::CompareTo(const BgpAttribute &rhs_attr) const {
829 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
830 3 : if (ret != 0) return ret;
831 3 : KEY_COMPARE(source_rd,
832 : static_cast<const BgpAttrSourceRd &>(rhs_attr).source_rd);
833 1 : return 0;
834 : }
835 :
836 28675 : void BgpAttrSourceRd::ToCanonical(BgpAttr *attr) {
837 28675 : attr->set_source_rd(source_rd);
838 28675 : }
839 :
840 1 : string BgpAttrSourceRd::ToString() const {
841 : char repr[80];
842 1 : snprintf(repr, sizeof(repr), "SourceRd <subcode: %d> : %s",
843 2 : subcode, source_rd.ToString().c_str());
844 1 : return string(repr);
845 : }
846 :
847 3 : int BgpAttrEsi::CompareTo(const BgpAttribute &rhs_attr) const {
848 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
849 3 : if (ret != 0) return ret;
850 3 : KEY_COMPARE(esi, static_cast<const BgpAttrEsi &>(rhs_attr).esi);
851 1 : return 0;
852 : }
853 :
854 59 : void BgpAttrEsi::ToCanonical(BgpAttr *attr) {
855 59 : attr->set_esi(esi);
856 59 : }
857 :
858 1 : string BgpAttrEsi::ToString() const {
859 : char repr[80];
860 1 : snprintf(repr, sizeof(repr), "Esi <subcode: %d> : %s",
861 2 : subcode, esi.ToString().c_str());
862 1 : return string(repr);
863 : }
864 :
865 3 : int BgpAttrParams::CompareTo(const BgpAttribute &rhs_attr) const {
866 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
867 3 : if (ret != 0) return ret;
868 3 : KEY_COMPARE(params, static_cast<const BgpAttrParams &>(rhs_attr).params);
869 1 : return 0;
870 : }
871 :
872 3 : void BgpAttrParams::ToCanonical(BgpAttr *attr) {
873 3 : attr->set_params(params);
874 3 : }
875 :
876 1 : string BgpAttrParams::ToString() const {
877 : char repr[80];
878 1 : snprintf(repr, sizeof(repr), "Params <subcode: %d> : 0x%016jx",
879 1 : subcode, params);
880 1 : return string(repr);
881 : }
882 :
883 3 : int BgpAttrSubProtocol::CompareTo(const BgpAttribute &rhs_attr) const {
884 3 : int ret = BgpAttribute::CompareTo(rhs_attr);
885 3 : if (ret != 0) return ret;
886 3 : KEY_COMPARE(sbp, static_cast<const BgpAttrSubProtocol &>(rhs_attr).sbp);
887 1 : return 0;
888 : }
889 :
890 21992 : void BgpAttrSubProtocol::ToCanonical(BgpAttr *attr) {
891 21992 : attr->set_sub_protocol(sbp);
892 21992 : }
893 :
894 1 : string BgpAttrSubProtocol::ToString() const {
895 : char repr[80];
896 1 : snprintf(repr, sizeof(repr), "SubProtocol <subcode: %d> : %s",
897 1 : subcode, sbp.c_str());
898 1 : return string(repr);
899 : }
900 :
901 38 : BgpAttr::BgpAttr()
902 38 : : attr_db_(NULL), origin_(BgpAttrOrigin::INCOMPLETE), nexthop_(),
903 38 : med_(0), local_pref_(0), atomic_aggregate_(false),
904 38 : aggregator_as_num_(0), aggregator_as4_num_(0), params_(0) {
905 38 : refcount_ = 0;
906 38 : }
907 :
908 5953 : BgpAttr::BgpAttr(BgpAttrDB *attr_db)
909 5953 : : attr_db_(attr_db), origin_(BgpAttrOrigin::INCOMPLETE),
910 11906 : nexthop_(), med_(0), local_pref_(0), atomic_aggregate_(false),
911 5953 : aggregator_as_num_(0), aggregator_as4_num_(0), params_(0) {
912 5953 : refcount_ = 0;
913 5953 : }
914 :
915 388986 : BgpAttr::BgpAttr(BgpAttrDB *attr_db, const BgpAttrSpec &spec)
916 388986 : : attr_db_(attr_db), origin_(BgpAttrOrigin::INCOMPLETE),
917 777968 : nexthop_(), med_(0),
918 388982 : local_pref_(BgpAttrLocalPref::kDefault),
919 388982 : atomic_aggregate_(false),
920 388982 : aggregator_as_num_(0), aggregator_as4_num_(0),
921 388986 : aggregator_address_(), params_(0) {
922 388973 : refcount_ = 0;
923 388988 : for (vector<BgpAttribute *>::const_iterator it = spec.begin();
924 1466347 : it < spec.end(); it++) {
925 1077357 : (*it)->ToCanonical(this);
926 : }
927 388979 : }
928 :
929 3021974 : BgpAttr::BgpAttr(const BgpAttr &rhs)
930 3021974 : : attr_db_(rhs.attr_db_), origin_(rhs.origin_), nexthop_(rhs.nexthop_),
931 3021616 : med_(rhs.med_), local_pref_(rhs.local_pref_),
932 3021616 : atomic_aggregate_(rhs.atomic_aggregate_),
933 3021616 : aggregator_as_num_(rhs.aggregator_as_num_),
934 3021616 : aggregator_as4_num_(rhs.aggregator_as4_num_),
935 3021616 : aggregator_address_(rhs.aggregator_address_),
936 3021435 : originator_id_(rhs.originator_id_),
937 3021363 : source_rd_(rhs.source_rd_), esi_(rhs.esi_), params_(rhs.params_),
938 3021284 : as_path_(rhs.as_path_),
939 3021952 : aspath_4byte_(rhs.aspath_4byte_),
940 3021855 : as4_path_(rhs.as4_path_),
941 3021709 : cluster_list_(rhs.cluster_list_),
942 3021623 : community_(rhs.community_),
943 3021569 : ext_community_(rhs.ext_community_),
944 3022412 : large_community_(rhs.large_community_),
945 3021967 : deferred_large_spec_(nullptr),
946 3021338 : origin_vn_path_(rhs.origin_vn_path_),
947 3021218 : pmsi_tunnel_(rhs.pmsi_tunnel_),
948 3021103 : edge_discovery_(rhs.edge_discovery_),
949 3020906 : edge_forwarding_(rhs.edge_forwarding_),
950 3020897 : label_block_(rhs.label_block_),
951 3020851 : olist_(rhs.olist_),
952 3020777 : leaf_olist_(rhs.leaf_olist_),
953 6042687 : sub_protocol_(rhs.sub_protocol_) {
954 3020488 : refcount_ = 0;
955 3022536 : }
956 :
957 19236 : void BgpAttr::set_as_path(AsPathPtr aspath) {
958 19236 : as_path_ = aspath;
959 19236 : }
960 :
961 361617 : void BgpAttr::set_as_path(const AsPathSpec *spec) {
962 361617 : if (spec) {
963 361487 : as_path_ = attr_db_->server()->aspath_db()->Locate(*spec);
964 : } else {
965 130 : as_path_ = NULL;
966 : }
967 361662 : }
968 :
969 0 : void BgpAttr::set_as4_path(As4PathPtr aspath) {
970 0 : as4_path_ = aspath;
971 0 : }
972 :
973 490 : void BgpAttr::set_as4_path(const As4PathSpec *spec) {
974 490 : if (spec) {
975 456 : as4_path_ = attr_db_->server()->as4path_db()->Locate(*spec);
976 : } else {
977 34 : as4_path_ = NULL;
978 : }
979 490 : }
980 :
981 0 : void BgpAttr::set_aspath_4byte(AsPath4BytePtr aspath) {
982 0 : aspath_4byte_ = aspath;
983 0 : }
984 :
985 544 : void BgpAttr::set_aspath_4byte(const AsPath4ByteSpec *spec) {
986 544 : if (spec) {
987 485 : aspath_4byte_ = attr_db_->server()->aspath_4byte_db()->Locate(*spec);
988 : } else {
989 59 : aspath_4byte_ = NULL;
990 : }
991 544 : }
992 :
993 202862 : void BgpAttr::set_cluster_list(const ClusterListSpec *spec) {
994 202862 : if (spec) {
995 18 : cluster_list_ = attr_db_->server()->cluster_list_db()->Locate(*spec);
996 : } else {
997 202844 : cluster_list_ = NULL;
998 : }
999 202860 : }
1000 :
1001 25337 : void BgpAttr::set_community(CommunityPtr comm) {
1002 25337 : community_ = comm;
1003 25337 : }
1004 :
1005 4331 : void BgpAttr::set_community(const CommunitySpec *comm) {
1006 4331 : if (comm) {
1007 4331 : community_ = attr_db_->server()->comm_db()->Locate(*comm);
1008 : } else {
1009 0 : community_ = NULL;
1010 : }
1011 4331 : }
1012 :
1013 1005309 : void BgpAttr::set_ext_community(ExtCommunityPtr extcomm) {
1014 1005309 : ext_community_ = extcomm;
1015 1005319 : }
1016 :
1017 183241 : void BgpAttr::set_ext_community(const ExtCommunitySpec *extcomm) {
1018 183241 : if (extcomm) {
1019 183241 : LargeCommunitySpec large_spec = LargeCommunitySpec::FromTag(*extcomm);
1020 183239 : set_large_community(&large_spec);
1021 :
1022 183240 : if (large_spec.communities.size()) {
1023 : ExtCommunitySpec new_comms =
1024 4 : LargeCommunitySpec::RemoveTags(*extcomm);
1025 : ext_community_ =
1026 4 : attr_db_->server()->extcomm_db()->Locate(new_comms);
1027 4 : return;
1028 4 : }
1029 183236 : ext_community_ = attr_db_->server()->extcomm_db()->Locate(*extcomm);
1030 183241 : } else {
1031 0 : ext_community_ = NULL;
1032 : }
1033 : }
1034 :
1035 18940 : void BgpAttr::set_large_community(LargeCommunityPtr largecomm) {
1036 18940 : large_community_ = largecomm;
1037 18940 : }
1038 :
1039 183283 : void BgpAttr::set_large_community(const LargeCommunitySpec *largecomm) {
1040 183283 : if (largecomm) {
1041 183283 : if (!bool(deferred_large_spec_)) {
1042 183239 : deferred_large_spec_.reset(new LargeCommunitySpec);
1043 183238 : deferred_large_spec_->communities = largecomm->communities;
1044 : } else {
1045 88 : deferred_large_spec_->communities.insert(
1046 88 : deferred_large_spec_->communities.end(),
1047 : largecomm->communities.begin(),
1048 : largecomm->communities.end());
1049 : }
1050 : large_community_ =
1051 183283 : attr_db_->server()->largecomm_db()->Locate(*deferred_large_spec_);
1052 : } else {
1053 0 : large_community_ = NULL;
1054 : }
1055 183284 : }
1056 :
1057 18941 : void BgpAttr::set_origin_vn_path(OriginVnPathPtr ovnpath) {
1058 18941 : origin_vn_path_ = ovnpath;
1059 18941 : }
1060 :
1061 725 : void BgpAttr::set_origin_vn_path(const OriginVnPathSpec *spec) {
1062 725 : if (spec) {
1063 725 : origin_vn_path_ = attr_db_->server()->ovnpath_db()->Locate(*spec);
1064 : } else {
1065 0 : origin_vn_path_ = NULL;
1066 : }
1067 725 : }
1068 :
1069 16892 : void BgpAttr::set_pmsi_tunnel(const PmsiTunnelSpec *pmsi_spec) {
1070 16892 : if (pmsi_spec) {
1071 16892 : pmsi_tunnel_ = attr_db_->server()->pmsi_tunnel_db()->Locate(*pmsi_spec);
1072 : } else {
1073 0 : pmsi_tunnel_ = NULL;
1074 : }
1075 16892 : }
1076 :
1077 2107 : void BgpAttr::set_edge_discovery(const EdgeDiscoverySpec *edspec) {
1078 2107 : if (edspec) {
1079 : edge_discovery_ =
1080 2107 : attr_db_->server()->edge_discovery_db()->Locate(*edspec);
1081 : } else {
1082 0 : edge_discovery_ = NULL;
1083 : }
1084 2107 : }
1085 :
1086 3388 : void BgpAttr::set_edge_forwarding(const EdgeForwardingSpec *efspec) {
1087 3388 : if (efspec) {
1088 : edge_forwarding_ =
1089 3388 : attr_db_->server()->edge_forwarding_db()->Locate(*efspec);
1090 : } else {
1091 0 : edge_forwarding_ = NULL;
1092 : }
1093 3388 : }
1094 :
1095 1034 : void BgpAttr::set_label_block(LabelBlockPtr label_block) {
1096 1034 : label_block_ = label_block;
1097 1034 : }
1098 :
1099 8344 : void BgpAttr::set_olist(const BgpOListSpec *olist_spec) {
1100 8344 : if (olist_spec) {
1101 8344 : olist_ = attr_db_->server()->olist_db()->Locate(*olist_spec);
1102 : } else {
1103 0 : olist_ = NULL;
1104 : }
1105 8344 : }
1106 :
1107 4618 : void BgpAttr::set_leaf_olist(const BgpOListSpec *leaf_olist_spec) {
1108 4618 : if (leaf_olist_spec) {
1109 4618 : leaf_olist_ = attr_db_->server()->olist_db()->Locate(*leaf_olist_spec);
1110 : } else {
1111 0 : leaf_olist_ = NULL;
1112 : }
1113 4618 : }
1114 :
1115 9331039 : int BgpAttr::max_as_path_count() const {
1116 9331039 : int as_count = as_path_count();
1117 9331067 : int as4_count = aspath_4byte_count();
1118 9331069 : return as_count > as4_count ? as_count : as4_count;
1119 : }
1120 :
1121 99098 : bool BgpAttr::IsAsPathEmpty() const {
1122 99098 : if (as_path_ && !as_path_->empty())
1123 332 : return false;
1124 98766 : if (aspath_4byte_ && !aspath_4byte_->empty())
1125 86 : return false;
1126 98679 : return true;
1127 : }
1128 :
1129 5604 : string BgpAttr::OriginToString(BgpAttrOrigin::OriginType origin) {
1130 5604 : switch (origin) {
1131 5249 : case BgpAttrOrigin::IGP:
1132 5249 : return "igp";
1133 : break;
1134 3 : case BgpAttrOrigin::EGP:
1135 3 : return "egp";
1136 : break;
1137 351 : case BgpAttrOrigin::INCOMPLETE:
1138 351 : return "incomplete";
1139 : break;
1140 : }
1141 1 : return "unknown";
1142 : }
1143 :
1144 20418 : BgpAttrOrigin::OriginType BgpAttr::OriginFromString(
1145 : const string &bgp_origin_type) {
1146 20418 : if(bgp_origin_type == "IGP"){
1147 20415 : return BgpAttrOrigin::IGP;
1148 3 : } else if( bgp_origin_type == "EGP") {
1149 2 : return BgpAttrOrigin::EGP;
1150 : }
1151 1 : return BgpAttrOrigin::INCOMPLETE;
1152 : }
1153 :
1154 5602 : string BgpAttr::origin_string() const {
1155 5602 : return OriginToString(origin());
1156 : }
1157 :
1158 2874 : Address::Family BgpAttr::nexthop_family() const {
1159 2874 : if (nexthop_.is_v6()) {
1160 325 : return Address::INET6;
1161 : } else {
1162 2549 : return Address::INET;
1163 : }
1164 : }
1165 :
1166 2652399 : as_t BgpAttr::neighbor_as() const {
1167 2652399 : if (as_path_.get())
1168 492097 : return as_path_->neighbor_as();
1169 2160295 : if (aspath_4byte_.get())
1170 224 : return aspath_4byte_->neighbor_as();
1171 2160076 : return 0;
1172 : }
1173 :
1174 9423512 : uint32_t BgpAttr::sequence_number() const {
1175 9423512 : if (!ext_community_)
1176 1136653 : return 0;
1177 8286876 : for (ExtCommunity::ExtCommunityList::const_iterator it =
1178 8286875 : ext_community_->communities().begin();
1179 28987650 : it != ext_community_->communities().end(); ++it) {
1180 20704943 : if (ExtCommunity::is_mac_mobility(*it)) {
1181 4321 : MacMobility mm(*it);
1182 4321 : return mm.sequence_number();
1183 : }
1184 : }
1185 8282461 : return 0;
1186 : }
1187 :
1188 4711856 : bool BgpAttr::evpn_sticky_mac() const {
1189 4711856 : if (!ext_community_)
1190 568328 : return 0;
1191 4143520 : for (ExtCommunity::ExtCommunityList::const_iterator it =
1192 4143543 : ext_community_->communities().begin();
1193 14494581 : it != ext_community_->communities().end(); ++it) {
1194 10353225 : if (ExtCommunity::is_mac_mobility(*it)) {
1195 2180 : MacMobility mm(*it);
1196 2180 : return mm.sticky();
1197 : }
1198 : }
1199 4141310 : return 0;
1200 : }
1201 :
1202 4711908 : bool BgpAttr::etree_leaf() const {
1203 4711908 : if (!ext_community_)
1204 568328 : return 0;
1205 4143580 : for (ExtCommunity::ExtCommunityList::const_iterator it =
1206 4143580 : ext_community_->communities().begin();
1207 14350360 : it != ext_community_->communities().end(); ++it) {
1208 10279901 : if (ExtCommunity::is_etree(*it)) {
1209 73151 : ETree etree(*it);
1210 73152 : return etree.leaf();
1211 : }
1212 : }
1213 4070368 : return 0;
1214 : }
1215 :
1216 63 : bool BgpAttr::evpn_single_active() const {
1217 63 : if (!ext_community_)
1218 0 : return false;
1219 63 : for (ExtCommunity::ExtCommunityList::const_iterator it =
1220 63 : ext_community_->communities().begin();
1221 126 : it != ext_community_->communities().end(); ++it) {
1222 126 : if (ExtCommunity::is_esi_label(*it)) {
1223 63 : EsiLabel esi_label(*it);
1224 63 : return esi_label.single_active();
1225 : }
1226 : }
1227 0 : return true;
1228 : }
1229 :
1230 463292 : MacAddress BgpAttr::mac_address() const {
1231 463292 : if (!ext_community_)
1232 13692 : return MacAddress::kZeroMac;
1233 449581 : for (ExtCommunity::ExtCommunityList::const_iterator it =
1234 449595 : ext_community_->communities().begin();
1235 3188462 : it != ext_community_->communities().end(); ++it) {
1236 2738907 : if (ExtCommunity::is_router_mac(*it)) {
1237 22 : RouterMac router_mac(*it);
1238 22 : return router_mac.mac_address();
1239 : }
1240 : }
1241 449484 : return MacAddress::kZeroMac;
1242 : }
1243 :
1244 438943 : void BgpAttr::Remove() {
1245 438943 : attr_db_->Delete(this);
1246 438947 : }
1247 :
1248 451775 : int BgpAttr::IsAsPathLoop(as_t asn, uint8_t max_loop_count) const {
1249 451775 : if (as_path() && as_path()->path().AsPathLoop(asn, max_loop_count)) {
1250 137728 : return true;
1251 : }
1252 314257 : if (aspath_4byte() &&
1253 192 : aspath_4byte()->path().AsPathLoop(asn, max_loop_count)) {
1254 49 : return true;
1255 : }
1256 314035 : if (asn > AS2_MAX && as4_path() &&
1257 20 : as4_path()->path().AsPathLoop(asn, max_loop_count)) {
1258 4 : return true;
1259 : }
1260 314011 : return false;
1261 : }
1262 :
1263 25545656 : int BgpAttr::CompareTo(const BgpAttr &rhs) const {
1264 25545656 : KEY_COMPARE(origin_, rhs.origin_);
1265 23818300 : KEY_COMPARE(nexthop_, rhs.nexthop_);
1266 16007276 : KEY_COMPARE(med_, rhs.med_);
1267 15245117 : KEY_COMPARE(local_pref_, rhs.local_pref_);
1268 14845462 : KEY_COMPARE(atomic_aggregate_, rhs.atomic_aggregate_);
1269 14845462 : KEY_COMPARE(aggregator_as_num_, rhs.aggregator_as_num_);
1270 14845270 : KEY_COMPARE(aggregator_as4_num_, rhs.aggregator_as4_num_);
1271 14845142 : KEY_COMPARE(aggregator_address_, rhs.aggregator_address_);
1272 14847746 : KEY_COMPARE(originator_id_, rhs.originator_id_);
1273 14846373 : KEY_COMPARE(pmsi_tunnel_.get(), rhs.pmsi_tunnel_.get());
1274 14491354 : KEY_COMPARE(edge_discovery_.get(), rhs.edge_discovery_.get());
1275 14429345 : KEY_COMPARE(edge_forwarding_.get(), rhs.edge_forwarding_.get());
1276 14366164 : KEY_COMPARE(esi_, rhs.esi_);
1277 14365963 : KEY_COMPARE(params_, rhs.params_);
1278 14365958 : KEY_COMPARE(source_rd_, rhs.source_rd_);
1279 10343629 : KEY_COMPARE(label_block_.get(), rhs.label_block_.get());
1280 10345691 : KEY_COMPARE(olist_.get(), rhs.olist_.get());
1281 10287074 : KEY_COMPARE(leaf_olist_.get(), rhs.leaf_olist_.get());
1282 10267215 : KEY_COMPARE(as_path_.get(), rhs.as_path_.get());
1283 9973127 : KEY_COMPARE(aspath_4byte_.get(), rhs.aspath_4byte_.get());
1284 9972083 : KEY_COMPARE(as4_path_.get(), rhs.as4_path_.get());
1285 9971139 : KEY_COMPARE(cluster_list_.get(), rhs.cluster_list_.get());
1286 9971122 : KEY_COMPARE(community_.get(), rhs.community_.get());
1287 9722999 : KEY_COMPARE(ext_community_.get(), rhs.ext_community_.get());
1288 7226339 : KEY_COMPARE(large_community_.get(), rhs.large_community_.get());
1289 7211581 : KEY_COMPARE(origin_vn_path_.get(), rhs.origin_vn_path_.get());
1290 7134572 : KEY_COMPARE(sub_protocol_, rhs.sub_protocol_);
1291 7118615 : return 0;
1292 : }
1293 :
1294 0 : std::size_t hash_value(BgpAttr const &attr) {
1295 0 : size_t hash = 0;
1296 :
1297 0 : boost::hash_combine(hash, attr.origin_);
1298 0 : boost::hash_combine(hash, attr.nexthop_.to_string());
1299 0 : boost::hash_combine(hash, attr.med_);
1300 0 : boost::hash_combine(hash, attr.local_pref_);
1301 0 : boost::hash_combine(hash, attr.atomic_aggregate_);
1302 0 : boost::hash_combine(hash, attr.aggregator_as_num_);
1303 0 : boost::hash_combine(hash, attr.aggregator_as4_num_);
1304 0 : boost::hash_combine(hash, attr.aggregator_address_.to_string());
1305 0 : boost::hash_combine(hash, attr.originator_id_.to_string());
1306 0 : boost::hash_combine(hash, attr.params_);
1307 0 : boost::hash_combine(hash, attr.source_rd_.ToString());
1308 0 : boost::hash_combine(hash, attr.esi_.ToString());
1309 :
1310 0 : if (attr.label_block_) {
1311 0 : boost::hash_combine(hash, attr.label_block_->first());
1312 0 : boost::hash_combine(hash, attr.label_block_->last());
1313 : }
1314 :
1315 0 : if (attr.olist_) {
1316 0 : boost::hash_range(hash, attr.olist_->elements().begin(),
1317 0 : attr.olist_->elements().end());
1318 : }
1319 0 : if (attr.leaf_olist_) {
1320 0 : boost::hash_range(hash, attr.leaf_olist_->elements().begin(),
1321 0 : attr.leaf_olist_->elements().end());
1322 : }
1323 :
1324 0 : if (attr.as_path_) boost::hash_combine(hash, *attr.as_path_);
1325 0 : if (attr.aspath_4byte_) boost::hash_combine(hash, *attr.aspath_4byte_);
1326 0 : if (attr.as4_path_) boost::hash_combine(hash, *attr.as4_path_);
1327 0 : if (attr.community_) boost::hash_combine(hash, *attr.community_);
1328 0 : if (attr.ext_community_) boost::hash_combine(hash, *attr.ext_community_);
1329 0 : if (attr.large_community_) boost::hash_combine(hash, *attr.large_community_);
1330 0 : if (attr.origin_vn_path_) boost::hash_combine(hash, *attr.origin_vn_path_);
1331 0 : if (!attr.sub_protocol_.empty()) {
1332 0 : boost::hash_combine(hash, attr.sub_protocol_);
1333 : }
1334 :
1335 0 : return hash;
1336 : }
1337 :
1338 9742 : BgpAttrDB::BgpAttrDB(BgpServer *server) : server_(server) {
1339 9742 : }
1340 :
1341 : // Return a clone of attribute with updated aspath.
1342 19236 : BgpAttrPtr BgpAttrDB::ReplaceAsPathAndLocate(const BgpAttr *attr,
1343 : AsPathPtr aspath) {
1344 19236 : BgpAttr *clone = new BgpAttr(*attr);
1345 19236 : clone->set_as_path(aspath);
1346 19236 : return Locate(clone);
1347 : }
1348 :
1349 : // Return a clone of attribute with updated community.
1350 19266 : BgpAttrPtr BgpAttrDB::ReplaceCommunityAndLocate(const BgpAttr *attr,
1351 : CommunityPtr community) {
1352 19266 : BgpAttr *clone = new BgpAttr(*attr);
1353 19266 : clone->set_community(community);
1354 19266 : return Locate(clone);
1355 : }
1356 :
1357 : //Return a clone of attribute with updated origin
1358 4 : BgpAttrPtr BgpAttrDB::ReplaceOriginAndLocate(const BgpAttr *attr,
1359 : BgpAttrOrigin::OriginType origin) {
1360 4 : BgpAttr *clone = new BgpAttr(*attr);
1361 4 : clone->set_origin(origin);
1362 4 : return Locate(clone);
1363 : }
1364 :
1365 : // Return a clone of attribute with updated extended community.
1366 1005286 : BgpAttrPtr BgpAttrDB::ReplaceExtCommunityAndLocate(const BgpAttr *attr,
1367 : ExtCommunityPtr extcomm) {
1368 1005286 : BgpAttr *clone = new BgpAttr(*attr);
1369 1005324 : clone->set_ext_community(extcomm);
1370 1005310 : return Locate(clone);
1371 : }
1372 :
1373 : // Return a clone of attribute with updated large community.
1374 18940 : BgpAttrPtr BgpAttrDB::ReplaceLargeCommunityAndLocate(
1375 : const BgpAttr *attr, LargeCommunityPtr largecomm) {
1376 18940 : BgpAttr *clone = new BgpAttr(*attr);
1377 18940 : clone->set_large_community(largecomm);
1378 18940 : return Locate(clone);
1379 : }
1380 :
1381 : // Return a clone of attribute with updated origin vn path.
1382 18941 : BgpAttrPtr BgpAttrDB::ReplaceOriginVnPathAndLocate(const BgpAttr *attr,
1383 : OriginVnPathPtr ovnpath) {
1384 18941 : BgpAttr *clone = new BgpAttr(*attr);
1385 18941 : clone->set_origin_vn_path(ovnpath);
1386 18941 : return Locate(clone);
1387 : }
1388 :
1389 : // Return a clone of attribute with updated local preference.
1390 18 : BgpAttrPtr BgpAttrDB::ReplaceLocalPreferenceAndLocate(const BgpAttr *attr,
1391 : uint32_t local_pref) {
1392 18 : BgpAttr *clone = new BgpAttr(*attr);
1393 18 : clone->set_local_pref(local_pref);
1394 18 : return Locate(clone);
1395 : }
1396 :
1397 : // Return a clone of attribute with updated originator id.
1398 1 : BgpAttrPtr BgpAttrDB::ReplaceOriginatorIdAndLocate(const BgpAttr *attr,
1399 : Ip4Address originator_id) {
1400 1 : BgpAttr *clone = new BgpAttr(*attr);
1401 1 : clone->set_originator_id(originator_id);
1402 1 : return Locate(clone);
1403 : }
1404 :
1405 : // Return a clone of attribute with updated source rd.
1406 604146 : BgpAttrPtr BgpAttrDB::ReplaceSourceRdAndLocate(const BgpAttr *attr,
1407 : const RouteDistinguisher &source_rd) {
1408 604146 : BgpAttr *clone = new BgpAttr(*attr);
1409 604179 : clone->set_source_rd(source_rd);
1410 604168 : return Locate(clone);
1411 : }
1412 :
1413 : // Return a clone of attribute with updated esi.
1414 17801 : BgpAttrPtr BgpAttrDB::ReplaceEsiAndLocate(const BgpAttr *attr,
1415 : const EthernetSegmentId &esi) {
1416 17801 : BgpAttr *clone = new BgpAttr(*attr);
1417 17801 : clone->set_esi(esi);
1418 17801 : return Locate(clone);
1419 : }
1420 :
1421 : // Return a clone of attribute with updated olist.
1422 4771 : BgpAttrPtr BgpAttrDB::ReplaceOListAndLocate(const BgpAttr *attr,
1423 : const BgpOListSpec *olist_spec) {
1424 4771 : assert(olist_spec->subcode == BgpAttribute::OList);
1425 4771 : BgpAttr *clone = new BgpAttr(*attr);
1426 4771 : clone->set_olist(olist_spec);
1427 4771 : return Locate(clone);
1428 : }
1429 :
1430 : // Return a clone of attribute with updated leaf olist.
1431 4618 : BgpAttrPtr BgpAttrDB::ReplaceLeafOListAndLocate(const BgpAttr *attr,
1432 : const BgpOListSpec *leaf_olist_spec) {
1433 4618 : assert(leaf_olist_spec->subcode == BgpAttribute::LeafOList);
1434 4618 : BgpAttr *clone = new BgpAttr(*attr);
1435 4618 : clone->set_leaf_olist(leaf_olist_spec);
1436 4618 : return Locate(clone);
1437 : }
1438 :
1439 : // Return a clone of attribute with updated sub-protocol.
1440 2048 : BgpAttrPtr BgpAttrDB::ReplaceSubProtocolAndLocate(const BgpAttr *attr,
1441 : const string &sbp) {
1442 2048 : BgpAttr *clone = new BgpAttr(*attr);
1443 2048 : clone->set_sub_protocol(sbp);
1444 2048 : return Locate(clone);
1445 : }
1446 :
1447 :
1448 : // Return a clone of attribute with updated pmsi tunnel.
1449 8759 : BgpAttrPtr BgpAttrDB::ReplacePmsiTunnelAndLocate(const BgpAttr *attr,
1450 : const PmsiTunnelSpec *pmsi_spec) {
1451 8759 : BgpAttr *clone = new BgpAttr(*attr);
1452 8759 : clone->set_pmsi_tunnel(pmsi_spec);
1453 8759 : return Locate(clone);
1454 : }
1455 :
1456 : // Return a clone of attribute with updated nexthop.
1457 339090 : BgpAttrPtr BgpAttrDB::ReplaceNexthopAndLocate(const BgpAttr *attr,
1458 : const IpAddress &addr) {
1459 339090 : BgpAttr *clone = new BgpAttr(*attr);
1460 339097 : clone->set_nexthop(addr);
1461 339096 : return Locate(clone);
1462 : }
|