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 3901362 : void DBTablePartBase::Notify(DBEntryBase *entry) { 18 3901362 : if (entry->is_onlist()) { 19 1399637 : return; 20 : } 21 2501775 : entry->set_onlist(); 22 2505291 : bool was_empty = change_list_.empty(); 23 2505036 : change_list_.push_back(*entry); 24 2506229 : if (was_empty) { 25 1329880 : DB *db = parent()->database(); 26 1329897 : DBPartition *partition = db->GetPartition(index_); 27 1329884 : 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 1329990 : bool DBTablePartBase::RunNotify() { 41 3837147 : for (int i = 0; ((i < kMaxIterations) && !change_list_.empty()); ++i) { 42 2507001 : DBEntryBase *entry = &change_list_.front(); 43 2506924 : change_list_.pop_front(); 44 : 45 2506863 : parent()->RunNotify(this, entry); 46 2507049 : 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 2971170 : if (entry->IsDeleted() && entry->is_state_empty(this) && 57 464009 : !entry->IsOnRemoveQ()) { 58 463946 : Remove(entry); 59 : } 60 : } 61 : 62 1329966 : if (!change_list_.empty()) { 63 40 : DB *db = parent()->database(); 64 41 : DBPartition *partition = db->GetPartition(index_); 65 41 : partition->OnTableChange(this); 66 41 : return false; 67 : } 68 1329969 : return true; 69 : } 70 : 71 1148737 : void DBTablePartBase::Delete(DBEntryBase *entry) { 72 1148737 : if (parent_->HasListeners()) { 73 850596 : entry->MarkDelete(); 74 850567 : Notify(entry); 75 : } else { 76 : // Remove from change_list 77 298293 : if (entry->is_onlist()) { 78 1464 : change_list_.erase(change_list_.iterator_to(*entry)); 79 : } 80 298293 : Remove(entry); 81 : } 82 1148769 : } 83 : 84 2075532 : DBTablePartition::DBTablePartition(DBTable *table, int index) 85 2075532 : : DBTablePartBase(table, index) { 86 2075430 : } 87 : 88 644479 : void DBTablePartition::Process(DBClient *client, DBRequest *req) { 89 644479 : DBTable *table = static_cast<DBTable *>(parent()); 90 644471 : table->incr_input_count(); 91 644469 : table->Input(this, client, req); 92 644450 : } 93 : 94 1128145 : void DBTablePartition::Add(DBEntry *entry) { 95 1128145 : std::scoped_lock lock(mutex_); 96 1128245 : std::pair<Tree::iterator, bool> ret = tree_.insert(*entry); 97 1128125 : assert(ret.second); 98 1128125 : entry->set_table_partition(static_cast<DBTablePartBase *>(this)); 99 1128121 : Notify(entry); 100 1128178 : parent()->AddRemoveCallback(entry, true); 101 1128180 : } 102 : 103 224035 : void DBTablePartition::Change(DBEntry *entry) { 104 224035 : std::scoped_lock lock(mutex_); 105 224035 : Notify(entry); 106 224035 : } 107 : 108 1128138 : void DBTablePartition::Remove(DBEntryBase *db_entry) { 109 1128138 : std::scoped_lock lock(mutex_); 110 1128173 : DBEntry *entry = static_cast<DBEntry *>(db_entry); 111 1128173 : parent()->AddRemoveCallback(entry, false); 112 : 113 1128144 : bool success = tree_.erase(*entry); 114 1127902 : 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 1127902 : 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 1128562 : if (tree_.empty()) 124 134466 : table()->RetryDelete(); 125 1128158 : } 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 2918932 : DBEntry *DBTablePartition::FindInternal(const DBEntry *entry) { 145 2918932 : Tree::iterator loc = tree_.find(*entry); 146 5837423 : if (loc != tree_.end()) { 147 1363880 : return loc.operator->(); 148 : } 149 1554818 : return NULL; 150 : } 151 : 152 24085 : const DBEntry *DBTablePartition::FindInternal(const DBEntry *entry) const { 153 24085 : Tree::const_iterator loc = tree_.find(*entry); 154 48170 : if (loc != tree_.end()) { 155 16723 : return loc.operator->(); 156 : } 157 7362 : 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 2200811 : DBEntry *DBTablePartition::Find(const DBEntry *entry) { 167 2200811 : std::scoped_lock lock(mutex_); 168 4402079 : return FindInternal(entry); 169 2200803 : } 170 : 171 24085 : const DBEntry *DBTablePartition::Find(const DBEntry *entry) const { 172 24085 : std::scoped_lock lock(mutex_); 173 48170 : return FindInternal(entry); 174 24085 : } 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 717900 : DBEntry *DBTablePartition::Find(const DBRequestKey *key) { 185 717900 : DBTable *table = static_cast<DBTable *>(parent()); 186 717900 : std::unique_ptr<DBEntry> entry_ptr = table->AllocEntry(key); 187 717900 : std::scoped_lock lock(mutex_); 188 1435800 : return FindInternal(entry_ptr.get()); 189 717900 : } 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 665917 : DBEntry *DBTablePartition::lower_bound(const DBEntryBase *key) { 205 665917 : const DBEntry *entry = static_cast<const DBEntry *>(key); 206 665917 : std::scoped_lock lock(mutex_); 207 : 208 666164 : Tree::iterator it = tree_.lower_bound(*entry); 209 1332088 : if (it != tree_.end()) { 210 665958 : return (it.operator->()); 211 : } 212 82 : return NULL; 213 666040 : } 214 : 215 1375546 : DBEntry *DBTablePartition::GetFirst() { 216 1375546 : std::scoped_lock lock(mutex_); 217 1375655 : Tree::iterator it = tree_.begin(); 218 2751260 : if (it == tree_.end()) { 219 1206668 : return NULL; 220 : } 221 168953 : return it.operator->(); 222 1375621 : } 223 : 224 : // Returns the next entry (Doesn't search). Threaded walk 225 1747512 : DBEntry *DBTablePartition::GetNext(const DBEntryBase *key) { 226 1747512 : const DBEntry *entry = static_cast<const DBEntry *>(key); 227 1747512 : std::scoped_lock lock(mutex_); 228 : 229 1747782 : Tree::const_iterator it = tree_.iterator_to(*entry); 230 1747590 : it++; 231 3494151 : if (it != tree_.end()) { 232 1520272 : return const_cast<DBEntry *>(it.operator->()); 233 : } 234 226694 : return NULL; 235 1746966 : } 236 : 237 142004 : DBTable *DBTablePartition::table() { 238 142004 : return static_cast<DBTable *>(parent()); 239 : }