Optimize swapping nodes with equivalent lighting

This commit is contained in:
Jude Melton-Houghton 2022-03-29 12:06:44 -04:00 committed by luk3yx
parent 745b858bae
commit ad2e053f17
2 changed files with 52 additions and 21 deletions

View File

@ -174,22 +174,30 @@ MapNode Map::getNode(v3s16 p, bool *is_valid_position)
return node; return node;
} }
static void set_node_in_block(MapBlock *block, v3s16 relpos, MapNode n)
{
// Never allow placing CONTENT_IGNORE, it causes problems
if(n.getContent() == CONTENT_IGNORE){
const NodeDefManager *nodedef = block->getParent()->getNodeDefManager();
v3s16 blockpos = block->getPos();
v3s16 p = blockpos * MAP_BLOCKSIZE + relpos;
bool temp_bool;
errorstream<<"Not allowing to place CONTENT_IGNORE"
<<" while trying to replace \""
<<nodedef->get(block->getNodeNoCheck(relpos, &temp_bool)).name
<<"\" at "<<PP(p)<<" (block "<<PP(blockpos)<<")"<<std::endl;
return;
}
block->setNodeNoCheck(relpos, n);
}
// throws InvalidPositionException if not found // throws InvalidPositionException if not found
void Map::setNode(v3s16 p, MapNode & n) void Map::setNode(v3s16 p, MapNode & n)
{ {
v3s16 blockpos = getNodeBlockPos(p); v3s16 blockpos = getNodeBlockPos(p);
MapBlock *block = getBlockNoCreate(blockpos); MapBlock *block = getBlockNoCreate(blockpos);
v3s16 relpos = p - blockpos*MAP_BLOCKSIZE; v3s16 relpos = p - blockpos*MAP_BLOCKSIZE;
// Never allow placing CONTENT_IGNORE, it causes problems set_node_in_block(block, relpos, n);
if(n.getContent() == CONTENT_IGNORE){
bool temp_bool;
errorstream<<"Map::setNode(): Not allowing to place CONTENT_IGNORE"
<<" while trying to replace \""
<<m_nodedef->get(block->getNodeNoCheck(relpos, &temp_bool)).name
<<"\" at "<<PP(p)<<" (block "<<PP(blockpos)<<")"<<std::endl;
return;
}
block->setNodeNoCheck(relpos, n);
} }
void Map::addNodeAndUpdate(v3s16 p, MapNode n, void Map::addNodeAndUpdate(v3s16 p, MapNode n,
@ -201,8 +209,14 @@ void Map::addNodeAndUpdate(v3s16 p, MapNode n,
RollbackNode rollback_oldnode(this, p, m_gamedef); RollbackNode rollback_oldnode(this, p, m_gamedef);
#endif #endif
v3s16 blockpos = getNodeBlockPos(p);
MapBlock *block = getBlockNoCreate(blockpos);
if (block->isDummy())
throw InvalidPositionException();
v3s16 relpos = p - blockpos * MAP_BLOCKSIZE;
// This is needed for updating the lighting // This is needed for updating the lighting
MapNode oldnode = getNode(p); MapNode oldnode = block->getNodeUnsafe(relpos);
// Remove node metadata // Remove node metadata
if (remove_metadata) { if (remove_metadata) {
@ -210,18 +224,29 @@ void Map::addNodeAndUpdate(v3s16 p, MapNode n,
} }
// Set the node on the map // Set the node on the map
// Ignore light (because calling voxalgo::update_lighting_nodes) const ContentFeatures &cf = m_nodedef->get(n);
n.setLight(LIGHTBANK_DAY, 0, m_nodedef); const ContentFeatures &oldcf = m_nodedef->get(oldnode);
n.setLight(LIGHTBANK_NIGHT, 0, m_nodedef); if (cf.lightingEquivalent(oldcf)) {
setNode(p, n); // No light update needed, just copy over the old light.
n.setLight(LIGHTBANK_DAY, oldnode.getLightRaw(LIGHTBANK_DAY, oldcf), cf);
n.setLight(LIGHTBANK_NIGHT, oldnode.getLightRaw(LIGHTBANK_NIGHT, oldcf), cf);
set_node_in_block(block, relpos, n);
// Update lighting modified_blocks[blockpos] = block;
std::vector<std::pair<v3s16, MapNode> > oldnodes; } else {
oldnodes.emplace_back(p, oldnode); // Ignore light (because calling voxalgo::update_lighting_nodes)
voxalgo::update_lighting_nodes(this, oldnodes, modified_blocks); n.setLight(LIGHTBANK_DAY, 0, cf);
n.setLight(LIGHTBANK_NIGHT, 0, cf);
set_node_in_block(block, relpos, n);
for (auto &modified_block : modified_blocks) { // Update lighting
modified_block.second->expireDayNightDiff(); std::vector<std::pair<v3s16, MapNode> > oldnodes;
oldnodes.emplace_back(p, oldnode);
voxalgo::update_lighting_nodes(this, oldnodes, modified_blocks);
for (auto &modified_block : modified_blocks) {
modified_block.second->expireDayNightDiff();
}
} }
#if USE_SQLITE #if USE_SQLITE

View File

@ -471,6 +471,12 @@ struct ContentFeatures
return (liquid_alternative_flowing_id == f.liquid_alternative_flowing_id); return (liquid_alternative_flowing_id == f.liquid_alternative_flowing_id);
} }
bool lightingEquivalent(const ContentFeatures &other) const {
return light_propagates == other.light_propagates
&& sunlight_propagates == other.sunlight_propagates
&& light_source == other.light_source;
}
int getGroup(const std::string &group) const int getGroup(const std::string &group) const
{ {
return itemgroup_get(groups, group); return itemgroup_get(groups, group);