Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include <vector>
6 : #include <atomic>
7 :
8 : #include <tbb/spin_rw_mutex.h>
9 :
10 : #include <boost/bind/bind.hpp>
11 : #include <boost/foreach.hpp>
12 : #include <boost/dynamic_bitset.hpp>
13 : #include <boost/type_traits.hpp>
14 :
15 : #include "base/compiler.h"
16 : #include "base/logging.h"
17 : #include "base/task_annotations.h"
18 : #include "base/time_util.h"
19 : #include "db/db.h"
20 : #include "db/db_partition.h"
21 : #include "db/db_table.h"
22 : #include "db/db_table_partition.h"
23 : #include "db/db_table_walk_mgr.h"
24 : #include "db/db_types.h"
25 :
26 : class DBEntry;
27 : class DBEntryBase;
28 :
29 : using namespace std;
30 : using namespace boost::placeholders;
31 :
32 1276429 : DBRequest::DBRequest() : oper(static_cast<DBOperation>(0)) {
33 1276415 : }
34 :
35 1276345 : DBRequest::~DBRequest() {
36 : #if defined(__GNUC__)
37 : #if (__GNUC_PREREQ(4, 2) > 0)
38 : boost::has_virtual_destructor<DBRequestKey>::type key_has_destructor;
39 : boost::has_virtual_destructor<DBRequestData>::type data_has_destructor;
40 1276345 : assert(key_has_destructor && data_has_destructor);
41 : #endif
42 : #endif
43 1276271 : }
44 :
45 645966 : void DBRequest::Swap(DBRequest *rhs) {
46 645966 : swap(oper, rhs->oper);
47 645964 : swap(key, rhs->key);
48 645959 : swap(data, rhs->data);
49 645952 : }
50 :
51 : // we need copy to be able to resize vector of atomics
52 : // therefore we don't need the same value in both instances
53 : template<typename _Tp>
54 : struct AtomicWithCopy : public std::atomic<_Tp> {
55 : // Inherit constructors
56 : using std::atomic<_Tp>::atomic;
57 :
58 : // Bring in base class operators and methods
59 : using std::atomic<_Tp>::operator=;
60 : using std::atomic<_Tp>::load;
61 : using std::atomic<_Tp>::store;
62 :
63 : // Custom constructor to handle copy from another AtomicWithCopy (must perform a load/store)
64 1010601 : AtomicWithCopy(const AtomicWithCopy& other) : std::atomic<_Tp>(other.load()) {}
65 :
66 : // Custom assignment operator
67 791805 : AtomicWithCopy& operator=(const AtomicWithCopy& other) {
68 1583608 : this->store(other.load());
69 791808 : return *this;
70 : }
71 : };
72 :
73 : class DBTableBase::ListenerInfo {
74 : public:
75 : typedef vector<ChangeCallback> CallbackList;
76 : typedef vector<string> NameList;
77 : typedef vector<AtomicWithCopy<uint64_t>> StateCountList;
78 :
79 1529629 : explicit ListenerInfo(const string &table_name) :
80 1529629 : db_state_accounting_(true) {
81 1529603 : if (table_name.find("__ifmap_") != string::npos) {
82 : // TODO need to have unconditional DB state accounting
83 : // for now skipp DB State accounting for ifmap tables
84 1232701 : db_state_accounting_ = false;
85 : }
86 1529606 : }
87 :
88 792290 : DBTableBase::ListenerId Register(ChangeCallback callback,
89 : const string &name) {
90 792290 : tbb::spin_rw_mutex::scoped_lock write_lock(rw_mutex_, true);
91 792409 : size_t i = bmap_.find_first();
92 792154 : if (i == bmap_.npos) {
93 791739 : i = callbacks_.size();
94 791732 : callbacks_.push_back(callback);
95 791838 : names_.push_back(name);
96 791801 : state_count_.resize(i + 1);
97 791822 : state_count_[i] = AtomicWithCopy<uint64_t>(0);
98 : } else {
99 415 : bmap_.reset(i);
100 413 : if (bmap_.none()) {
101 300 : bmap_.clear();
102 : }
103 413 : callbacks_[i] = callback;
104 413 : names_[i] = name;
105 413 : state_count_[i] = 0;
106 : }
107 792423 : return i;
108 792446 : }
109 :
110 792448 : void Unregister(ListenerId listener) {
111 792448 : tbb::spin_rw_mutex::scoped_lock write_lock(rw_mutex_, true);
112 792448 : callbacks_[listener] = NULL;
113 792447 : names_[listener] = "";
114 : // During Unregister Listener should have cleaned up,
115 : // DB states from all the entries in this table.
116 792445 : assert(state_count_[listener] == 0);
117 792445 : if ((size_t) listener == callbacks_.size() - 1) {
118 1256327 : while (!callbacks_.empty() && callbacks_.back() == NULL) {
119 792029 : callbacks_.pop_back();
120 792029 : names_.pop_back();
121 792031 : state_count_.pop_back();
122 : }
123 464296 : if (bmap_.size() > callbacks_.size()) {
124 225811 : bmap_.resize(callbacks_.size());
125 : }
126 : } else {
127 328149 : if ((size_t) listener >= bmap_.size()) {
128 326390 : bmap_.resize(listener + 1);
129 : }
130 328150 : bmap_.set(listener);
131 : }
132 792446 : }
133 :
134 : // concurrency: called from DBPartition task.
135 2508662 : void RunNotify(DBTablePartBase *tpart, DBEntryBase *entry) {
136 2508662 : tbb::spin_rw_mutex::scoped_lock read_lock(rw_mutex_, false);
137 2509237 : for (CallbackList::iterator iter = callbacks_.begin();
138 8666041 : iter != callbacks_.end(); ++iter) {
139 6156394 : if (*iter != NULL) {
140 6133286 : ChangeCallback cb = *iter;
141 6132452 : (cb)(tpart, entry);
142 6133766 : }
143 : }
144 2508458 : }
145 :
146 3836369 : void AddToDBStateCount(ListenerId listener, int count) {
147 3836369 : if (db_state_accounting_ && listener != DBTableBase::kInvalidId) {
148 3620205 : tbb::spin_rw_mutex::scoped_lock read_lock(rw_mutex_, false);
149 3620940 : state_count_[listener] += count;
150 3621549 : }
151 3837605 : }
152 :
153 1292539 : uint64_t GetDBStateCount(ListenerId listener) {
154 1292539 : assert(db_state_accounting_ && listener != DBTableBase::kInvalidId);
155 1292560 : tbb::spin_rw_mutex::scoped_lock read_lock(rw_mutex_, false);
156 2585410 : return state_count_[listener];
157 1292613 : }
158 :
159 6594 : void FillListeners(vector<ShowTableListener> *listeners) const {
160 6594 : tbb::spin_rw_mutex::scoped_lock read_lock(rw_mutex_, false);
161 6706 : ListenerId id = 0;
162 6706 : for (CallbackList::const_iterator iter = callbacks_.begin();
163 15875 : iter != callbacks_.end(); ++iter, ++id) {
164 9234 : if (*iter != NULL) {
165 9224 : ShowTableListener item;
166 9214 : item.id = id;
167 9214 : item.name = names_[id];
168 9224 : item.state_count = state_count_[id];
169 9215 : listeners->push_back(item);
170 9179 : }
171 : }
172 6639 : }
173 :
174 3755029 : bool empty() const {
175 3755029 : tbb::spin_rw_mutex::scoped_lock read_lock(rw_mutex_, false);
176 7510333 : return callbacks_.empty();
177 3755007 : }
178 :
179 10317 : size_t size() const {
180 10317 : tbb::spin_rw_mutex::scoped_lock read_lock(rw_mutex_, false);
181 20634 : return (callbacks_.size() - bmap_.count());
182 10317 : }
183 :
184 : private:
185 : bool db_state_accounting_;
186 : CallbackList callbacks_;
187 : NameList names_;
188 : StateCountList state_count_;
189 : mutable tbb::spin_rw_mutex rw_mutex_;
190 : boost::dynamic_bitset<> bmap_; // free list.
191 : };
192 :
193 1529619 : DBTableBase::DBTableBase(DB *db, const string &name)
194 1529614 : : db_(db), name_(name), info_(new ListenerInfo(name)),
195 1529619 : enqueue_count_(0), input_count_(0), notify_count_(0) {
196 1529601 : walker_count_ = 0;
197 1529658 : walk_request_count_ = 0;
198 1529644 : walk_complete_count_ = 0;
199 1529655 : walk_cancel_count_ = 0;
200 1529657 : walk_again_count_ = 0;
201 1529658 : walk_count_ = 0;
202 1529649 : }
203 :
204 1529660 : DBTableBase::~DBTableBase() {
205 1529660 : }
206 :
207 792394 : DBTableBase::ListenerId DBTableBase::Register(ChangeCallback callback,
208 : const string &name) {
209 792394 : return info_->Register(callback, name);
210 : }
211 :
212 792448 : void DBTableBase::Unregister(ListenerId listener) {
213 792448 : info_->Unregister(listener);
214 : // If a table is marked for deletion, then we may trigger the deletion
215 : // process when the last client is removed
216 792448 : if (info_->empty())
217 360566 : RetryDelete();
218 792448 : }
219 :
220 644815 : bool DBTableBase::Enqueue(DBRequest *req) {
221 644815 : DBTablePartBase *tpart = GetTablePartition(req->key.get());
222 644805 : DBPartition *partition = db_->GetPartition(tpart->index());
223 644805 : enqueue_count_++;
224 644805 : return partition->EnqueueRequest(tpart, NULL, req);
225 : }
226 :
227 368881 : void DBTableBase::EnqueueRemove(DBEntryBase *db_entry) {
228 368881 : DBTablePartBase *tpart = GetTablePartition(db_entry);
229 368806 : DBPartition *partition = db_->GetPartition(tpart->index());
230 368800 : partition->EnqueueRemove(tpart, db_entry);
231 368942 : }
232 :
233 2508859 : void DBTableBase::RunNotify(DBTablePartBase *tpart, DBEntryBase *entry) {
234 2508859 : notify_count_++;
235 2508859 : info_->RunNotify(tpart, entry);
236 2509159 : }
237 :
238 3836735 : void DBTableBase::AddToDBStateCount(ListenerId listener, int count) {
239 3836735 : info_->AddToDBStateCount(listener, count);
240 3837517 : }
241 :
242 1292517 : uint64_t DBTableBase::GetDBStateCount(ListenerId listener) {
243 1292517 : return info_->GetDBStateCount(listener);
244 : }
245 :
246 288563 : bool DBTableBase::MayDelete() const {
247 288563 : if (HasListeners()) {
248 0 : return false;
249 : }
250 288563 : if (HasWalkers()) {
251 0 : return false;
252 : }
253 288563 : if (!empty()) {
254 0 : return false;
255 : }
256 :
257 288563 : return true;
258 : }
259 :
260 2962564 : bool DBTableBase::HasListeners() const {
261 2962564 : return !info_->empty();
262 : }
263 :
264 10317 : size_t DBTableBase::GetListenerCount() const {
265 10317 : return info_->size();
266 : }
267 :
268 6601 : void DBTableBase::FillListeners(vector<ShowTableListener> *listeners) const {
269 6601 : info_->FillListeners(listeners);
270 6708 : }
271 :
272 : class DBTable::WalkWorker : public Task {
273 : public:
274 : WalkWorker(TableWalker *walker, int db_partition_id);
275 :
276 : virtual bool Run();
277 :
278 0 : std::string Description() const { return "DBTable::WalkWorker"; }
279 :
280 : private:
281 : // Store the last visited node to continue walk
282 : std::unique_ptr<DBRequestKey> walk_ctx_;
283 :
284 : // Table partition for which this worker was created
285 : DBTablePartition *tbl_partition_;
286 :
287 : TableWalker *walker_;
288 : };
289 :
290 : class DBTable::TableWalker {
291 : public:
292 1529638 : TableWalker(DBTable *table) : table_(table) {
293 1529626 : pending_workers_ = 0;
294 1529655 : }
295 :
296 : void StartWalk();
297 :
298 1054438 : DBTable *table() {
299 1054438 : return table_;
300 : }
301 :
302 226155 : void ClearWalkWorks() {
303 226155 : worker_tasks_.clear();
304 226155 : }
305 :
306 : DBTable *table_;
307 : // check whether iteration is completed on all Table Partition
308 : std::atomic<uint16_t> pending_workers_;
309 : // For debugging purpose. Few of the tasks in this list could has finished
310 : // executing and destroyed. List of workers are useful in debugging with
311 : // gdb/gcore to see the current state of the walk and walk_context
312 : std::list<Task *> worker_tasks_;
313 : };
314 :
315 793546 : bool DBTable::WalkWorker::Run() {
316 793546 : int count = 0;
317 793546 : DBRequestKey *key_resume = walk_ctx_.get();
318 793547 : DBTable *table = walker_->table();
319 793534 : int max_walk_entry_count = table->GetWalkIterationToYield();
320 : DBEntry *entry;
321 :
322 793532 : if (key_resume != NULL) {
323 663195 : std::unique_ptr<const DBEntryBase> start;
324 663195 : start = table->AllocEntry(key_resume);
325 : // Find matching or next in sort order
326 663008 : entry = tbl_partition_->lower_bound(start.get());
327 663242 : } else {
328 130337 : entry = tbl_partition_->GetFirst();
329 : }
330 793475 : if (entry == NULL) {
331 664 : goto walk_done;
332 : }
333 :
334 1585677 : for (DBEntry *next = NULL; entry; entry = next) {
335 1455930 : next = tbl_partition_->GetNext(entry);
336 1455930 : if (count == max_walk_entry_count) {
337 : // store the context
338 663223 : walk_ctx_ = entry->GetDBRequestKey();
339 662645 : return false;
340 : }
341 :
342 : // Invoke walker function
343 792707 : bool more = table->InvokeWalkCb(tbl_partition_, entry);
344 792987 : if (!more) {
345 135 : break;
346 : }
347 :
348 792852 : db_walker_wait();
349 792866 : count++;
350 : }
351 :
352 129747 : walk_done:
353 : // Check whether all other walks on the table is completed
354 130546 : long num_walkers_on_tpart = walker_->pending_workers_.fetch_sub(1);
355 130546 : if (num_walkers_on_tpart == 1) {
356 106636 : table->WalkDone();
357 : }
358 130448 : return true;
359 : }
360 :
361 130449 : DBTable::WalkWorker::WalkWorker(TableWalker *walker, int db_partition_id)
362 130449 : : Task(walker->table()->GetWalkerTaskId(), db_partition_id), walker_(walker) {
363 130449 : tbl_partition_ = static_cast<DBTablePartition *>
364 130449 : (walker_->table()->GetTablePartition(db_partition_id));
365 130449 : }
366 :
367 226155 : void DBTable::TableWalker::StartWalk() {
368 226155 : CHECK_CONCURRENCY("db::Walker");
369 226155 : assert(pending_workers_ == 0);
370 738021 : for (int i = 0; i < table_->PartitionCount(); i++) {
371 : DBTablePartition *partition = static_cast<DBTablePartition *>(
372 511866 : table_->GetTablePartition(i));
373 511866 : if (!partition->size()) continue;
374 130449 : worker_tasks_.push_back(new WalkWorker(this, i));
375 130449 : pending_workers_++;
376 : }
377 226155 : if (pending_workers_ == 0) {
378 119519 : table_->WalkDone();
379 : } else {
380 106636 : TaskScheduler *scheduler = TaskScheduler::GetInstance();
381 237085 : for (auto *task : worker_tasks_) scheduler->Enqueue(task);
382 : }
383 226155 : }
384 :
385 : ///////////////////////////////////////////////////////////
386 : // Implementation of DBTable methods
387 : ///////////////////////////////////////////////////////////
388 1529616 : DBTable::DBTable(DB *db, const string &name)
389 : : DBTableBase(db, name),
390 1529650 : walker_(new TableWalker(this)),
391 3059213 : walker_task_id_(db->task_id()) {
392 :
393 : static bool init_ = false;
394 : static int iter_to_yield_env_ = 0;
395 :
396 1529588 : if (!init_) {
397 : // XXX To be used for testing purposes only.
398 157 : char *count_str = getenv("DB_ITERATION_TO_YIELD");
399 157 : if (count_str) {
400 157 : iter_to_yield_env_ = strtol(count_str, NULL, 0);
401 : } else {
402 0 : iter_to_yield_env_ = kIterationToYield;
403 : }
404 157 : init_ = true;
405 : }
406 1529588 : max_walk_iteration_to_yield_ = iter_to_yield_env_;
407 1529588 : }
408 :
409 1529660 : DBTable::~DBTable() {
410 1529660 : STLDeleteValues(&partitions_);
411 1529660 : }
412 :
413 1521517 : void DBTable::Init() {
414 3596601 : for (int i = 0; i < PartitionCount(); i++) {
415 2075244 : partitions_.push_back(AllocPartition(i));
416 : }
417 1521307 : }
418 :
419 2075243 : DBTablePartition *DBTable::AllocPartition(int index) {
420 2075243 : return new DBTablePartition(this, index);
421 : }
422 :
423 226155 : void DBTable::StartWalk() {
424 226155 : CHECK_CONCURRENCY("db::Walker");
425 226155 : incr_walk_count();
426 226155 : walker_->StartWalk();
427 226155 : }
428 :
429 212727 : DBEntry *DBTable::Add(const DBRequest *req) {
430 212727 : return AllocEntry(req->key.get()).release();
431 : }
432 :
433 1468 : void DBTable::Change(DBEntryBase *entry) {
434 1468 : DBTablePartBase *tpart = GetTablePartition(entry);
435 1468 : tpart->Notify(entry);
436 1468 : }
437 :
438 0 : bool DBTable::OnChange(DBEntry *entry, const DBRequest *req) {
439 0 : return true;
440 : }
441 :
442 0 : bool DBTable::Delete(DBEntry *entry, const DBRequest *req) {
443 0 : return true;
444 : }
445 :
446 1413598 : int DBTable::PartitionCount() const {
447 1413598 : return DB::PartitionCount();
448 : }
449 :
450 16158690 : static size_t HashToPartition(size_t hash) {
451 16158690 : return hash % DB::PartitionCount();
452 : }
453 :
454 19614955 : DBTablePartBase *DBTable::GetTablePartition(const int index) {
455 19614955 : return partitions_[index];
456 : }
457 :
458 24087 : const DBTablePartBase *DBTable::GetTablePartition(const int index) const {
459 24087 : return partitions_[index];
460 : }
461 :
462 704482 : DBTablePartBase *DBTable::GetTablePartition(const DBRequestKey *key) {
463 704482 : int id = HashToPartition(Hash(key));
464 704473 : return GetTablePartition(id);
465 : }
466 :
467 0 : const DBTablePartBase *DBTable::GetTablePartition(
468 : const DBRequestKey *key) const {
469 0 : int id = HashToPartition(Hash(key));
470 0 : return GetTablePartition(id);
471 : }
472 :
473 14974719 : DBTablePartBase *DBTable::GetTablePartition(const DBEntryBase *entry) {
474 14974719 : const DBEntry *gentry = static_cast<const DBEntry *>(entry);
475 14974719 : size_t id = HashToPartition(Hash(gentry));
476 14966616 : return GetTablePartition(id);
477 : }
478 :
479 24087 : const DBTablePartBase *DBTable::GetTablePartition(
480 : const DBEntryBase *entry) const {
481 24087 : const DBEntry *gentry = static_cast<const DBEntry *>(entry);
482 24087 : size_t id = HashToPartition(Hash(gentry));
483 24087 : return GetTablePartition(id);
484 : }
485 :
486 : // Find DB Entry without taking lock. Calling routine must ensure its
487 : // running in exclusion with DB task
488 0 : DBEntry *DBTable::FindNoLock(const DBEntry *entry) {
489 0 : size_t id = HashToPartition(Hash(entry));
490 : DBTablePartition *tbl_partition =
491 0 : static_cast<DBTablePartition *>(GetTablePartition(id));
492 0 : return tbl_partition->FindNoLock(entry);
493 : }
494 :
495 382029 : DBEntry *DBTable::Find(const DBEntry *entry) {
496 382029 : size_t id = HashToPartition(Hash(entry));
497 : DBTablePartition *tbl_partition =
498 382029 : static_cast<DBTablePartition *>(GetTablePartition(id));
499 382029 : return tbl_partition->Find(entry);
500 : }
501 :
502 0 : const DBEntry *DBTable::Find(const DBEntry *entry) const {
503 0 : return const_cast<DBTable *>(this)->Find(entry);
504 : }
505 :
506 : // Find DB Entry without taking lock. Calling routine must ensure its
507 : // running in exclusion with DB task
508 0 : DBEntry *DBTable::FindNoLock(const DBRequestKey *key) {
509 0 : int id = HashToPartition(Hash(key));
510 : DBTablePartition *tbl_partition =
511 0 : static_cast<DBTablePartition *>(GetTablePartition(id));
512 0 : return tbl_partition->FindNoLock(key);
513 : }
514 :
515 81937 : DBEntry *DBTable::Find(const DBRequestKey *key, int id) {
516 81937 : if (id == -1)
517 81909 : id = HashToPartition(Hash(key));
518 : DBTablePartition *tbl_partition =
519 81937 : static_cast<DBTablePartition *>(GetTablePartition(id));
520 81937 : return tbl_partition->Find(key);
521 : }
522 :
523 260 : const DBEntry *DBTable::Find(const DBRequestKey *key, int id) const {
524 260 : return const_cast<DBTable *>(this)->Find(key, id);
525 : }
526 :
527 : //
528 : // Concurrency: called from task that's mutually exclusive with db::DBTable
529 : // or db::IFMapTable as applicable.
530 : //
531 : // Calculate the size across all partitions.
532 : //
533 1884035 : size_t DBTable::Size() const {
534 1884035 : size_t total = 0;
535 1884035 : for (vector<DBTablePartition *>::const_iterator iter = partitions_.begin();
536 6978094 : iter != partitions_.end(); iter++) {
537 5094089 : total += (*iter)->size();
538 : }
539 1884009 : return total;
540 : }
541 :
542 48 : void DBTable::Input(DBTablePartition *tbl_partition, DBClient *client,
543 : DBRequest *req) {
544 : DBRequestKey *key =
545 48 : static_cast<DBRequestKey *>(req->key.get());
546 48 : DBEntry *entry = NULL;
547 :
548 48 : entry = tbl_partition->Find(key);
549 48 : if (req->oper == DBRequest::DB_ENTRY_ADD_CHANGE) {
550 26 : if (entry) {
551 5 : if (OnChange(entry, req) || entry->IsDeleted()) {
552 : // The entry may currently be marked as deleted.
553 5 : entry->ClearDelete();
554 5 : tbl_partition->Change(entry);
555 : }
556 : } else {
557 21 : if ((entry = Add(req)) != NULL) {
558 21 : tbl_partition->Add(entry);
559 : }
560 : }
561 22 : } else if (req->oper == DBRequest::DB_ENTRY_DELETE) {
562 22 : if (entry) {
563 22 : if (Delete(entry, req)) {
564 22 : tbl_partition->Delete(entry);
565 : }
566 : }
567 0 : } else if (req->oper == DBRequest::DB_ENTRY_NOTIFY) {
568 0 : if (entry) {
569 0 : tbl_partition->Notify(entry);
570 : }
571 : } else {
572 0 : assert(0);
573 : }
574 48 : }
575 :
576 0 : void DBTable::DBStateClear(DBTable *table, ListenerId id) {
577 0 : DBEntryBase *next = NULL;
578 :
579 0 : for (int i = 0; i < table->PartitionCount(); ++i) {
580 : DBTablePartition *partition = static_cast<DBTablePartition *>(
581 0 : table->GetTablePartition(i));
582 :
583 0 : for (DBEntryBase *entry = partition->GetFirst(); entry; entry = next) {
584 0 : next = partition->GetNext(entry);
585 0 : DBState *state = entry->GetState(table, id);
586 0 : if (state) {
587 0 : entry->ClearState(table, id);
588 0 : delete state;
589 : }
590 : }
591 : }
592 0 : }
593 :
594 : //
595 : // Callback for table walk triggered by NotifyAllEntries.
596 : //
597 129502 : bool DBTable::WalkCallback(DBTablePartBase *tpart, DBEntryBase *entry) {
598 129502 : tpart->Notify(entry);
599 129507 : return true;
600 : }
601 :
602 : //
603 : // Callback for completion of table walk triggered by NotifyAllEntries.
604 : //
605 61319 : void DBTable::WalkCompleteCallback(DBTableBase *tbl_base) {
606 61319 : walk_ref_.reset();
607 61319 : }
608 :
609 : //
610 : // Concurrency: called from task that's mutually exclusive with db::DBTable
611 : // or db::IFMapTable as applicable.
612 : //
613 : // Trigger notification of all entries to all listeners.
614 : // Should be used sparingly e.g. to handle significant configuration change.
615 : //
616 : // The walk callback just turns around and puts the DBentryBase on the change
617 : // list.
618 : //
619 : // If the walk is already running, it is allowed to complete and WalkAgain API
620 : // is invoked to trigger walk on current walk completion.
621 : //
622 64214 : void DBTable::NotifyAllEntries() {
623 64214 : CHECK_CONCURRENCY("bgp::Config", "bgp::ConfigHelper", "bgp::RTFilter",
624 : "db::DBTable");
625 :
626 64213 : if (empty())
627 2881 : return;
628 :
629 61332 : if (walk_ref_ == NULL) {
630 : walk_ref_ =
631 122638 : AllocWalker(boost::bind(&DBTable::WalkCallback, this, _1, _2),
632 61319 : boost::bind(&DBTable::WalkCompleteCallback, this, _2));
633 61319 : WalkTable(walk_ref_);
634 : } else {
635 13 : WalkAgain(walk_ref_);
636 : }
637 : }
638 :
639 218999 : DBTable::DBTableWalkRef DBTable::AllocWalker(WalkFn walk_fn,
640 : WalkCompleteFn walk_complete) {
641 218999 : DBTableWalkMgr *walk_mgr = database()->GetWalkMgr();
642 218999 : return walk_mgr->AllocWalker(this, walk_fn, walk_complete);
643 : }
644 :
645 157601 : void DBTable::ReleaseWalker(DBTable::DBTableWalkRef &walk) {
646 157601 : DBTableWalkMgr *walk_mgr = database()->GetWalkMgr();
647 157601 : walk_mgr->ReleaseWalker(walk);
648 157601 : return;
649 : }
650 :
651 226924 : void DBTable::WalkTable(DBTable::DBTableWalkRef walk) {
652 226924 : DBTableWalkMgr *walk_mgr = database()->GetWalkMgr();
653 226924 : walk_mgr->WalkTable(walk);
654 226924 : return;
655 : }
656 :
657 28746 : void DBTable::WalkAgain(DBTable::DBTableWalkRef walk) {
658 28746 : DBTableWalkMgr *walk_mgr = database()->GetWalkMgr();
659 28746 : walk_mgr->WalkAgain(walk);
660 28746 : return;
661 : }
662 :
663 792852 : bool DBTable::InvokeWalkCb(DBTablePartBase *part, DBEntryBase *entry) {
664 792852 : DBTableWalkMgr *walk_mgr = database()->GetWalkMgr();
665 792785 : return walk_mgr->InvokeWalkCb(part, entry);
666 : }
667 :
668 226155 : void DBTable::WalkDone() {
669 226155 : incr_walk_complete_count();
670 226155 : walker_->ClearWalkWorks();
671 226155 : DBTableWalkMgr *walk_mgr = database()->GetWalkMgr();
672 226155 : return walk_mgr->WalkDone();
673 : }
|