Merge #18710: Add local thread pool to CCheckQueue

bb6fcc75d1 refactor: Drop boost::thread stuff in CCheckQueue (Hennadii Stepanov)
6784ac471b bench: Use CCheckQueue local thread pool (Hennadii Stepanov)
dba30695fc test: Use CCheckQueue local thread pool (Hennadii Stepanov)
01511776ac Add local thread pool to CCheckQueue (Hennadii Stepanov)
0ef938685b refactor: Use member initializers in CCheckQueue (Hennadii Stepanov)

Pull request description:

  This PR:
  - gets rid of `boost::thread_group` in the `CCheckQueue` class
  - allows thread safety annotation usage in the `CCheckQueue` class
  - is alternative to #14464 (https://github.com/bitcoin/bitcoin/pull/18710#issuecomment-616618525, https://github.com/bitcoin/bitcoin/pull/18710#issuecomment-617291612)

  Also, with this PR (I hope) it could be easier to resurrect a bunch of brilliant ideas from #9938.

  Related: #17307

ACKs for top commit:
  laanwj:
    Code review ACK bb6fcc75d1
  LarryRuane:
    ACK bb6fcc75d1
  jonatack:
    Code review ACK bb6fcc75d1 and verified rebase to master builds cleanly with unit/functional tests green

Tree-SHA512: fddeb720d5a391b48bb4c6fa58ed34ccc3f57862fdb8e641745c021841c8340e35c5126338271446cbd98f40bd5484f27926aa6c3e76fa478ba1efafe72e73c1
This commit is contained in:
Wladimir J. van der Laan 2021-01-25 13:35:12 +01:00
commit b386d37360
No known key found for this signature in database
GPG key ID: 1E4AED62986CD25D
9 changed files with 98 additions and 97 deletions

View file

@ -10,8 +10,6 @@
#include <random.h> #include <random.h>
#include <util/system.h> #include <util/system.h>
#include <boost/thread/thread.hpp>
#include <vector> #include <vector>
static const size_t BATCHES = 101; static const size_t BATCHES = 101;
@ -44,12 +42,9 @@ static void CCheckQueueSpeedPrevectorJob(benchmark::Bench& bench)
void swap(PrevectorJob& x){p.swap(x.p);}; void swap(PrevectorJob& x){p.swap(x.p);};
}; };
CCheckQueue<PrevectorJob> queue {QUEUE_BATCH_SIZE}; CCheckQueue<PrevectorJob> queue {QUEUE_BATCH_SIZE};
boost::thread_group tg;
// The main thread should be counted to prevent thread oversubscription, and // The main thread should be counted to prevent thread oversubscription, and
// to decrease the variance of benchmark results. // to decrease the variance of benchmark results.
for (auto x = 0; x < GetNumCores() - 1; ++x) { queue.StartWorkerThreads(GetNumCores() - 1);
tg.create_thread([&]{queue.Thread();});
}
// create all the data once, then submit copies in the benchmark. // create all the data once, then submit copies in the benchmark.
FastRandomContext insecure_rand(true); FastRandomContext insecure_rand(true);
@ -70,8 +65,7 @@ static void CCheckQueueSpeedPrevectorJob(benchmark::Bench& bench)
// it is done explicitly here for clarity // it is done explicitly here for clarity
control.Wait(); control.Wait();
}); });
tg.interrupt_all(); queue.StopWorkerThreads();
tg.join_all();
ECC_Stop(); ECC_Stop();
} }
BENCHMARK(CCheckQueueSpeedPrevectorJob); BENCHMARK(CCheckQueueSpeedPrevectorJob);

View file

