summaryrefslogtreecommitdiff
path: root/mesecons
diff options
context:
space:
mode:
authorChristopher Head <chead@chead.ca>2016-08-20 21:53:02 -0700
committerChristopher Head <chead@chead.ca>2016-08-24 00:42:38 -0700
commit6d79272ed4966b63e2035b19af07d13f2a0be2a3 (patch)
tree4230afdd39c312b46f6ccff1715568cb438e1e38 /mesecons
parent564cee346afcf94339472e75995394f0363a5543 (diff)
downloadmesecons-6d79272ed4966b63e2035b19af07d13f2a0be2a3.tar
mesecons-6d79272ed4966b63e2035b19af07d13f2a0be2a3.tar.gz
mesecons-6d79272ed4966b63e2035b19af07d13f2a0be2a3.tar.bz2
mesecons-6d79272ed4966b63e2035b19af07d13f2a0be2a3.tar.xz
mesecons-6d79272ed4966b63e2035b19af07d13f2a0be2a3.zip
Implement VoxelManipulator-based transactions.
Rather than calling out through the Lua-to-C API for each node that needs to be read or written, a group of map reads (and optionally writes) can be grouped into a transaction. Access to map data within the transaction is provided at high speed by means of VoxelManipulators. Once the reads and writes are finished, the transaction can be committed or aborted.
Diffstat (limited to 'mesecons')
-rw-r--r--mesecons/util.lua131
1 files changed, 123 insertions, 8 deletions
diff --git a/mesecons/util.lua b/mesecons/util.lua
index 502b269..a4960a9 100644
--- a/mesecons/util.lua
+++ b/mesecons/util.lua
@@ -236,16 +236,131 @@ local function unhash_blockpos(hash)
return vector.multiply(minetest.get_position_from_hash(hash), BLOCKSIZE)
end
--- get node and force-load area
+-- Maps from a hashed mapblock position (as returned by hash_blockpos) to a
+-- table.
+--
+-- Contents of the table are:
+-- “vm” → the VoxelManipulator
+-- “va” → the VoxelArea
+-- “data” → the data array
+-- “param1” → the param1 array
+-- “param2” → the param2 array
+-- “dirty” → true if data has been modified
+--
+-- Nil if no VM-based transaction is in progress.
+local vm_cache = nil
+
+-- Starts a VoxelManipulator-based transaction.
+--
+-- During a VM transaction, calls to vm_get_node and vm_swap_node operate on a
+-- cached copy of the world loaded via VoxelManipulators. That cache can later
+-- be committed to the real map by means of vm_commit or discarded by means of
+-- vm_abort.
+function mesecon.vm_begin()
+ vm_cache = {}
+end
+
+-- Finishes a VoxelManipulator-based transaction, freeing the VMs and map data
+-- and writing back any modified areas.
+function mesecon.vm_commit()
+ for hash, tbl in pairs(vm_cache) do
+ if tbl.dirty then
+ local vm = tbl.vm
+ vm:set_data(tbl.data)
+ vm:write_to_map()
+ vm:update_map()
+ end
+ end
+ vm_cache = nil
+end
+
+-- Finishes a VoxelManipulator-based transaction, freeing the VMs and throwing
+-- away any modified areas.
+function mesecon.vm_abort()
+ vm_cache = nil
+end
+
+-- Gets the cache entry covering a position, populating it if necessary.
+local function vm_get_or_create_entry(pos)
+ local hash = hash_blockpos(pos)
+ local tbl = vm_cache[hash]
+ if not tbl then
+ local vm = minetest.get_voxel_manip(pos, pos)
+ local min_pos, max_pos = vm:get_emerged_area()
+ local va = VoxelArea:new{MinEdge = min_pos, MaxEdge = max_pos}
+ tbl = {vm = vm, va = va, data = vm:get_data(), param1 = vm:get_light_data(), param2 = vm:get_param2_data(), dirty = false}
+ vm_cache[hash] = tbl
+ end
+ return tbl
+end
+
+-- Gets the node at a given position during a VoxelManipulator-based
+-- transaction.
+function mesecon.vm_get_node(pos)
+ local tbl = vm_get_or_create_entry(pos)
+ local index = tbl.va:indexp(pos)
+ local node_value = tbl.data[index]
+ if node_value == core.CONTENT_IGNORE then
+ return nil
+ else
+ local node_param1 = tbl.param1[index]
+ local node_param2 = tbl.param2[index]
+ return {name = minetest.get_name_from_content_id(node_value), param1 = node_param1, param2 = node_param2}
+ end
+end
+
+-- Sets a node’s name during a VoxelManipulator-based transaction.
+--
+-- Existing param1, param2, and metadata are left alone.
+function mesecon.vm_swap_node(pos, name)
+ local tbl = vm_get_or_create_entry(pos)
+ local index = tbl.va:indexp(pos)
+ tbl.data[index] = minetest.get_content_id(name)
+ tbl.dirty = true
+end
+
+-- Gets the node at a given position, regardless of whether it is loaded or
+-- not, respecting a transaction if one is in progress.
+--
+-- Outside a VM transaction, if the mapblock is not loaded, it is pulled into
+-- the server’s main map data cache and then accessed from there.
+--
+-- Inside a VM transaction, the transaction’s VM cache is used.
function mesecon.get_node_force(pos)
- local node = minetest.get_node_or_nil(pos)
- if node == nil then
- -- Node is not currently loaded; use a VoxelManipulator to prime
- -- the mapblock cache and try again.
- minetest.get_voxel_manip(pos, pos)
- node = minetest.get_node_or_nil(pos)
+ if vm_cache then
+ return mesecon.vm_get_node(pos)
+ else
+ local node = minetest.get_node_or_nil(pos)
+ if node == nil then
+ -- Node is not currently loaded; use a VoxelManipulator to prime
+ -- the mapblock cache and try again.
+ minetest.get_voxel_manip(pos, pos)
+ node = minetest.get_node_or_nil(pos)
+ end
+ return node
+ end
+end
+
+-- Swaps the node at a given position, regardless of whether it is loaded or
+-- not, respecting a transaction if one is in progress.
+--
+-- Outside a VM transaction, if the mapblock is not loaded, it is pulled into
+-- the server’s main map data cache and then accessed from there.
+--
+-- Inside a VM transaction, the transaction’s VM cache is used.
+--
+-- This function can only be used to change the node’s name, not its parameters
+-- or metadata.
+function mesecon.swap_node_force(pos, name)
+ if vm_cache then
+ return mesecon.vm_swap_node(pos, name)
+ else
+ -- This serves to both ensure the mapblock is loaded and also hand us
+ -- the old node table so we can preserve param2.
+ local node = mesecon.get_node_force(pos)
+ node.name = name
+ minetest.swap_node(pos, node)
end
- return node
end
-- Un-forceload any forceloaded mapblocks from older versions of Mesecons which