fef32435ed
Now the 26 blocks around an inside block are emerged before the inside is constructed. This prevents set_lighting() calls from mapgen of the neighboring chunks from overwriting port param1 values. This problem is usually avoided before it comes up now because I shifter the x and z base positions to chunk centers. Even before this commit, the problem only occurred for certain Lua mapgens such as mapgen_rivers. I also corrected the mapgen_limit assertions.
263 lines
9.4 KiB
Lua
263 lines
9.4 KiB
Lua
--[[
|
|
Copyright (C) 2021 Jude Melton-Houghton
|
|
|
|
This file is part of area_containers. It implements the allocation of
|
|
unused mapblocks in which to place the inner chambers of area containers.
|
|
|
|
area_containers is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published
|
|
by the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
area_containers is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with area_containers. If not, see <https://www.gnu.org/licenses/>.
|
|
]]
|
|
|
|
--[[
|
|
OVERVIEW
|
|
|
|
There must be a mapping between the insides of containers and the container
|
|
nodes themselves. A "relation" encodes an inside position and a container
|
|
position. The inside position is the minimum coordinate of the inside chamber.
|
|
Chambers are block-aligned, and are pretty much assumed to fill exactly one
|
|
block. The container position is stored in the metadata at the inside
|
|
position, and may or may not be set. A relation consists of param1 and param2
|
|
values that can be stored in a node. Relations are allocated and freed using
|
|
functions in this file. An allocated/freed parameter pair must never have both
|
|
parameters be 0, but all possible parameter combinations are considered valid
|
|
relations.
|
|
]]
|
|
|
|
local use = ...
|
|
local storage, blockpos_in_range = use("misc", {"storage", "blockpos_in_range"})
|
|
local Y_LEVEL_BLOCKS, MAX_CONTAINER_CACHE_SIZE = use("settings", {
|
|
"Y_LEVEL_BLOCKS", "MAX_CACHE_SIZE",
|
|
})
|
|
|
|
local exports = {}
|
|
|
|
-- Persistent settings/state (some state isn't declared here) --
|
|
|
|
-- The period between insides measured in node lengths.
|
|
local INSIDE_SPACING = tonumber(storage:get("INSIDE_SPACING") or 240)
|
|
-- The y value of all inside positions.
|
|
local Y_LEVEL = tonumber(storage:get("Y_LEVEL") or (16 * Y_LEVEL_BLOCKS))
|
|
-- The minimum x and z values of all inside positions.
|
|
local X_BASE = tonumber(storage:get("X_BASE") or -30640)
|
|
local Z_BASE = tonumber(storage:get("Z_BASE") or -30640)
|
|
-- The next param values to be allocated if no other free spaces are available.
|
|
local param1_next = tonumber(storage:get("param1_next") or 1)
|
|
local param2_next = tonumber(storage:get("param2_next") or 0)
|
|
|
|
-- Check that the positioning settings are within bounds:
|
|
assert(blockpos_in_range(vector.new(X_BASE / 16, 0, Z_BASE / 16)),
|
|
"The area_containers minimum position is outside the mapgen_limit")
|
|
assert(
|
|
blockpos_in_range(
|
|
vector.new(
|
|
(X_BASE + INSIDE_SPACING * 255) / 16, 0,
|
|
(Z_BASE + INSIDE_SPACING * 255) / 16)),
|
|
"The area_containers maximum position is outside the mapgen_limit")
|
|
assert(blockpos_in_range(vector.new(0, Y_LEVEL / 16, 0)),
|
|
"The area_containers Y-level is outside the mapgen_limit")
|
|
|
|
-- Persist, any newly created values:
|
|
storage:set_int("INSIDE_SPACING", INSIDE_SPACING)
|
|
storage:set_int("Y_LEVEL", Y_LEVEL)
|
|
storage:set_int("X_BASE", X_BASE)
|
|
storage:set_int("Z_BASE", Z_BASE)
|
|
storage:set_int("param1_next", param1_next)
|
|
storage:set_int("param2_next", param2_next)
|
|
|
|
-- Container position caching --
|
|
|
|
-- The cache of container positions (indexed by get_params_index.)
|
|
local cached_containers = {}
|
|
-- The number of entries.
|
|
local container_cache_size = 0
|
|
|
|
-- Cache the container position to associate with the given parameter index.
|
|
local function cache_container(index, pos)
|
|
local old_value = cached_containers[index]
|
|
if pos and not old_value then
|
|
-- Add.
|
|
if container_cache_size >= MAX_CONTAINER_CACHE_SIZE then
|
|
-- Just purge everything and start again.
|
|
cached_containers = {}
|
|
container_cache_size = 0
|
|
end
|
|
container_cache_size = container_cache_size + 1
|
|
elseif old_value and not pos then
|
|
-- Remove.
|
|
container_cache_size = container_cache_size - 1
|
|
end
|
|
cached_containers[index] = pos
|
|
end
|
|
|
|
-- Parameter Interpretation --
|
|
|
|
-- Returns a numeric index unique to the parameter pair.
|
|
local function get_params_index(param1, param2)
|
|
return param1 + param2 * 256
|
|
end
|
|
exports.get_params_index = get_params_index
|
|
|
|
-- Returns the related inside position (the minimum coordinate of the chamber.)
|
|
local function get_related_inside(param1, param2)
|
|
return vector.new(
|
|
X_BASE + param1 * INSIDE_SPACING,
|
|
Y_LEVEL,
|
|
Z_BASE + param2 * INSIDE_SPACING
|
|
)
|
|
end
|
|
exports.get_related_inside = get_related_inside
|
|
|
|
-- Returns the two params associated with the position if it is a position that
|
|
-- could be returned from get_related_inside, or two nil values otherwise.
|
|
function exports.get_params_from_inside(inside_pos)
|
|
if inside_pos.y ~= Y_LEVEL then return nil, nil end
|
|
local param1 = (inside_pos.x - X_BASE) / INSIDE_SPACING
|
|
local param2 = (inside_pos.z - Z_BASE) / INSIDE_SPACING
|
|
if param1 >= 0 and param1 <= 255 and param2 >= 0 and param2 <= 255 and
|
|
param1 % 1 == 0 and param2 % 1 == 0 then
|
|
return param1, param2
|
|
end
|
|
return nil, nil
|
|
end
|
|
|
|
-- The actual Y-level (in nodes) of all inside positions (container bottoms.)
|
|
exports.INSIDE_Y_LEVEL = Y_LEVEL
|
|
|
|
-- Gets the related container position. Returns nil if it isn't set.
|
|
function exports.get_related_container(param1, param2)
|
|
local idx = get_params_index(param1, param2)
|
|
local container_pos = cached_containers[idx]
|
|
if not container_pos then
|
|
local inside_pos = get_related_inside(param1, param2)
|
|
local inside_meta = minetest.get_meta(inside_pos)
|
|
container_pos = minetest.string_to_pos(
|
|
inside_meta:get_string("area_containers:container_pos"))
|
|
cache_container(idx, container_pos)
|
|
end
|
|
return container_pos
|
|
end
|
|
|
|
-- Sets the related container, or unsets it if container_pos is nil.
|
|
function exports.set_related_container(param1, param2, container_pos)
|
|
local inside_pos = get_related_inside(param1, param2)
|
|
local inside_meta = minetest.get_meta(inside_pos)
|
|
inside_meta:set_string("area_containers:container_pos",
|
|
container_pos and minetest.pos_to_string(container_pos) or "")
|
|
cache_container(get_params_index(param1, param2), container_pos)
|
|
end
|
|
|
|
-- Parameter String Encoding and Decoding --
|
|
|
|
--[[
|
|
The storage key "freed" is a string representation of a stack. It consists of
|
|
concatenated fixed-length records representing parameter pairs. Each record is
|
|
PARAMS_STRING_LENGTH characters long.
|
|
]]
|
|
|
|
-- Set up the bi-directional mapping between characters and segments of 6 bits:
|
|
local SEG2BYTE = {string.byte(
|
|
"123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ+-",
|
|
1, -1
|
|
)}
|
|
assert(#SEG2BYTE == 63)
|
|
SEG2BYTE[0] = string.byte("0") -- The indices must be [0-63]
|
|
local BYTE2SEG = {}
|
|
for i = 0, 63 do
|
|
BYTE2SEG[SEG2BYTE[i]] = i
|
|
end
|
|
|
|
local PARAMS_STRING_LENGTH = 3
|
|
|
|
local function params_to_string(param1, param2)
|
|
local seg1 = param1 % 64
|
|
local seg2 = param2 % 64
|
|
local seg3 = math.floor(param1 / 64) + math.floor(param2 / 64) * 4
|
|
return string.char(SEG2BYTE[seg1], SEG2BYTE[seg2], SEG2BYTE[seg3])
|
|
end
|
|
|
|
local function string_to_params(str)
|
|
local byte1, byte2, byte3 = string.byte(str, 1, 3)
|
|
local seg1 = BYTE2SEG[byte1]
|
|
local seg2 = BYTE2SEG[byte2]
|
|
local seg3 = BYTE2SEG[byte3]
|
|
local param1 = seg1 + (seg3 % 4) * 64
|
|
local param2 = seg2 + math.floor(seg3 / 4) * 64
|
|
return param1, param2
|
|
end
|
|
|
|
-- Allocation and Deallocation (with No Map Alteration) --
|
|
|
|
-- Returns a newly allocated param1, param2. Returns nil, nil if there is no
|
|
-- space left.
|
|
function exports.alloc_relation()
|
|
local param1, param2
|
|
local freed = storage:get_string("freed")
|
|
if #freed >= PARAMS_STRING_LENGTH then
|
|
-- Pop a space off the freed stack if one is available.
|
|
param1, param2 = string_to_params(
|
|
string.sub(freed, -PARAMS_STRING_LENGTH))
|
|
freed = string.sub(freed, 1, #freed - PARAMS_STRING_LENGTH)
|
|
storage:set_string("freed", freed)
|
|
elseif param2_next < 256 then
|
|
-- Add a new space to the pool if no space is available.
|
|
param1 = param1_next
|
|
param2 = param2_next
|
|
param1_next = param1_next + 1
|
|
if param1_next >= 256 then
|
|
-- Wrap around.
|
|
param1_next = 0
|
|
param2_next = param2_next + 1
|
|
end
|
|
storage:set_int("param1_next", param1_next)
|
|
storage:set_int("param2_next", param2_next)
|
|
end
|
|
return param1, param2
|
|
end
|
|
|
|
-- Adds the relation to the freed list to be reused later.
|
|
function exports.free_relation(param1, param2)
|
|
-- Push the params:
|
|
local freed = storage:get_string("freed")
|
|
freed = freed .. params_to_string(param1, param2)
|
|
storage:set_string("freed", freed)
|
|
end
|
|
|
|
-- Tries to reclaim the specific relation from the freed list. Returned is
|
|
-- whether the relation could be reclaimed and removed from the freed list.
|
|
function exports.reclaim_relation(param1, param2)
|
|
local find_params = params_to_string(param1, param2)
|
|
local freed = storage:get_string("freed")
|
|
-- A special case for when the reclaimed is the most recently freed:
|
|
if string.sub(freed, -PARAMS_STRING_LENGTH) == find_params then
|
|
freed = string.sub(freed, 1, #freed - PARAMS_STRING_LENGTH)
|
|
storage:set_string("freed", freed)
|
|
return true
|
|
end
|
|
-- Search through all the other records backward (more recent first):
|
|
for i = 1, #freed - PARAMS_STRING_LENGTH + 1, PARAMS_STRING_LENGTH do
|
|
local start = -i - PARAMS_STRING_LENGTH + 1
|
|
local finish = -i
|
|
local check_params = string.sub(freed, start, finish)
|
|
if check_params == find_params then
|
|
-- Found!
|
|
freed = string.sub(freed, 1, start - 1) ..
|
|
string.sub(freed, finish + 1)
|
|
storage:set_string("freed", freed)
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
return exports
|