/* Minetest Copyright (C) 2013, 2017 celeron55, Perttu Ahola This program is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #pragma once #include #include #include #include #include "mapblock_mesh.h" #include "threading/mutex_auto_lock.h" #include "util/thread.h" #include #include struct CachedMapBlockData { v3s16 p = v3s16(-1337, -1337, -1337); MapNode *data = nullptr; // A copy of the MapBlock's data member int refcount_from_queue = 0; std::time_t last_used_timestamp = std::time(0); CachedMapBlockData() = default; ~CachedMapBlockData(); }; struct QueuedMeshUpdate { v3s16 p = v3s16(-1337, -1337, -1337); bool ack_block_to_server = false; int crack_level = -1; v3s16 crack_pos; MeshMakeData *data = nullptr; // This is generated in MeshUpdateQueue::pop() bool urgent = false; QueuedMeshUpdate() = default; ~QueuedMeshUpdate(); }; /* A thread-safe queue of mesh update tasks and a cache of MapBlock data */ class MeshUpdateQueue { enum UpdateMode { FORCE_UPDATE, SKIP_UPDATE_IF_ALREADY_CACHED, }; public: MeshUpdateQueue(Client *client); ~MeshUpdateQueue(); // Caches the block at p and its neighbors (if needed) and queues a mesh // update for the block at p bool addBlock(Map *map, v3s16 p, bool ack_block_to_server, bool urgent); // Returned pointer must be deleted // Returns NULL if queue is empty QueuedMeshUpdate *pop(); // Marks a position as finished, unblocking the next update void done(v3s16 pos); u32 size() { MutexAutoLock lock(m_mutex); return m_queue.size(); } private: Client *m_client; std::vector m_queue; std::unordered_set m_urgents; std::unordered_map m_cache; std::unordered_set m_inflight_blocks; u64 m_next_cache_cleanup; // milliseconds std::mutex m_mutex; // TODO: Add callback to update these when g_settings changes bool m_cache_enable_shaders; bool m_cache_smooth_lighting; int m_meshgen_block_cache_size; CachedMapBlockData *cacheBlock(Map *map, v3s16 p, UpdateMode mode, size_t *cache_hit_counter = NULL); CachedMapBlockData *getCachedBlock(const v3s16 &p); void fillDataFromMapBlockCache(QueuedMeshUpdate *q); void cleanupCache(); }; struct MeshUpdateResult { v3s16 p = v3s16(-1338, -1338, -1338); MapBlockMesh *mesh = nullptr; bool ack_block_to_server = false; bool urgent = false; MeshUpdateResult() = default; }; class MeshUpdateManager; class MeshUpdateWorkerThread : public UpdateThread { public: MeshUpdateWorkerThread(MeshUpdateQueue *queue_in, MeshUpdateManager *manager, v3s16 *camera_offset); protected: virtual void doUpdate(); private: MeshUpdateQueue *m_queue_in; MeshUpdateManager *m_manager; v3s16 *m_camera_offset; // TODO: Add callback to update these when g_settings changes int m_generation_interval; }; class MeshUpdateManager { public: MeshUpdateManager(Client *client); // Caches the block at p and its neighbors (if needed) and queues a mesh // update for the block at p void updateBlock(Map *map, v3s16 p, bool ack_block_to_server, bool urgent, bool update_neighbors = false); void putResult(const MeshUpdateResult &r); bool getNextResult(MeshUpdateResult &r); v3s16 m_camera_offset; void start(); void stop(); void wait(); bool isRunning(); private: void deferUpdate(); MeshUpdateQueue m_queue_in; MutexedQueue m_queue_out; MutexedQueue m_queue_out_urgent; std::vector> m_workers; };