/* * This file is part of the AzerothCore Project. See AUTHORS file for Copyright information * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU Affero General Public License as published by the * Free Software Foundation; either version 3 of the License, or (at your * option) any later version. * * This program 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 Affero 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 . */ #include "DatabaseEnv.h" #include "LFGMgr.h" #include "Map.h" #include "MapUpdater.h" #include "Metric.h" class UpdateRequest { public: UpdateRequest() = default; virtual ~UpdateRequest() = default; virtual void call() = 0; }; class MapUpdateRequest : public UpdateRequest { public: MapUpdateRequest(Map& m, MapUpdater& u, uint32 d, uint32 sd) : m_map(m), m_updater(u), m_diff(d), s_diff(sd) { } void call() override { METRIC_TIMER("map_update_time_diff", METRIC_TAG("map_id", std::to_string(m_map.GetId()))); m_map.Update(m_diff, s_diff); m_updater.update_finished(); } private: Map& m_map; MapUpdater& m_updater; uint32 m_diff; uint32 s_diff; }; class LFGUpdateRequest : public UpdateRequest { public: LFGUpdateRequest(MapUpdater& u, uint32 d) : m_updater(u), m_diff(d) {} void call() override { sLFGMgr->Update(m_diff, 1); m_updater.update_finished(); } private: MapUpdater& m_updater; uint32 m_diff; }; MapUpdater::MapUpdater(): pending_requests(0) { } void MapUpdater::activate(size_t num_threads) { _workerThreads.reserve(num_threads); for (size_t i = 0; i < num_threads; ++i) { _workerThreads.push_back(std::thread(&MapUpdater::WorkerThread, this)); } } void MapUpdater::deactivate() { _cancelationToken = true; wait(); _queue.Cancel(); for (auto& thread : _workerThreads) { if (thread.joinable()) { thread.join(); } } } void MapUpdater::wait() { std::unique_lock guard(_lock); while (pending_requests > 0) _condition.wait(guard); guard.unlock(); } void MapUpdater::schedule_update(Map& map, uint32 diff, uint32 s_diff) { std::lock_guard guard(_lock); ++pending_requests; _queue.Push(new MapUpdateRequest(map, *this, diff, s_diff)); } void MapUpdater::schedule_lfg_update(uint32 diff) { std::lock_guard guard(_lock); ++pending_requests; _queue.Push(new LFGUpdateRequest(*this, diff)); } bool MapUpdater::activated() { return _workerThreads.size() > 0; } void MapUpdater::update_finished() { std::lock_guard lock(_lock); --pending_requests; _condition.notify_all(); } void MapUpdater::WorkerThread() { LoginDatabase.WarnAboutSyncQueries(true); CharacterDatabase.WarnAboutSyncQueries(true); WorldDatabase.WarnAboutSyncQueries(true); while (1) { UpdateRequest* request = nullptr; _queue.WaitAndPop(request); if (_cancelationToken) return; request->call(); delete request; } }