OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
vrf_ovsdb.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2015 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 extern "C" {
6 #include <ovsdb_wrapper.h>
7 };
9 #include <ovsdb_client.h>
10 #include <ovsdb_client_idl.h>
11 #include <ovsdb_client_session.h>
12 #include <logical_switch_ovsdb.h>
13 #include <vrf_ovsdb.h>
15 #include <physical_locator_ovsdb.h>
16 
17 #include <oper/vn.h>
18 #include <oper/vrf.h>
19 #include <oper/nexthop.h>
20 #include <oper/tunnel_nh.h>
21 #include <oper/agent_path.h>
22 #include <oper/bridge_route.h>
23 #include <ovsdb_sandesh.h>
24 #include <ovsdb_types.h>
25 
26 using namespace OVSDB;
27 
29  const std::string &logical_switch) : OvsdbDBEntry(table),
30  logical_switch_name_(logical_switch), route_table_(NULL),
31  oper_route_table_(NULL) {
32 }
33 
35  assert(route_table_ == NULL);
36 }
37 
39  // if table is scheduled for delete, return from here
40  // and wait for delete callback
41  if (table_->delete_scheduled()) {
42  return true;
43  }
44 
45  // create route table and register
46  if (route_table_ == NULL) {
48  logical_switch_name_, this);
49  }
50 
51  if (!stale() && route_table_->GetDBTable() == NULL &&
52  oper_route_table_ != NULL) {
54  }
55  return true;
56 }
57 
59  return Add();
60 }
61 
63  // delete route table.
64  if (route_table_ != NULL) {
66  // while triggering a delete for UnicastMacRemoteTable, wait
67  // for table cleanup to complete to proceed with the KSync
68  // state machine, so that we maintain consistency for remote
69  // MAC table by not allowing two transient route table, one
70  // in process of deletion and other in process of addition
71  // which can lead to inconsistent state in OVSDB-server
72  return false;
73  }
74 
75  return true;
76 }
77 
78 bool VrfOvsdbEntry::Sync(DBEntry *db_entry) {
79  // check if route table is available.
80  const VrfEntry *vrf = static_cast<const VrfEntry *>(db_entry);
81  assert(logical_switch_name_ == UuidToString(vrf->vn()->GetUuid()));
82  if (oper_route_table_ != vrf->GetBridgeRouteTable()) {
84  return true;
85  }
86  return false;
87 }
88 
89 bool VrfOvsdbEntry::IsLess(const KSyncEntry &entry) const {
90  const VrfOvsdbEntry &vrf_entry =
91  static_cast<const VrfOvsdbEntry &>(entry);
92  return (logical_switch_name_.compare(vrf_entry.logical_switch_name_) < 0);
93 }
94 
96  return NULL;
97 }
98 
100  OvsdbDBObject *object = static_cast<OvsdbDBObject*>(GetObject());
101  assert(route_table_ == table);
102  route_table_ = NULL;
103  object->SafeNotifyEvent(this, KSyncEntry::DEL_ACK);
104 }
105 
108  boost::bind(&VrfOvsdbObject::OvsdbNotify, this, _1, _2));
109 }
110 
112 }
113 
115  struct ovsdb_idl_row *row) {
116  const char *mac = ovsdb_wrapper_ucast_mac_remote_mac(row);
117  const char *logical_switch =
119  /* if logical switch is not available ignore notification */
120  if (logical_switch == NULL)
121  return;
122  VrfOvsdbEntry vrf_key(this, logical_switch);
123  VrfOvsdbEntry *vrf_ovsdb = static_cast<VrfOvsdbEntry *>(Find(&vrf_key));
124 
125  if (vrf_ovsdb == NULL) {
126  if (op == OvsdbClientIdl::OVSDB_DEL) {
127  // nothing to do return from here.
128  return;
129  }
130  // if vrf is not available create a stale entry
131  // to accomodate stale unicast remote route
132  vrf_ovsdb = static_cast<VrfOvsdbEntry *>(CreateStale(&vrf_key));
133  }
134 
135  const char *dest_ip = ovsdb_wrapper_ucast_mac_remote_dst_ip(row);
136  UnicastMacRemoteTable *table= vrf_ovsdb->route_table_;
137  UnicastMacRemoteEntry key(table, mac);
138  if (op == OvsdbClientIdl::OVSDB_DEL) {
139  table->NotifyDeleteOvsdb((OvsdbDBEntry*)&key, row);
140  if (dest_ip)
141  key.dest_ip_ = std::string(dest_ip);
143  } else if (op == OvsdbClientIdl::OVSDB_ADD) {
144  table->NotifyAddOvsdb((OvsdbDBEntry*)&key, row);
145  if (dest_ip)
146  key.dest_ip_ = std::string(dest_ip);
148  }
149 }
150 
151 KSyncEntry *VrfOvsdbObject::Alloc(const KSyncEntry *key, uint32_t index) {
152  const VrfOvsdbEntry *k_entry =
153  static_cast<const VrfOvsdbEntry *>(key);
154  VrfOvsdbEntry *entry =
155  new VrfOvsdbEntry(this, k_entry->logical_switch_name_);
156  return entry;
157 }
158 
160  const VrfEntry *entry = static_cast<const VrfEntry *>(db_entry);
161  VrfOvsdbEntry *key =
162  new VrfOvsdbEntry(this, UuidToString(entry->vn()->GetUuid()));
163  return static_cast<KSyncEntry *>(key);
164 }
165 
166 OvsdbDBEntry *VrfOvsdbObject::AllocOvsEntry(struct ovsdb_idl_row *row) {
167  return NULL;
168 }
169 
171  const DBEntry *entry, const OvsdbDBEntry *ovsdb_entry) {
172  const VrfEntry *vrf = static_cast<const VrfEntry *>(entry);
173  // Delete Vrf if vn goes NULL
174  if (vrf->vn() == NULL) {
175  return DBFilterDelete;
176  }
177 
178  if (ovsdb_entry != NULL) {
179  const VrfOvsdbEntry *o_vrf =
180  static_cast<const VrfOvsdbEntry *>(ovsdb_entry);
181  if (o_vrf->logical_switch_name_ != UuidToString(vrf->vn()->GetUuid())) {
182  return DBFilterDelAdd;
183  }
184  }
185 
186  return DBFilterAccept;
187 }
188 
190 // Sandesh routines
193  AgentSandeshArguments &args) :
194  OvsdbSandeshTask(resp_ctx, args) {
195  if (false == args.Get("logical_switch", &logical_switch_)) {
196  logical_switch_ = "";
197  }
198  if (false == args.Get("mac", &mac_)) {
199  mac_ = "";
200  }
201 }
202 
204  std::string resp_ctx, const std::string &ip, uint32_t port,
205  const std::string &logical_switch, const std::string &mac) :
206  OvsdbSandeshTask(resp_ctx, ip, port), logical_switch_(logical_switch),
207  mac_(mac) {
208 }
209 
211 }
212 
214  if (!logical_switch_.empty()) {
215  args.Add("logical_switch", logical_switch_);
216  }
217  if (!mac_.empty()) {
218  args.Add("mac", mac_);
219  }
220 }
221 
223  if (!logical_switch_.empty()) {
224  VrfOvsdbEntry *entry = static_cast<VrfOvsdbEntry *>(kentry);
225  if (entry->logical_switch_name().find(
226  logical_switch_) != std::string::npos) {
227  return FilterAllow;
228  }
229  return FilterDeny;
230  }
231  return FilterAllow;
232 }
233 
235  SandeshResponse *resp) {
236  VrfOvsdbEntry *entry = static_cast<VrfOvsdbEntry *>(kentry);
237  OvsdbVrfEntry ventry;
238  ventry.set_state(entry->StateString());
239  ventry.set_logical_switch(entry->logical_switch_name());
241  entry->logical_switch_name(), mac_);
242  ventry.set_unicast_remote_table(task.EncodeFirstPage());
243  OvsdbVrfResp *vrf_resp = static_cast<OvsdbVrfResp *>(resp);
244 
245  std::vector<OvsdbVrfEntry> &vrf_list =
246  const_cast<std::vector<OvsdbVrfEntry>&>(vrf_resp->get_vrfs());
247  vrf_list.push_back(ventry);
248 }
249 
251  return static_cast<SandeshResponse *>(new OvsdbVrfResp());
252 }
253 
255  return static_cast<KSyncObject *>(session->client_idl()->vrf_ovsdb());
256 }
257 
258 void OvsdbVrfReq::HandleRequest() const {
260  new OvsdbVrfSandeshTask(context(), get_session_remote_ip(),
261  get_session_remote_port(),
262  get_logical_switch(), get_mac());
264  scheduler->Enqueue(task);
265 }
266 
KSyncEntry * UnresolvedReference()
Definition: vrf_ovsdb.cc:95
char * ovsdb_wrapper_ucast_mac_remote_mac(struct ovsdb_idl_row *row)
OvsdbClientIdl * client_idl()
Definition: ovsdb_object.h:76
DBFilterResp OvsdbDBEntryFilter(const DBEntry *entry, const OvsdbDBEntry *ovsdb_entry)
Definition: vrf_ovsdb.cc:170
The TaskScheduler keeps track of what tasks are currently schedulable. When a task is enqueued it is ...
Definition: task.h:178
Definition: vrf.h:86
VrfOvsdbObject(OvsdbClientIdl *idl)
Definition: vrf_ovsdb.cc:106
std::string logical_switch_name_
Definition: vrf_ovsdb.h:66
AgentRouteTable * oper_route_table_
Definition: vrf_ovsdb.h:68
OvsdbClientIdlPtr client_idl_
Definition: ovsdb_object.h:83
bool IsLess(const KSyncEntry &) const
Definition: vrf_ovsdb.cc:89
std::string StateString() const
void EncodeArgs(AgentSandeshArguments &args)
Definition: vrf_ovsdb.cc:213
char * ovsdb_wrapper_ucast_mac_remote_logical_switch(struct ovsdb_idl_row *row)
UnicastMacRemoteTable * route_table_
Definition: vrf_ovsdb.h:67
char * ovsdb_wrapper_ucast_mac_remote_dst_ip(struct ovsdb_idl_row *row)
static std::string UuidToString(const boost::uuids::uuid &id)
Definition: string_util.h:138
FilterResp Filter(KSyncEntry *entry)
Definition: vrf_ovsdb.cc:222
bool delete_scheduled()
Definition: ksync_object.h:146
void OvsdbNotify(OvsdbClientIdl::Op, struct ovsdb_idl_row *)
Definition: vrf_ovsdb.cc:114
void TriggerAck(UnicastMacRemoteTable *table)
Definition: vrf_ovsdb.cc:99
bool Get(const std::string &key, std::string *val) const
void SafeNotifyEvent(KSyncEntry *entry, KSyncEntry::KSyncEvent event)
static TaskScheduler * GetInstance()
Definition: task.cc:547
void Enqueue(Task *task)
Enqueues a task for running. Starts task if all policy rules are met else puts task in waitq...
Definition: task.cc:636
KSyncEntry * Alloc(const KSyncEntry *key, uint32_t index)
Definition: vrf_ovsdb.cc:151
DBTableBase * GetDBTable()
Definition: ksync_object.h:244
void NotifyDeleteOvsdb(OvsdbDBEntry *key, struct ovsdb_idl_row *row)
KSyncObject * GetObject() const
Definition: ovsdb_entry.cc:194
bool stale() const
Definition: ksync_entry.h:161
AgentRouteTable * GetBridgeRouteTable() const
Definition: vrf.cc:334
bool Sync(DBEntry *)
Definition: vrf_ovsdb.cc:78
KSyncEntry * Find(const KSyncEntry *key)
Definition: ksync_object.cc:99
bool Add(const std::string &key, const std::string &val)
OvsdbVrfSandeshTask(std::string resp_ctx, AgentSandeshArguments &args)
Definition: vrf_ovsdb.cc:192
VrfOvsdbObject * vrf_ovsdb()
const std::string & logical_switch_name()
Definition: vrf_ovsdb.h:58
KSyncEntry * CreateStale(const KSyncEntry *key)
KSyncObject * GetObject(OvsdbClientSession *session)
Definition: vrf_ovsdb.cc:254
VrfOvsdbEntry(OvsdbDBObject *table, const std::string &logical_switch)
Definition: vrf_ovsdb.cc:28
VnEntry * vn() const
Definition: vrf.h:101
OvsdbDBEntry * AllocOvsEntry(struct ovsdb_idl_row *row)
Definition: vrf_ovsdb.cc:166
virtual ~VrfOvsdbObject()
Definition: vrf_ovsdb.cc:111
KSyncEntry * DBToKSyncEntry(const DBEntry *)
Definition: vrf_ovsdb.cc:159
const boost::uuids::uuid & GetUuid() const
Definition: vn.h:161
virtual void OvsdbRegisterDBTable(DBTable *tbl)
OvsdbDBObject * table_
Definition: ovsdb_entry.h:110
void UpdateResp(KSyncEntry *kentry, SandeshResponse *resp)
Definition: vrf_ovsdb.cc:234
SandeshResponse * Alloc()
Definition: vrf_ovsdb.cc:250
void NotifyAddOvsdb(OvsdbDBEntry *key, struct ovsdb_idl_row *row)
Definition: ovsdb_object.cc:92
struct task_ task