mirror of
https://github.com/luanti-org/luanti.git
synced 2025-08-06 17:41:04 +00:00
Add a bit of debug code around MapBlock refcounting
This commit is contained in:
parent
39417cf7a7
commit
0c12c1f400
6 changed files with 74 additions and 31 deletions
|
@ -202,8 +202,14 @@ void ClientMap::onSettingChanged(std::string_view name, bool all)
|
||||||
|
|
||||||
ClientMap::~ClientMap()
|
ClientMap::~ClientMap()
|
||||||
{
|
{
|
||||||
|
verbosestream << FUNCTION_NAME << std::endl;
|
||||||
|
|
||||||
g_settings->deregisterAllChangedCallbacks(this);
|
g_settings->deregisterAllChangedCallbacks(this);
|
||||||
|
|
||||||
|
// avoid refcount warning from ~Map()
|
||||||
|
clearDrawList();
|
||||||
|
clearDrawListShadow();
|
||||||
|
|
||||||
for (auto &it : m_dynamic_buffers)
|
for (auto &it : m_dynamic_buffers)
|
||||||
it.second.drop();
|
it.second.drop();
|
||||||
}
|
}
|
||||||
|
@ -351,23 +357,29 @@ private:
|
||||||
v3s16 volume;
|
v3s16 volume;
|
||||||
};
|
};
|
||||||
|
|
||||||
void ClientMap::updateDrawList()
|
void ClientMap::clearDrawList()
|
||||||
{
|
{
|
||||||
ScopeProfiler sp(g_profiler, "CM::updateDrawList()", SPT_AVG);
|
|
||||||
|
|
||||||
m_needs_update_drawlist = false;
|
|
||||||
|
|
||||||
for (auto &i : m_drawlist) {
|
for (auto &i : m_drawlist) {
|
||||||
MapBlock *block = i.second;
|
MapBlock *block = i.second;
|
||||||
block->refDrop();
|
block->refDrop();
|
||||||
}
|
}
|
||||||
m_drawlist.clear();
|
m_drawlist.clear();
|
||||||
|
|
||||||
for (auto &block : m_keeplist) {
|
for (auto &block : m_keeplist)
|
||||||
block->refDrop();
|
block->refDrop();
|
||||||
}
|
|
||||||
m_keeplist.clear();
|
m_keeplist.clear();
|
||||||
|
|
||||||
|
m_needs_update_drawlist = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ClientMap::updateDrawList()
|
||||||
|
{
|
||||||
|
ScopeProfiler sp(g_profiler, "CM::updateDrawList()", SPT_AVG);
|
||||||
|
|
||||||
|
clearDrawList();
|
||||||
|
|
||||||
|
m_needs_update_drawlist = false;
|
||||||
|
|
||||||
const v3s16 cam_pos_nodes = floatToInt(m_camera_position, BS);
|
const v3s16 cam_pos_nodes = floatToInt(m_camera_position, BS);
|
||||||
|
|
||||||
v3s16 p_blocks_min;
|
v3s16 p_blocks_min;
|
||||||
|
@ -1519,6 +1531,15 @@ void ClientMap::renderMapShadows(video::IVideoDriver *driver,
|
||||||
g_profiler->avg(prefix + "material swaps [#]", material_swaps);
|
g_profiler->avg(prefix + "material swaps [#]", material_swaps);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ClientMap::clearDrawListShadow()
|
||||||
|
{
|
||||||
|
for (auto &i : m_drawlist_shadow) {
|
||||||
|
MapBlock *block = i.second;
|
||||||
|
block->refDrop();
|
||||||
|
}
|
||||||
|
m_drawlist_shadow.clear();
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Custom update draw list for the pov of shadow light.
|
Custom update draw list for the pov of shadow light.
|
||||||
*/
|
*/
|
||||||
|
@ -1526,11 +1547,7 @@ void ClientMap::updateDrawListShadow(v3f shadow_light_pos, v3f shadow_light_dir,
|
||||||
{
|
{
|
||||||
ScopeProfiler sp(g_profiler, "CM::updateDrawListShadow()", SPT_AVG);
|
ScopeProfiler sp(g_profiler, "CM::updateDrawListShadow()", SPT_AVG);
|
||||||
|
|
||||||
for (auto &i : m_drawlist_shadow) {
|
clearDrawListShadow();
|
||||||
MapBlock *block = i.second;
|
|
||||||
block->refDrop();
|
|
||||||
}
|
|
||||||
m_drawlist_shadow.clear();
|
|
||||||
|
|
||||||
// Number of blocks currently loaded by the client
|
// Number of blocks currently loaded by the client
|
||||||
u32 blocks_loaded = 0;
|
u32 blocks_loaded = 0;
|
||||||
|
|
|
@ -89,12 +89,20 @@ public:
|
||||||
|
|
||||||
void getBlocksInViewRange(v3s16 cam_pos_nodes,
|
void getBlocksInViewRange(v3s16 cam_pos_nodes,
|
||||||
v3s16 *p_blocks_min, v3s16 *p_blocks_max, float range=-1.0f);
|
v3s16 *p_blocks_min, v3s16 *p_blocks_max, float range=-1.0f);
|
||||||
|
|
||||||
void updateDrawList();
|
void updateDrawList();
|
||||||
// @brief Calculate statistics about the map and keep the blocks alive
|
/// @brief clears m_drawlist and m_keeplist
|
||||||
|
void clearDrawList();
|
||||||
|
|
||||||
|
/// @brief Calculate statistics about the map and keep the blocks alive
|
||||||
void touchMapBlocks();
|
void touchMapBlocks();
|
||||||
|
|
||||||
void updateDrawListShadow(v3f shadow_light_pos, v3f shadow_light_dir, float radius, float length);
|
void updateDrawListShadow(v3f shadow_light_pos, v3f shadow_light_dir, float radius, float length);
|
||||||
|
void clearDrawListShadow();
|
||||||
|
|
||||||
// Returns true if draw list needs updating before drawing the next frame.
|
// Returns true if draw list needs updating before drawing the next frame.
|
||||||
bool needsUpdateDrawList() { return m_needs_update_drawlist; }
|
bool needsUpdateDrawList() { return m_needs_update_drawlist; }
|
||||||
|
|
||||||
void renderMap(video::IVideoDriver* driver, s32 pass);
|
void renderMap(video::IVideoDriver* driver, s32 pass);
|
||||||
|
|
||||||
void renderMapShadows(video::IVideoDriver *driver,
|
void renderMapShadows(video::IVideoDriver *driver,
|
||||||
|
|
17
src/map.cpp
17
src/map.cpp
|
@ -30,12 +30,23 @@ Map::Map(IGameDef *gamedef):
|
||||||
|
|
||||||
Map::~Map()
|
Map::~Map()
|
||||||
{
|
{
|
||||||
/*
|
// Free all sectors
|
||||||
Free all MapSectors
|
size_t used = 0;
|
||||||
*/
|
|
||||||
for (auto §or : m_sectors) {
|
for (auto §or : m_sectors) {
|
||||||
|
sector.second->deleteBlocks(&used);
|
||||||
delete sector.second;
|
delete sector.second;
|
||||||
}
|
}
|
||||||
|
m_sectors.clear();
|
||||||
|
|
||||||
|
if (used > 0) {
|
||||||
|
#ifdef NDEBUG
|
||||||
|
std::ostream &to = infostream;
|
||||||
|
#else
|
||||||
|
std::ostream &to = warningstream;
|
||||||
|
#endif
|
||||||
|
PrintInfo(to);
|
||||||
|
to << used << " blocks deleted despite reference count > 0. Potential bug." << std::endl;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Map::addEventReceiver(MapEventReceiver *event_receiver)
|
void Map::addEventReceiver(MapEventReceiver *event_receiver)
|
||||||
|
|
|
@ -187,26 +187,28 @@ public:
|
||||||
//// Position stuff
|
//// Position stuff
|
||||||
////
|
////
|
||||||
|
|
||||||
|
/// @return map position of block
|
||||||
inline v3s16 getPos()
|
inline v3s16 getPos()
|
||||||
{
|
{
|
||||||
return m_pos;
|
return m_pos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// @return in-world position of the block (== pos * MAP_BLOCKSIZE)
|
||||||
inline v3s16 getPosRelative()
|
inline v3s16 getPosRelative()
|
||||||
{
|
{
|
||||||
return m_pos_relative;
|
return m_pos_relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline core::aabbox3d<s16> getBox() {
|
/// @return in-world box of the block
|
||||||
|
inline core::aabbox3d<s16> getBox()
|
||||||
|
{
|
||||||
return getBox(getPosRelative());
|
return getBox(getPosRelative());
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline core::aabbox3d<s16> getBox(const v3s16 &pos_relative)
|
static inline core::aabbox3d<s16> getBox(v3s16 pos_relative)
|
||||||
{
|
{
|
||||||
return core::aabbox3d<s16>(pos_relative,
|
return core::aabbox3d<s16>(pos_relative,
|
||||||
pos_relative
|
pos_relative + v3s16(MAP_BLOCKSIZE - 1));
|
||||||
+ v3s16(MAP_BLOCKSIZE, MAP_BLOCKSIZE, MAP_BLOCKSIZE)
|
|
||||||
- v3s16(1,1,1));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////
|
////
|
||||||
|
@ -360,7 +362,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
////
|
////
|
||||||
//// Reference counting (see m_refcount)
|
//// Reference counting (different purposes on client vs. server)
|
||||||
////
|
////
|
||||||
|
|
||||||
inline void refGrab()
|
inline void refGrab()
|
||||||
|
@ -470,10 +472,6 @@ private:
|
||||||
*/
|
*/
|
||||||
v3s16 m_pos_relative;
|
v3s16 m_pos_relative;
|
||||||
|
|
||||||
/*
|
|
||||||
Reference count; currently used for determining if this block is in
|
|
||||||
the list of blocks to be drawn.
|
|
||||||
*/
|
|
||||||
short m_refcount = 0;
|
short m_refcount = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -19,12 +19,18 @@ MapSector::~MapSector()
|
||||||
deleteBlocks();
|
deleteBlocks();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MapSector::deleteBlocks()
|
void MapSector::deleteBlocks(size_t *used_count)
|
||||||
{
|
{
|
||||||
// Clear cache
|
|
||||||
m_block_cache = nullptr;
|
m_block_cache = nullptr;
|
||||||
|
|
||||||
// Delete all blocks
|
size_t u = 0;
|
||||||
|
for (auto &it : m_blocks) {
|
||||||
|
if (it.second->refGet() > 0)
|
||||||
|
u++;
|
||||||
|
it.second.reset();
|
||||||
|
}
|
||||||
|
if (used_count)
|
||||||
|
*used_count += u;
|
||||||
m_blocks.clear();
|
m_blocks.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,7 +29,9 @@ public:
|
||||||
MapSector(Map *parent, v2s16 pos, IGameDef *gamedef);
|
MapSector(Map *parent, v2s16 pos, IGameDef *gamedef);
|
||||||
virtual ~MapSector();
|
virtual ~MapSector();
|
||||||
|
|
||||||
void deleteBlocks();
|
/// @brief Deletes all blocks (regardless of reference count).
|
||||||
|
/// @param used_count output: number of blocks which were still ref'd
|
||||||
|
void deleteBlocks(size_t *used_count = nullptr);
|
||||||
|
|
||||||
v2s16 getPos() const
|
v2s16 getPos() const
|
||||||
{
|
{
|
||||||
|
@ -60,7 +62,8 @@ public:
|
||||||
|
|
||||||
bool empty() const { return m_blocks.empty(); }
|
bool empty() const { return m_blocks.empty(); }
|
||||||
|
|
||||||
int size() const { return m_blocks.size(); }
|
size_t size() const { return m_blocks.size(); }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
// The pile of MapBlocks
|
// The pile of MapBlocks
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue