From 5f9ee9909cc743ce6201fb21480051d3f5ad2879 Mon Sep 17 00:00:00 2001
From: Jens Georg <jens.georg@desy.de>
Date: Tue, 11 Jun 2019 15:48:38 +0200
Subject: [PATCH] Improve location locking in updater

 - Preallocate the unordered set with worst-case size before the loop
 - Perform insert and lock in one step using the return value of insert
---
 src/DoocsUpdater.cc | 12 ++++++++----
 1 file changed, 8 insertions(+), 4 deletions(-)

diff --git a/src/DoocsUpdater.cc b/src/DoocsUpdater.cc
index ddcd99c..47242dc 100644
--- a/src/DoocsUpdater.cc
+++ b/src/DoocsUpdater.cc
@@ -36,22 +36,26 @@ namespace ChimeraTK {
       return;
     }
 
+    // Worst-case: We need to lock all locations, so pre-allocate this here
+    std::unordered_set<EqFct*> locationsToLock;
+    locationsToLock.reserve(_toDoocsEqFctMap.size());
+
     ReadAnyGroup group(_elementsToRead.begin(), _elementsToRead.end());
     while(true) {
       // Wait until any variable got an update
       auto notification = group.waitAny();
       auto updatedElement = notification.getId();
       // Gather all involved locations in a unique set
-      std::unordered_set<EqFct*> locationsToLock;
-      for(auto& location : _toDoocsEqFctMap[updatedElement]) locationsToLock.insert(location);
-      // Lock all involved locations
-      for(auto& location : locationsToLock) location->lock();
+      for(auto& location : _toDoocsEqFctMap[updatedElement]) {
+        if(locationsToLock.insert(location).second) location->lock();
+      }
       // Complete the read transfer of the process variable
       notification.accept();
       // Call all updater functions
       for(auto& updaterFunction : _toDoocsUpdateMap[updatedElement]) updaterFunction();
       // Unlock all involved locations
       for(auto& location : locationsToLock) location->unlock();
+      locationsToLock.clear();
       // Allow shutting down this thread...
       boost::this_thread::interruption_point();
     }
-- 
GitLab