improve the efficiency of giant mycelium growth using flat node array, fewer dereferences

This commit is contained in:
FaceDeer 2020-02-10 13:38:27 -07:00
parent d9b3903e66
commit 336c0849aa
2 changed files with 123 additions and 100 deletions

View File

@ -104,7 +104,7 @@ local mushroom_warren_ceiling = function(abs_cracks, vi, area, data, data_param2
data[vi] = c_mycelial_dirt data[vi] = c_mycelial_dirt
if abs_cracks < 0.2 then if abs_cracks < 0.2 then
local rand = math.random() local rand = math.random()
if rand < 0.002 then if rand < 0.001 then
local mycelium_index = vi-ystride local mycelium_index = vi-ystride
data[mycelium_index] = c_giant_mycelium data[mycelium_index] = c_giant_mycelium
minetest.get_node_timer(area:position(mycelium_index)):start(math.random(1,giant_mycelium_timer_spread)) minetest.get_node_timer(area:position(mycelium_index)):start(math.random(1,giant_mycelium_timer_spread))
@ -354,7 +354,7 @@ local decorate_primordial = function(minp, maxp, seed, vm, node_arrays, area, da
minetest.get_node_timer(area:position(vi)):start(math.random(30, 120)) minetest.get_node_timer(area:position(vi)):start(math.random(30, 120))
else else
data[vi] = c_mycelial_dirt data[vi] = c_mycelial_dirt
if math.random() < 0.025 then if math.random() < 0.05 then
local rand_vi = vi + random_dir[math.random(1,4)] local rand_vi = vi + random_dir[math.random(1,4)]
if data[rand_vi] == c_air then if data[rand_vi] == c_air then
data[rand_vi] = c_giant_mycelium data[rand_vi] = c_giant_mycelium

View File

@ -111,29 +111,37 @@ minetest.register_craft({
-- By growing with these conditions hyphae will hug the ground and will not immediately loop back on themselves -- By growing with these conditions hyphae will hug the ground and will not immediately loop back on themselves
-- (though they can run into other pre-existing growths, forming larger loops - which is fine, large loops are nice) -- (though they can run into other pre-existing growths, forming larger loops - which is fine, large loops are nice)
local ystride = 3
local zstride = 9
local get_item_group = minetest.get_item_group
local get_node = minetest.get_node
local registered_nodes = minetest.registered_nodes
local math_random = math.random
local find_mycelium_growth_targets = function(pos) local find_mycelium_growth_targets = function(pos)
local nodes = {} local nodes = {}
local pos_x = pos.x
local pos_y = pos.y
local pos_z = pos.z
for x = -1, 1 do for x = -1, 1 do
nodes[x] = {}
for y = -1, 1 do for y = -1, 1 do
nodes[x][y] = {}
for z = -1, 1 do for z = -1, 1 do
if not (x == y and y == z) then -- we don't care about the diagonals or the center node if not (x == y and y == z) then -- we don't care about the diagonals or the center node
local node = minetest.get_node({x=pos.x+x, y=pos.y+y, z=pos.z+z}) local node = get_node({x=pos_x+x, y=pos_y+y, z=pos_z+z})
if node.name == "ignore" then local node_name = node.name
if node_name == "ignore" then
-- Pause growth! We're at the edge of the known world. -- Pause growth! We're at the edge of the known world.
return nil return nil
end end
local state = {} if get_item_group(node_name, "soil") > 0 or
if minetest.get_item_group(node.name, "soil") > 0 or get_item_group(node_name, "stone") > 0 and math_random() < 0.5 then -- let hyphae explore out over stone
minetest.get_item_group(node.name, "stone") > 0 and math.random() < 0.5 then -- let hyphae explore out over stone nodes[x + y*ystride + z*zstride] = "soil"
state.soil = true elseif get_item_group(node_name, "hypha") > 0 then
elseif minetest.get_item_group(node.name, "hypha") > 0 then nodes[x + y*ystride + z*zstride] = "hypha"
state.hypha = true elseif registered_nodes[node_name] and registered_nodes[node_name].buildable_to then
elseif minetest.registered_nodes[node.name] and minetest.registered_nodes[node.name].buildable_to then nodes[x + y*ystride + z*zstride] = "buildable"
state.buildable = true
end end
nodes[x][y][z] = state
end end
end end
end end
@ -143,119 +151,119 @@ local find_mycelium_growth_targets = function(pos)
--copy and pasting is easy and nobody's going to decide whether to hire or fire me based on this --copy and pasting is easy and nobody's going to decide whether to hire or fire me based on this
--particular snippet of code so what the hell. I'll fix it later when that clever way comes to me. --particular snippet of code so what the hell. I'll fix it later when that clever way comes to me.
local valid_targets = {} local valid_targets = {}
if nodes[-1][0][0].buildable and if nodes[-1] == "buildable" and
-- test for soil to directly support new growth -- test for soil to directly support new growth
(nodes[-1][-1][0].soil or (nodes[-1 -ystride] == "soil" or
nodes[-1][1][0].soil or nodes[-1 +ystride] == "soil" or
nodes[-1][0][-1].soil or nodes[-1 -zstride] == "soil" or
nodes[-1][0][1].soil or nodes[-1 +zstride] == "soil" or
-- test for soil "around the corner" to allow for growth over an edge -- test for soil "around the corner" to allow for growth over an edge
nodes[0][-1][0].soil or nodes[-ystride] == "soil" or
nodes[0][1][0].soil or nodes[ystride] == "soil" or
nodes[0][0][-1].soil or nodes[-zstride] == "soil" or
nodes[0][0][1].soil) nodes[zstride] == "soil")
and not -- no adjacent hypha and not -- no adjacent hypha
(nodes[-1][-1][0].hypha or (nodes[-1 -ystride] == "hypha" or
nodes[-1][1][0].hypha or nodes[-1 +ystride] == "hypha" or
nodes[-1][0][-1].hypha or nodes[-1 -zstride] == "hypha" or
nodes[-1][0][1].hypha) nodes[-1 +zstride] == "hypha")
then then
table.insert(valid_targets, {x=pos.x-1, y=pos.y, z=pos.z}) table.insert(valid_targets, {x=pos_x-1, y=pos_y, z=pos_z})
end end
if nodes[1][0][0].buildable and if nodes[1] == "buildable" and
-- test for soil to directly support new growth -- test for soil to directly support new growth
(nodes[1][-1][0].soil or (nodes[1 -ystride] == "soil" or
nodes[1][1][0].soil or nodes[1 +ystride] == "soil" or
nodes[1][0][-1].soil or nodes[1 -zstride] == "soil" or
nodes[1][0][1].soil or nodes[1 +zstride] == "soil" or
-- test for soil "around the corner" to allow for growth over an edge -- test for soil "around the corner" to allow for growth over an edge
nodes[0][-1][0].soil or nodes[-ystride] == "soil" or
nodes[0][1][0].soil or nodes[ystride] == "soil" or
nodes[0][0][-1].soil or nodes[-zstride] == "soil" or
nodes[0][0][1].soil) nodes[zstride] == "soil")
and not -- no adjacent hypha and not -- no adjacent hypha
(nodes[1][-1][0].hypha or (nodes[1 -ystride] == "hypha" or
nodes[1][1][0].hypha or nodes[1 +ystride] == "hypha" or
nodes[1][0][-1].hypha or nodes[1 -zstride] == "hypha" or
nodes[1][0][1].hypha) nodes[1 +zstride] == "hypha")
then then
table.insert(valid_targets, {x=pos.x+1, y=pos.y, z=pos.z}) table.insert(valid_targets, {x=pos_x+1, y=pos_y, z=pos_z})
end end
if nodes[0][-1][0].buildable and if nodes[-ystride] == "buildable" and
-- test for soil to directly support new growth -- test for soil to directly support new growth
(nodes[-1][-1][0].soil or (nodes[-1 -ystride] == "soil" or
nodes[1][-1][0].soil or nodes[1 -ystride] == "soil" or
nodes[0][-1][-1].soil or nodes[-ystride -zstride] == "soil" or
nodes[0][-1][1].soil or nodes[-ystride +zstride] == "soil" or
-- test for soil "around the corner" to allow for growth over an edge -- test for soil "around the corner" to allow for growth over an edge
nodes[-1][0][0].soil or nodes[-1] == "soil" or
nodes[1][0][0].soil or nodes[1] == "soil" or
nodes[0][0][-1].soil or nodes[-zstride] == "soil" or
nodes[0][0][1].soil) nodes[zstride] == "soil")
and not -- no adjacent hypha and not -- no adjacent hypha
(nodes[-1][-1][0].hypha or (nodes[-1 -ystride] == "hypha" or
nodes[1][-1][0].hypha or nodes[1 -ystride] == "hypha" or
nodes[0][-1][-1].hypha or nodes[-ystride -zstride] == "hypha" or
nodes[0][-1][1].hypha) nodes[-ystride +zstride] == "hypha")
then then
table.insert(valid_targets, {x=pos.x, y=pos.y-1, z=pos.z}) table.insert(valid_targets, {x=pos_x, y=pos_y-1, z=pos_z})
end end
if nodes[0][1][0].buildable and if nodes[ystride] == "buildable" and
-- test for soil to directly support new growth -- test for soil to directly support new growth
(nodes[-1][1][0].soil or (nodes[-1 +ystride] == "soil" or
nodes[1][1][0].soil or nodes[1 +ystride] == "soil" or
nodes[0][1][-1].soil or nodes[ystride -zstride] == "soil" or
nodes[0][1][1].soil or nodes[ystride +zstride] == "soil" or
-- test for soil "around the corner" to allow for growth over an edge -- test for soil "around the corner" to allow for growth over an edge
nodes[-1][0][0].soil or nodes[-1] == "soil" or
nodes[1][0][0].soil or nodes[1] == "soil" or
nodes[0][0][-1].soil or nodes[-zstride] == "soil" or
nodes[0][0][1].soil) nodes[zstride] == "soil")
and not -- no adjacent hypha and not -- no adjacent hypha
(nodes[-1][1][0].hypha or (nodes[-1] == "hypha" or
nodes[1][1][0].hypha or nodes[1 + ystride] == "hypha" or
nodes[0][1][-1].hypha or nodes[ystride -zstride] == "hypha" or
nodes[0][1][1].hypha) nodes[ystride +zstride] == "hypha")
then then
table.insert(valid_targets, {x=pos.x, y=pos.y+1, z=pos.z}) table.insert(valid_targets, {x=pos_x, y=pos_y+1, z=pos_z})
end end
if nodes[0][0][-1].buildable and if nodes[-zstride] == "buildable" and
-- test for soil to directly support new growth -- test for soil to directly support new growth
(nodes[-1][0][-1].soil or (nodes[-1 -zstride] == "soil" or
nodes[1][0][-1].soil or nodes[1 -zstride] == "soil" or
nodes[0][-1][-1].soil or nodes[-ystride -zstride] == "soil" or
nodes[0][1][-1].soil or nodes[ystride -zstride] == "soil" or
-- test for soil "around the corner" to allow for growth over an edge -- test for soil "around the corner" to allow for growth over an edge
nodes[-1][0][0].soil or nodes[-1] == "soil" or
nodes[1][0][0].soil or nodes[1] == "soil" or
nodes[0][-1][0].soil or nodes[-ystride] == "soil" or
nodes[0][1][0].soil) nodes[ystride] == "soil")
and not -- no adjacent hypha and not -- no adjacent hypha
(nodes[-1][0][-1].hypha or (nodes[-1 -zstride] == "hypha" or
nodes[1][0][-1].hypha or nodes[1 -zstride] == "hypha" or
nodes[0][-1][-1].hypha or nodes[-ystride -zstride] == "hypha" or
nodes[0][1][-1].hypha) nodes[ystride -zstride] == "hypha")
then then
table.insert(valid_targets, {x=pos.x, y=pos.y, z=pos.z-1}) table.insert(valid_targets, {x=pos_x, y=pos_y, z=pos_z-1})
end end
if nodes[0][0][1].buildable and if nodes[zstride] == "buildable" and
-- test for soil to directly support new growth -- test for soil to directly support new growth
(nodes[-1][0][1].soil or (nodes[-1 +zstride] == "soil" or
nodes[1][0][1].soil or nodes[1 +zstride] == "soil" or
nodes[0][-1][1].soil or nodes[-ystride +zstride] == "soil" or
nodes[0][1][1].soil or nodes[ystride +zstride] == "soil" or
-- test for soil "around the corner" to allow for growth over an edge -- test for soil "around the corner" to allow for growth over an edge
nodes[-1][0][0].soil or nodes[-1] == "soil" or
nodes[1][0][0].soil or nodes[1] == "soil" or
nodes[0][-1][0].soil or nodes[-ystride] == "soil" or
nodes[0][1][0].soil) nodes[ystride] == "soil")
and not -- no adjacent hypha and not -- no adjacent hypha
(nodes[-1][0][1].hypha or (nodes[-1 +zstride] == "hypha" or
nodes[1][0][1].hypha or nodes[1 +zstride] == "hypha" or
nodes[0][-1][1].hypha or nodes[-ystride + zstride] == "hypha" or
nodes[0][1][1].hypha) nodes[ystride +zstride] == "hypha")
then then
table.insert(valid_targets, {x=pos.x, y=pos.y, z=pos.z+1}) table.insert(valid_targets, {x=pos_x, y=pos_y, z=pos_z+1})
end end
return valid_targets return valid_targets
@ -371,11 +379,12 @@ local grow_mycelium_immediately = function(pos)
local pos = table.remove(stack) local pos = table.remove(stack)
if not (df_farming and df_farming.kill_if_sunlit(pos)) then if not (df_farming and df_farming.kill_if_sunlit(pos)) then
local new_poses = grow_mycelium(pos, "df_primordial_items:giant_hypha_apical_mapgen") local new_poses = grow_mycelium(pos, "df_primordial_items:giant_hypha_apical_mapgen")
if new_poses then -- if we hit the end of the world, just stop. There'll be a mapgen meristem left here, re-trigger it. if new_poses then
for _, new_pos in ipairs(new_poses) do for _, new_pos in ipairs(new_poses) do
table.insert(stack, new_pos) table.insert(stack, new_pos)
end end
else else
-- if we hit the end of the world, just stop. There'll be a mapgen meristem left here, re-trigger it.
minetest.get_node_timer(pos):start(math.random(10,60)) minetest.get_node_timer(pos):start(math.random(10,60))
end end
end end
@ -408,3 +417,17 @@ minetest.register_node("df_primordial_items:giant_hypha_apical_mapgen", {
minetest.get_node_timer(pos):stop() minetest.get_node_timer(pos):stop()
end, end,
}) })
-- Just in case mapgen fails to trigger the timer on a mapgen mycelium this ABM will clean up.
minetest.register_abm({
label = "df_primordial_items ensure giant mycelium growth",
nodenames = {"df_primordial_items:giant_hypha_apical_mapgen"},
interval = 10.0,
chance = 1,
action = function(pos, node, active_object_count, active_object_count_wider)
local timer = minetest.get_node_timer(pos)
if not timer:is_started() then
timer:start(math.random(1,10))
end
end,
})