OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
service_chaining.h
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 #ifndef SRC_BGP_ROUTING_INSTANCE_SERVICE_CHAINING_H_
6 #define SRC_BGP_ROUTING_INSTANCE_SERVICE_CHAINING_H_
7 
8 #include <boost/ptr_container/ptr_map.hpp>
9 #include <boost/shared_ptr.hpp>
10 #include <tbb/mutex.h>
11 
12 #include <list>
13 #include <map>
14 #include <set>
15 #include <string>
16 #include <vector>
17 
18 #include "base/lifetime.h"
19 #include "base/queue_task.h"
21 #include "bgp/inet/inet_route.h"
22 #include "bgp/inet6/inet6_route.h"
23 #include "bgp/evpn/evpn_route.h"
25 
26 class BgpRoute;
27 class BgpTable;
28 class RoutingInstance;
29 class BgpTable;
30 class BgpServer;
31 class InetVpnRoute;
32 class Inet6VpnRoute;
33 class SandeshResponse;
34 class ServiceChainConfig;
35 class ServiceChainGroup;
36 class ShowServicechainInfo;
37 
38 template <typename T> class ServiceChainMgr;
39 
40 template <typename T1, typename T2, typename T3, typename T4>
42  typedef T1 RouteT;
43  typedef T2 VpnRouteT;
44  typedef T3 PrefixT;
45  typedef T4 AddressT;
46 };
47 
49  InetRoute, InetVpnRoute, Ip4Prefix, Ip4Address> {
50 };
51 
53  Inet6Route, Inet6VpnRoute, Inet6Prefix, Ip6Address> {
54 };
55 
57  EvpnRoute, InetVpnRoute, EvpnPrefix, Ip4Address> {
58 };
59 
61  EvpnRoute, Inet6VpnRoute, EvpnPrefix, Ip6Address> {
62 };
63 
65 
67 public:
69  }
70  ServiceChainPtr info() { return info_; }
71 
72 private:
75 };
76 
77 template <typename T>
79 public:
80  typedef typename T::PrefixT PrefixT;
81 
82  enum RequestType {
92  };
93 
95  PrefixT aggregate_match, ServiceChainPtr info)
96  : type_(type),
97  table_(table),
98  rt_(route),
99  aggregate_match_(aggregate_match),
100  info_(info),
101  snh_resp_(NULL) {
102  }
103 
105  : type_(type),
106  table_(NULL),
107  rt_(NULL),
108  snh_resp_(resp) {
109  }
110 
117 
118 private:
120 };
121 
122 template <typename T>
123 class ServiceChain : public ConditionMatch {
124 public:
125  typedef typename T::RouteT RouteT;
126  typedef typename T::VpnRouteT VpnRouteT;
127  typedef typename T::PrefixT PrefixT;
128  typedef typename T::AddressT AddressT;
131 
132  // List of more specific routes resulted in Aggregate route
133  typedef std::set<BgpRoute *> RouteList;
134 
135  // Map of Virtual Network subnet prefix to List of More Specific routes
136  typedef std::map<PrefixT, RouteList> PrefixToRouteListMap;
137 
138  // Map of External Connecting route to Service Chain Route
139  typedef std::set<BgpRoute *> ExtConnectRouteList;
140 
141  // List of path ids for the connected route
142  typedef std::set<uint32_t> ConnectedPathIdList;
143 
145  RoutingInstance *src, RoutingInstance *dest, RoutingInstance *connected,
146  const std::vector<std::string> &subnets, AddressT addr, bool head,
147  bool retain_as_path);
150  return manager_->GetConnectedFamily();
151  }
153 
154  // Delete is triggered from configuration, not via LifetimeManager.
155  void ManagedDelete() { }
156 
157  bool CompareServiceChainConfig(const ServiceChainConfig &config);
158  void RemoveMatchState(BgpRoute *route, ServiceChainState *state);
159 
160  void SetConnectedRoute(BgpRoute *connected);
161  bool IsConnectedRouteValid() const;
163  return connected_path_ids_;
164  }
165 
171  void UpdateServiceChainRoute(PrefixT prefix, const RouteT *orig_route,
172  const ConnectedPathIdList &old_path_ids, bool aggregate);
173  void DeleteServiceChainRoute(PrefixT prefix, bool aggregate);
174 
175  bool AddMoreSpecific(PrefixT aggregate, BgpRoute *more_specific);
176  bool DeleteMoreSpecific(PrefixT aggregate, BgpRoute *more_specific);
177 
178  BgpTable *src_table() const;
179  BgpTable *connected_table() const;
180  BgpTable *dest_table() const;
181 
182  ServiceChainGroup *group() const { return group_; }
183  void clear_group() { group_ = NULL; }
184 
186  return &prefix_to_routelist_map_;
187  }
189  return &prefix_to_routelist_map_;
190  }
191 
192  virtual bool Match(BgpServer *server, BgpTable *table, BgpRoute *route,
193  bool deleted);
194  virtual std::string ToString() const;
195 
196  void FillServiceChainInfo(ShowServicechainInfo *info) const;
197 
200  }
203  }
204  bool dest_table_unregistered() const {
206  }
209  }
210  bool unregistered() const {
212  }
213 
215  return ext_connect_routes_;
216  }
218  return &ext_connect_routes_;
219  }
220 
221  bool aggregate_enable() const { return aggregate_; }
222  bool is_sc_head() const { return sc_head_; }
223  bool retain_as_path() const { return retain_as_path_; }
224  void set_aggregate_enable() { aggregate_ = true; }
225  bool group_oper_state_up() const { return group_oper_state_up_; }
227 
228 private:
242  bool aggregate_; // Whether the host route needs to be aggregated
243  bool sc_head_; // Whether this SI is at the head of the chain
248 
249  // Helper function to match
250  bool IsMoreSpecific(BgpRoute *route, PrefixT *aggregate_match) const;
251  bool IsAggregate(BgpRoute *route) const;
252  bool IsConnectedRoute(BgpRoute *route, bool is_conn_table=false) const;
253  bool IsEvpnType5Route(BgpRoute *route) const;
255  BgpRoute *&route, BgpTable *&table, PrefixT prefix, bool create);
256  void ProcessServiceChainPath(uint32_t path_id, BgpPath *path,
257  BgpAttrPtr attr, BgpRoute *&route, DBTablePartition *&partition,
258  bool aggregate, BgpTable *bgptable);
259  void UpdateServiceChainRouteInternal(const RouteT *orig_route,
260  const ConnectedPathIdList &old_path_ids, BgpRoute *sc_route,
261  DBTablePartition *partition, BgpTable *bgptable, bool aggregate);
262  void DeleteServiceChainRouteInternal(BgpRoute *service_chain_route,
263  DBTablePartition *partition,
264  BgpTable *bgptable, bool aggregate);
265 
267 };
268 
269 //
270 // This represents a service chain group within a ServiceChainMgr. All the
271 // individual ServiceChains that are part of the same logical service chain
272 // in the configuration are part of a given ServiceChainGroup.
273 //
274 // A ServiceChainGroup maintains operational state for itself based on the
275 // operational state of the individual ServiceChains that are part of the
276 // group. This is used to force fate sharing for all the ServiceChains in
277 // the group. If the operational status of the group is down, re-originated
278 // routes for all ServiceChainTs are deleted/withdrawn.
279 //
280 // A ServiceChainGroup gets created when the ServiceChainMgr processes a
281 // ServiceChainConfig with a non-empty service_chain_id. It gets deleted
282 // when there are no more ServiceChainTs or pending chains that belong to
283 // the ServiceChainGroup.
284 //
285 // The set of member chains is tracked using RoutingInstance pointers (as
286 // opposed to ServiceChainT pointers) so that the state of pending chains
287 // can also be tracked. Note that ServiceChainT objects are not allocated
288 // for pending chains.
289 //
290 // The membership of RoutingInstances in a ServiceChainGroup is updated as
291 // the group in the ServiceChainConfig for the RoutingInstances is updated.
292 //
293 // The operational state of the group is updated whenever a RoutingInstance
294 // is added or deleted to/from the ServiceChainGroup and when the connected
295 // route for the ServiceChainT is added/updated/deleted.
296 //
298 public:
299  ServiceChainGroup(IServiceChainMgr *manager, const std::string &name);
301 
302  void AddRoutingInstance(RoutingInstance *rtinstance);
303  void DeleteRoutingInstance(RoutingInstance *rtinstance);
304  void UpdateOperState();
305  std::string name() const { return name_; }
306  bool empty() const { return chain_set_.empty(); }
307  bool oper_state_up() const { return oper_state_up_; }
308 
309 private:
310  typedef std::set<RoutingInstance *> ServiceChainSet;
311 
313  std::string name_;
316 };
317 
318 template <typename T>
319 class ServiceChainMgr : public IServiceChainMgr {
320 public:
321  typedef typename T::RouteT RouteT;
322  typedef typename T::PrefixT PrefixT;
323  typedef typename T::AddressT AddressT;
326 
327  explicit ServiceChainMgr(BgpServer *server);
328  virtual ~ServiceChainMgr();
329 
330  void Terminate();
331  void ManagedDelete();
332  bool MayDelete() const;
333  void RetryDelete();
334 
335  // Creates a new service chain between two Virtual network
336  // If the two routing instance is already connected, it updates the
337  // connected route address for existing service chain
338  virtual bool LocateServiceChain(RoutingInstance *rtinstance,
339  const ServiceChainConfig &config);
340 
341  // Remove the existing service chain between from routing instance
342  virtual void StopServiceChain(RoutingInstance *rtinstance);
343  virtual void UpdateServiceChain(RoutingInstance *rtinstance,
344  bool group_oper_state_up);
346 
347  virtual size_t PendingQueueSize() const { return pending_chains_.size(); }
348  virtual size_t ResolvedQueueSize() const { return chain_set_.size(); }
349  virtual uint32_t GetDownServiceChainCount() const;
350  virtual bool IsQueueEmpty() const { return process_queue_->IsQueueEmpty(); }
351  virtual bool ServiceChainIsPending(RoutingInstance *rtinstance,
352  std::string *reason = NULL) const;
353  virtual bool ServiceChainIsUp(RoutingInstance *rtinstance) const;
354 
355  Address::Family GetFamily() const;
358  void Enqueue(ServiceChainRequestT *req);
359  virtual bool FillServiceChainInfo(RoutingInstance *rtinstance,
360  ShowServicechainInfo *info) const;
362 private:
363  template <typename U> friend class ServiceChainIntegrationTest;
364  template <typename U> friend class ServiceChainTest;
365  class DeleteActor;
366 
367  // All service chain related actions are performed in the context
368  // of this task. This task has exclusion with db::DBTable task.
369  static int service_chain_task_id_;
370 
373  }
375  : group(group), reason(reason) {
376  }
378  std::string reason;
379  };
380 
381  // Set of service chains created in the system
382  typedef std::map<RoutingInstance *, ServiceChainPtr> ServiceChainMap;
383 
384  // At the time of processing, service chain request, all required info
385  // may not be available (e.g. dest routing instance may not be created,
386  // or marked deleted etc). Create a list of pending service chains that
387  // are waiting to get created and maintain a reason string for why the
388  // service chain is on the pending list.
389  typedef std::map<RoutingInstance *, PendingChainState> PendingChainList;
390 
391  typedef boost::ptr_map<std::string, ServiceChainGroup> GroupMap;
392  typedef std::set<ServiceChainGroup *> GroupSet;
393 
395  ServiceChainGroup *FindServiceChainGroup(const std::string &group_name);
396  ServiceChainGroup *LocateServiceChainGroup(const std::string &group_name);
398 
400  void StopServiceChainDone(BgpTable *table, ConditionMatch *info);
401  ServiceChainT *FindServiceChain(const std::string &instance) const;
402  ServiceChainT *FindServiceChain(RoutingInstance *rtinstance) const;
403 
405  ServiceChainGroup *group, std::string reason) {
406  PendingChainState state(group, reason);
407  pending_chains_.insert(std::make_pair(rtinstance, state));
408  }
410  pending_chains_.erase(rtinstance);
411  }
413  typename PendingChainList::const_iterator loc =
414  pending_chains_.find(rtinstance);
415  if (loc != pending_chains_.end()) {
416  return loc->second;
417  } else {
418  return PendingChainState();
419  }
420  }
421 
423  const typename ServiceChainT::ConnectedPathIdList &old_path_ids);
425 
426  void StartResolve();
428  void RoutingInstanceCallback(std::string name, int op);
429  void PeerRegistrationCallback(IPeer *peer, BgpTable *table,
430  bool unregister);
431 
433  virtual void set_aggregate_host_route(bool value) {
434  aggregate_host_route_ = value;
435  }
436 
437  virtual void DisableResolveTrigger();
438  virtual void EnableResolveTrigger();
439  virtual void DisableGroupTrigger();
440  virtual void EnableGroupTrigger();
441 
442  // Work Queue to handle requests posted from Match function, called
443  // in the context of db::DBTable task.
444  // The actions are performed in the bgp::ServiceChain task context.
445  virtual void DisableQueue() { process_queue_->set_disable(true); }
446  virtual void EnableQueue() { process_queue_->set_disable(false); }
447 
448  // Mutex is used to serialize access from multiple bgp::ConfigHelper tasks.
450  tbb::mutex mutex_;
452  boost::scoped_ptr<TaskTrigger> resolve_trigger_;
453  boost::scoped_ptr<WorkQueue<ServiceChainRequestT *> > process_queue_;
456  boost::scoped_ptr<TaskTrigger> group_trigger_;
460  int id_;
462  boost::scoped_ptr<DeleteActor> deleter_;
464 
466 };
467 
472 
473 #endif // SRC_BGP_ROUTING_INSTANCE_SERVICE_CHAINING_H_
std::set< RoutingInstance * > ServiceChainSet
bool AddMoreSpecific(PrefixT aggregate, BgpRoute *more_specific)
boost::scoped_ptr< WorkQueue< ServiceChainRequestT * > > process_queue_
virtual size_t ResolvedQueueSize() const
virtual void EnableGroupTrigger()
ServiceChain(ServiceChainMgrT *manager, ServiceChainGroup *group, RoutingInstance *src, RoutingInstance *dest, RoutingInstance *connected, const std::vector< std::string > &subnets, AddressT addr, bool head, bool retain_as_path)
ServiceChainGroup(IServiceChainMgr *manager, const std::string &name)
ServiceChainT * FindServiceChain(const std::string &instance) const
void Enqueue(ServiceChainRequestT *req)
std::set< BgpRoute * > RouteList
RoutingInstance * src_
void PeerRegistrationCallback(IPeer *peer, BgpTable *table, bool unregister)
bool dest_table_unregistered() const
DISALLOW_COPY_AND_ASSIGN(ServiceChain)
virtual void EnableQueue()
T::RouteT RouteT
LifetimeRef< ServiceChain > dest_table_delete_ref_
bool aggregate_enable() const
void set_group_oper_state_up(bool up)
ServiceChainRequest< T > ServiceChainRequestT
virtual void StopServiceChain(RoutingInstance *rtinstance)
bool is_sc_head() const
ServiceChainMgr< ServiceChainEvpn6 > ServiceChainMgrEvpn6
void GetReplicationFamilyInfo(DBTablePartition *&partition, BgpRoute *&route, BgpTable *&table, PrefixT prefix, bool create)
const ExtConnectRouteList & ext_connecting_routes() const
void UpdateServiceChainRoute(PrefixT prefix, const RouteT *orig_route, const ConnectedPathIdList &old_path_ids, bool aggregate)
LifetimeRef< ServiceChain > src_table_delete_ref_
ServiceChainPtr info_
boost::intrusive_ptr< ConditionMatch > ConditionMatchPtr
virtual std::string ToString() const
ExtConnectRouteList ext_connect_routes_
bool IsConnectedRouteValid() const
virtual bool ServiceChainIsPending(RoutingInstance *rtinstance, std::string *reason=NULL) const
const ConnectedPathIdList & GetConnectedPathIds()
BgpRoute * connected_route() const
BgpTable * connected_table() const
friend class ServiceChainTest
void FillServiceChainInfo(ShowServicechainInfo *info) const
ServiceChainPtr info()
ServiceChainMgr< ServiceChainInet > ServiceChainMgrInet
ServiceChainMgr< ServiceChainEvpn > ServiceChainMgrEvpn
Family
Definition: address.h:24
bool dest_table_unregistered_
virtual void UpdateServiceChain(RoutingInstance *rtinstance, bool group_oper_state_up)
std::string name() const
ServiceChainRequest< T > ServiceChainRequestT
PendingChainState GetPendingServiceChain(RoutingInstance *rtinstance)
ServiceChainGroup * group_
RoutingInstance * dest_routing_instance() const
virtual bool IsQueueEmpty() const
bool unregistered() const
RoutingInstance * connected_routing_instance() const
void RemoveMatchState(BgpRoute *route, ServiceChainState *state)
virtual void DisableGroupTrigger()
BgpTable * dest_table() const
RoutingInstance * src_routing_instance() const
DISALLOW_COPY_AND_ASSIGN(ServiceChainMgr)
ServiceChainSet chain_set_
virtual uint32_t GetDownServiceChainCount() const
std::set< BgpRoute * > ExtConnectRouteList
bool IsMoreSpecific(BgpRoute *route, PrefixT *aggregate_match) const
ConnectedPathIdList connected_path_ids_
bool oper_state_up() const
LifetimeRef< ServiceChain > connected_table_delete_ref_
virtual void DisableQueue()
Definition: ipeer.h:186
static int service_chain_task_id_
ServiceChainGroup * LocateServiceChainGroup(const std::string &group_name)
ServiceChainMgr(BgpServer *server)
bool IsEvpnType5Route(BgpRoute *route) const
LifetimeRef< ServiceChainMgr > server_delete_ref_
DISALLOW_COPY_AND_ASSIGN(ServiceChainState)
ServiceChainGroup * FindServiceChainGroup(RoutingInstance *rtinstance)
void StopServiceChainDone(BgpTable *table, ConditionMatch *info)
ServiceChainMap chain_set_
boost::intrusive_ptr< const BgpAttr > BgpAttrPtr
Definition: bgp_attr.h:991
PrefixToRouteListMap prefix_to_routelist_map_
T::AddressT AddressT
void AddPendingServiceChain(RoutingInstance *rtinstance, ServiceChainGroup *group, std::string reason)
PendingChainList pending_chains_
bool IsAggregate(BgpRoute *route) const
void set_aggregate_enable()
void set_dest_table_unregistered()
uint8_t type
Definition: load_balance.h:109
BgpConditionListener * listener_
RoutingInstance * dest_
bool connected_table_unregistered() const
bool MayDelete() const
SCAddress::Family GetSCFamily() const
void DeleteRoutingInstance(RoutingInstance *rtinstance)
ServiceChainRequest(RequestType type, SandeshResponse *resp)
std::set< ServiceChainGroup * > GroupSet
virtual void set_aggregate_host_route(bool value)
virtual void DisableResolveTrigger()
ServiceChainMgr< ServiceChainInet6 > ServiceChainMgrInet6
friend class ServiceChainIntegrationTest
Address::Family GetConnectedFamily() const
virtual bool FillServiceChainInfo(RoutingInstance *rtinstance, ShowServicechainInfo *info) const
T::PrefixT PrefixT
DISALLOW_COPY_AND_ASSIGN(ServiceChainRequest)
boost::scoped_ptr< TaskTrigger > group_trigger_
virtual BgpConditionListener * GetListener()
void UpdateServiceChainRouteInternal(const RouteT *orig_route, const ConnectedPathIdList &old_path_ids, BgpRoute *sc_route, DBTablePartition *partition, BgpTable *bgptable, bool aggregate)
void DeletePendingServiceChain(RoutingInstance *rtinstance)
PrefixToRouteListMap * prefix_to_route_list_map()
bool RequestHandler(ServiceChainRequestT *req)
PendingChainState(ServiceChainGroup *group, std::string reason)
BgpTable * src_table() const
virtual bool ServiceChainIsUp(RoutingInstance *rtinstance) const
ServiceChainGroup * group() const
RoutingInstance * connected_
void ProcessServiceChainPath(uint32_t path_id, BgpPath *path, BgpAttrPtr attr, BgpRoute *&route, DBTablePartition *&partition, bool aggregate, BgpTable *bgptable)
virtual bool LocateServiceChain(RoutingInstance *rtinstance, const ServiceChainConfig &config)
bool connected_table_unregistered_
void RoutingInstanceCallback(std::string name, int op)
void UpdateServiceChainGroup(ServiceChainGroup *group)
bool retain_as_path() const
void UpdateServiceChainRoutes(ServiceChainT *chain, const typename ServiceChainT::ConnectedPathIdList &old_path_ids)
ServiceChain< T > ServiceChainT
virtual void EnableResolveTrigger()
void DeleteServiceChainRoutes(ServiceChainT *chain)
const PrefixToRouteListMap * prefix_to_route_list_map() const
bool ResolvePendingServiceChain()
const AddressT & service_chain_addr() const
bool IsConnectedRoute(BgpRoute *route, bool is_conn_table=false) const
std::map< PrefixT, RouteList > PrefixToRouteListMap
bool aggregate_host_route() const
Address::Family GetFamily() const
bool CompareServiceChainConfig(const ServiceChainConfig &config)
ServiceChainPtr info_
ConditionMatchPtr ServiceChainPtr
void DeleteServiceChainRouteInternal(BgpRoute *service_chain_route, DBTablePartition *partition, BgpTable *bgptable, bool aggregate)
ExtConnectRouteList * ext_connecting_routes()
std::map< RoutingInstance *, PendingChainState > PendingChainList
SCAddress::Family GetSCFamily() const
AddressT service_chain_addr_
boost::ptr_map< std::string, ServiceChainGroup > GroupMap
IServiceChainMgr * manager_
bool group_oper_state_up() const
virtual bool Match(BgpServer *server, BgpTable *table, BgpRoute *route, bool deleted)
BgpRoute * connected_route_
boost::scoped_ptr< TaskTrigger > resolve_trigger_
ServiceChainMgr< T > ServiceChainMgrT
Address::Family GetFamily() const
ServiceChainMgrT * manager_
T::AddressT AddressT
bool empty() const
void AddRoutingInstance(RoutingInstance *rtinstance)
bool ProcessServiceChainGroups()
SandeshResponse * snh_resp_
ServiceChainRequest(RequestType type, BgpTable *table, BgpRoute *route, PrefixT aggregate_match, ServiceChainPtr info)
std::map< RoutingInstance *, ServiceChainPtr > ServiceChainMap
BgpServer * server_
Address::Family GetConnectedFamily() const
ServiceChainState(ServiceChainPtr info)
std::set< uint32_t > ConnectedPathIdList
void SetConnectedRoute(BgpRoute *connected)
virtual size_t PendingQueueSize() const
void set_connected_table_unregistered()
boost::scoped_ptr< DeleteActor > deleter_
bool DeleteMoreSpecific(PrefixT aggregate, BgpRoute *more_specific)
T::VpnRouteT VpnRouteT
void DeleteServiceChainRoute(PrefixT prefix, bool aggregate)
virtual ~ServiceChainMgr()