LCOV - code coverage report
Current view: top level - db - db_table_partition.cc (source / functions) Hit Total Coverage
Test: OpenSDN C/C++ coverage (all TARGET_SET jobs) Lines: 113 148 76.4 %
Date: 2026-06-18 01:51:13 Functions: 17 22 77.3 %
Legend: Lines: hit not hit

          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             : }

Generated by: LCOV version 1.14