Add min_y and max_y checks for Active Block Modifiers (ABM) (#11333)

This check can be used by ABM to reduce CPU usage.
This commit is contained in:
sfence 2021-06-20 17:21:35 +02:00 committed by GitHub
parent 1805775f3d
commit b10091be9b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 41 additions and 3 deletions

@ -20,6 +20,7 @@ core.features = {
direct_velocity_on_players = true, direct_velocity_on_players = true,
use_texture_alpha_string_modes = true, use_texture_alpha_string_modes = true,
degrotate_240_steps = true, degrotate_240_steps = true,
abm_min_max_y = true,
} }
function core.has_feature(arg) function core.has_feature(arg)

@ -4484,6 +4484,8 @@ Utilities
-- degrotate param2 rotates in units of 1.5° instead of 2° -- degrotate param2 rotates in units of 1.5° instead of 2°
-- thus changing the range of values from 0-179 to 0-240 (5.5.0) -- thus changing the range of values from 0-179 to 0-240 (5.5.0)
degrotate_240_steps = true, degrotate_240_steps = true,
-- ABM supports min_y and max_y fields in definition (5.5.0)
abm_min_max_y = true,
} }
* `minetest.has_feature(arg)`: returns `boolean, missing_features` * `minetest.has_feature(arg)`: returns `boolean, missing_features`
@ -7188,6 +7190,11 @@ Used by `minetest.register_abm`.
-- Chance of triggering `action` per-node per-interval is 1.0 / this -- Chance of triggering `action` per-node per-interval is 1.0 / this
-- value -- value
min_y = -32768,
max_y = 32767,
-- min and max height levels where ABM will be processed
-- can be used to reduce CPU usage
catch_up = true, catch_up = true,
-- If true, catch-up behaviour is enabled: The `chance` value is -- If true, catch-up behaviour is enabled: The `chance` value is
-- temporarily reduced when returning to an area to simulate time lost -- temporarily reduced when returning to an area to simulate time lost

@ -151,12 +151,18 @@ void ScriptApiEnv::initializeEnvironment(ServerEnvironment *env)
bool simple_catch_up = true; bool simple_catch_up = true;
getboolfield(L, current_abm, "catch_up", simple_catch_up); getboolfield(L, current_abm, "catch_up", simple_catch_up);
s16 min_y = INT16_MIN;
getintfield(L, current_abm, "min_y", min_y);
s16 max_y = INT16_MAX;
getintfield(L, current_abm, "max_y", max_y);
lua_getfield(L, current_abm, "action"); lua_getfield(L, current_abm, "action");
luaL_checktype(L, current_abm + 1, LUA_TFUNCTION); luaL_checktype(L, current_abm + 1, LUA_TFUNCTION);
lua_pop(L, 1); lua_pop(L, 1);
LuaABM *abm = new LuaABM(L, id, trigger_contents, required_neighbors, LuaABM *abm = new LuaABM(L, id, trigger_contents, required_neighbors,
trigger_interval, trigger_chance, simple_catch_up); trigger_interval, trigger_chance, simple_catch_up, min_y, max_y);
env->addActiveBlockModifier(abm); env->addActiveBlockModifier(abm);

@ -223,17 +223,21 @@ private:
float m_trigger_interval; float m_trigger_interval;
u32 m_trigger_chance; u32 m_trigger_chance;
bool m_simple_catch_up; bool m_simple_catch_up;
s16 m_min_y;
s16 m_max_y;
public: public:
LuaABM(lua_State *L, int id, LuaABM(lua_State *L, int id,
const std::vector<std::string> &trigger_contents, const std::vector<std::string> &trigger_contents,
const std::vector<std::string> &required_neighbors, const std::vector<std::string> &required_neighbors,
float trigger_interval, u32 trigger_chance, bool simple_catch_up): float trigger_interval, u32 trigger_chance, bool simple_catch_up, s16 min_y, s16 max_y):
m_id(id), m_id(id),
m_trigger_contents(trigger_contents), m_trigger_contents(trigger_contents),
m_required_neighbors(required_neighbors), m_required_neighbors(required_neighbors),
m_trigger_interval(trigger_interval), m_trigger_interval(trigger_interval),
m_trigger_chance(trigger_chance), m_trigger_chance(trigger_chance),
m_simple_catch_up(simple_catch_up) m_simple_catch_up(simple_catch_up),
m_min_y(min_y),
m_max_y(max_y)
{ {
} }
virtual const std::vector<std::string> &getTriggerContents() const virtual const std::vector<std::string> &getTriggerContents() const
@ -256,6 +260,14 @@ public:
{ {
return m_simple_catch_up; return m_simple_catch_up;
} }
virtual s16 getMinY()
{
return m_min_y;
}
virtual s16 getMaxY()
{
return m_max_y;
}
virtual void trigger(ServerEnvironment *env, v3s16 p, MapNode n, virtual void trigger(ServerEnvironment *env, v3s16 p, MapNode n,
u32 active_object_count, u32 active_object_count_wider); u32 active_object_count, u32 active_object_count_wider);
}; };

@ -729,6 +729,8 @@ struct ActiveABM
int chance; int chance;
std::vector<content_t> required_neighbors; std::vector<content_t> required_neighbors;
bool check_required_neighbors; // false if required_neighbors is known to be empty bool check_required_neighbors; // false if required_neighbors is known to be empty
s16 min_y;
s16 max_y;
}; };
class ABMHandler class ABMHandler
@ -773,6 +775,9 @@ public:
} else { } else {
aabm.chance = chance; aabm.chance = chance;
} }
// y limits
aabm.min_y = abm->getMinY();
aabm.max_y = abm->getMaxY();
// Trigger neighbors // Trigger neighbors
const std::vector<std::string> &required_neighbors_s = const std::vector<std::string> &required_neighbors_s =
@ -885,6 +890,9 @@ public:
v3s16 p = p0 + block->getPosRelative(); v3s16 p = p0 + block->getPosRelative();
for (ActiveABM &aabm : *m_aabms[c]) { for (ActiveABM &aabm : *m_aabms[c]) {
if ((p.Y < aabm.min_y) || (p.Y > aabm.max_y))
continue;
if (myrand() % aabm.chance != 0) if (myrand() % aabm.chance != 0)
continue; continue;

@ -67,6 +67,10 @@ public:
virtual u32 getTriggerChance() = 0; virtual u32 getTriggerChance() = 0;
// Whether to modify chance to simulate time lost by an unnattended block // Whether to modify chance to simulate time lost by an unnattended block
virtual bool getSimpleCatchUp() = 0; virtual bool getSimpleCatchUp() = 0;
// get min Y for apply abm
virtual s16 getMinY() = 0;
// get max Y for apply abm
virtual s16 getMaxY() = 0;
// This is called usually at interval for 1/chance of the nodes // This is called usually at interval for 1/chance of the nodes
virtual void trigger(ServerEnvironment *env, v3s16 p, MapNode n){}; virtual void trigger(ServerEnvironment *env, v3s16 p, MapNode n){};
virtual void trigger(ServerEnvironment *env, v3s16 p, MapNode n, virtual void trigger(ServerEnvironment *env, v3s16 p, MapNode n,