fix wireless pos comparison and update jammer, there might be a possibility to somehow cause a not ending loop

This commit is contained in:
HybridDog 2015-09-10 21:45:08 +02:00
parent 9938d13412
commit 680e21af42

View File

@ -3,39 +3,44 @@ local JAMMER_MAX_DISTANCE = 15
local wireless = {} local wireless = {}
local wireless_rids = {} local wireless_rids = {}
-- localize these functions with small names because they work fairly fast
local get = vector.get_data_from_pos
local set = vector.set_data_to_pos
local remove = vector.remove_data_from_pos
local register = function(pos) -- if the wireless at pos isn't stored yet, put it into the tables
local RID = vector.get_data_from_pos(wireless_rids, pos.z,pos.y,pos.x) local function register_RID(pos)
if not RID then if get(wireless_rids, pos.z,pos.y,pos.x) then
table.insert(wireless, pos) return
vector.set_data_to_pos(wireless_rids, pos.z,pos.y,pos.x, #wireless)
end end
local RID = #wireless+1
wireless[RID] = pos
set(wireless_rids, pos.z,pos.y,pos.x, RID)
end end
local wireless_activate = function(pos) local is_jammed
if not minetest.registered_nodes["moremesecons_wireless:wireless"] then return end local function wireless_activate(pos)
local channel_first_wireless = nil if is_jammed(pos) then
-- jamming doesn't disallow receiving signals, only sending them
return
end
local channel_first_wireless = minetest.get_meta(pos):get_string("channel")
for i = 1, #wireless do for i = 1, #wireless do
meta = minetest.get_meta(pos) if not vector.equals(wireless[i], pos)
channel_first_wireless = meta:get_string("channel") and minetest.get_meta(wireless[i]):get_string("channel") == channel_first_wireless then
meta = minetest.get_meta(wireless[i])
if wireless[i] ~= pos and meta:get_string("channel") == channel_first_wireless and not minetest.find_node_near(pos, JAMMER_MAX_DISTANCE, {"moremesecons_wireless:jammer_on"}) then
mesecon.receptor_on(wireless[i]) mesecon.receptor_on(wireless[i])
end end
end end
end end
local wireless_deactivate = function(pos) local function wireless_deactivate(pos)
if not minetest.registered_nodes["moremesecons_wireless:wireless"] then return end if is_jammed(pos) then
local meta = minetest.get_meta(pos) return
local channel_first_wireless = nil end
local channel_first_wireless = minetest.get_meta(pos):get_string("channel")
for i = 1, #wireless do for i = 1, #wireless do
meta = minetest.get_meta(pos) if not vector.equals(wireless[i], pos)
channel_first_wireless = meta:get_string("channel") and minetest.get_meta(wireless[i]):get_string("channel") == channel_first_wireless then
meta = minetest.get_meta(wireless[i])
if wireless[i] ~= pos and meta:get_string("channel") == channel_first_wireless and not minetest.find_node_near(pos, JAMMER_MAX_DISTANCE, {"moremesecons_wireless:jammer_on"}) then
mesecon.receptor_off(wireless[i]) mesecon.receptor_off(wireless[i])
end end
end end
@ -56,10 +61,10 @@ minetest.register_node("moremesecons_wireless:wireless", {
on_construct = function(pos) on_construct = function(pos)
local meta = minetest.get_meta(pos) local meta = minetest.get_meta(pos)
meta:set_string("formspec", "field[channel;channel;${channel}]") meta:set_string("formspec", "field[channel;channel;${channel}]")
register(pos) register_RID(pos)
end, end,
on_destruct = function(pos) on_destruct = function(pos)
local RID = vector.get_data_from_pos(wireless_rids, pos.z,pos.y,pos.x) local RID = get(wireless_rids, pos.z,pos.y,pos.x)
if RID then if RID then
table.remove(wireless, RID) table.remove(wireless, RID)
vector.remove_data_from_pos(wireless_rids, pos.z,pos.y,pos.x) vector.remove_data_from_pos(wireless_rids, pos.z,pos.y,pos.x)
@ -71,6 +76,38 @@ minetest.register_node("moremesecons_wireless:wireless", {
end, end,
}) })
local jammers = {}
local function add_jammer(pos)
if get(jammers, pos.z,pos.y,pos.x) then
return
end
set(jammers, pos.z,pos.y,pos.x, true)
end
local function remove_jammer(pos)
remove(jammers, pos.z,pos.y,pos.x)
end
-- looks big, but should work fast
function is_jammed(pos)
local pz,py,px = vector.unpack(pos)
for z,yxs in pairs(jammers) do
if math.abs(pz-z) <= JAMMER_MAX_DISTANCE then
for y,xs in pairs(yxs) do
if math.abs(py-y) <= JAMMER_MAX_DISTANCE then
for x in pairs(xs) do
if math.abs(px-x) <= JAMMER_MAX_DISTANCE
and (px-x)^2+(py-y)^2+(pz-z)^2 <= JAMMER_MAX_DISTANCE^2 then
return true
end
end
end
end
end
end
return false
end
mesecon.register_node("moremesecons_wireless:jammer", { mesecon.register_node("moremesecons_wireless:jammer", {
description="Wireless Jammer", description="Wireless Jammer",
paramtype = "light", paramtype = "light",
@ -79,16 +116,21 @@ mesecon.register_node("moremesecons_wireless:jammer", {
groups = {dig_immediate=2}, groups = {dig_immediate=2},
mesecons = {effector = { mesecons = {effector = {
action_on = function(pos) action_on = function(pos)
table.foreach(pos, print) add_jammer(pos)
minetest.swap_node(pos, {name="moremesecons_wireless:jammer_on"}) minetest.swap_node(pos, {name="moremesecons_wireless:jammer_on"})
end }} end
}}
},{ },{
tiles = {"moremesecons_jammer_on.png"}, tiles = {"moremesecons_jammer_on.png"},
groups = {dig_immediate=2, not_in_creative_inventory=1}, groups = {dig_immediate=2, not_in_creative_inventory=1},
mesecons = {effector = { mesecons = {effector = {
action_off = function(pos) action_off = function(pos)
remove_jammer(pos)
minetest.swap_node(pos, {name="moremesecons_wireless:jammer_off"}) minetest.swap_node(pos, {name="moremesecons_wireless:jammer_off"})
end }} end
}},
on_destruct = remove_jammer,
on_construct = add_jammer,
}) })
minetest.register_craft({ minetest.register_craft({
@ -108,8 +150,15 @@ minetest.register_craft({
}) })
minetest.register_abm({ minetest.register_abm({
nodenames = {"moremesecons_wireless:wireless"}, nodenames = {"moremesecons_wireless:jammer_on"},
interval=1, interval = 5,
chance = 1, chance = 1,
action = register action = add_jammer
})
minetest.register_abm({
nodenames = {"moremesecons_wireless:wireless"},
interval = 5,
chance = 1,
action = register_RID
}) })