Line data Source code
1 : /* 2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved. 3 : */ 4 : 5 : #include "base/logging.h" 6 : #include "base/task_annotations.h" 7 : #include "base/time_util.h" 8 : #include "db/db.h" 9 : #include "db/db_entry.h" 10 : #include "db/db_partition.h" 11 : #include "db/db_table.h" 12 : #include "db/db_table_partition.h" 13 : 14 : using namespace std; 15 : 16 : // concurrency: called from DBPartition task. 17 3908175 : void DBTablePartBase::Notify(DBEntryBase *entry) { 18 3908175 : if (entry->is_onlist()) { 19 1405614 : return; 20 : } 21 2502326 : entry->set_onlist(); 22 2507170 : bool was_empty = change_list_.empty(); 23 2506963 : change_list_.push_back(*entry); 24 2508240 : if (was_empty) { 25 1325349 : DB *db = parent()->database(); 26 1325330 : DBPartition *partition = db->GetPartition(index_); 27 1325319 : partition->OnTableChange(this); 28 : } 29 : } 30 : 31 : // 32 : // Concurrency: called from db::DBTable/db::IFMapTable task. 33 : // 34 : // Evaluate concurrency issues with DBEntryBase::ClearState when making 35 : // changes to this method. We expect that either this method or ClearState 36 : // is responsible for removing the DBEntryBase when they run concurrently, 37 : // assuming the DBEntryBase is eligible for removal. The dbstate_mutex is 38 : // used for synchronization. 39 : // 40 1325401 : bool DBTablePartBase::RunNotify() { 41 3834607 : for (int i = 0; ((i < kMaxIterations) && !change_list_.empty()); ++i) { 42 2509074 : DBEntryBase *entry = &change_list_.front(); 43 2508945 : change_list_.pop_front(); 44 : 45 2508735 : parent()->RunNotify(this, entry); 46 2509177 : entry->clear_onlist(); 47 : 48 : // If the entry is marked deleted and all DBStates are removed 49 : // and it's not already on the remove queue, it can be removed 50 : // from the tree right away. 51 : // 52 : // Note that IsOnRemoveQ must be called after is_state_empty as 53 : // synchronization with DBEntryBase::ClearState happens via the 54 : // call to is_state_empty, and ClearState can set the OnRemoveQ 55 : // bit in the entry. 56 2970423 : if (entry->IsDeleted() && entry->is_state_empty(this) && 57 461183 : !entry->IsOnRemoveQ()) { 58 461135 : Remove(entry); 59 : } 60 : } 61 : 62 1325396 : if (!change_list_.empty()) { 63 39 : DB *db = parent()->database(); 64 40 : DBPartition *partition = db->GetPartition(index_); 65 40 : partition->OnTableChange(this); 66 40 : return false; 67 : } 68 1325406 : return true; 69 : } 70 : 71 1149142 : void DBTablePartBase::Delete(DBEntryBase *entry) { 72 1149142 : if (parent_->HasListeners()) { 73 850988 : entry->MarkDelete(); 74 850968 : Notify(entry); 75 : } else { 76 : // Remove from change_list 77 298264 : if (entry->is_onlist()) { 78 1464 : change_list_.erase(change_list_.iterator_to(*entry)); 79 : } 80 298264 : Remove(entry); 81 : } 82 1149179 : } 83 : 84 2075344 : DBTablePartition::DBTablePartition(DBTable *table, int index) 85 2075344 : : DBTablePartBase(table, index) { 86 2075234 : } 87 : 88 644811 : void DBTablePartition::Process(DBClient *client, DBRequest *req) { 89 644811 : DBTable *table = static_cast<DBTable *>(parent()); 90 644805 : table->incr_input_count(); 91 644799 : table->Input(this, client, req); 92 644775 : } 93 : 94 1128333 : void DBTablePartition::Add(DBEntry *entry) { 95 1128333 : std::scoped_lock lock(mutex_); 96 1128418 : std::pair<Tree::iterator, bool> ret = tree_.insert(*entry); 97 1128293 : assert(ret.second); 98 1128293 : entry->set_table_partition(static_cast<DBTablePartBase *>(this)); 99 1128288 : Notify(entry); 100 1128357 : parent()->AddRemoveCallback(entry, true); 101 1128353 : } 102 : 103 223932 : void DBTablePartition::Change(DBEntry *entry) { 104 223932 : std::scoped_lock lock(mutex_); 105 223932 : Notify(entry); 106 223932 : } 107 : 108 1128308 : void DBTablePartition::Remove(DBEntryBase *db_entry) { 109 1128308 : std::scoped_lock lock(mutex_); 110 1128364 : DBEntry *entry = static_cast<DBEntry *>(db_entry); 111 1128364 : parent()->AddRemoveCallback(entry, false); 112 : 113 1128336 : bool success = tree_.erase(*entry); 114 1128052 : if (!success) { 115 0 : LOG(FATAL, "ABORT: DB node erase failed for table " + parent()->name()); 116 0 : LOG(FATAL, "Invalid node " + db_entry->ToString()); 117 0 : abort(); 118 : } 119 1128052 : delete entry; 120 : 121 : // If a table is marked for deletion, then we may trigger the deletion 122 : // process when the last prefix is deleted 123 1128488 : if (tree_.empty()) 124 134370 : table()->RetryDelete(); 125 1128311 : } 126 : 127 0 : void DBTablePartition::AddWithoutAlloc(DBEntry *entry) { 128 0 : std::scoped_lock lock(mutex_); 129 0 : tree_.insert(*entry); 130 0 : entry->set_table_partition(static_cast<DBTablePartBase *>(this)); 131 0 : Notify(entry); 132 0 : parent()->AddRemoveCallback(entry, true); 133 0 : } 134 : 135 0 : void DBTablePartition::RemoveWithoutDelete(DBEntry *entry) { 136 0 : std::scoped_lock lock(mutex_); 137 0 : bool success = tree_.erase(*entry); 138 0 : if (!success) { 139 0 : LOG(FATAL, "ABORT: DB node erase failed for table " + parent()->name()); 140 0 : abort(); 141 : } 142 0 : } 143 : 144 2908883 : DBEntry *DBTablePartition::FindInternal(const DBEntry *entry) { 145 2908883 : Tree::iterator loc = tree_.find(*entry); 146 5817280 : if (loc != tree_.end()) { 147 1354859 : return loc.operator->(); 148 : } 149 1553774 : return NULL; 150 : } 151 : 152 24087 : const DBEntry *DBTablePartition::FindInternal(const DBEntry *entry) const { 153 24087 : Tree::const_iterator loc = tree_.find(*entry); 154 48174 : if (loc != tree_.end()) { 155 16723 : return loc.operator->(); 156 : } 157 7364 : return NULL; 158 : } 159 : 160 0 : DBEntry *DBTablePartition::FindNoLock(const DBEntry *entry) { 161 0 : CHECK_CONCURRENCY("db::DBTable", "db::IFMapTable", 162 : "Agent::FlowEvent", "Agent::FlowUpdate"); 163 0 : return FindInternal(entry); 164 : } 165 : 166 2191195 : DBEntry *DBTablePartition::Find(const DBEntry *entry) { 167 2191195 : std::scoped_lock lock(mutex_); 168 4383025 : return FindInternal(entry); 169 2191269 : } 170 : 171 24087 : const DBEntry *DBTablePartition::Find(const DBEntry *entry) const { 172 24087 : std::scoped_lock lock(mutex_); 173 48174 : return FindInternal(entry); 174 24087 : } 175 : 176 0 : DBEntry *DBTablePartition::FindNoLock(const DBRequestKey *key) { 177 0 : CHECK_CONCURRENCY("db::DBTable", "db::IFMapTable", 178 : "Agent::FlowEvent", "Agent::FlowUpdate"); 179 0 : DBTable *table = static_cast<DBTable *>(parent()); 180 0 : std::unique_ptr<DBEntry> entry_ptr = table->AllocEntry(key); 181 0 : return FindInternal(entry_ptr.get()); 182 0 : } 183 : 184 717373 : DBEntry *DBTablePartition::Find(const DBRequestKey *key) { 185 717373 : DBTable *table = static_cast<DBTable *>(parent()); 186 717373 : std::unique_ptr<DBEntry> entry_ptr = table->AllocEntry(key); 187 717373 : std::scoped_lock lock(mutex_); 188 1434746 : return FindInternal(entry_ptr.get()); 189 717373 : } 190 : 191 0 : DBEntry *DBTablePartition::FindNext(const DBRequestKey *key) { 192 0 : std::scoped_lock lock(mutex_); 193 0 : DBTable *table = static_cast<DBTable *>(parent()); 194 0 : std::unique_ptr<DBEntry> entry_ptr = table->AllocEntry(key); 195 : 196 0 : Tree::iterator loc = tree_.upper_bound(*(entry_ptr.get())); 197 0 : if (loc != tree_.end()) { 198 0 : return loc.operator->(); 199 : } 200 0 : return NULL; 201 0 : } 202 : 203 : // Returns the matching entry or next in lex order 204 663128 : DBEntry *DBTablePartition::lower_bound(const DBEntryBase *key) { 205 663128 : const DBEntry *entry = static_cast<const DBEntry *>(key); 206 663128 : std::scoped_lock lock(mutex_); 207 : 208 663415 : Tree::iterator it = tree_.lower_bound(*entry); 209 1326531 : if (it != tree_.end()) { 210 663186 : return (it.operator->()); 211 : } 212 77 : return NULL; 213 663263 : } 214 : 215 1375206 : DBEntry *DBTablePartition::GetFirst() { 216 1375206 : std::scoped_lock lock(mutex_); 217 1375332 : Tree::iterator it = tree_.begin(); 218 2750592 : if (it == tree_.end()) { 219 1206477 : return NULL; 220 : } 221 168808 : return it.operator->(); 222 1375285 : } 223 : 224 : // Returns the next entry (Doesn't search). Threaded walk 225 1741663 : DBEntry *DBTablePartition::GetNext(const DBEntryBase *key) { 226 1741663 : const DBEntry *entry = static_cast<const DBEntry *>(key); 227 1741663 : std::scoped_lock lock(mutex_); 228 : 229 1742141 : Tree::const_iterator it = tree_.iterator_to(*entry); 230 1741978 : it++; 231 3482917 : if (it != tree_.end()) { 232 1514987 : return const_cast<DBEntry *>(it.operator->()); 233 : } 234 226376 : return NULL; 235 1741363 : } 236 : 237 141801 : DBTable *DBTablePartition::table() { 238 141801 : return static_cast<DBTable *>(parent()); 239 : }