Aaron Suen bf744aab8b Fix excess meta writes in soaking API
It didn't seem to affect many test cases, but avoid
excess calls to meta:set_*() due to 0/nil disagreement.
2021-12-05 09:10:43 -05:00

168 lines
4.4 KiB
Lua

-- LUALOCALS < ---------------------------------------------------------
local error, math, minetest, nodecore, pairs, type
= error, math, minetest, nodecore, pairs, type
local math_floor, math_sqrt
= math.floor, math.sqrt
-- LUALOCALS > ---------------------------------------------------------
local metacache = {}
nodecore.register_on_nodeupdate(function(pos)
metacache[minetest.hash_node_position(pos)] = nil
end)
local function metaget(meta, def, nodekey)
local cached = nodekey and metacache[nodekey]
if cached then return cached end
cached = {
qty = meta:get_float(def.fieldname .. "qty"),
time = meta:get_float(def.fieldname .. "time")
}
if cached.qty == 0 then cached.qty = nil end
if cached.time == 0 then cached.time = nil end
if nodekey then metacache = cached end
return cached
end
local function metaset_core(meta, field, value)
if value then
return meta:set_float(field, value)
else
return meta:set_string(field, "")
end
end
local function metaset(meta, def, nodekey, qty, time)
local cached = metaget(meta, def, nodekey)
if qty == 0 then qty = nil end
if cached.qty ~= qty then
metaset_core(meta, def.fieldname .. "qty", qty)
cached.qty = qty
end
if time == 0 then time = nil end
if cached.time ~= time then
metaset_core(meta, def.fieldname .. "time", time)
cached.time = time
end
end
local function soaking_core(def, reg, getmeta, getnodekey)
if not def.fieldname then error("soaking def missing fieldname") end
def.interval = def.interval or 1
def.chance = def.chance or 1
def.soakinterval = def.soakinterval or (def.interval * def.chance)
if not def.soakrate then error("soaking missing soakrate callback") end
if not def.soakcheck then error("soaking missing soakcheck callback") end
def.soakvary = def.soakvary or 0.25
if not def.soakrand then
if def.soakvary then
def.soakrand = function(rate, ticks)
return rate * (1 + def.soakvary * nodecore.boxmuller()
/ math_sqrt(ticks)) * ticks
end
else
def.soakrand = function(rate, ticks) return rate * ticks end
end
end
local rateadj = nodecore.rate_adjustment("speed", "soaking", def.label)
def.action = function(...)
local nodekey = getnodekey(...)
local meta = getmeta(...)
if def.quickcheck and not def.quickcheck(...) then
metaset(meta, def, nodekey)
return ...
end
local now = nodecore.gametime
local metadata = metaget(meta, def, nodekey)
local total = metadata.qty or 0
local start = metadata.time
start = start and start ~= 0 and start or now
local rate = 0
local delta = 0
if start <= now then
rate = def.soakrate(...)
if rate == false then
metaset(meta, def, nodekey)
return ...
end
rate = rate or 0
local ticks = 1 + math_floor((now - start) / def.soakinterval)
delta = def.soakrand(rate, ticks)
total = total + delta * rateadj
start = start + ticks * def.soakinterval
end
local function helper(set, ...)
if set == false then
metaset(meta, def, nodekey)
return ...
end
metaset(meta, def, nodekey,
set and type(set) == "number" and set or total,
start)
return ...
end
return helper(def.soakcheck({
rate = rate,
delta = delta,
total = total
}, ...))
end
return reg(def)
end
local soaking_abm_by_fieldname = {}
function nodecore.register_soaking_abm(def)
def.nodeidx = nodecore.group_expand(def.nodenames, true)
soaking_abm_by_fieldname[def.fieldname] = def
return soaking_core(def,
minetest.register_abm,
function(pos) return minetest.get_meta(pos) end,
minetest.hash_node_position
)
end
function nodecore.register_soaking_aism(def)
return soaking_core(def,
nodecore.register_aism,
function(stack) return stack:get_meta() end,
function() end
)
end
local pending
function nodecore.soaking_abm_push(pos, fieldname, qty)
local abm = soaking_abm_by_fieldname[fieldname]
if not abm then return end
local node = minetest.get_node(pos)
if not abm.nodeidx[node.name] then return end
local meta = minetest.get_meta(pos)
local qf = fieldname .. "qty"
meta:set_float(qf, (meta:get_float(qf) or 0) + qty)
local tf = fieldname .. "time"
if (meta:get_float(tf) or 0) == 0 then meta:set_float(tf, nodecore.gametime) end
local func = abm.action
if pending then
pending[#pending + 1] = function() return func(pos, node) end
else
pending = {}
func(pos, node)
while #pending > 0 do
local batch = pending
pending = {}
for _, f in pairs(batch) do f() end
end
pending = nil
end
end