LCOV - code coverage report
Current view: top level - bgp - xmpp_message_builder.cc (source / functions) Hit Total Coverage
Test: OpenSDN C/C++ coverage (all TARGET_SET jobs) Lines: 397 413 96.1 %
Date: 2026-06-11 01:56:02 Functions: 31 31 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*
       2             :  * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
       3             :  */
       4             : 
       5             : #include "bgp/xmpp_message_builder.h"
       6             : 
       7             : #include <boost/foreach.hpp>
       8             : 
       9             : #include <algorithm>
      10             : 
      11             : #include "bgp/ipeer.h"
      12             : #include "bgp/bgp_server.h"
      13             : #include "bgp/bgp_table.h"
      14             : #include "bgp/extended-community/etree.h"
      15             : #include "bgp/extended-community/mac_mobility.h"
      16             : #include "bgp/ermvpn/ermvpn_route.h"
      17             : #include "bgp/evpn/evpn_route.h"
      18             : #include "bgp/mvpn/mvpn_route.h"
      19             : #include "bgp/routing-instance/routing_instance.h"
      20             : #include "bgp/security_group/security_group.h"
      21             : #include "db/db.h"
      22             : #include "net/community_type.h"
      23             : #include "schema/xmpp_multicast_types.h"
      24             : #include "schema/xmpp_mvpn_types.h"
      25             : #include "schema/xmpp_enet_types.h"
      26             : #include "xmpp/xmpp_init.h"
      27             : 
      28             : using pugi::xml_attribute;
      29             : using pugi::xml_document;
      30             : using pugi::xml_node;
      31             : using std::ostringstream;
      32             : using std::copy;
      33             : using std::fill;
      34             : using std::string;
      35             : using std::stringstream;
      36             : using std::vector;
      37             : 
      38      472827 : static inline const char *AfiName(uint16_t afi) {
      39      472827 :     switch (afi) {
      40      264551 :     case BgpAf::IPv4:
      41      264551 :         return "1";
      42             :         break;
      43      197906 :     case BgpAf::IPv6:
      44      197906 :         return "2";
      45             :         break;
      46       10397 :     case BgpAf::L2Vpn:
      47       10397 :         return "25";
      48             :         break;
      49             :     }
      50           0 :     assert(false);
      51             :     return NULL;
      52             : }
      53             : 
      54      472831 : static inline const char *XmppSafiName(uint8_t safi) {
      55      472831 :     switch (safi) {
      56      453255 :     case BgpAf::Unicast:
      57      453255 :         return "1";
      58             :         break;
      59           6 :     case BgpAf::Mpls:
      60           6 :         return "4";
      61             :         break;
      62        6478 :     case BgpAf::MVpn:
      63        6478 :         return "5";
      64             :         break;
      65        2699 :     case BgpAf::Mcast:
      66        2699 :         return "241";
      67             :         break;
      68       10397 :     case BgpAf::Enet:
      69       10397 :         return "242";
      70             :         break;
      71             :     }
      72           0 :     assert(false);
      73             :     return NULL;
      74             : }
      75             : 
      76         143 : BgpXmppMessage::BgpXmppMessage()
      77         144 :     : table_(NULL),
      78         144 :       writer_(XmlWriter(&repr_)),
      79         144 :       is_reachable_(false),
      80         144 :       cache_routes_(false),
      81         144 :       repr_valid_(false),
      82         144 :       mobility_(0, false),
      83         287 :       etree_leaf_(false) {
      84         144 :     msg_begin_.reserve(kMaxFromToLength);
      85         144 : }
      86             : 
      87         288 : BgpXmppMessage::~BgpXmppMessage() {
      88         288 : }
      89             : 
      90      472754 : void BgpXmppMessage::Reset() {
      91      472754 :     Message::Reset();
      92      472757 :     table_ = NULL;
      93      472757 :     is_reachable_ = false;
      94      472757 :     cache_routes_ = false;
      95      472757 :     repr_valid_ = false;
      96      472757 :     repr_.clear();
      97      472747 : }
      98             : 
      99      472741 : bool BgpXmppMessage::Start(const RibOut *ribout, bool cache_routes,
     100             :     const RibOutAttr *roattr, const BgpRoute *route) {
     101      472741 :     Reset();
     102      472743 :     table_ = ribout->table();
     103      472740 :     is_reachable_ = roattr->IsReachable();
     104      472735 :     cache_routes_ = cache_routes;
     105      472735 :     Address::Family family = table_->family();
     106             : 
     107      472737 :     if (is_reachable_) {
     108      317143 :         const BgpAttr *attr = roattr->attr();
     109      317144 :         ProcessCommunity(attr->community());
     110      317141 :         ProcessExtCommunity(attr->ext_community());
     111             :     }
     112             : 
     113             :     // Reserve space for the begin line that contains the message opening tag
     114             :     // with from and to attributes. Actual value gets patched in when GetData
     115             :     // is called.
     116      472786 :     repr_.append(kMaxFromToLength, ' ');
     117             : 
     118             :     // Add opening tags for event and items. The closing tags are added when
     119             :     // GetData is called.
     120      472791 :     repr_ += "\n\t<event xmlns=\"http://jabber.org/protocol/pubsub\">";
     121      472814 :     repr_ += "\n\t\t<items node=\"";
     122      472834 :     repr_ += AfiName(BgpAf::FamilyToAfi(family));
     123      472840 :     repr_ += "/";
     124      472851 :     repr_ += XmppSafiName(BgpAf::FamilyToXmppSafi(family));
     125      472833 :     repr_ += "/";
     126      472832 :     repr_ += table_->routing_instance()->name();
     127      472827 :     repr_ += "\">\n";
     128             : 
     129      472832 :     if (table_->family() == Address::ERMVPN) {
     130        2699 :         AddMcastRoute(route, roattr);
     131      470125 :     } else if (table_->family() == Address::EVPN) {
     132       10397 :         AddEnetRoute(route, roattr);
     133      459712 :     } else if (table_->family() == Address::INET6) {
     134      197898 :         AddInet6Route(route, roattr);
     135      261822 :     } else if (table_->family() == Address::MVPN) {
     136        6478 :         AddMvpnRoute(route, roattr);
     137             :     } else {
     138      255362 :         AddInetRoute(route, roattr);
     139             :     }
     140      472883 :     return true;
     141             : }
     142             : 
     143      472896 : void BgpXmppMessage::Finish() {
     144      472896 : }
     145             : 
     146      615940 : bool BgpXmppMessage::AddRoute(const BgpRoute *route, const RibOutAttr *roattr) {
     147      615940 :     assert(is_reachable_ == roattr->IsReachable());
     148      615932 :     if (is_reachable_ && num_reach_route_ >= kMaxReachCount)
     149         191 :         return false;
     150      615741 :     if (!is_reachable_ && num_unreach_route_ >= kMaxUnreachCount)
     151           8 :         return false;
     152             : 
     153      615733 :     if (is_reachable_) {
     154      519163 :         const BgpAttr *attr = roattr->attr();
     155      519160 :         ProcessCommunity(attr->community());
     156      519160 :         ProcessExtCommunity(attr->ext_community());
     157             :     }
     158             : 
     159      615739 :     if (table_->family() == Address::ERMVPN) {
     160         139 :         return AddMcastRoute(route, roattr);
     161      615596 :     } else if (table_->family() == Address::EVPN) {
     162        2237 :         return AddEnetRoute(route, roattr);
     163      613355 :     } else if (table_->family() == Address::INET6) {
     164      121355 :         return AddInet6Route(route, roattr);
     165      491998 :     } else if (table_->family() == Address::MVPN) {
     166      190464 :         return AddMvpnRoute(route, roattr);
     167             :     } else {
     168      301579 :         return AddInetRoute(route, roattr);
     169             :     }
     170             : }
     171             : 
     172      530485 : void BgpXmppMessage::EncodeNextHop(const BgpRoute *route,
     173             :                                    const RibOutAttr::NextHop &nexthop,
     174             :                                    autogen::ItemType *item) {
     175      530485 :     autogen::NextHopType item_nexthop;
     176             : 
     177      530472 :     const IpAddress &address = nexthop.address();
     178      530469 :     if (address.is_v4()) {
     179      527172 :         item_nexthop.af = BgpAf::IPv4;
     180      527172 :         item_nexthop.address = address.to_v4().to_string();
     181             :     } else {
     182        3296 :         item_nexthop.af = BgpAf::IPv6;
     183        3296 :         item_nexthop.address = address.to_v6().to_string();
     184             :     }
     185      530735 :     item_nexthop.label = nexthop.label();
     186      530728 :     item_nexthop.virtual_network = GetVirtualNetwork(nexthop);
     187      530727 :     item_nexthop.tag_list.tag = nexthop.tag_list();
     188      530553 :     item_nexthop.is_new_tags_list = true;
     189             : 
     190             :     // If there's a non-zero label and encap list is empty use mpls over gre
     191             :     // as default encap.
     192      530553 :     if (item_nexthop.label) {
     193      521527 :         vector<string> &encap_list =
     194             :             item_nexthop.tunnel_encapsulation_list.tunnel_encapsulation;
     195      521527 :         if (nexthop.encap().empty()) {
     196      100616 :             encap_list.push_back(string("gre"));
     197             :         } else {
     198      420928 :             encap_list = nexthop.encap();
     199             :         }
     200             :     }
     201             : 
     202      530483 :     item->entry.next_hops.next_hop.push_back(item_nexthop);
     203      530546 : }
     204             : 
     205      626507 : void BgpXmppMessage::AddIpReach(const BgpRoute *route,
     206             :                                 const RibOutAttr *roattr) {
     207      626507 :     if (!roattr->repr().empty()) {
     208       98208 :         repr_ += roattr->repr();
     209       98208 :         return;
     210             :     }
     211      528313 :     Address::Family family = table_->family();
     212             : 
     213      528306 :     autogen::ItemType item;
     214      528279 :     item.entry.nlri.af = BgpAf::FamilyToAfi(family);
     215      528274 :     item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
     216      528274 :     item.entry.nlri.address = route->ToString();
     217      528377 :     item.entry.version = 1;
     218      528377 :     item.entry.virtual_network = GetVirtualNetwork(route, roattr);
     219      528382 :     item.entry.local_preference = roattr->attr()->local_pref();
     220      528345 :     item.entry.med = roattr->attr()->med();
     221      528332 :     item.entry.sequence_number = mobility_.sequence_number;
     222      528332 :     item.entry.mobility.seqno = mobility_.sequence_number;
     223      528332 :     item.entry.mobility.sticky = mobility_.sticky;
     224             : 
     225      528332 :     assert(!roattr->nexthop_list().empty());
     226             : 
     227             :     // Encode all next-hops in the list.
     228     1589352 :     BOOST_FOREACH(const RibOutAttr::NextHop &nexthop, roattr->nexthop_list()) {
     229      530497 :         EncodeNextHop(route, nexthop, &item);
     230             :     }
     231             : 
     232      528246 :     for (vector<int>::const_iterator it = security_group_list_.begin();
     233      850719 :          it != security_group_list_.end(); ++it) {
     234      322515 :         item.entry.security_group_list.security_group.push_back(*it);
     235             :     }
     236             : 
     237      528207 :     for (vector<string>::const_iterator it = community_list_.begin();
     238      540284 :          it != community_list_.end(); ++it) {
     239       12086 :         item.entry.community_tag_list.community_tag.push_back(*it);
     240             :     }
     241             : 
     242             :     // Encode load balance attribute.
     243      528190 :     if (!load_balance_attribute_.IsDefault())
     244           4 :         load_balance_attribute_.Encode(&item.entry.load_balance);
     245             : 
     246      528187 :     xml_node node = doc_.append_child("item");
     247      528123 :     node.append_attribute("id") = route->ToXmppIdString().c_str();
     248             : 
     249             :     // Remember the previous size.
     250             :     // Using remove_child instead of reset allows memory pages allocated for
     251             :     // the xml_document to be reused during the lifetime of the xml_document.
     252      528225 :     size_t pos = repr_.size();
     253      528222 :     item.Encode(&node);
     254      528239 :     doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
     255      528359 :     doc_.remove_child(node);
     256             : 
     257             :     // Cache the substring starting at the previous size.
     258      528346 :     if (cache_routes_)
     259       29650 :         roattr->set_repr(repr_, pos);
     260      528350 : }
     261             : 
     262      249610 : void BgpXmppMessage::AddIpUnreach(const BgpRoute *route) {
     263      249610 :     repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
     264      249709 : }
     265             : 
     266      556942 : bool BgpXmppMessage::AddInetRoute(const BgpRoute *route,
     267             :                                   const RibOutAttr *roattr) {
     268      556942 :     if (is_reachable_) {
     269      424582 :         num_reach_route_++;
     270      424582 :         AddIpReach(route, roattr);
     271             :     } else {
     272      132360 :         num_unreach_route_++;
     273      132360 :         AddIpUnreach(route);
     274             :     }
     275      556993 :     return true;
     276             : }
     277             : 
     278      319245 : bool BgpXmppMessage::AddInet6Route(const BgpRoute *route,
     279             :                                    const RibOutAttr *roattr) {
     280      319245 :     if (is_reachable_) {
     281      201961 :         num_reach_route_++;
     282      201961 :         AddIpReach(route, roattr);
     283             :     } else {
     284      117284 :         num_unreach_route_++;
     285      117284 :         AddIpUnreach(route);
     286             :     }
     287      319302 :     return true;
     288             : }
     289             : 
     290        9083 : void BgpXmppMessage::EncodeEnetNextHop(const BgpRoute *route,
     291             :                                        const RibOutAttr::NextHop &nexthop,
     292             :                                        autogen::EnetItemType *item) {
     293        9083 :     autogen::EnetNextHopType item_nexthop;
     294             : 
     295        9083 :     item_nexthop.af = BgpAf::IPv4;
     296        9083 :     item_nexthop.address = nexthop.address().to_v4().to_string();
     297        9083 :     item_nexthop.label = nexthop.label();
     298        9083 :     item_nexthop.l3_label = nexthop.l3_label();
     299        9083 :     if (!nexthop.mac().IsZero())
     300          20 :         item_nexthop.mac = nexthop.mac().ToString();
     301             : 
     302             :     // If encap list is empty use mpls over gre as default encap.
     303        9083 :     vector<string> &encap_list =
     304             :         item_nexthop.tunnel_encapsulation_list.tunnel_encapsulation;
     305        9083 :     if (nexthop.encap().empty()) {
     306           0 :         encap_list.push_back(string("gre"));
     307             :     } else {
     308        9083 :         encap_list = nexthop.encap();
     309             :     }
     310        9083 :     item_nexthop.tag_list.tag = nexthop.tag_list();
     311        9083 :     item_nexthop.is_new_tags_list = true;
     312        9083 :     item->entry.next_hops.next_hop.push_back(item_nexthop);
     313        9083 : }
     314             : 
     315       10645 : void BgpXmppMessage::AddEnetReach(const BgpRoute *route,
     316             :                                   const RibOutAttr *roattr) {
     317       10645 :     if (!roattr->repr().empty()) {
     318           0 :         repr_ += roattr->repr();
     319           0 :         return;
     320             :     }
     321       10645 :     Address::Family family = table_->family();
     322             : 
     323       10645 :     autogen::EnetItemType item;
     324       10645 :     item.entry.nlri.af = BgpAf::FamilyToAfi(family);
     325       10645 :     item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
     326             : 
     327       10645 :     EvpnRoute *evpn_route =
     328             :         static_cast<EvpnRoute *>(const_cast<BgpRoute *>(route));
     329       10645 :     const EvpnPrefix &evpn_prefix = evpn_route->GetPrefix();
     330       10645 :     item.entry.nlri.ethernet_tag = evpn_prefix.tag();
     331       10645 :     item.entry.nlri.mac = evpn_prefix.mac_addr().ToString();
     332       21290 :     item.entry.nlri.address = evpn_prefix.ip_address().to_string() + "/" +
     333       31935 :         integerToString(evpn_prefix.ip_address_length());
     334       10645 :     item.entry.nlri.source = evpn_prefix.source().to_string();
     335       10645 :     item.entry.nlri.group = evpn_prefix.group().to_string();
     336             : 
     337       10645 :     item.entry.virtual_network = GetVirtualNetwork(route, roattr);
     338       10645 :     item.entry.local_preference = roattr->attr()->local_pref();
     339       10645 :     item.entry.med = roattr->attr()->med();
     340       10645 :     item.entry.sequence_number = mobility_.sequence_number;
     341       10645 :     item.entry.mobility.seqno = mobility_.sequence_number;
     342       10645 :     item.entry.mobility.sticky = mobility_.sticky;
     343       10645 :     item.entry.etree_leaf = etree_leaf_;
     344             : 
     345       10645 :     for (vector<int>::const_iterator it = security_group_list_.begin();
     346       21298 :          it != security_group_list_.end(); ++it) {
     347       10653 :         item.entry.security_group_list.security_group.push_back(*it);
     348             :     }
     349             : 
     350       10645 :     const BgpOList *olist = roattr->attr()->olist().get();
     351       10645 :     assert((olist == NULL) != roattr->nexthop_list().empty());
     352             : 
     353       10645 :     if (olist) {
     354        1690 :         assert(olist->olist().subcode == BgpAttribute::OList);
     355       15832 :         BOOST_FOREACH(const BgpOListElem *elem, olist->elements()) {
     356        7071 :             autogen::EnetNextHopType nh;
     357        7071 :             nh.af = BgpAf::IPv4;
     358        7071 :             nh.address = elem->address.to_string();
     359        7071 :             nh.label = elem->label;
     360        7071 :             nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
     361        7071 :             item.entry.olist.next_hop.push_back(nh);
     362        7071 :         }
     363             :     }
     364             : 
     365       10645 :     const BgpOList *leaf_olist = roattr->attr()->leaf_olist().get();
     366       10645 :     assert((leaf_olist == NULL) != roattr->nexthop_list().empty());
     367             : 
     368       10645 :     if (leaf_olist) {
     369        1690 :         assert(leaf_olist->olist().subcode == BgpAttribute::LeafOList);
     370        1690 :         BOOST_FOREACH(const BgpOListElem *elem, leaf_olist->elements()) {
     371           0 :             autogen::EnetNextHopType nh;
     372           0 :             nh.af = BgpAf::IPv4;
     373           0 :             nh.address = elem->address.to_string();
     374           0 :             nh.label = elem->label;
     375           0 :             nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
     376           0 :             item.entry.leaf_olist.next_hop.push_back(nh);
     377           0 :         }
     378             :     }
     379             : 
     380       28811 :     BOOST_FOREACH(const RibOutAttr::NextHop &nexthop, roattr->nexthop_list()) {
     381        9083 :         EncodeEnetNextHop(route, nexthop, &item);
     382             :     }
     383             : 
     384       11540 :     for (const auto &peer_name : route->peer_sources()) {
     385         895 :         item.entry.peers.peer.push_back(peer_name);
     386             :     }
     387             : 
     388       10645 :     xml_node node = doc_.append_child("item");
     389       10645 :     node.append_attribute("id") = route->ToXmppIdString().c_str();
     390             : 
     391             :     // Remember the previous size.
     392             :     // Using remove_child instead of reset allows memory pages allocated for
     393             :     // the xml_document to be reused during the lifetime of the xml_document.
     394       10645 :     size_t pos = repr_.size();
     395       10645 :     item.Encode(&node);
     396       10645 :     doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
     397       10645 :     doc_.remove_child(node);
     398             : 
     399             :     // Cache the substring starting at the previous size.
     400       10645 :     if (cache_routes_)
     401        1969 :         roattr->set_repr(repr_, pos);
     402       10645 : }
     403             : 
     404        1989 : void BgpXmppMessage::AddEnetUnreach(const BgpRoute *route) {
     405        1989 :     repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
     406        1989 : }
     407             : 
     408       12634 : bool BgpXmppMessage::AddEnetRoute(const BgpRoute *route,
     409             :     const RibOutAttr *roattr) {
     410       12634 :     if (is_reachable_) {
     411       10645 :         num_reach_route_++;
     412       10645 :         AddEnetReach(route, roattr);
     413             :     } else {
     414        1989 :         num_unreach_route_++;
     415        1989 :         AddEnetUnreach(route);
     416             :     }
     417       12634 :     return true;
     418             : }
     419             : 
     420             : //
     421             : // Note that there's no need to cache the string representation since a given
     422             : // mcast route is sent to exactly one xmpp peer.
     423             : //
     424        2377 : void BgpXmppMessage::AddMcastReach(const BgpRoute *route,
     425             :                                    const RibOutAttr *roattr) {
     426        2377 :     Address::Family family = table_->family();
     427        2377 :     autogen::McastItemType item;
     428        2377 :     item.entry.nlri.af = BgpAf::FamilyToAfi(family);
     429        2377 :     item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
     430             : 
     431        2377 :     ErmVpnRoute *ermvpn_route =
     432             :         static_cast<ErmVpnRoute *>(const_cast<BgpRoute *>(route));
     433        2377 :     item.entry.nlri.group = ermvpn_route->GetPrefix().group().to_string();
     434        2377 :     item.entry.nlri.source =  ermvpn_route->GetPrefix().source().to_string();
     435        2377 :     item.entry.nlri.source_label = roattr->label();
     436        2377 :     if (!roattr->source_address().is_unspecified()) {
     437             :         item.entry.nlri.source_address =
     438         149 :            roattr->source_address().to_string();
     439             :     }
     440             : 
     441        2377 :     const BgpOList *olist = roattr->attr()->olist().get();
     442        2377 :     assert(olist->olist().subcode == BgpAttribute::OList);
     443        7461 :     BOOST_FOREACH(const BgpOListElem *elem, olist->elements()) {
     444        2542 :         autogen::McastNextHopType nh;
     445        2542 :         nh.af = BgpAf::IPv4;
     446        2542 :         nh.address = elem->address.to_string();
     447        2542 :         nh.label = integerToString(elem->label);
     448        2542 :         nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
     449        2542 :         item.entry.olist.next_hop.push_back(nh);
     450        2542 :     }
     451             : 
     452             :     // Using remove_child instead of reset allows memory pages allocated for
     453             :     // the xml_document to be reused during the lifetime of the xml_document.
     454        2377 :     xml_node node = doc_.append_child("item");
     455        2377 :     node.append_attribute("id") = route->ToXmppIdString().c_str();
     456        2377 :     item.Encode(&node);
     457        2377 :     doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
     458        2377 :     doc_.remove_child(node);
     459        2377 : }
     460             : 
     461         461 : void BgpXmppMessage::AddMcastUnreach(const BgpRoute *route) {
     462         461 :     repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
     463         461 : }
     464             : 
     465        2838 : bool BgpXmppMessage::AddMcastRoute(const BgpRoute *route,
     466             :                                    const RibOutAttr *roattr) {
     467        2838 :     if (is_reachable_) {
     468        2377 :         num_reach_route_++;
     469        2377 :         AddMcastReach(route, roattr);
     470             :     } else {
     471         461 :         num_unreach_route_++;
     472         461 :         AddMcastUnreach(route);
     473             :     }
     474        2838 :     return true;
     475             : }
     476             : 
     477      196818 : void BgpXmppMessage::AddMvpnReach(const BgpRoute *route,
     478             :                                   const RibOutAttr *roattr) {
     479      196818 :     Address::Family family = table_->family();
     480      196818 :     autogen::MvpnItemType item;
     481      196818 :     item.entry.nlri.af = BgpAf::FamilyToAfi(family);
     482      196818 :     item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
     483             : 
     484      196818 :     MvpnRoute *mvpn_route =
     485             :         static_cast<MvpnRoute *>(const_cast<BgpRoute *>(route));
     486      196818 :     item.entry.nlri.group = mvpn_route->GetPrefix().group().to_string();
     487      196818 :     item.entry.nlri.source =  mvpn_route->GetPrefix().source().to_string();
     488      196818 :     item.entry.nlri.route_type = mvpn_route->GetPrefix().type();
     489      196818 :     assert((item.entry.nlri.route_type == MvpnPrefix::SourceActiveADRoute) ||
     490             :             (item.entry.nlri.route_type == MvpnPrefix::SourceTreeJoinRoute));
     491             : 
     492      196818 :     if (item.entry.nlri.route_type == MvpnPrefix::SourceActiveADRoute) {
     493      196724 :         const BgpOList *olist = roattr->attr()->olist().get();
     494      196724 :         assert(olist->olist().subcode == BgpAttribute::OList);
     495      590172 :         BOOST_FOREACH(const BgpOListElem *elem, olist->elements()) {
     496      196724 :             autogen::MvpnNextHopType nh;
     497      196724 :             nh.af = BgpAf::IPv4;
     498      196724 :             nh.address = elem->address.to_string();
     499      196724 :             nh.label = elem->label;
     500      196724 :             nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
     501      196724 :             item.entry.olist.next_hop.push_back(nh);
     502      196724 :         }
     503             :     }
     504             : 
     505             :     // Using remove_child instead of reset allows memory pages allocated for
     506             :     // the xml_document to be reused during the lifetime of the xml_document.
     507      196818 :     xml_node node = doc_.append_child("item");
     508      196818 :     node.append_attribute("id") = route->ToXmppIdString().c_str();
     509      196818 :     item.Encode(&node);
     510      196818 :     doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
     511      196818 :     doc_.remove_child(node);
     512      196818 : }
     513             : 
     514         124 : void BgpXmppMessage::AddMvpnUnreach(const BgpRoute *route) {
     515         124 :     repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
     516         124 : }
     517             : 
     518      196942 : bool BgpXmppMessage::AddMvpnRoute(const BgpRoute *route,
     519             :                                    const RibOutAttr *roattr) {
     520      196942 :     if (is_reachable_) {
     521      196818 :         num_reach_route_++;
     522      196818 :         AddMvpnReach(route, roattr);
     523             :     } else {
     524         124 :         num_unreach_route_++;
     525         124 :         AddMvpnUnreach(route);
     526             :     }
     527      196942 :     return true;
     528             : }
     529             : 
     530    14168274 : const uint8_t *BgpXmppMessage::GetData(IPeerUpdate *peer, size_t *lenp,
     531             :     const string **msg_str, string *temp) {
     532             :     // Build begin line that contains message opening tag with from and to
     533             :     // attributes.
     534    14168274 :     msg_begin_.clear();
     535    14168332 :     msg_begin_ += "\n<message from=\"";
     536    14168403 :     msg_begin_ += XmppInit::kControlNodeJID;
     537    14168481 :     msg_begin_ += "\" to=\"";
     538    14168564 :     msg_begin_ += peer->ToString();
     539    14168608 :     msg_begin_ += "/";
     540    14168620 :     msg_begin_ += XmppInit::kBgpPeer;
     541    14168575 :     msg_begin_ += "\">";
     542             : 
     543             :     // Add closing tags if this is the first peer to which the message will
     544             :     // be sent.
     545    14168599 :     if (!repr_valid_) {
     546      472898 :         repr_ += "\t\t</items>\n\t</event>\n</message>\n";
     547      472885 :         repr_valid_ = true;
     548             :     }
     549             : 
     550             :     // Replace the begin line if it fits in the space reserved at the start
     551             :     // of repr_. Use fill and copy instead of string::replace as the latter
     552             :     // seems to construct a new temporary string to hold the input data to
     553             :     // be copied.
     554             :     // Otherwise build a new string with the begin line and the rest of the
     555             :     // message in repr_.
     556    14168586 :     size_t begin_size = msg_begin_.size();
     557    14168491 :     if (begin_size <= kMaxFromToLength) {
     558     9974227 :         size_t extra = kMaxFromToLength - begin_size;
     559     9974227 :         char *data = const_cast<char *>(repr_.c_str());
     560     9974222 :         fill(data, data + extra, ' ');
     561     9974251 :         copy(msg_begin_.c_str(), msg_begin_.c_str() + begin_size, data + extra);
     562     9974094 :         *lenp = repr_.size() - extra;
     563     9974088 :         *msg_str = &repr_;
     564     9974088 :         return reinterpret_cast<const uint8_t *>(repr_.c_str()) + extra;
     565             :     } else {
     566     4194264 :         *temp = msg_begin_ + string(repr_, kMaxFromToLength);
     567     4194304 :         *lenp = temp->size();
     568     4194304 :         *msg_str = NULL;
     569     4194304 :         return reinterpret_cast<const uint8_t *>(temp->c_str());
     570             :     }
     571             : }
     572             : 
     573      836276 : void BgpXmppMessage::ProcessCommunity(const Community *community) {
     574      836276 :     community_list_.clear();
     575      836296 :     if (community == NULL)
     576      824232 :         return;
     577       36182 :     BOOST_FOREACH(uint32_t value, community->communities()) {
     578       12085 :         community_list_.push_back(CommunityType::CommunityToString(value));
     579             :     }
     580             : }
     581             : 
     582      836282 : void BgpXmppMessage::ProcessExtCommunity(const ExtCommunity *ext_community) {
     583      836282 :     mobility_.sequence_number = 0;
     584      836282 :     mobility_.sticky = false;
     585      836282 :     etree_leaf_ = false;
     586      836282 :     security_group_list_.clear();
     587      836289 :     load_balance_attribute_ = LoadBalance::LoadBalanceAttribute();
     588      836498 :     if (ext_community == NULL)
     589      208122 :         return;
     590             : 
     591      628376 :     as_t as_number =  table_->server()->autonomous_system();
     592      628323 :     bool sg_asn_match = true;
     593      628288 :     for (ExtCommunity::ExtCommunityList::const_iterator iter =
     594      628323 :         ext_community->communities().begin();
     595     3681211 :         iter != ext_community->communities().end(); ++iter) {
     596     3052980 :         if (ExtCommunity::is_security_group(*iter)) {
     597      431522 :             SecurityGroup sg(*iter);
     598      431517 :             if (as_number <= AS2_MAX)
     599      234909 :                 if (sg.as_number() != as_number && !sg.IsGlobal())
     600           2 :                     continue;
     601      431523 :             security_group_list_.push_back(sg.security_group_id());
     602     2621550 :         } else if (ExtCommunity::is_security_group4(*iter)) {
     603      196608 :             SecurityGroup4ByteAs sg(*iter);
     604      196608 :             if (sg.as_number() != as_number)
     605           0 :                 sg_asn_match = false;
     606     2424902 :         } else if (ExtCommunity::is_mac_mobility(*iter)) {
     607        1079 :             MacMobility mm(*iter);
     608        1079 :             mobility_.sequence_number = mm.sequence_number();
     609        1079 :             mobility_.sticky = mm.sticky();
     610     2423785 :         } else if (ExtCommunity::is_load_balance(*iter)) {
     611           4 :             LoadBalance load_balance(*iter);
     612           4 :             load_balance.FillAttribute(&load_balance_attribute_);
     613     2423755 :         } else if (ExtCommunity::is_etree(*iter)) {
     614        9526 :             ETree etree(*iter);
     615        9526 :             etree_leaf_ = etree.leaf();
     616             :         }
     617             :     }
     618      628209 :     if (!sg_asn_match)
     619           0 :         security_group_list_.clear();
     620             : }
     621             : 
     622     1067909 : string BgpXmppMessage::GetVirtualNetwork(
     623             :     const RibOutAttr::NextHop &nexthop) const {
     624     1067909 :     int index = nexthop.origin_vn_index();
     625     1067913 :     if (index > 0) {
     626             :         const RoutingInstanceMgr *manager =
     627       69136 :             table_->routing_instance()->manager();
     628       69043 :         return manager->GetVirtualNetworkByVnIndex(index);
     629      998777 :     } else if (index == 0) {
     630       61248 :         return table_->routing_instance()->GetVirtualNetworkName();
     631             :     } else {
     632      937529 :         return "unresolved";
     633             :     }
     634             : }
     635             : 
     636      539004 : string BgpXmppMessage::GetVirtualNetwork(const BgpRoute *route,
     637             :     const RibOutAttr *roattr) const {
     638      539004 :     if (!is_reachable_) {
     639           0 :         return "unresolved";
     640      539004 :     } else if (roattr->nexthop_list().empty()) {
     641        1690 :         if (roattr->vrf_originated()) {
     642        1690 :             return table_->routing_instance()->GetVirtualNetworkName();
     643             :         } else {
     644           0 :             return "unresolved";
     645             :         }
     646             :     } else {
     647      537255 :         return GetVirtualNetwork(roattr->nexthop_list().front());
     648             :     }
     649             : }
     650             : 
     651          50 : BgpXmppMessageBuilder::BgpXmppMessageBuilder() {
     652          51 : }
     653             : 
     654         144 : Message *BgpXmppMessageBuilder::Create() const {
     655         144 :     return new BgpXmppMessage;
     656             : }

Generated by: LCOV version 1.14