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 3894335 : void DBTablePartBase::Notify(DBEntryBase *entry) { 18 3894335 : if (entry->is_onlist()) { 19 1399177 : return; 20 : } 21 2495070 : entry->set_onlist(); 22 2499575 : bool was_empty = change_list_.empty(); 23 2499342 : change_list_.push_back(*entry); 24 2500452 : if (was_empty) { 25 1332251 : DB *db = parent()->database(); 26 1332223 : DBPartition *partition = db->GetPartition(index_); 27 1332205 : 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 1332314 : bool DBTablePartBase::RunNotify() { 41 3834207 : for (int i = 0; ((i < kMaxIterations) && !change_list_.empty()); ++i) { 42 2501691 : DBEntryBase *entry = &change_list_.front(); 43 2501594 : change_list_.pop_front(); 44 : 45 2501493 : parent()->RunNotify(this, entry); 46 2501832 : 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 2961766 : if (entry->IsDeleted() && entry->is_state_empty(this) && 57 459834 : !entry->IsOnRemoveQ()) { 58 459758 : Remove(entry); 59 : } 60 : } 61 : 62 1332306 : if (!change_list_.empty()) { 63 41 : DB *db = parent()->database(); 64 41 : DBPartition *partition = db->GetPartition(index_); 65 41 : partition->OnTableChange(this); 66 41 : return false; 67 : } 68 1332323 : return true; 69 : } 70 : 71 1148597 : void DBTablePartBase::Delete(DBEntryBase *entry) { 72 1148597 : if (parent_->HasListeners()) { 73 850546 : entry->MarkDelete(); 74 850492 : Notify(entry); 75 : } else { 76 : // Remove from change_list 77 298263 : if (entry->is_onlist()) { 78 1462 : change_list_.erase(change_list_.iterator_to(*entry)); 79 : } 80 298264 : Remove(entry); 81 : } 82 1148628 : } 83 : 84 2075899 : DBTablePartition::DBTablePartition(DBTable *table, int index) 85 2075899 : : DBTablePartBase(table, index) { 86 2075730 : } 87 : 88 647945 : void DBTablePartition::Process(DBClient *client, DBRequest *req) { 89 647945 : DBTable *table = static_cast<DBTable *>(parent()); 90 647936 : table->incr_input_count(); 91 647920 : table->Input(this, client, req); 92 647873 : } 93 : 94 1127887 : void DBTablePartition::Add(DBEntry *entry) { 95 1127887 : std::scoped_lock lock(mutex_); 96 1128016 : std::pair<Tree::iterator, bool> ret = tree_.insert(*entry); 97 1127837 : assert(ret.second); 98 1127837 : entry->set_table_partition(static_cast<DBTablePartBase *>(this)); 99 1127859 : Notify(entry); 100 1127938 : parent()->AddRemoveCallback(entry, true); 101 1127935 : } 102 : 103 224159 : void DBTablePartition::Change(DBEntry *entry) { 104 224159 : std::scoped_lock lock(mutex_); 105 224159 : Notify(entry); 106 224159 : } 107 : 108 1127868 : void DBTablePartition::Remove(DBEntryBase *db_entry) { 109 1127868 : std::scoped_lock lock(mutex_); 110 1127906 : DBEntry *entry = static_cast<DBEntry *>(db_entry); 111 1127906 : parent()->AddRemoveCallback(entry, false); 112 : 113 1127861 : bool success = tree_.erase(*entry); 114 1127571 : 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 1127571 : 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 1127948 : if (tree_.empty()) 124 134509 : table()->RetryDelete(); 125 1127916 : } 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 2914799 : DBEntry *DBTablePartition::FindInternal(const DBEntry *entry) { 145 2914799 : Tree::iterator loc = tree_.find(*entry); 146 5829167 : if (loc != tree_.end()) { 147 1359284 : return loc.operator->(); 148 : } 149 1555290 : return NULL; 150 : } 151 : 152 24100 : const DBEntry *DBTablePartition::FindInternal(const DBEntry *entry) const { 153 24100 : Tree::const_iterator loc = tree_.find(*entry); 154 48200 : if (loc != tree_.end()) { 155 16727 : return loc.operator->(); 156 : } 157 7373 : return NULL; 158 : } 159 : 160 98 : DBEntry *DBTablePartition::FindNoLock(const DBEntry *entry) { 161 98 : CHECK_CONCURRENCY("db::DBTable", "db::IFMapTable", 162 : "Agent::FlowEvent", "Agent::FlowUpdate"); 163 98 : return FindInternal(entry); 164 : } 165 : 166 2190069 : DBEntry *DBTablePartition::Find(const DBEntry *entry) { 167 2190069 : std::scoped_lock lock(mutex_); 168 4380797 : return FindInternal(entry); 169 2190108 : } 170 : 171 24100 : const DBEntry *DBTablePartition::Find(const DBEntry *entry) const { 172 24100 : std::scoped_lock lock(mutex_); 173 48200 : return FindInternal(entry); 174 24100 : } 175 : 176 17 : DBEntry *DBTablePartition::FindNoLock(const DBRequestKey *key) { 177 17 : CHECK_CONCURRENCY("db::DBTable", "db::IFMapTable", 178 : "Agent::FlowEvent", "Agent::FlowUpdate"); 179 17 : DBTable *table = static_cast<DBTable *>(parent()); 180 17 : std::unique_ptr<DBEntry> entry_ptr = table->AllocEntry(key); 181 34 : return FindInternal(entry_ptr.get()); 182 17 : } 183 : 184 724357 : DBEntry *DBTablePartition::Find(const DBRequestKey *key) { 185 724357 : DBTable *table = static_cast<DBTable *>(parent()); 186 724357 : std::unique_ptr<DBEntry> entry_ptr = table->AllocEntry(key); 187 724357 : std::scoped_lock lock(mutex_); 188 1448713 : return FindInternal(entry_ptr.get()); 189 724356 : } 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 659890 : DBEntry *DBTablePartition::lower_bound(const DBEntryBase *key) { 205 659890 : const DBEntry *entry = static_cast<const DBEntry *>(key); 206 659890 : std::scoped_lock lock(mutex_); 207 : 208 660104 : Tree::iterator it = tree_.lower_bound(*entry); 209 1320000 : if (it != tree_.end()) { 210 659919 : return (it.operator->()); 211 : } 212 74 : return NULL; 213 659993 : } 214 : 215 1375961 : DBEntry *DBTablePartition::GetFirst() { 216 1375961 : std::scoped_lock lock(mutex_); 217 1376039 : Tree::iterator it = tree_.begin(); 218 2752013 : if (it == tree_.end()) { 219 1207014 : return NULL; 220 : } 221 168982 : return it.operator->(); 222 1375996 : } 223 : 224 : // Returns the next entry (Doesn't search). Threaded walk 225 1735349 : DBEntry *DBTablePartition::GetNext(const DBEntryBase *key) { 226 1735349 : const DBEntry *entry = static_cast<const DBEntry *>(key); 227 1735349 : std::scoped_lock lock(mutex_); 228 : 229 1735746 : Tree::const_iterator it = tree_.iterator_to(*entry); 230 1735508 : it++; 231 3470162 : if (it != tree_.end()) { 232 1508369 : return const_cast<DBEntry *>(it.operator->()); 233 : } 234 226604 : return NULL; 235 1734973 : } 236 : 237 141928 : DBTable *DBTablePartition::table() { 238 141928 : return static_cast<DBTable *>(parent()); 239 : }