OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
logical_switch_ovsdb.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2014 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 <physical_switch_ovsdb.h>
13 #include <logical_switch_ovsdb.h>
14 #include <physical_locator_ovsdb.h>
17 
18 #include <oper/vn.h>
19 #include <oper/vrf.h>
21 #include <ovsdb_sandesh.h>
22 #include <ovsdb_types.h>
23 
24 using namespace OVSDB;
25 
26 namespace OVSDB {
27 
29  p->back_ref_set_.insert(ref);
30 }
31 
33  p->back_ref_set_.erase(ref);
34 }
35 };
36 
38  const std::string &name) :
39  OvsdbDBEntry(table), name_(name), device_name_(), vxlan_id_(0),
40  mcast_local_row_list_(), mcast_remote_row_(NULL), delete_ovs_(false),
41  res_vxlan_id_(table->client_idl()->vxlan_table(), this), del_task_(NULL) {
42 }
43 
45  const PhysicalDeviceVn *entry) : OvsdbDBEntry(table),
46  name_(UuidToString(entry->vn()->GetUuid())), mcast_local_row_list_(),
47  mcast_remote_row_(NULL), delete_ovs_(false),
48  res_vxlan_id_(table->client_idl()->vxlan_table(), this), del_task_(NULL) {
49  vxlan_id_ = entry->vxlan_id();
51 }
52 
54  const LogicalSwitchEntry *entry) : OvsdbDBEntry(table),
55  mcast_local_row_list_(), mcast_remote_row_(NULL), delete_ovs_(false),
56  res_vxlan_id_(table->client_idl()->vxlan_table(), this), del_task_(NULL) {
57  name_ = entry->name_;
58  vxlan_id_ = entry->vxlan_id_;;
59  device_name_ = entry->device_name_;
60 }
61 
63  struct ovsdb_idl_row *entry) : OvsdbDBEntry(table, entry),
64  name_(ovsdb_wrapper_logical_switch_name(entry)), device_name_(""),
66  mcast_remote_row_(NULL), delete_ovs_(false),
67  res_vxlan_id_(table->client_idl()->vxlan_table(), this), del_task_(NULL) {
68 }
69 
71  assert(pl_create_ref_.get() == NULL);
72 }
73 
75  PhysicalSwitchEntry *p_switch =
76  static_cast<PhysicalSwitchEntry *>(physical_switch_.get());
77  return p_switch->tunnel_ip();
78 }
79 
80 void LogicalSwitchEntry::AddMsg(struct ovsdb_idl_txn *txn) {
82  PhysicalSwitchEntry key(p_table, device_name_.c_str());
83  physical_switch_ = p_table->GetReference(&key);
84 
85  if (stale()) {
86  // skip add encoding for stale entry
87  return;
88  }
89 
90  struct ovsdb_idl_row *row =
93 
94  // Encode Delete for Old remote multicast entries
96 
97  // Add remote multicast entry if not already present
98  // and if old mcast remote MAC list is empty
99  if (old_mcast_remote_row_list_.empty() && mcast_remote_row_ == NULL) {
100  std::string dest_ip = table_->client_idl()->tsn_ip().to_string();
101  PhysicalLocatorTable *pl_table =
103  PhysicalLocatorEntry pl_key(pl_table, dest_ip);
104  /*
105  * we don't take reference to physical locator, just use if locator
106  * is existing or we will create a new one.
107  */
108  PhysicalLocatorEntry *pl_entry =
109  static_cast<PhysicalLocatorEntry *>(pl_table->Find(&pl_key));
110  struct ovsdb_idl_row *pl_row = NULL;
111  if (pl_entry)
112  pl_row = pl_entry->ovs_entry();
113  ovsdb_wrapper_add_mcast_mac_remote(txn, NULL, "unknown-dst", row,
114  pl_row, dest_ip.c_str());
115  }
116 
118 }
119 
120 void LogicalSwitchEntry::ChangeMsg(struct ovsdb_idl_txn *txn) {
121  AddMsg(txn);
122 }
123 
124 void LogicalSwitchEntry::DeleteMsg(struct ovsdb_idl_txn *txn) {
125  physical_switch_ = NULL;
126 
127  // encode delete of entry if it is non-NULL
128  if (mcast_remote_row_ != NULL) {
130  }
132 
133  if (ovs_entry_ != NULL) {
135  }
136 
137  OvsdbIdlRowList::iterator it;
138  for (it = mcast_local_row_list_.begin();
139  it != mcast_local_row_list_.end(); ++it) {
141  }
142  for (it = ucast_local_row_list_.begin();
143  it != ucast_local_row_list_.end(); ++it) {
145  }
146 
148 }
149 
151  if (!IsResolved())
153 }
154 
155 const std::string &LogicalSwitchEntry::name() const {
156  return name_;
157 }
158 
159 const std::string &LogicalSwitchEntry::device_name() const {
160  return device_name_;
161 }
162 
164  return vxlan_id_;
165 }
166 
169 }
170 
172  return res_vxlan_id_;
173 }
174 
176  return delete_ovs_;
177 }
178 
180  PhysicalDeviceVn *entry =
181  static_cast<PhysicalDeviceVn *>(db_entry);
182  bool change = false;
183  if (vxlan_id_ != entry->vxlan_id()) {
184  vxlan_id_ = entry->vxlan_id();
185  change = true;
186  }
187  if (device_name_ != entry->device_display_name()) {
189  change = true;
190  }
191  return change;
192 }
193 
194 bool LogicalSwitchEntry::IsLess(const KSyncEntry &entry) const {
195  const LogicalSwitchEntry &ps_entry =
196  static_cast<const LogicalSwitchEntry&>(entry);
197  return (name_.compare(ps_entry.name_) < 0);
198 }
199 
201  assert(pl_create_ref_.get() == NULL);
202 
203  if (stale()) {
204  // while creating stale entry we should not wait for physical
205  // switch object since it will not be available till config
206  // comes up
207 
208  // for stale entry we should always be able to acquire vxlan id
209  // However in certain cases, where OVSDB database is already
210  // in a state where two Logical switch entries exists with
211  // same VxLAN ID, we need to recover by deleting the entry
212  // from OVSDB database
213  bool ret = res_vxlan_id_.AcquireVxLanId((uint32_t)vxlan_id_);
214  if (!ret) {
216  }
217 
218  return NULL;
219  }
220 
222  PhysicalSwitchEntry key(p_table, device_name_.c_str());
223  PhysicalSwitchEntry *p_switch =
224  static_cast<PhysicalSwitchEntry *>(p_table->GetReference(&key));
225  if (!p_switch->IsResolved()) {
226  return p_switch;
227  }
228 
229  bool ret = res_vxlan_id_.AcquireVxLanId((uint32_t)vxlan_id_);
230  if (!ret) {
231  // failed to get vxlan-id hold entry in defer state
232  // and delete ovs
233  DeleteOvs(true);
235  }
236 
237  // cancel running delete process
238  CancelDeleteOvs();
239 
240  // check if physical locator is available
241  std::string dest_ip = table_->client_idl()->tsn_ip().to_string();
242  PhysicalLocatorTable *pl_table =
244  PhysicalLocatorEntry pl_key(pl_table, dest_ip);
245  PhysicalLocatorEntry *pl_entry =
246  static_cast<PhysicalLocatorEntry *>(pl_table->GetReference(&pl_key));
247  if (!pl_entry->IsResolved()) {
248  if (!pl_entry->AcquireCreateRequest(this)) {
249  // failed to Acquire Create Request, wait for physical locator
250  return pl_entry;
251  }
252  pl_create_ref_ = pl_entry;
253  }
254 
255  return NULL;
256 }
257 
259  return (local_mac_ref_.get() != NULL);
260 }
261 
262 void LogicalSwitchEntry::Ack(bool success) {
263  if (success) {
264  if (ovs_entry_ != NULL) {
265  uint32_t active_vxlan_id =
267  res_vxlan_id_.set_active_vxlan_id(active_vxlan_id);
268  } else {
270  // trigger delete ovs completed/cancel
271  CancelDeleteOvs();
272  }
273  } else {
274  if (delete_ovs_) {
275  // trigger delete ovs completed/cancel
276  CancelDeleteOvs();
277  // try delete OVS again
278  DeleteOvs(false);
279  }
280  }
282  OvsdbDBEntry::Ack(success);
283 }
284 
286  if (ovs_entry_ != NULL) {
287  uint32_t active_vxlan_id =
289  res_vxlan_id_.set_active_vxlan_id(active_vxlan_id);
290  } else {
292  // trigger delete ovs completed/cancel
293  CancelDeleteOvs();
294  }
296 }
297 
298 void LogicalSwitchEntry::DeleteOvs(bool add_change_in_progress) {
299  if (ovs_entry_ == NULL || delete_ovs_ == true) {
300  return;
301  }
302 
303  delete_ovs_ = true;
304  assert(del_task_ == NULL);
305  // should not be triggered on stale entry
306  assert(!stale());
307 
308  // skip assert if this API is called inline from an Add/Change
309  // processing, where the state may say not active while this API
310  // is called and will result in false failures
311  if (!add_change_in_progress) {
312  // this API internally triggers DELADD_REQ on KSync entry
313  // which will activate the entry, if it needs to handle this
314  // operation for an inactive entry, it also needs to ensure
315  // that eventually an ADD should not be triggered
316  assert(IsActive());
317  }
318 
320  TaskScheduler *scheduler = table_->client_idl()->agent()->task_scheduler();
321  scheduler->Enqueue(del_task_);
322 }
323 
325  LogicalSwitchEntry *entry) :
326  Task((entry->table()->client_idl()->agent()->task_scheduler()\
327  ->GetTaskId("Agent::KSync")), 0), entry_(entry) {
328 }
329 
331 }
332 
334  LogicalSwitchEntry *entry = static_cast<LogicalSwitchEntry*>(entry_.get());
335 
336  // start from the entry where the last iteration stopped
337  // entries are removed from this tree in PostDelete which
338  // is triggered only on Delete Acks, using begin_ref_ helps
339  // skip already delete-add processed entries.
340  IntrusiveReferrer reffer(begin_ref_.get(), NULL);
341  std::set<IntrusiveReferrer>::const_iterator it =
342  entry->back_ref_set_.lower_bound(reffer);
343  OvsdbDBEntry *ref_entry = NULL;
344 
345  for (int i = 0; i < kEntriesPerIteration; i++) {
346  if (it == entry->back_ref_set_.end()) {
347  break;
348  }
349  ref_entry = static_cast<OvsdbDBEntry*>((*it).first);
350  it++;
351  ref_entry->TriggerDeleteAdd();
352  }
353 
354  if (it != entry->back_ref_set_.end()) {
355  // keep reference to the entry to be processes next
356  begin_ref_ = static_cast<OvsdbDBEntry*>((*it).first);
357  } else {
358  // finished with the complete set reset begin_ref_ as NULL
359  begin_ref_ = NULL;
360  }
361 
362  if (!entry->back_ref_set_.empty()) {
363  return false;
364  }
365 
366  entry->del_task_ = NULL;
367 
368  if (entry->ovs_entry() != NULL && !entry->IsLocalMacsRef()) {
369  entry->TriggerDeleteAdd();
370  }
371 
372  // on task completion reset intrusive pointer in context of task itself
373  // rather than processing it in the Task destructor, since in case of
374  // task completion task destructor will be called in context of scheduler
375  entry_ = NULL;
376  return true;
377 }
378 
380  delete_ovs_ = false;
381  if (del_task_ != NULL) {
382  TaskScheduler *scheduler = table_->client_idl()->agent()->task_scheduler();
383  assert(scheduler->Cancel(del_task_) == TaskScheduler::CANCELLED);
384  del_task_ = NULL;
385  }
386 }
387 
389  SandeshLogicalSwitchInfo info;
390  switch (event) {
391  case ADD_REQ:
392  info.set_op("Add Requested");
393  break;
394  case DEL_REQ:
395  info.set_op("Delete Requested");
396  break;
397  case ADD_ACK:
398  info.set_op("Add Received");
399  break;
400  case DEL_ACK:
401  info.set_op("Delete Received");
402  break;
403  case DUP_TUNNEL_KEY_ADD:
404  info.set_op("Add Request with Duplicate tunnel key");
405  break;
406  default:
407  info.set_op("unknown");
408  }
409  info.set_name(name_);
410  info.set_device_name(device_name_);
411  info.set_vxlan(vxlan_id_);
412  OVSDB_TRACE(LogicalSwitch, info);
413 }
414 
416  OvsdbIdlRowList::iterator it;
417  for (it = old_mcast_remote_row_list_.begin();
418  it != old_mcast_remote_row_list_.end(); ++it) {
420  }
421 }
422 
424  // on Ack Release the physical locator create ref, if present
425  if (pl_create_ref_.get() != NULL) {
426  // release creator reference on txn complete
427  PhysicalLocatorEntry *pl_entry =
428  static_cast<PhysicalLocatorEntry *>(pl_create_ref_.get());
429  pl_entry->ReleaseCreateRequest(this);
430  pl_create_ref_ = NULL;
431  }
432 }
433 
434 void LogicalSwitchEntry::NotifyDelete(struct ovsdb_idl_row *row) {
435  OvsdbIdlRowList::iterator it;
436  //Since logical-switch delete is received from ovsdb node,
437  //aggresively delete the local macs.
438  for (it = mcast_local_row_list_.begin();
439  it != mcast_local_row_list_.end(); ++it) {
441  Notify(OvsdbClientIdl::OVSDB_DEL, (*it));
443  }
444  for (it = ucast_local_row_list_.begin();
445  it != ucast_local_row_list_.end(); ++it) {
447  Notify(OvsdbClientIdl::OVSDB_DEL, (*it));
449  }
450 
452  return;
453 }
454 
455 // No config seen for this logical switch, so do the cleanup so that delete can
456 // proceed.
458  //Release self reference.
459  local_mac_ref_ = NULL;
460 }
461 
463  OvsdbDBObject(idl, true) {
465  boost::bind(&LogicalSwitchTable::OvsdbNotify, this, _1, _2));
468  this, _1, _2));
469 }
470 
472 }
473 
475  struct ovsdb_idl_row *row) {
476  LogicalSwitchEntry key(this, row);
477  if (op == OvsdbClientIdl::OVSDB_DEL) {
478  NotifyDeleteOvsdb((OvsdbDBEntry*)&key, row);
480  } else if (op == OvsdbClientIdl::OVSDB_ADD) {
481  NotifyAddOvsdb((OvsdbDBEntry*)&key, row);
483  } else {
484  assert(0);
485  }
486 }
487 
489  struct ovsdb_idl_row *row) {
490  const char *mac = ovsdb_wrapper_mcast_mac_local_mac(row);
491  const char *ls = ovsdb_wrapper_mcast_mac_local_logical_switch(row);
492  LogicalSwitchEntry *entry = NULL;
493  if (ls) {
494  LogicalSwitchEntry key(this, ls);
495  entry = static_cast<LogicalSwitchEntry *>(Find(&key));
496  }
497  struct ovsdb_idl_row *l_set =
499  // physical locator set is immutable, multicast row with NULL
500  // physical locator is not valid, and it may not observe further
501  // delete trigger, so trigger delete for row and wait for
502  // locator set to be available
503  if (op == OvsdbClientIdl::OVSDB_DEL || l_set == NULL) {
504  // trigger deletion based on the entry for which add was triggered
505  OvsdbIdlRowMap::iterator idl_it = idl_row_map_.find(row);
506  if (idl_it != idl_row_map_.end()) {
507  entry = idl_it->second;
508  idl_row_map_.erase(idl_it);
509  }
510 
511  OVSDB_TRACE(Trace, "Delete : Local Mcast MAC " + std::string(mac) +
512  ", logical switch " + (entry != NULL ? entry->name() : ""));
513  if (entry) {
514  entry->mcast_local_row_list_.erase(row);
515  }
516  } else if (op == OvsdbClientIdl::OVSDB_ADD) {
517  OVSDB_TRACE(Trace, "Add : Local Mcast MAC " + std::string(mac) +
518  ", logical switch " + (ls ? std::string(ls) : ""));
519  if (entry) {
520  idl_row_map_[row] = entry;
521  entry->mcast_local_row_list_.insert(row);
522  }
523  } else {
524  assert(0);
525  }
526 
527  if (entry) {
528  if (!entry->mcast_local_row_list_.empty() ||
529  !entry->ucast_local_row_list_.empty()) {
530  if (entry->IsActive())
531  entry->local_mac_ref_ = entry;
532  } else {
533  entry->local_mac_ref_ = NULL;
534  if (entry->delete_ovs_ && entry->ovs_entry() != NULL) {
535  entry->TriggerDeleteAdd();
536  }
537  }
538  }
539 }
540 
542  struct ovsdb_idl_row *row) {
543  const char *mac = ovsdb_wrapper_mcast_mac_remote_mac(row);
544  const char *ls = ovsdb_wrapper_mcast_mac_remote_logical_switch(row);
545  LogicalSwitchEntry *entry = NULL;
546  if (ls) {
547  LogicalSwitchEntry key(this, ls);
548  entry = static_cast<LogicalSwitchEntry *>(Find(&key));
549  }
550  if (op == OvsdbClientIdl::OVSDB_DEL) {
551  // trigger deletion based on the entry for which add was triggered
552  OvsdbIdlRowMap::iterator idl_it = idl_row_map_.find(row);
553  if (idl_it != idl_row_map_.end()) {
554  entry = idl_it->second;
555  idl_row_map_.erase(idl_it);
556  }
557 
558  OVSDB_TRACE(Trace, "Delete : Remote Mcast MAC " + std::string(mac) +
559  ", logical switch " + (entry != NULL ? entry->name() : ""));
560  if (entry) {
561  entry->old_mcast_remote_row_list_.erase(row);
562  if (entry->mcast_remote_row_ == row) {
563  entry->mcast_remote_row_ = NULL;
564  }
565 
566  // trigger change for active entry, once we are done deleting
567  // all old mcast remote rows
568  if (entry->IsActive() && entry->old_mcast_remote_row_list_.empty())
569  Change(entry);
570  }
571  } else if (op == OvsdbClientIdl::OVSDB_ADD) {
572  OVSDB_TRACE(Trace, "Add : Remote Mcast MAC " + std::string(mac) +
573  ", logical switch " + (ls ? std::string(ls) : ""));
574  if (entry) {
575  idl_row_map_[row] = entry;
576  if (entry->mcast_remote_row_ != row) {
577  if (entry->mcast_remote_row_) {
578  // if we already had an entry move old and current
579  // entry to old remote mac list to trigger delete
580  // for both the mcast remote mac entries.
581  // once both are deleted, we will trigger add
582  // for new entry
583  entry->old_mcast_remote_row_list_.insert(
584  entry->mcast_remote_row_);
585  entry->mcast_remote_row_ = NULL;
586  entry->old_mcast_remote_row_list_.insert(row);
587  } else {
588  entry->mcast_remote_row_ = row;
589  }
590  }
591 
592  std::string dest_ip = ovsdb_wrapper_mcast_mac_remote_dst_ip(row);
593  std::string tsn_ip = client_idl()->tsn_ip().to_string();
594  if (dest_ip.compare(tsn_ip) != 0) {
595  // dest ip is different from tsn ip
596  // move this row to old mcast list, to delete curreny
597  // entry and reprogram new entry with correct TSN IP
598  entry->old_mcast_remote_row_list_.insert(row);
599  entry->mcast_remote_row_ = NULL;
600  }
601 
602  // trigger change on active entry to delete all old
603  // mcast remote rows.
604  if (entry->IsActive() && !entry->old_mcast_remote_row_list_.empty())
605  Change(entry);
606  }
607  } else {
608  assert(0);
609  }
610 }
611 
613  struct ovsdb_idl_row *row) {
614  const char *mac = ovsdb_wrapper_ucast_mac_local_mac(row);
615  const char *ls = ovsdb_wrapper_ucast_mac_local_logical_switch(row);
616  LogicalSwitchEntry *entry = NULL;
617  if (ls) {
618  LogicalSwitchEntry key(this, ls);
619  entry = static_cast<LogicalSwitchEntry *>(Find(&key));
620  }
621  if (op == OvsdbClientIdl::OVSDB_DEL) {
622  // trigger deletion based on the entry for which add was triggered
623  OvsdbIdlRowMap::iterator idl_it = idl_row_map_.find(row);
624  if (idl_it != idl_row_map_.end()) {
625  entry = idl_it->second;
626  idl_row_map_.erase(idl_it);
627  }
628 
629  OVSDB_TRACE(Trace, "Delete : Local Ucast MAC " + std::string(mac) +
630  ", logical switch " + (entry != NULL ? entry->name() : ""));
631  if (entry) {
632  entry->ucast_local_row_list_.erase(row);
633  }
634  } else if (op == OvsdbClientIdl::OVSDB_ADD) {
635  OVSDB_TRACE(Trace, "Add : Local Ucast MAC " + std::string(mac) +
636  ", logical switch " + (ls ? std::string(ls) : ""));
637  if (entry) {
638  idl_row_map_[row] = entry;
639  entry->ucast_local_row_list_.insert(row);
640  }
641  } else {
642  assert(0);
643  }
644 
645  if (entry) {
646  if (!entry->mcast_local_row_list_.empty() ||
647  !entry->ucast_local_row_list_.empty()) {
648  if (entry->IsActive())
649  entry->local_mac_ref_ = entry;
650  } else {
651  entry->local_mac_ref_ = NULL;
652  if (entry->delete_ovs_ && entry->ovs_entry() != NULL) {
653  entry->TriggerDeleteAdd();
654  }
655  }
656  }
657 }
658 
659 KSyncEntry *LogicalSwitchTable::Alloc(const KSyncEntry *key, uint32_t index) {
660  const LogicalSwitchEntry *k_entry =
661  static_cast<const LogicalSwitchEntry *>(key);
662  LogicalSwitchEntry *entry = new LogicalSwitchEntry(this, k_entry);
663  return entry;
664 }
665 
667  const PhysicalDeviceVn *entry =
668  static_cast<const PhysicalDeviceVn *>(db_entry);
669  LogicalSwitchEntry *key = new LogicalSwitchEntry(this, entry);
670  return static_cast<KSyncEntry *>(key);
671 }
672 
673 OvsdbDBEntry *LogicalSwitchTable::AllocOvsEntry(struct ovsdb_idl_row *row) {
674  LogicalSwitchEntry key(this, row);
675  return static_cast<OvsdbDBEntry *>(CreateStale(&key));
676 }
677 
679  const DBEntry *db_entry, const OvsdbDBEntry *ovsdb_entry) {
680  const PhysicalDeviceVn *entry =
681  static_cast<const PhysicalDeviceVn *>(db_entry);
682 
683  // Delete the entry which has invalid VxLAN id associated.
684  if (entry->vxlan_id() == 0) {
685  return DBFilterDelete;
686  }
687  return DBFilterAccept;
688 }
689 
692  new ProcessDeleteTableReqTask(this);
693  TaskScheduler *scheduler = client_idl()->agent()->task_scheduler();
694  scheduler->Enqueue(task);
695 }
696 
698  LogicalSwitchTable *table) :
699  Task((table->client_idl()->agent()->task_scheduler()\
700  ->GetTaskId("Agent::KSync")), 0), table_(table), entry_(NULL) {
701 }
702 
704 }
705 
707  KSyncEntry *kentry = entry_.get();
708  if (kentry == NULL) {
709  kentry = table_->Next(kentry);
710  }
711 
712  int count = 0;
713  while (kentry != NULL) {
714  LogicalSwitchEntry *entry = static_cast<LogicalSwitchEntry *>(kentry);
715  count++;
716  kentry = table_->Next(kentry);
717  // while table is set for deletion reset the local_mac_ref
718  // since there will be no trigger from OVSDB database
719  entry->local_mac_ref_ = NULL;
720 
721  // check for yield
722  if (count == kEntriesPerIteration && kentry != NULL) {
723  entry_ = kentry;
724  return false;
725  }
726  }
727 
728  entry_ = NULL;
729  // Done processing delete request, schedule delete table
730  table_->DeleteTable();
731  return true;
732 }
733 
735 // Sandesh routines
738  std::string resp_ctx, AgentSandeshArguments &args) :
739  OvsdbSandeshTask(resp_ctx, args), name_("") {
740  if (false == args.Get("name", &name_)) {
741  name_ = "";
742  }
743  int vxlan_id = 0;
744  if (false == args.Get("vxlan_id", &vxlan_id)) {
745  vxlan_id = 0;
746  }
747  vxlan_id_ = vxlan_id;
748 }
749 
751  const std::string &ip,
752  uint32_t port,
753  const std::string &name,
754  uint32_t vxlan_id) :
755  OvsdbSandeshTask(resp_ctx, ip, port), name_(name), vxlan_id_(vxlan_id) {
756 }
757 
759 }
760 
762  if (!name_.empty()) {
763  args.Add("name", name_);
764  }
765  if (vxlan_id_ != 0) {
766  args.Add("vxlan_id", vxlan_id_);
767  }
768 }
769 
772  if (!name_.empty()) {
773  LogicalSwitchEntry *entry = static_cast<LogicalSwitchEntry *>(kentry);
774  if (entry->name().find(name_) == std::string::npos) {
775  return FilterDeny;
776  }
777  }
778  if (vxlan_id_ != 0) {
779  LogicalSwitchEntry *entry = static_cast<LogicalSwitchEntry *>(kentry);
780  const OvsdbResourceVxLanId &res = entry->res_vxlan_id();
781  if (entry->vxlan_id() != vxlan_id_ &&
782  res.active_vxlan_id() != vxlan_id_) {
783  return FilterDeny;
784  }
785  }
786  return FilterAllow;
787 }
788 
790  SandeshResponse *resp) {
791  LogicalSwitchEntry *entry = static_cast<LogicalSwitchEntry *>(kentry);
792  OvsdbLogicalSwitchEntry lentry;
793  lentry.set_state(entry->StateString());
794  lentry.set_name(entry->name());
795  lentry.set_physical_switch(entry->device_name());
796  lentry.set_vxlan_id(entry->vxlan_id());
797  lentry.set_tor_service_node(entry->tor_service_node());
798  const OvsdbResourceVxLanId &res = entry->res_vxlan_id();
799  lentry.set_vxlan_id_available(res.VxLanId() != 0);
800  lentry.set_ovs_vxlan_id(res.active_vxlan_id());
801  lentry.set_delete_in_progress(entry->IsDeleteOvsInProgress());
802 
803  //Debugs to find out what keeps logical switch pending.
804  lentry.set_mcast_local_size(entry->mcast_local_row_list_size());
805  lentry.set_ucast_local_size(entry->ucast_local_row_list_size());
806  lentry.set_old_mcast_remote_size(entry->old_mcast_remote_row_list_size());
807  lentry.set_mcast_remote_set(entry->is_mcast_remote_set());
808  lentry.set_local_mac_ref_set(entry->is_local_mac_ref_set());
809 
810  if ((entry->IsDeleted() || entry->IsDeleteOvsInProgress()) &&
811  entry->IsLocalMacsRef()) {
812  lentry.set_message("Waiting for Local Macs Cleanup");
813  }
814  OvsdbLogicalSwitchResp *ls_resp =
815  static_cast<OvsdbLogicalSwitchResp *>(resp);
816  std::vector<OvsdbLogicalSwitchEntry> &lswitch =
817  const_cast<std::vector<OvsdbLogicalSwitchEntry>&>(
818  ls_resp->get_lswitch());
819  lswitch.push_back(lentry);
820 }
821 
823  return static_cast<SandeshResponse *>(new OvsdbLogicalSwitchResp());
824 }
825 
827  return static_cast<KSyncObject *>(
828  session->client_idl()->logical_switch_table());
829 }
830 
831 void OvsdbLogicalSwitchReq::HandleRequest() const {
833  new LogicalSwitchSandeshTask(context(), get_session_remote_ip(),
834  get_session_remote_port(),
835  get_name(), get_vxlan_id());
837  scheduler->Enqueue(task);
838 }
struct ovsdb_idl_row * ovsdb_wrapper_add_logical_switch(struct ovsdb_idl_txn *, struct ovsdb_idl_row *, const char *, int64_t)
ProcessDeleteOvsReqTask * del_task_
const std::string & device_display_name() const
#define OVSDB_TRACE(obj,...)
void Change(KSyncEntry *entry)
char * ovsdb_wrapper_mcast_mac_local_logical_switch(struct ovsdb_idl_row *row)
OvsdbClientIdl * client_idl()
Definition: ovsdb_object.h:76
The TaskScheduler keeps track of what tasks are currently schedulable. When a task is enqueued it is ...
Definition: task.h:178
OvsdbIdlRowList old_mcast_remote_row_list_
bool IsResolved()
uint32_t ucast_local_row_list_size() const
KSyncEntry * Alloc(const KSyncEntry *key, uint32_t index)
virtual void NotifyDelete(struct ovsdb_idl_row *)
LogicalSwitchTable(OvsdbClientIdl *idl)
std::string StateString() const
KSyncEntry * DBToKSyncEntry(const DBEntry *)
struct ovsdb_idl_row * ovsdb_wrapper_mcast_mac_local_physical_locator_set(struct ovsdb_idl_row *row)
char * ovsdb_wrapper_logical_switch_name(struct ovsdb_idl_row *row)
char * ovsdb_wrapper_mcast_mac_remote_logical_switch(struct ovsdb_idl_row *row)
bool AcquireCreateRequest(KSyncEntry *creator)
Agent * agent() const
void UpdateResp(KSyncEntry *kentry, SandeshResponse *resp)
std::pair< void *, void * > IntrusiveReferrer
static std::string UuidToString(const boost::uuids::uuid &id)
Definition: string_util.h:138
PhysicalLocatorTable * physical_locator_table()
int vxlan_id() const
bool IsLess(const KSyncEntry &) const
void Register(EntryType type, NotifyCB cb)
MulticastMacLocalOvsdb * multicast_mac_local_ovsdb()
DBFilterResp OvsdbDBEntryFilter(const DBEntry *entry, const OvsdbDBEntry *ovsdb_entry)
uint32_t old_mcast_remote_row_list_size() const
bool Run()
Code to execute. Returns true if task is completed. Return false to reschedule the task...
void ChangeMsg(struct ovsdb_idl_txn *)
CancelReturnCode Cancel(Task *task)
Cancels a Task that can be in RUN/WAIT state. The caller needs to ensure that the task exists when Ca...
Definition: task.cc:699
struct ovsdb_idl_row * ovs_entry()
Definition: ovsdb_entry.h:94
void ovsdb_wrapper_delete_mcast_mac_remote(struct ovsdb_idl_row *row)
bool AcquireVxLanId(uint32_t vxlan_id)
TaskScheduler * task_scheduler() const
Definition: agent.h:1120
bool Get(const std::string &key, std::string *val) const
char * ovsdb_wrapper_mcast_mac_remote_mac(struct ovsdb_idl_row *row)
OvsdbDBEntry * AllocOvsEntry(struct ovsdb_idl_row *row)
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
std::string tor_service_node() const
void OvsdbMcastRemoteMacNotify(OvsdbClientIdl::Op, struct ovsdb_idl_row *)
int64_t ovsdb_wrapper_logical_switch_tunnel_key(struct ovsdb_idl_row *row)
void set_active_vxlan_id(uint32_t vxlan_id)
void NotifyDeleteOvsdb(OvsdbDBEntry *key, struct ovsdb_idl_row *row)
Definition: trace.h:220
const std::string & name() const
void AddMsg(struct ovsdb_idl_txn *)
void ovsdb_wrapper_delete_mcast_mac_local(struct ovsdb_idl_row *row)
bool stale() const
Definition: ksync_entry.h:161
struct ovsdb_idl_row * ovs_entry_
Definition: ovsdb_entry.h:111
std::set< IntrusiveReferrer > back_ref_set_
void OvsdbNotify(OvsdbClientIdl::Op, struct ovsdb_idl_row *)
KSyncEntry * Find(const KSyncEntry *key)
Definition: ksync_object.cc:99
struct ovsdb_idl_row * ovs_entry()
Definition: ovsdb_entry.h:47
bool Add(const std::string &key, const std::string &val)
KSyncEntry * GetReference(const KSyncEntry *key)
bool IsActive()
Definition: ksync_entry.h:172
void OvsdbMcastLocalMacNotify(OvsdbClientIdl::Op, struct ovsdb_idl_row *)
boost::asio::ip::address_v4 Ip4Address
Definition: address.h:14
Ip4Address & physical_switch_tunnel_ip()
virtual void NotifyDelete(struct ovsdb_idl_row *)
Definition: ovsdb_entry.cc:190
LogicalSwitchTable * logical_switch_table()
LogicalSwitchSandeshTask(std::string resp_ctx, AgentSandeshArguments &args)
bool IsDeleted()
Definition: ksync_entry.h:163
uint32_t mcast_local_row_list_size() const
KSyncEntry * CreateStale(const KSyncEntry *key)
void ovsdb_wrapper_add_mcast_mac_remote(struct ovsdb_idl_txn *txn, struct ovsdb_idl_row *row, const char *mac, struct ovsdb_idl_row *ls, struct ovsdb_idl_row *pl, const char *dst_ip)
void ovsdb_wrapper_delete_logical_switch(struct ovsdb_idl_row *)
void SendTrace(Trace event) const
void EncodeArgs(AgentSandeshArguments &args)
OvsdbResourceVxLanId & res_vxlan_id()
virtual void Ack(bool success)
Definition: ovsdb_entry.cc:198
void DeleteMsg(struct ovsdb_idl_txn *)
UnicastMacLocalOvsdb * unicast_mac_local_ovsdb()
static KSyncEntry * default_defer_entry()
FilterResp Filter(KSyncEntry *entry)
KSyncObject * GetObject(OvsdbClientSession *session)
void DeleteOvs(bool add_change_in_progress)
const std::string & device_name() const
void intrusive_ptr_add_back_ref(IntrusiveReferrer ref, LogicalSwitchEntry *p)
char * ovsdb_wrapper_ucast_mac_local_logical_switch(struct ovsdb_idl_row *row)
char * ovsdb_wrapper_ucast_mac_local_mac(struct ovsdb_idl_row *row)
bool Run()
Code to execute. Returns true if task is completed. Return false to reschedule the task...
struct ovsdb_idl_row * mcast_remote_row_
void ReleaseCreateRequest(KSyncEntry *creator)
OvsdbDBObject * table_
Definition: ovsdb_entry.h:110
char * ovsdb_wrapper_mcast_mac_remote_dst_ip(struct ovsdb_idl_row *row)
OvsdbResourceVxLanId res_vxlan_id_
char * ovsdb_wrapper_mcast_mac_local_mac(struct ovsdb_idl_row *row)
Task is a wrapper over tbb::task to support policies.
Definition: task.h:86
LogicalSwitchEntry(OvsdbDBObject *table, const std::string &name)
void NotifyAddOvsdb(OvsdbDBEntry *key, struct ovsdb_idl_row *row)
Definition: ovsdb_object.cc:92
void OvsdbUcastLocalMacNotify(OvsdbClientIdl::Op, struct ovsdb_idl_row *)
DBTableBase * table_
Definition: ksync_object.h:262
void intrusive_ptr_del_back_ref(IntrusiveReferrer ref, LogicalSwitchEntry *p)
void ovsdb_wrapper_delete_ucast_mac_local(struct ovsdb_idl_row *row)
struct task_ task
PhysicalSwitchTable * physical_switch_table()