Compress node ids and indices

This commit is contained in:
Hybrid Dog 2017-11-12 20:32:57 +01:00
parent 3ff134c6ca
commit 34b39ec4e7
3 changed files with 59 additions and 15 deletions

@ -12,7 +12,6 @@ Ignored chatcommands:
TODO: TODO:
* Compress nodeids and indices (string.byte, nodeid bits)
* Implement more commands * Implement more commands
* Add mod load time info * Add mod load time info
* Add settingtypes.txt for the settings * Add settingtypes.txt for the settings

@ -1 +1,2 @@
worldedit
worldedit_commands worldedit_commands

@ -266,6 +266,7 @@ undo_info_funcs.marker = function(data)
end end
-- override the worldedit vmanip finish function to catch the data table
local we_data = false local we_data = false
local we_manip_end = worldedit.manip_helpers.finish local we_manip_end = worldedit.manip_helpers.finish
function worldedit.manip_helpers.finish(manip, data) function worldedit.manip_helpers.finish(manip, data)
@ -275,6 +276,46 @@ function worldedit.manip_helpers.finish(manip, data)
return we_manip_end(manip, data) return we_manip_end(manip, data)
end end
local function compress_nodedata(indices, nodeids)
-- indices contain 3 * 16 = 48 bit values
-- nodeids contain 16 bit values (see mapnode.h)
-- big endian here
local data = {}
for i = 1,#indices do
local v = ""
for f = 5,0,-1 do
v = v .. string.char(math.floor(indices[i] * 2^(-8*f)) % 0x100)
data[i] = v
end
end
for i = 1,#nodeids do
data[#data+1] = string.char(math.floor(nodeids[i] * 2^-8)) ..
string.char(nodeids[i] % 0x100)
end
return minetest.compress(table.concat(data))
end
local function decompress_nodedata(compressed_data, count)
local indices = {}
local nodeids = {}
local data = minetest.decompress(compressed_data)
assert(#data == count * (6 + 2), "invalid decompressed data")
local p = 1
for i = 1,count do
local v = 0
for f = 5,0,-1 do
v = v + 2^(8*f) * data:byte(p)
p = p+1
end
indices[i] = v
end
for i = 1,count do
nodeids[i] = data:byte(p) * 0x100 + data:byte(p+1)
p = p + 2
end
return indices, nodeids
end
local we_set = worldedit.set local we_set = worldedit.set
local my_we_set = function(pos1, pos2, ...) local my_we_set = function(pos1, pos2, ...)
assert(command_invoker, "Player not known") assert(command_invoker, "Player not known")
@ -310,14 +351,16 @@ local my_we_set = function(pos1, pos2, ...)
end end
we_data = false we_data = false
local compressed_data = compress_nodedata(indices, nodeids)
add_to_history({ add_to_history({
type = "nodeids", type = "nodeids",
mem_use = 9 * (2 * 7 + 2 * #nodeids + 2), mem_use = 9 * (2 * 7 + #compressed_data),
pos1 = pos1, pos1 = pos1,
pos2 = pos2, pos2 = pos2,
nodeids = nodeids, count = #nodeids,
indices = indices, compressed_data = compressed_data
}, command_invoker) }, command_invoker)
-- Note: param1, param2 and metadata are not changed by worldedit.set
return rv return rv
end end
@ -336,6 +379,9 @@ undo_funcs.nodeids = function(name, data)
local ylen = pos2.y - pos1.y + 1 local ylen = pos2.y - pos1.y + 1
local ystride = pos2.x - pos1.x + 1 local ystride = pos2.x - pos1.x + 1
local indices, nodeids = decompress_nodedata(data.compressed_data,
data.count)
local manip = minetest.get_voxel_manip() local manip = minetest.get_voxel_manip()
local e1, e2 = manip:read_from_map(pos1, pos2) local e1, e2 = manip:read_from_map(pos1, pos2)
local area = VoxelArea:new{MinEdge=e1, MaxEdge=e2} local area = VoxelArea:new{MinEdge=e1, MaxEdge=e2}
@ -343,27 +389,25 @@ undo_funcs.nodeids = function(name, data)
-- swap the nodes in the world and history data -- swap the nodes in the world and history data
local new_nodeids = {} local new_nodeids = {}
for k = 1,#data.indices do for k = 1,#indices do
local i = data.indices[k] local i = indices[k]
local x = i % ystride local x = i % ystride
local y = math.floor(i / ystride) % ylen local y = math.floor(i / ystride) % ylen
local z = math.floor(i / (ystride * ylen)) local z = math.floor(i / (ystride * ylen))
local vi = area:index(pos1.x + x, pos1.y + y, pos1.z + z) local vi = area:index(pos1.x + x, pos1.y + y, pos1.z + z)
new_nodeids[k] = mdata[vi] new_nodeids[k] = mdata[vi]
mdata[vi] = data.nodeids[k] mdata[vi] = nodeids[k]
end end
manip:set_data(mdata) manip:set_data(mdata)
manip:write_to_map() manip:write_to_map()
data.nodeids = new_nodeids data.compressed_data = compress_nodedata(indices, new_nodeids)
worldedit.player_notify(name, #data.indices .. " nodes set") worldedit.player_notify(name, data.count .. " nodes set")
end end
undo_info_funcs.marker = function(data) undo_info_funcs.nodeids = function(data)
if not data.pos then return "pos1: " .. minetest.pos_to_string(data.pos1) .. ", pos2: " ..
return "Set pos" .. data.id minetest.pos_to_string(data.pos2) .. ", " .. data.count ..
end " nodes changed"
return "changed pos" .. data.id .. ", previous value: " ..
minetest.pos_to_string(data.pos)
end end