521 lines
15 KiB
Lua
Raw Normal View History

-- Buildat: builtin/voxelworld/client_lua/module.lua
-- http://www.apache.org/licenses/LICENSE-2.0
-- Copyright 2014 Perttu Ahola <celeron55@gmail.com>
local log = buildat.Logger("voxelworld")
2014-10-12 10:11:36 +03:00
local magic = require("buildat/extension/urho3d")
local replicate = require("buildat/extension/replicate")
local cereal = require("buildat/extension/cereal")
local dump = buildat.dump
local M = {}
local UPDATE_TIME_FRACTION = 0.10
2014-10-14 14:23:55 +03:00
--local LOD_DISTANCE = 140
--local LOD_DISTANCE = 100
local LOD_DISTANCE = 80
2014-10-14 14:23:55 +03:00
--local LOD_DISTANCE = 50
local LOD_THRESHOLD = 0.2
local PHYSICS_DISTANCE = 100
local INITIAL_PHYSICS_NEAR_WEIGHT = 1.0
2014-10-21 01:47:51 +03:00
local MODIFIED_PHYSICS_NEAR_WEIGHT = 1.0
local INITIAL_GEOMETRY_NEAR_WEIGHT = 0.15
2014-10-21 01:47:51 +03:00
local MODIFIED_GEOMETRY_NEAR_WEIGHT = 1.0
2014-10-15 04:14:20 +03:00
local MAX_LOD = 4
2014-10-12 10:11:36 +03:00
local camera_node = nil
2014-10-12 11:26:33 +03:00
local update_counter = -1
2014-10-14 14:23:55 +03:00
local camera_p = magic.Vector3(0, 0, 0)
local camera_dir = magic.Vector3(0, 0, 0)
2014-10-14 14:23:55 +03:00
local camera_last_p = camera_p
local camera_last_dir = camera_dir
2014-10-14 12:23:51 +03:00
local end_of_update_processing_us = 0
2014-10-12 11:26:33 +03:00
2014-10-19 01:32:26 +03:00
local voxel_reg = buildat.createVoxelRegistry()
2014-10-19 14:40:43 +03:00
local atlas_reg = buildat.createAtlasRegistry()
2014-10-12 11:26:33 +03:00
M.chunk_size_voxels = nil
M.section_size_chunks = nil
M.section_size_voxels = nil
2014-10-20 11:12:30 +03:00
-- Start higher than any conceivable value because otherwise things will never
-- be triggered to reflect this value
M.camera_far_clip = 1000
2014-10-12 10:11:36 +03:00
-- voxelworld is ready once the init and voxel_registry packets (and in the
-- future possibly some other ones) have been received)
local is_ready = false
local on_ready_callbacks = {}
local geometry_update_cbs = {} -- function(node)
-- TODO: Implement unload by timeout
local node_volume_cache = {} -- {node_id: {volume:, last_access_us:}}
-- NOTE: node can be nil, meaning that it was cached to be nil
local static_node_cache = {} -- {z: {y: {x: {node:, fetched:}}}} (chunk_p)
function on_ready()
if is_ready then
error("on_ready(): already ready")
end
is_ready = true
for _, v in ipairs(on_ready_callbacks) do
v()
end
on_ready_callbacks = nil
end
function M.init()
log:info("voxelworld.init()")
2014-10-12 10:11:36 +03:00
local node_update_queue = buildat.SpatialUpdateQueue()
local function queue_initial_node_update(node)
node_update_queue:put(node:GetWorldPosition(),
2014-10-20 11:12:30 +03:00
INITIAL_GEOMETRY_NEAR_WEIGHT, M.camera_far_clip * 1.2,
nil, nil, {
type = "geometry",
current_lod = 0,
node_id = node:GetID(),
})
node_update_queue:put(node:GetWorldPosition(),
INITIAL_PHYSICS_NEAR_WEIGHT, PHYSICS_DISTANCE, nil, nil, {
type = "physics",
node_id = node:GetID(),
})
end
2014-10-21 01:47:51 +03:00
local function queue_modified_node_update(node)
node_update_queue:put(node:GetWorldPosition(),
MODIFIED_GEOMETRY_NEAR_WEIGHT, M.camera_far_clip * 1.2,
nil, nil, {
type = "geometry",
current_lod = 0,
node_id = node:GetID(),
})
node_update_queue:put(node:GetWorldPosition(),
MODIFIED_PHYSICS_NEAR_WEIGHT, PHYSICS_DISTANCE, nil, nil, {
type = "physics",
node_id = node:GetID(),
})
end
2014-10-12 10:11:36 +03:00
buildat.sub_packet("voxelworld:init", function(data)
local values = cereal.binary_input(data, {"object",
{"chunk_size_voxels", {"object",
2014-10-20 14:05:58 +03:00
{"x", "int16_t"},
{"y", "int16_t"},
{"z", "int16_t"},
2014-10-12 10:11:36 +03:00
}},
{"section_size_chunks", {"object",
2014-10-20 14:05:58 +03:00
{"x", "int16_t"},
{"y", "int16_t"},
{"z", "int16_t"},
2014-10-12 10:11:36 +03:00
}},
})
log:info("voxelworld:init: "..dump(values))
2014-10-14 12:23:51 +03:00
M.chunk_size_voxels = buildat.Vector3(values.chunk_size_voxels)
M.section_size_chunks = buildat.Vector3(values.section_size_chunks)
2014-10-12 11:26:33 +03:00
M.section_size_voxels =
M.chunk_size_voxels:mul_components(M.section_size_chunks)
2014-10-12 10:11:36 +03:00
end)
buildat.sub_packet("voxelworld:voxel_registry", function(data)
voxel_reg:deserialize(data)
on_ready() -- This is the last packet
end)
buildat.sub_packet("voxelworld:node_voxel_data_updated", function(data)
local values = cereal.binary_input(data, {"object",
{"node_id", "int32_t"},
})
log:info("voxelworld:node_voxel_data_updated: "..dump(values))
node_volume_cache[values.node_id] = nil -- Clear cache
local node = replicate.main_scene:GetNode(values.node_id)
2014-10-21 01:47:51 +03:00
queue_modified_node_update(node)
end)
2014-10-14 14:23:55 +03:00
local function update_voxel_geometry(node)
local data = node:GetVar("buildat_voxel_data"):GetBuffer()
2014-10-14 14:23:55 +03:00
local node_p = node:GetWorldPosition()
local d = (node_p - camera_p):Length()
local lod_fraction = d / LOD_DISTANCE
local lod = math.floor(1 + lod_fraction)
2014-10-15 04:14:20 +03:00
if lod > MAX_LOD then lod = MAX_LOD end
2014-10-14 14:23:55 +03:00
2014-10-20 17:15:32 +03:00
log:debug("update_voxel_geometry(): lod="..lod..
", d="..math.floor(d)..", #data="..data:GetSize()..
", node="..node:GetID())
2014-10-14 14:23:55 +03:00
2014-10-20 11:12:30 +03:00
for _, cb in ipairs(geometry_update_cbs) do
cb(node)
2014-10-19 23:39:35 +03:00
end
2014-10-14 14:23:55 +03:00
local near_trigger_d = nil
local near_weight = nil
local far_trigger_d = nil
local far_weight = nil
2014-10-20 11:12:30 +03:00
if d >= M.camera_far_clip * 1.4 then
2014-10-20 17:15:32 +03:00
log:debug("Clearing voxel geometry outside camera far clip ("
2014-10-20 11:12:30 +03:00
..M.camera_far_clip..")")
buildat.clear_voxel_geometry(node)
-- clip out -> 4
2014-10-20 11:12:30 +03:00
near_trigger_d = 1.2 * M.camera_far_clip
near_weight = 0.4
elseif lod == 1 then
buildat.set_voxel_geometry(
node, data, voxel_reg, atlas_reg)
2014-10-14 14:23:55 +03:00
-- 1 -> 2
far_trigger_d = LOD_DISTANCE * (1.0 + LOD_THRESHOLD)
far_weight = 0.5
2014-10-14 14:23:55 +03:00
else
buildat.set_voxel_lod_geometry(lod, node, data, voxel_reg, atlas_reg)
2014-10-14 14:23:55 +03:00
if lod == 1 then
-- Shouldn't go here
elseif lod == 2 then
-- 2 -> 1
near_trigger_d = LOD_DISTANCE * (1.0 - LOD_THRESHOLD)
near_weight = 0.75
2014-10-14 14:23:55 +03:00
-- 2 -> 3
far_trigger_d = 2 * LOD_DISTANCE * (1.0 + LOD_THRESHOLD)
far_weight = 0.4
2014-10-14 14:23:55 +03:00
elseif lod == 3 then
-- 3 -> 2
near_trigger_d = 2 * LOD_DISTANCE * (1.0 - LOD_THRESHOLD)
near_weight = 0.6
2014-10-15 04:14:20 +03:00
-- 3 -> 4
far_trigger_d = 3 * LOD_DISTANCE * (1.0 + LOD_THRESHOLD)
far_weight = 0.4
2014-10-15 04:14:20 +03:00
elseif lod == 4 then
-- 4 -> 3
near_trigger_d = 3 * LOD_DISTANCE * (1.0 - LOD_THRESHOLD)
near_weight = 0.5
-- 4 -> clip out
2014-10-20 11:12:30 +03:00
far_trigger_d = M.camera_far_clip * 1.4
far_weight = 0.4
2014-10-14 14:23:55 +03:00
end
end
node_update_queue:put(node_p,
near_weight, near_trigger_d,
far_weight, far_trigger_d, {
type = "geometry",
current_lod = lod,
node_id = node:GetID(),
})
end
local function update_voxel_physics(node)
local data = node:GetVar("buildat_voxel_data"):GetBuffer()
2014-10-14 14:23:55 +03:00
local node_p = node:GetWorldPosition()
local d = (node_p - camera_p):Length()
2014-10-20 17:15:32 +03:00
log:debug("update_voxel_physics(): d="..math.floor(d)..
", #data="..data:GetSize()..", node="..node:GetID())
local near_trigger_d = nil
local near_weight = nil
local far_trigger_d = nil
local far_weight = nil
if d > PHYSICS_DISTANCE then
2014-10-20 17:15:32 +03:00
log:debug("Clearing physics boxes outside physics distance ("..
PHYSICS_DISTANCE..")")
buildat.clear_voxel_physics_boxes(node)
2014-10-14 14:23:55 +03:00
near_trigger_d = PHYSICS_DISTANCE * 0.8
near_weight = 1.0
else
buildat.set_voxel_physics_boxes(node, data, voxel_reg)
far_trigger_d = PHYSICS_DISTANCE * 1.2
far_weight = 0.2
end
node_update_queue:put(node_p,
near_weight, near_trigger_d,
far_weight, far_trigger_d, {
type = "physics",
node_id = node:GetID(),
})
end
2014-10-12 10:11:36 +03:00
magic.SubscribeToEvent("Update", function(event_type, event_data)
buildat.profiler_block_begin("Buildat|voxelworld:update")
-- Required for handling device resets (eg. fullscreen toggle)
atlas_reg:update()
2014-10-14 12:23:51 +03:00
--local t0 = buildat.get_time_us()
--local dt = event_data:GetFloat("TimeStep")
update_counter = update_counter + 1
2014-10-14 14:23:55 +03:00
if camera_node then
camera_dir = camera_node.direction
camera_p = camera_node:GetWorldPosition()
2014-10-20 11:12:30 +03:00
M.camera_far_clip = camera_node:GetComponent("Camera").farClip
2014-10-14 14:23:55 +03:00
end
2014-10-12 13:28:24 +03:00
if camera_node and M.section_size_voxels then
-- TODO: How should position information be sent to the server?
2014-10-14 14:23:55 +03:00
local p = camera_p
2014-10-12 10:11:36 +03:00
if update_counter % 60 == 0 then
2014-10-14 12:23:51 +03:00
local section_p = buildat.Vector3(p):div_components(
2014-10-12 11:26:33 +03:00
M.section_size_voxels):floor()
2014-10-12 13:28:24 +03:00
--log:info("p: "..p.x..", "..p.y..", "..p.z.." -> section_p: "..
-- section_p.x..", "..section_p.y..", "..section_p.z)
2014-10-14 12:23:51 +03:00
--[[send_get_section(section_p + buildat.Vector3( 0, 0, 0))
send_get_section(section_p + buildat.Vector3(-1, 0, 0))
send_get_section(section_p + buildat.Vector3( 1, 0, 0))
send_get_section(section_p + buildat.Vector3( 0, 1, 0))
send_get_section(section_p + buildat.Vector3( 0,-1, 0))
send_get_section(section_p + buildat.Vector3( 0, 0, 1))
send_get_section(section_p + buildat.Vector3( 0, 0,-1))]]
2014-10-12 10:11:36 +03:00
end
end
2014-10-14 12:23:51 +03:00
-- Node updates: Handle one or a few per frame
local current_us = buildat.get_time_us()
-- Spend time doing this proportionate to the rest of the update cycle
local last_outer_frame_us = (current_us - end_of_update_processing_us)
local max_handling_time_us = last_outer_frame_us * UPDATE_TIME_FRACTION
2014-10-14 12:23:51 +03:00
local stop_at_us = current_us + max_handling_time_us
node_update_queue:set_p(camera_p)
2014-10-14 14:23:55 +03:00
-- Scale queue update operations according to the handling time of the
-- rest of the processing
node_update_queue:update(max_handling_time_us / 50 + 1)
2014-10-14 12:23:51 +03:00
for i = 1, 10 do -- Usually there is time only for a few
2014-10-14 14:23:55 +03:00
local f = node_update_queue:peek_next_f()
local fw = node_update_queue:peek_next_fw()
local did_update = false
if f and f <= 1.0 then
local node_update = node_update_queue:get()
local node = replicate.main_scene:GetNode(node_update.node_id)
2014-10-20 17:15:32 +03:00
log:debug("Node update #"..
node_update_queue:get_length()..
2014-10-14 14:23:55 +03:00
" (f="..(math.floor(f*100)/100)..""..
", fw="..(math.floor(fw*100)/100)..")"..
": "..node:GetName())
if node_update.type == "geometry" then
update_voxel_geometry(node)
end
if node_update.type == "physics" then
update_voxel_physics(node)
2014-10-14 12:23:51 +03:00
end
2014-10-14 14:23:55 +03:00
did_update = true
else
2014-10-20 17:15:32 +03:00
log:debug("Poked update #"..
node_update_queue:get_length()..
2014-10-14 14:23:55 +03:00
" (f="..(math.floor((f or -1)*100)/100)..""..
", fw="..(math.floor((fw or -1)*100)/100)..")")
--node_update_queue.queue = {} -- For testing
end
2014-10-14 12:23:51 +03:00
-- Check this at the end of the loop so at least one is handled
2014-10-14 14:23:55 +03:00
if not did_update or buildat.get_time_us() >= stop_at_us then
log:debug("Handled "..i.." node updates")
2014-10-14 12:23:51 +03:00
break
end
end
2014-10-14 12:23:51 +03:00
end_of_update_processing_us = buildat.get_time_us()
if camera_node then
2014-10-13 01:32:58 +03:00
camera_last_dir = camera_dir
camera_last_p = camera_p
end
2014-10-14 12:23:51 +03:00
--log:info(buildat.get_time_us()-t0)
buildat.profiler_block_end()
2014-10-12 10:11:36 +03:00
end)
2014-10-12 11:26:33 +03:00
replicate.sub_sync_node_added({}, function(node)
if not node:GetVar("buildat_voxel_data"):IsEmpty() then
queue_initial_node_update(node)
2014-10-12 11:26:33 +03:00
end
2014-10-14 12:23:51 +03:00
--local name = node:GetName()
2014-10-12 11:26:33 +03:00
end)
2014-10-12 10:11:36 +03:00
end
function M.set_camera(new_camera_node)
camera_node = new_camera_node
end
function M.get_voxel_registry()
return voxel_reg
end
function M.get_atlas_registry()
return atlas_reg
end
function M.sub_ready(cb)
if is_ready then
cb()
else
table.insert(on_ready_callbacks, cb)
end
end
2014-10-20 11:12:30 +03:00
function M.sub_geometry_update(cb)
table.insert(geometry_update_cbs, cb)
end
function M.get_chunk_position(voxel_p)
local p = buildat.Vector3(voxel_p):round()
2014-10-21 12:57:57 +03:00
local chunk_p = p:div_components(M.chunk_size_voxels):floor()
local in_chunk_p = p - M.chunk_size_voxels:mul_components(chunk_p)
return chunk_p, in_chunk_p
end
function M.get_static_node_cache(chunk_p)
local ztable = static_node_cache[chunk_p.z]
if not ztable then
ztable = {}
static_node_cache[chunk_p.z] = ztable
end
local ytable = ztable[chunk_p.y]
if not ytable then
ytable = {}
ztable[chunk_p.y] = ytable
end
local xtable = ytable[chunk_p.x]
if not xtable then
xtable = {fetched=false}
xtable[chunk_p.x] = xtable
end
return xtable
end
function M.get_static_node(chunk_p)
local cache = M.get_static_node_cache(chunk_p)
if cache and cache.fetched then
log:debug("get_static_node(): chunk_p="..chunk_p:dump().." (cache)")
-- NOTE: cache.node can be nil, meaning that it was cached to be nil
return cache.node
end
log:debug("get_static_node(): chunk_p="..chunk_p:dump().." (no cache)")
2014-10-21 12:57:57 +03:00
-- NOTE: Chunks are positioned by their center position, and chunks are
-- aligned to a grid offset by half their size from global origin
local chunk_center_p = chunk_p:mul_components(M.chunk_size_voxels) +
M.chunk_size_voxels / 2
log:debug("get_static_node(): chunk_center_p="..chunk_center_p:dump())
2014-10-21 12:57:57 +03:00
-- Find the static node
local scene = replicate.main_scene
local octree = scene:GetComponent("Octree")
local clearance = magic.Vector3(1,1,1)
local find_lc = chunk_center_p - clearance
local find_uc = chunk_center_p + clearance
log:debug("get_static_node(): find_lc="..find_lc:dump())
log:debug("get_static_node(): find_uc="..find_uc:dump())
2014-10-21 12:57:57 +03:00
local result = octree:GetDrawables(magic.BoundingBox(
magic.Vector3.from_buildat(find_lc),
magic.Vector3.from_buildat(find_uc)))
log:debug("get_static_node(): result="..dump(result))
2014-10-21 12:57:57 +03:00
-- NOTE: The result will contain all kinds of global things like zones and
-- lights
-- NOTE: Static nodes can be distinguished by the user variable
-- buildat_static=true
local node = nil
for _, v in ipairs(result) do
--log:debug("get_static_node(): result node:GetName()="..v.node:GetName())
2014-10-21 12:57:57 +03:00
local buildat_static_var = v.node:GetVar("buildat_static")
if buildat_static_var:GetBool() == true then
node = v.node
break
end
end
cache.node = node
cache.fetched = true
2014-10-21 12:57:57 +03:00
if node == nil then
log:debug("get_static_node(): static node "..chunk_p:dump()..
" not found")
2014-10-21 12:57:57 +03:00
for _, v in ipairs(result) do
log:debug("get_static_node(): * not "..v.node:GetName())
2014-10-21 12:57:57 +03:00
end
end
return node
end
function M.get_volume(node)
local node_id = node:GetID()
local cache = node_volume_cache[node_id]
if not cache then
log:verbose("Fetching node "..node_id.." volume into cache")
local data = node:GetVar("buildat_voxel_data"):GetBuffer()
if data == nil then
log_w(MODULE, "get_volume(): Node "..node_id..
" does not contain buildat_voxel_data")
return nil
end
cache = {
volume = buildat.deserialize_volume(data),
last_access_us = buildat.get_time_us(),
}
node_volume_cache[node_id] = cache
2014-10-21 12:57:57 +03:00
end
return cache.volume
end
-- Return value: VoxelInstance (found), VoxelInstance(0) (not found)
function M.get_static_voxel(p)
p = buildat.Vector3(p):round()
log:debug("get_static_voxel(): p="..p:dump())
-- Calculate which chunk this voxel is in
local chunk_p, in_chunk_p = M.get_chunk_position(p)
log:debug("get_static_voxel(): chunk_p="..chunk_p:dump()..
", in_chunk_p="..in_chunk_p:dump())
-- Find the static node
local node = M.get_static_node(chunk_p)
if node == nil then
return buildat.VoxelInstance(0)
end
-- Get voxel from volume
local volume = M.get_volume(node)
2014-10-21 12:57:57 +03:00
log:debug("get_static_voxel(): volume="..dump(volume))
local v = volume:get_voxel_at(in_chunk_p.x, in_chunk_p.y, in_chunk_p.z)
log:debug("get_static_voxel(): v="..dump(v))
return v
end
-- TODO
-- NOTE: This does not synchronize the voxel to the server, because games have
-- to implement their own mechanisms for disallowing cheating
function M.set_static_voxel(x, y, z, v)
if type(x) == "table" then
z = x.z
y = x.y
x = x.x
end
end
2014-10-12 11:26:33 +03:00
function send_get_section(p)
local data = cereal.binary_output({
p = {
x = p.x,
y = p.y,
z = p.z,
},
}, {"object",
{"p", {"object",
{"x", "int32_t"},
{"y", "int32_t"},
{"z", "int32_t"},
}},
})
--log:info(dump(buildat.bytes(data)))
buildat.send_packet("voxelworld:get_section", data)
end
return M
-- vim: set noet ts=4 sw=4: