/* Stockfish, a UCI chess playing engine derived from Glaurung 2.1 Copyright (C) 2004-2020 The Stockfish developers (see AUTHORS file) Stockfish is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Stockfish is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #ifndef THREAD_H_INCLUDED #define THREAD_H_INCLUDED #include #include #include #include #include #include #include "material.h" #include "movepick.h" #include "pawns.h" #include "position.h" #include "search.h" #include "thread_win32_osx.h" /// Thread class keeps together all the thread-related stuff. We use /// per-thread pawn and material hash tables so that once we get a /// pointer to an entry its life time is unlimited and we don't have /// to care about someone changing the entry under our feet. namespace Detail { template struct TypeIdentity { using Type = T; }; } class Thread { std::mutex mutex; std::condition_variable cv; size_t idx; bool exit = false, searching = true; // Set before starting std::thread std::function worker; NativeThread stdThread; public: explicit Thread(size_t); virtual ~Thread(); virtual void search(); // The function object to be executed is taken by value to remove // the need for separate lvalue and rvalue overloads. // The worker thread needs to have ownership of the task // to be executed because otherwise there's no way to manage its lifetime. virtual void execute_with_worker(std::function t); void clear(); void idle_loop(); void start_searching(); void wait_for_search_finished(); void wait_for_worker_finished(); size_t thread_idx() const { return idx; } Pawns::Table pawnsTable; Material::Table materialTable; size_t pvIdx, pvLast; uint64_t ttHitAverage; int selDepth, nmpMinPly; Color nmpColor; std::atomic nodes, tbHits, bestMoveChanges; Position rootPos; StateInfo rootState; Search::RootMoves rootMoves; Depth rootDepth, completedDepth; CounterMoveHistory counterMoves; ButterflyHistory mainHistory; LowPlyHistory lowPlyHistory; CapturePieceToHistory captureHistory; ContinuationHistory continuationHistory[2][2]; Score contempt; int failedHighCnt; bool rootInTB; int Cardinality; bool UseRule50; Depth ProbeDepth; }; /// MainThread is a derived class specific for main thread struct MainThread : public Thread { using Thread::Thread; void search() override; void check_time(); double previousTimeReduction; Value bestPreviousScore; Value iterValue[4]; int callsCnt; bool stopOnPonderhit; std::atomic_bool ponder; }; /// ThreadPool struct handles all the threads-related stuff like init, starting, /// parking and, most importantly, launching a thread. All the access to threads /// is done through this class. struct ThreadPool : public std::vector { // Each thread gets its own copy of the `worker` function object. // This means that each worker thread will have exclusive access // to the state of the `worker` function object. void execute_with_workers(const std::function& worker); template void for_each_index_with_workers( IndexT begin, typename Detail::TypeIdentity::Type end, FuncT func) { // This value must outlive the function call. // It's fairly safe if we make it static // because for_each_index_with_workers // is not reentrant nor thread safe. static std::atomic i_atomic; i_atomic.store(begin); execute_with_workers( [end, func](Thread& th) mutable { for(;;) { const auto i = i_atomic.fetch_add(1); if (i >= end) break; func(th, i); } }); } void start_thinking(Position&, StateListPtr&, const Search::LimitsType&, bool = false); void clear(); void set(size_t); MainThread* main() const { return static_cast(front()); } uint64_t nodes_searched() const { return accumulate(&Thread::nodes); } uint64_t tb_hits() const { return accumulate(&Thread::tbHits); } Thread* get_best_thread() const; void start_searching(); void wait_for_search_finished() const; void wait_for_workers_finished() const; std::atomic_bool stop, increaseDepth; private: StateListPtr setupStates; uint64_t accumulate(std::atomic Thread::* member) const { uint64_t sum = 0; for (Thread* th : *this) sum += (th->*member).load(std::memory_order_relaxed); return sum; } }; extern ThreadPool Threads; #endif // #ifndef THREAD_H_INCLUDED