Line data Source code
1 : /* 2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved. 3 : */ 4 : 5 : #include "db/db_entry.h" 6 : 7 : #include <tbb/mutex.h> 8 : 9 : #include "base/time_util.h" 10 : #include "db/db_table_partition.h" 11 : 12 : using namespace std; 13 : 14 9490935 : DBEntryBase::DBEntryBase() 15 9490935 : : tpart_(NULL), flags(0), last_change_at_(UTCTimestampUsec()) { 16 9492122 : onremoveq_ = false; 17 9492116 : } 18 : 19 9478160 : DBEntryBase::~DBEntryBase() { 20 9478160 : } 21 : 22 3707421 : void DBEntryBase::SetState(DBTableBase *tbl_base, ListenerId listener, 23 : DBState *state) { 24 3707421 : DBTablePartBase *tpart = tbl_base->GetTablePartition(this); 25 3707233 : tbb::spin_rw_mutex::scoped_lock lock(tpart->dbstate_mutex(), true); 26 3709358 : pair<StateMap::iterator, bool> res = state_.insert( 27 3709227 : make_pair(listener, state)); 28 3709088 : if (!res.second) { 29 1227365 : res.first->second = state; 30 : } else { 31 2481723 : assert(!IsDeleted()); 32 : // Account for state addition for this listener. 33 2481704 : tbl_base->AddToDBStateCount(listener, 1); 34 : } 35 3708858 : } 36 : 37 12209892 : DBState *DBEntryBase::GetState(DBTableBase *tbl_base, ListenerId listener) const { 38 12209892 : DBTablePartBase *tpart = tbl_base->GetTablePartition(this); 39 12206320 : tbb::spin_rw_mutex::scoped_lock lock(tpart->dbstate_mutex(), false); 40 12221506 : StateMap::const_iterator loc = state_.find(listener); 41 12203449 : if (loc != state_.end()) { 42 7826697 : return loc->second; 43 : } 44 4379624 : return NULL; 45 12205955 : } 46 : 47 795 : const DBState *DBEntryBase::GetState(const DBTableBase *tbl_base, 48 : ListenerId listener) const { 49 795 : DBTableBase *table = const_cast<DBTableBase *>(tbl_base); 50 795 : DBTablePartBase *tpart = table->GetTablePartition(this); 51 795 : tbb::spin_rw_mutex::scoped_lock lock(tpart->dbstate_mutex(), false); 52 795 : StateMap::const_iterator loc = state_.find(listener); 53 795 : if (loc != state_.end()) { 54 768 : return loc->second; 55 : } 56 27 : return NULL; 57 795 : } 58 : 59 : // 60 : // Concurrency: called from arbitrary task. 61 : // 62 : // Evaluate concurrency issues with DBTablePartBase::RunNotify when making 63 : // changes to this method. We expect that either this method or RunNotify 64 : // is responsible for removing the DBEntryBase when they run concurrently, 65 : // assuming the DBEntryBase is eligible for removal. The dbstate_mutex in 66 : // in DBTablePartBase is used for synchronization. 67 : // 68 : // Remove DBState on this DBEntryBase for the given listener and enqueue 69 : // for removal if appropriate. 70 : // Note that the entry cannot be removed from DBTablePartBase here since 71 : // this method may be called from an arbitrary Task. 72 : // 73 2480534 : void DBEntryBase::ClearState(DBTableBase *tbl_base, ListenerId listener) { 74 2480534 : DBTablePartBase *tpart = tbl_base->GetTablePartition(this); 75 2480303 : tbb::spin_rw_mutex::scoped_lock lock(tpart->dbstate_mutex(), true); 76 : 77 2481712 : assert(state_.erase(listener) != 0); 78 : 79 : // Account for state removal for this listener. 80 2480566 : tbl_base->AddToDBStateCount(listener, -1); 81 : 82 2481542 : if (state_.empty() && IsDeleted() && !is_onlist() && !IsOnRemoveQ()) { 83 518735 : tbl_base->EnqueueRemove(this); 84 : } 85 2481347 : } 86 : 87 1058678 : bool DBEntryBase::is_state_empty(DBTablePartBase *tpart) { 88 1058678 : tbb::spin_rw_mutex::scoped_lock lock(tpart->dbstate_mutex(), false); 89 2117573 : return state_.empty(); 90 1058645 : } 91 : 92 518727 : bool DBEntryBase::is_state_empty_unlocked(DBTablePartBase *tpart) { 93 518727 : return state_.empty(); 94 : } 95 : 96 2187013 : void DBEntryBase::set_last_change_at_to_now() { 97 2187013 : last_change_at_ = UTCTimestampUsec(); 98 2187278 : } 99 : 100 0 : void DBEntryBase::set_last_change_at(uint64_t time) { 101 0 : last_change_at_ = time; 102 0 : } 103 : 104 2303010 : void DBEntryBase::set_last_update_at_to_now() { 105 2303010 : last_update_at_ = UTCTimestampUsec(); 106 2303202 : } 107 : 108 150 : void DBEntryBase::set_last_update_at(uint64_t time) { 109 150 : last_update_at_ = time; 110 150 : } 111 1432139 : void DBEntryBase::set_table_partition(DBTablePartBase *tpart) { 112 1432139 : tpart_ = tpart; 113 1432139 : } 114 : 115 3211549 : DBTablePartBase *DBEntryBase::get_table_partition() const { 116 3211549 : return tpart_; 117 : } 118 : 119 9331660 : DBTableBase *DBEntryBase::get_table() const { 120 9331660 : return (tpart_ ? tpart_->parent() : NULL); 121 : } 122 : 123 0 : const std::string DBEntryBase::last_change_at_str() const { 124 0 : return duration_usecs_to_string(UTCTimestampUsec() - last_change_at_); 125 : } 126 : 127 60140 : void DBEntryBase::Notify() { 128 60140 : tpart_->Notify(this); 129 60140 : } 130 : 131 48078 : void DBEntryBase::Delete() { 132 48078 : if (!IsDeleted()) 133 48078 : tpart_->Delete(this); 134 48078 : }