@ -6,13 +6,12 @@
#define BITCOIN_CHECKQUEUE_H #define BITCOIN_CHECKQUEUE_H
#include <sync.h> #include <sync.h>
#include <tinyformat.h>
#include <util/threadnames.h>
#include <algorithm> #include <algorithm>
#include <vector> #include <vector>
#include <boost/thread/condition_variable.hpp>
#include <boost/thread/mutex.hpp>
template <typename T> template <typename T>
class CCheckQueueControl; class CCheckQueueControl;
@ -31,61 +30,64 @@ class CCheckQueue
{ {
private: private:
//! Mutex to protect the inner state //! Mutex to protect the inner state
boost::mutex mutex; Mutex m_mutex;
//! Worker threads block on this when out of work //! Worker threads block on this when out of work
boost::condition_variable condWorker; std::condition_variable m_worker_cv;
//! Master thread blocks on this when out of work //! Master thread blocks on this when out of work
boost::condition_variable condMaster; std::condition_variable m_master_cv;
//! The queue of elements to be processed. //! The queue of elements to be processed.
//! As the order of booleans doesn't matter, it is used as a LIFO (stack) //! As the order of booleans doesn't matter, it is used as a LIFO (stack)
std::vector<T> queue; std::vector<T> queue GUARDED_BY(m_mutex);
//! The number of workers (including the master) that are idle. //! The number of workers (including the master) that are idle.
int nIdle; int nIdle GUARDED_BY(m_mutex){0};
//! The total number of workers (including the master). //! The total number of workers (including the master).
int nTotal; int nTotal GUARDED_BY(m_mutex){0};
//! The temporary evaluation result. //! The temporary evaluation result.
bool fAllOk; bool fAllOk GUARDED_BY(m_mutex){true};
/** /**
* Number of verifications that haven't completed yet. * Number of verifications that haven't completed yet.
* This includes elements that are no longer queued, but still in the * This includes elements that are no longer queued, but still in the
* worker's own batches. * worker's own batches.
*/ */
unsigned int nTodo; unsigned int nTodo GUARDED_BY(m_mutex){0};
//! The maximum number of elements to be processed in one batch //! The maximum number of elements to be processed in one batch
unsigned int nBatchSize; const unsigned int nBatchSize;
std::vector<std::thread> m_worker_threads;
bool m_request_stop GUARDED_BY(m_mutex){false};
/** Internal function that does bulk of the verification work. */ /** Internal function that does bulk of the verification work. */
bool Loop(bool fMaster = false) bool Loop(bool fMaster)
{ {
boost::condition_variable& cond = fMaster ? condMaster : condWorker; std::condition_variable& cond = fMaster ? m_master_cv : m_worker_cv;
std::vector<T> vChecks; std::vector<T> vChecks;
vChecks.reserve(nBatchSize); vChecks.reserve(nBatchSize);
unsigned int nNow = 0; unsigned int nNow = 0;
bool fOk = true; bool fOk = true;
do { do {
{ {
boost::unique_lock<boost::mutex> lock(mutex); WAIT_LOCK(m_mutex, lock);
// first do the clean-up of the previous loop run (allowing us to do it in the same critsect) // first do the clean-up of the previous loop run (allowing us to do it in the same critsect)
if (nNow) { if (nNow) {
fAllOk &= fOk; fAllOk &= fOk;
nTodo -= nNow; nTodo -= nNow;
if (nTodo == 0 && !fMaster) if (nTodo == 0 && !fMaster)
// We processed the last element; inform the master it can exit and return the result // We processed the last element; inform the master it can exit and return the result
condMaster.notify_one(); m_master_cv.notify_one();
} else { } else {
// first iteration // first iteration
nTotal++; nTotal++;
} }
// logically, the do loop starts here // logically, the do loop starts here
while (queue.empty()) { while (queue.empty() && !m_request_stop) {
if (fMaster && nTodo == 0) { if (fMaster && nTodo == 0) {
nTotal--; nTotal--;
bool fRet = fAllOk; bool fRet = fAllOk;
@ -98,6 +100,10 @@ private:
cond.wait(lock); // wait cond.wait(lock); // wait
nIdle--; nIdle--;
} }
if (m_request_stop) {
return false;
}
// Decide how many work units to process now. // Decide how many work units to process now.
// * Do not try to do everything at once, but aim for increasingly smaller batches so // * Do not try to do everything at once, but aim for increasingly smaller batches so
// all workers finish approximately simultaneously. // all workers finish approximately simultaneously.
@ -106,7 +112,7 @@ private:
nNow = std::max(1U, std::min(nBatchSize, (unsigned int)queue.size() / (nTotal + nIdle + 1))); nNow = std::max(1U, std::min(nBatchSize, (unsigned int)queue.size() / (nTotal + nIdle + 1)));
vChecks.resize(nNow); vChecks.resize(nNow);
for (unsigned int i = 0; i < nNow; i++) { for (unsigned int i = 0; i < nNow; i++) {
// We want the lock on the mutex to be as short as possible, so swap jobs from the global // We want the lock on the m_mutex to be as short as possible, so swap jobs from the global
// queue to the local batch vector instead of copying. // queue to the local batch vector instead of copying.
vChecks[i].swap(queue.back()); vChecks[i].swap(queue.back());
queue.pop_back(); queue.pop_back();
@ -124,40 +130,68 @@ private:
public: public:
//! Mutex to ensure only one concurrent CCheckQueueControl //! Mutex to ensure only one concurrent CCheckQueueControl
boost::mutex ControlMutex; Mutex m_control_mutex;
//! Create a new check queue //! Create a new check queue
explicit CCheckQueue(unsigned int nBatchSizeIn) : nIdle(0), nTotal(0), fAllOk(true), nTodo(0), nBatchSize(nBatchSizeIn) {} explicit CCheckQueue(unsigned int nBatchSizeIn)
: nBatchSize(nBatchSizeIn)
//! Worker thread
void Thread()
{ {
Loop(); }
//! Create a pool of new worker threads.
void StartWorkerThreads(const int threads_num)
{
{
LOCK(m_mutex);
nIdle = 0;
nTotal = 0;
fAllOk = true;
}
assert(m_worker_threads.empty());
for (int n = 0; n < threads_num; ++n) {
m_worker_threads.emplace_back([this, n]() {
util::ThreadRename(strprintf("scriptch.%i", n));
Loop(false /* worker thread */);
});
}
} }
//! Wait until execution finishes, and return whether all evaluations were successful. //! Wait until execution finishes, and return whether all evaluations were successful.
bool Wait() bool Wait()
{ {
return Loop(true); return Loop(true /* master thread */);
} }
//! Add a batch of checks to the queue //! Add a batch of checks to the queue
void Add(std::vector<T>& vChecks) void Add(std::vector<T>& vChecks)
{ {
boost::unique_lock<boost::mutex> lock(mutex); LOCK(m_mutex);
for (T& check : vChecks) { for (T& check : vChecks) {
queue.push_back(T()); queue.push_back(T());
check.swap(queue.back()); check.swap(queue.back());
} }
nTodo += vChecks.size(); nTodo += vChecks.size();
if (vChecks.size() == 1) if (vChecks.size() == 1)
condWorker.notify_one(); m_worker_cv.notify_one();
else if (vChecks.size() > 1) else if (vChecks.size() > 1)
condWorker.notify_all(); m_worker_cv.notify_all();
}
//! Stop all of the worker threads.
void StopWorkerThreads()
{
WITH_LOCK(m_mutex, m_request_stop = true);
m_worker_cv.notify_all();
for (std::thread& t : m_worker_threads) {
t.join();
}
m_worker_threads.clear();
WITH_LOCK(m_mutex, m_request_stop = false);
} }
~CCheckQueue() ~CCheckQueue()
{ {
assert(m_worker_threads.empty());
} }
}; };
@ -181,7 +215,7 @@ public:
{ {
// passed queue is supposed to be unused, or nullptr // passed queue is supposed to be unused, or nullptr
if (pqueue != nullptr) { if (pqueue != nullptr) {
ENTER_CRITICAL_SECTION(pqueue->ControlMutex); ENTER_CRITICAL_SECTION(pqueue->m_control_mutex);
} }
} }
@ -205,7 +239,7 @@ public:
if (!fDone) if (!fDone)
Wait(); Wait();
if (pqueue != nullptr) { if (pqueue != nullptr) {
LEAVE_CRITICAL_SECTION(pqueue->ControlMutex); LEAVE_CRITICAL_SECTION(pqueue->m_control_mutex);
} }
} }
}; };

View file

@ -223,6 +223,7 @@ void Shutdown(NodeContext& node)
if (g_load_block.joinable()) g_load_block.join(); if (g_load_block.joinable()) g_load_block.join();
threadGroup.interrupt_all(); threadGroup.interrupt_all();
threadGroup.join_all(); threadGroup.join_all();
StopScriptCheckWorkerThreads();
// After the threads that potentially access these pointers have been stopped, // After the threads that potentially access these pointers have been stopped,
// destruct and reset all to nullptr. // destruct and reset all to nullptr.
@ -1334,9 +1335,7 @@ bool AppInitMain(const util::Ref& context, NodeContext& node, interfaces::BlockA
LogPrintf("Script verification uses %d additional threads\n", script_threads); LogPrintf("Script verification uses %d additional threads\n", script_threads);
if (script_threads >= 1) { if (script_threads >= 1) {
g_parallel_script_checks = true; g_parallel_script_checks = true;
for (int i = 0; i < script_threads; ++i) { StartScriptCheckWorkerThreads(script_threads);
threadGroup.create_thread([i]() { return ThreadScriptCheck(i); });
}
} }
assert(!node.scheduler); assert(!node.scheduler);

View file

@ -148,10 +148,7 @@ typedef CCheckQueue<FrozenCleanupCheck> FrozenCleanup_Queue;
static void Correct_Queue_range(std::vector<size_t> range) static void Correct_Queue_range(std::vector<size_t> range)
{ {
auto small_queue = MakeUnique<Correct_Queue>(QUEUE_BATCH_SIZE); auto small_queue = MakeUnique<Correct_Queue>(QUEUE_BATCH_SIZE);
boost::thread_group tg; small_queue->StartWorkerThreads(SCRIPT_CHECK_THREADS);
for (auto x = 0; x < SCRIPT_CHECK_THREADS; ++x) {
tg.create_thread([&]{small_queue->Thread();});
}
// Make vChecks here to save on malloc (this test can be slow...) // Make vChecks here to save on malloc (this test can be slow...)
std::vector<FakeCheckCheckCompletion> vChecks; std::vector<FakeCheckCheckCompletion> vChecks;
for (const size_t i : range) { for (const size_t i : range) {
@ -168,8 +165,7 @@ static void Correct_Queue_range(std::vector<size_t> range)
BOOST_REQUIRE_EQUAL(FakeCheckCheckCompletion::n_calls, i); BOOST_REQUIRE_EQUAL(FakeCheckCheckCompletion::n_calls, i);
} }
} }
tg.interrupt_all(); small_queue->StopWorkerThreads();
tg.join_all();
} }
/** Test that 0 checks is correct /** Test that 0 checks is correct
@ -212,11 +208,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_Correct_Random)
BOOST_AUTO_TEST_CASE(test_CheckQueue_Catches_Failure) BOOST_AUTO_TEST_CASE(test_CheckQueue_Catches_Failure)
{ {
auto fail_queue = MakeUnique<Failing_Queue>(QUEUE_BATCH_SIZE); auto fail_queue = MakeUnique<Failing_Queue>(QUEUE_BATCH_SIZE);
fail_queue->StartWorkerThreads(SCRIPT_CHECK_THREADS);
boost::thread_group tg;
for (auto x = 0; x < SCRIPT_CHECK_THREADS; ++x) {
tg.create_thread([&]{fail_queue->Thread();});
}
for (size_t i = 0; i < 1001; ++i) { for (size_t i = 0; i < 1001; ++i) {
CCheckQueueControl<FailingCheck> control(fail_queue.get()); CCheckQueueControl<FailingCheck> control(fail_queue.get());
@ -237,18 +229,14 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_Catches_Failure)
BOOST_REQUIRE(success); BOOST_REQUIRE(success);
} }
} }
tg.interrupt_all(); fail_queue->StopWorkerThreads();
tg.join_all();
} }
// Test that a block validation which fails does not interfere with // Test that a block validation which fails does not interfere with
// future blocks, ie, the bad state is cleared. // future blocks, ie, the bad state is cleared.
BOOST_AUTO_TEST_CASE(test_CheckQueue_Recovers_From_Failure) BOOST_AUTO_TEST_CASE(test_CheckQueue_Recovers_From_Failure)
{ {
auto fail_queue = MakeUnique<Failing_Queue>(QUEUE_BATCH_SIZE); auto fail_queue = MakeUnique<Failing_Queue>(QUEUE_BATCH_SIZE);
boost::thread_group tg; fail_queue->StartWorkerThreads(SCRIPT_CHECK_THREADS);
for (auto x = 0; x < SCRIPT_CHECK_THREADS; ++x) {
tg.create_thread([&]{fail_queue->Thread();});
}
for (auto times = 0; times < 10; ++times) { for (auto times = 0; times < 10; ++times) {
for (const bool end_fails : {true, false}) { for (const bool end_fails : {true, false}) {
@ -263,8 +251,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_Recovers_From_Failure)
BOOST_REQUIRE(r != end_fails); BOOST_REQUIRE(r != end_fails);
} }
} }
tg.interrupt_all(); fail_queue->StopWorkerThreads();
tg.join_all();
} }
// Test that unique checks are actually all called individually, rather than // Test that unique checks are actually all called individually, rather than
@ -273,11 +260,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_Recovers_From_Failure)
BOOST_AUTO_TEST_CASE(test_CheckQueue_UniqueCheck) BOOST_AUTO_TEST_CASE(test_CheckQueue_UniqueCheck)
{ {
auto queue = MakeUnique<Unique_Queue>(QUEUE_BATCH_SIZE); auto queue = MakeUnique<Unique_Queue>(QUEUE_BATCH_SIZE);
boost::thread_group tg; queue->StartWorkerThreads(SCRIPT_CHECK_THREADS);
for (auto x = 0; x < SCRIPT_CHECK_THREADS; ++x) {
tg.create_thread([&]{queue->Thread();});
}
size_t COUNT = 100000; size_t COUNT = 100000;
size_t total = COUNT; size_t total = COUNT;
@ -300,8 +283,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_UniqueCheck)
} }
BOOST_REQUIRE(r); BOOST_REQUIRE(r);
} }
tg.interrupt_all(); queue->StopWorkerThreads();
tg.join_all();
} }
@ -313,10 +295,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_UniqueCheck)
BOOST_AUTO_TEST_CASE(test_CheckQueue_Memory) BOOST_AUTO_TEST_CASE(test_CheckQueue_Memory)
{ {
auto queue = MakeUnique<Memory_Queue>(QUEUE_BATCH_SIZE); auto queue = MakeUnique<Memory_Queue>(QUEUE_BATCH_SIZE);
boost::thread_group tg; queue->StartWorkerThreads(SCRIPT_CHECK_THREADS);
for (auto x = 0; x < SCRIPT_CHECK_THREADS; ++x) {
tg.create_thread([&]{queue->Thread();});
}
for (size_t i = 0; i < 1000; ++i) { for (size_t i = 0; i < 1000; ++i) {
size_t total = i; size_t total = i;
{ {
@ -335,8 +314,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_Memory)
} }
BOOST_REQUIRE_EQUAL(MemoryCheck::fake_allocated_memory, 0U); BOOST_REQUIRE_EQUAL(MemoryCheck::fake_allocated_memory, 0U);
} }
tg.interrupt_all(); queue->StopWorkerThreads();
tg.join_all();
} }
// Test that a new verification cannot occur until all checks // Test that a new verification cannot occur until all checks
@ -344,11 +322,8 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_Memory)
BOOST_AUTO_TEST_CASE(test_CheckQueue_FrozenCleanup) BOOST_AUTO_TEST_CASE(test_CheckQueue_FrozenCleanup)
{ {
auto queue = MakeUnique<FrozenCleanup_Queue>(QUEUE_BATCH_SIZE); auto queue = MakeUnique<FrozenCleanup_Queue>(QUEUE_BATCH_SIZE);
boost::thread_group tg;
bool fails = false; bool fails = false;
for (auto x = 0; x < SCRIPT_CHECK_THREADS; ++x) { queue->StartWorkerThreads(SCRIPT_CHECK_THREADS);
tg.create_thread([&]{queue->Thread();});
}
std::thread t0([&]() { std::thread t0([&]() {
CCheckQueueControl<FrozenCleanupCheck> control(queue.get()); CCheckQueueControl<FrozenCleanupCheck> control(queue.get());
std::vector<FrozenCleanupCheck> vChecks(1); std::vector<FrozenCleanupCheck> vChecks(1);
@ -367,7 +342,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_FrozenCleanup)
} }
// Try to get control of the queue a bunch of times // Try to get control of the queue a bunch of times
for (auto x = 0; x < 100 && !fails; ++x) { for (auto x = 0; x < 100 && !fails; ++x) {
fails = queue->ControlMutex.try_lock(); fails = queue->m_control_mutex.try_lock();
} }
{ {
// Unfreeze (we need lock n case of spurious wakeup) // Unfreeze (we need lock n case of spurious wakeup)
@ -378,9 +353,8 @@ BOOST_AUTO_TEST_CASE(test_CheckQueue_FrozenCleanup)
FrozenCleanupCheck::cv.notify_one(); FrozenCleanupCheck::cv.notify_one();
// Wait for control to finish // Wait for control to finish
t0.join(); t0.join();
tg.interrupt_all();
tg.join_all();
BOOST_REQUIRE(!fails); BOOST_REQUIRE(!fails);
queue->StopWorkerThreads();
} }
@ -431,7 +405,7 @@ BOOST_AUTO_TEST_CASE(test_CheckQueueControl_Locks)
cv.wait(l, [&](){return has_lock;}); cv.wait(l, [&](){return has_lock;});
bool fails = false; bool fails = false;
for (auto x = 0; x < 100 && !fails; ++x) { for (auto x = 0; x < 100 && !fails; ++x) {
fails = queue->ControlMutex.try_lock(); fails = queue->m_control_mutex.try_lock();
} }
has_tried = true; has_tried = true;
cv.notify_one(); cv.notify_one();
@ -445,4 +419,3 @@ BOOST_AUTO_TEST_CASE(test_CheckQueueControl_Locks)
} }
} }
BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_SUITE_END()

View file

@ -428,12 +428,10 @@ BOOST_AUTO_TEST_CASE(test_big_witness_transaction)
// check all inputs concurrently, with the cache // check all inputs concurrently, with the cache
PrecomputedTransactionData txdata(tx); PrecomputedTransactionData txdata(tx);
boost::thread_group threadGroup;
CCheckQueue<CScriptCheck> scriptcheckqueue(128); CCheckQueue<CScriptCheck> scriptcheckqueue(128);
CCheckQueueControl<CScriptCheck> control(&scriptcheckqueue); CCheckQueueControl<CScriptCheck> control(&scriptcheckqueue);
for (int i=0; i<20; i++) scriptcheckqueue.StartWorkerThreads(20);
threadGroup.create_thread(std::bind(&CCheckQueue<CScriptCheck>::Thread, std::ref(scriptcheckqueue)));
std::vector<Coin> coins; std::vector<Coin> coins;
for(uint32_t i = 0; i < mtx.vin.size(); i++) { for(uint32_t i = 0; i < mtx.vin.size(); i++) {
@ -455,9 +453,7 @@ BOOST_AUTO_TEST_CASE(test_big_witness_transaction)
bool controlCheck = control.Wait(); bool controlCheck = control.Wait();
assert(controlCheck); assert(controlCheck);
scriptcheckqueue.StopWorkerThreads();
threadGroup.interrupt_all();
threadGroup.join_all();
} }
SignatureData CombineSignatures(const CMutableTransaction& input1, const CMutableTransaction& input2, const CTransactionRef tx) SignatureData CombineSignatures(const CMutableTransaction& input1, const CMutableTransaction& input2, const CTransactionRef tx)

View file

@ -143,9 +143,7 @@ ChainTestingSetup::ChainTestingSetup(const std::string& chainName, const std::ve
// Start script-checking threads. Set g_parallel_script_checks to true so they are used. // Start script-checking threads. Set g_parallel_script_checks to true so they are used.
constexpr int script_check_threads = 2; constexpr int script_check_threads = 2;
for (int i = 0; i < script_check_threads; ++i) { StartScriptCheckWorkerThreads(script_check_threads);
threadGroup.create_thread([i]() { return ThreadScriptCheck(i); });
}
g_parallel_script_checks = true; g_parallel_script_checks = true;
} }
@ -154,6 +152,7 @@ ChainTestingSetup::~ChainTestingSetup()
if (m_node.scheduler) m_node.scheduler->stop(); if (m_node.scheduler) m_node.scheduler->stop();
threadGroup.interrupt_all(); threadGroup.interrupt_all();
threadGroup.join_all(); threadGroup.join_all();
StopScriptCheckWorkerThreads();
GetMainSignals().FlushBackgroundCallbacks(); GetMainSignals().FlushBackgroundCallbacks();
GetMainSignals().UnregisterBackgroundSignalScheduler(); GetMainSignals().UnregisterBackgroundSignalScheduler();
m_node.connman.reset(); m_node.connman.reset();

View file

@ -1763,9 +1763,14 @@ static bool WriteUndoDataForBlock(const CBlockUndo& blockundo, BlockValidationSt
static CCheckQueue<CScriptCheck> scriptcheckqueue(128); static CCheckQueue<CScriptCheck> scriptcheckqueue(128);
void ThreadScriptCheck(int worker_num) { void StartScriptCheckWorkerThreads(int threads_num)
util::ThreadRename(strprintf("scriptch.%i", worker_num)); {
scriptcheckqueue.Thread(); scriptcheckqueue.StartWorkerThreads(threads_num);
}
void StopScriptCheckWorkerThreads()
{
scriptcheckqueue.StopWorkerThreads();
} }
VersionBitsCache versionbitscache GUARDED_BY(cs_main); VersionBitsCache versionbitscache GUARDED_BY(cs_main);

View file

@ -149,8 +149,10 @@ void LoadExternalBlockFile(const CChainParams& chainparams, FILE* fileIn, FlatFi
bool LoadGenesisBlock(const CChainParams& chainparams); bool LoadGenesisBlock(const CChainParams& chainparams);
/** Unload database information */ /** Unload database information */
void UnloadBlockIndex(CTxMemPool* mempool, ChainstateManager& chainman); void UnloadBlockIndex(CTxMemPool* mempool, ChainstateManager& chainman);
/** Run an instance of the script checking thread */ /** Run instances of script checking worker threads */
void ThreadScriptCheck(int worker_num); void StartScriptCheckWorkerThreads(int threads_num);
/** Stop all of the script checking worker threads */
void StopScriptCheckWorkerThreads();
/** /**
* Return transaction from the block at block_index. * Return transaction from the block at block_index.
* If block_index is not provided, fall back to mempool. * If block_index is not provided, fall back to mempool.

View file

@ -68,7 +68,6 @@ EXPECTED_BOOST_INCLUDES=(
boost/signals2/signal.hpp boost/signals2/signal.hpp
boost/test/unit_test.hpp boost/test/unit_test.hpp
boost/thread/condition_variable.hpp boost/thread/condition_variable.hpp
boost/thread/mutex.hpp
boost/thread/shared_mutex.hpp boost/thread/shared_mutex.hpp
boost/thread/thread.hpp boost/thread/thread.hpp
) )