mirror of
https://github.com/bjc/prosody.git
synced 2025-04-04 21:57:45 +03:00
net.server_epoll: Optimize timer handling
This commit is contained in:
parent
b246b00f85
commit
50f89a9f96
1 changed files with 33 additions and 56 deletions
|
@ -6,9 +6,7 @@
|
||||||
--
|
--
|
||||||
|
|
||||||
|
|
||||||
local t_sort = table.sort;
|
|
||||||
local t_insert = table.insert;
|
local t_insert = table.insert;
|
||||||
local t_remove = table.remove;
|
|
||||||
local t_concat = table.concat;
|
local t_concat = table.concat;
|
||||||
local setmetatable = setmetatable;
|
local setmetatable = setmetatable;
|
||||||
local tostring = tostring;
|
local tostring = tostring;
|
||||||
|
@ -20,6 +18,7 @@ local log = require "util.logger".init("server_epoll");
|
||||||
local socket = require "socket";
|
local socket = require "socket";
|
||||||
local luasec = require "ssl";
|
local luasec = require "ssl";
|
||||||
local gettime = require "util.time".now;
|
local gettime = require "util.time".now;
|
||||||
|
local indexedbheap = require "util.indexedbheap";
|
||||||
local createtable = require "util.table".create;
|
local createtable = require "util.table".create;
|
||||||
local inet = require "util.net";
|
local inet = require "util.net";
|
||||||
local inet_pton = inet.pton;
|
local inet_pton = inet.pton;
|
||||||
|
@ -69,22 +68,24 @@ local fds = createtable(10, 0); -- FD -> conn
|
||||||
|
|
||||||
-- Timer and scheduling --
|
-- Timer and scheduling --
|
||||||
|
|
||||||
local timers = {};
|
local timers = indexedbheap.create();
|
||||||
|
|
||||||
local function noop() end
|
local function noop() end
|
||||||
local function closetimer(t)
|
local function closetimer(t)
|
||||||
t[1] = 0;
|
t[1] = 0;
|
||||||
t[2] = noop;
|
t[2] = noop;
|
||||||
|
timers:remove(t.id);
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Set to true when timers have changed
|
local function reschedule(t, time)
|
||||||
local resort_timers = false;
|
t[1] = time;
|
||||||
|
timers:reprioritize(t.id, time);
|
||||||
|
end
|
||||||
|
|
||||||
-- Add absolute timer
|
-- Add absolute timer
|
||||||
local function at(time, f)
|
local function at(time, f)
|
||||||
local timer = { time, f, close = closetimer };
|
local timer = { time, f, close = closetimer, reschedule = reschedule, id = nil };
|
||||||
t_insert(timers, timer);
|
timer.id = timers:insert(timer, time);
|
||||||
resort_timers = true;
|
|
||||||
return timer;
|
return timer;
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -97,54 +98,32 @@ end
|
||||||
-- Return time until next timeout
|
-- Return time until next timeout
|
||||||
local function runtimers(next_delay, min_wait)
|
local function runtimers(next_delay, min_wait)
|
||||||
-- Any timers at all?
|
-- Any timers at all?
|
||||||
if not timers[1] then
|
local now = gettime();
|
||||||
|
local peek = timers:peek();
|
||||||
|
while peek do
|
||||||
|
|
||||||
|
if peek > now then
|
||||||
|
next_delay = peek - now;
|
||||||
|
break;
|
||||||
|
end
|
||||||
|
|
||||||
|
local _, timer, id = timers:pop();
|
||||||
|
local ok, ret = pcall(timer[2], now);
|
||||||
|
if ok and type(ret) == "number" then
|
||||||
|
local next_time = now+ret;
|
||||||
|
timer[1] = next_time;
|
||||||
|
timers:insert(timer, next_time);
|
||||||
|
end
|
||||||
|
|
||||||
|
peek = timers:peek();
|
||||||
|
end
|
||||||
|
if peek == nil then
|
||||||
return next_delay;
|
return next_delay;
|
||||||
end
|
end
|
||||||
|
|
||||||
if resort_timers then
|
if next_delay < min_wait then
|
||||||
-- Sort earliest timers to the end
|
return min_wait;
|
||||||
t_sort(timers, function (a, b) return a[1] > b[1]; end);
|
|
||||||
resort_timers = false;
|
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Iterate from the end and remove completed timers
|
|
||||||
for i = #timers, 1, -1 do
|
|
||||||
local timer = timers[i];
|
|
||||||
local t, f = timer[1], timer[2];
|
|
||||||
-- Get time for every iteration to increase accuracy
|
|
||||||
local now = gettime();
|
|
||||||
if t > now then
|
|
||||||
-- This timer should not fire yet
|
|
||||||
local diff = t - now;
|
|
||||||
if diff < next_delay then
|
|
||||||
next_delay = diff;
|
|
||||||
end
|
|
||||||
break;
|
|
||||||
end
|
|
||||||
local new_timeout = f(now);
|
|
||||||
if new_timeout then
|
|
||||||
-- Schedule for 'delay' from the time actually scheduled, not from now,
|
|
||||||
-- in order to prevent timer drift, unless it already drifted way out of sync.
|
|
||||||
if (t + new_timeout) > ( now - new_timeout ) then
|
|
||||||
timer[1] = t + new_timeout;
|
|
||||||
else
|
|
||||||
timer[1] = now + new_timeout;
|
|
||||||
end
|
|
||||||
resort_timers = true;
|
|
||||||
else
|
|
||||||
t_remove(timers, i);
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
if resort_timers or next_delay < min_wait then
|
|
||||||
-- Timers may be added from within a timer callback.
|
|
||||||
-- Those would not be considered for next_delay,
|
|
||||||
-- and we might sleep for too long, so instead
|
|
||||||
-- we return a shorter timeout so we can
|
|
||||||
-- properly sort all new timers.
|
|
||||||
next_delay = min_wait;
|
|
||||||
end
|
|
||||||
|
|
||||||
return next_delay;
|
return next_delay;
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -251,8 +230,7 @@ function interface:setreadtimeout(t)
|
||||||
end
|
end
|
||||||
t = t or cfg.read_timeout;
|
t = t or cfg.read_timeout;
|
||||||
if self._readtimeout then
|
if self._readtimeout then
|
||||||
self._readtimeout[1] = gettime() + t;
|
self._readtimeout:reschedule(gettime() + t);
|
||||||
resort_timers = true;
|
|
||||||
else
|
else
|
||||||
self._readtimeout = addtimer(t, function ()
|
self._readtimeout = addtimer(t, function ()
|
||||||
if self:on("readtimeout") then
|
if self:on("readtimeout") then
|
||||||
|
@ -276,8 +254,7 @@ function interface:setwritetimeout(t)
|
||||||
end
|
end
|
||||||
t = t or cfg.send_timeout;
|
t = t or cfg.send_timeout;
|
||||||
if self._writetimeout then
|
if self._writetimeout then
|
||||||
self._writetimeout[1] = gettime() + t;
|
self._writetimeout:reschedule(gettime() + t);
|
||||||
resort_timers = true;
|
|
||||||
else
|
else
|
||||||
self._writetimeout = addtimer(t, function ()
|
self._writetimeout = addtimer(t, function ()
|
||||||
self:on("disconnect", "write timeout");
|
self:on("disconnect", "write timeout");
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue