2020-08-31 22:23:09 +00:00
|
|
|
local Device = require("device")
|
|
|
|
|
|
|
|
if not Device:isPocketBook() --[[and not Device:isKobo()]] then
|
|
|
|
return { disabled = true }
|
|
|
|
end
|
|
|
|
|
|
|
|
local PowerD = Device:getPowerDevice()
|
|
|
|
local DataStorage = require("datastorage")
|
|
|
|
local LuaSettings = require("luasettings")
|
|
|
|
local UIManager = require("ui/uimanager")
|
|
|
|
local WidgetContainer = require("ui/widget/container/widgetcontainer")
|
|
|
|
local SpinWidget = require("ui/widget/spinwidget")
|
|
|
|
local logger = require("logger")
|
|
|
|
local _ = require("gettext")
|
|
|
|
|
|
|
|
local AutoStandby = WidgetContainer:new{
|
|
|
|
is_doc_only = false,
|
|
|
|
name = "autostandby",
|
2020-09-17 11:18:14 +00:00
|
|
|
|
|
|
|
-- static for all plugin instances
|
2020-08-31 22:23:09 +00:00
|
|
|
settings = LuaSettings:open(DataStorage:getSettingsDir() .. "/autostandby.lua"),
|
|
|
|
delay = 0,
|
|
|
|
lastInput = 0,
|
|
|
|
preventing = false,
|
|
|
|
}
|
|
|
|
|
|
|
|
function AutoStandby:init()
|
2020-09-17 11:18:14 +00:00
|
|
|
logger.dbg("AutoStandby:init() instance=", tostring(self))
|
2020-08-31 22:23:09 +00:00
|
|
|
if not self.settings:has("filter") then
|
|
|
|
logger.dbg("AutoStandby: No settings found, initializing defaults")
|
|
|
|
self.settings.data = {
|
|
|
|
forbidden = false, -- If forbidden, standby is never allowed to occur
|
|
|
|
filter = 1, -- Consider input only further than this many seconds apart
|
|
|
|
min = 1, -- Initial delay period during which we won't standby
|
|
|
|
mul = 1.5, -- Multiply the delay with each subsequent input that happens, scales up to max
|
|
|
|
max = 30, -- Input that happens further than 30 seconds since last input one reset delay back to 'min'
|
|
|
|
win = 5, -- Additional time window to consider input contributing to standby delay
|
|
|
|
bat = 60, -- If battery is below this percent, make auto-standby aggressive again (disables scaling by mul)
|
|
|
|
}
|
|
|
|
self.settings:flush()
|
|
|
|
end
|
|
|
|
|
|
|
|
UIManager.event_hook:registerWidget("InputEvent", self)
|
|
|
|
self.ui.menu:registerToMainMenu(self)
|
|
|
|
end
|
|
|
|
|
2021-04-02 23:48:35 +00:00
|
|
|
function AutoStandby:onCloseWidget()
|
|
|
|
logger.dbg("AutoStandby:onCloseWidget() instance=", tostring(self))
|
|
|
|
UIManager:unschedule(AutoStandby.allow)
|
|
|
|
end
|
|
|
|
|
2020-08-31 22:23:09 +00:00
|
|
|
function AutoStandby:addToMainMenu(menu_items)
|
|
|
|
menu_items.autostandby = {
|
2020-10-09 05:40:23 +00:00
|
|
|
sorting_hint = "device",
|
2020-08-31 22:23:09 +00:00
|
|
|
text = _("Auto-standby settings"),
|
|
|
|
sub_item_table = {
|
|
|
|
{
|
|
|
|
keep_menu_open = true,
|
|
|
|
text = _("Allow auto-standby"),
|
|
|
|
checked_func = function() return self:isAllowedByConfig() end,
|
|
|
|
callback = function() self.settings:saveSetting("forbidden", self:isAllowedByConfig()):flush() end,
|
|
|
|
},
|
|
|
|
self:genSpinMenuItem(_("Min input idle seconds"), "min", function() return 0 end, function() return self.settings:readSetting("max") end),
|
|
|
|
self:genSpinMenuItem(_("Max input idle seconds"), "max", function() return 0 end),
|
|
|
|
self:genSpinMenuItem(_("Input window seconds"), "win", function() return 0 end, function() return self.settings:readSetting("max") end),
|
|
|
|
self:genSpinMenuItem(_("Always standby if battery below"), "bat", function() return 0 end, function() return 100 end),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
The great Input/GestureDetector/TimeVal spring cleanup (a.k.a., a saner main loop) (#7415)
* ReaderDictionary: Port delay computations to TimeVal
* ReaderHighlight: Port delay computations to TimeVal
* ReaderView: Port delay computations to TimeVal
* Android: Reset gesture detection state on APP_CMD_TERM_WINDOW.
This prevents potentially being stuck in bogus gesture states when switching apps.
* GestureDetector:
* Port delay computations to TimeVal
* Fixed delay computations to handle time warps (large and negative deltas).
* Simplified timed callback handling to invalidate timers much earlier, preventing accumulating useless timers that no longer have any chance of ever detecting a gesture.
* Fixed state clearing to handle the actual effective slots, instead of hard-coding slot 0 & slot 1.
* Simplified timed callback handling in general, and added support for a timerfd backend for better performance and accuracy.
* The improved timed callback handling allows us to detect and honor (as much as possible) the three possible clock sources usable by Linux evdev events.
The only case where synthetic timestamps are used (and that only to handle timed callbacks) is limited to non-timerfd platforms where input events use
a clock source that is *NOT* MONOTONIC.
AFAICT, that's pretty much... PocketBook, and that's it?
* Input:
* Use the <linux/input.h> FFI module instead of re-declaring every constant
* Fixed (verbose) debug logging of input events to actually translate said constants properly.
* Completely reset gesture detection state on suspend. This should prevent bogus gesture detection on resume.
* Refactored the waitEvent loop to make it easier to comprehend (hopefully) and much more efficient.
Of specific note, it no longer does a crazy select spam every 100µs, instead computing and relying on sane timeouts,
as afforded by switching the UI event/input loop to the MONOTONIC time base, and the refactored timed callbacks in GestureDetector.
* reMarkable: Stopped enforcing synthetic timestamps on input events, as it should no longer be necessary.
* TimeVal:
* Refactored and simplified, especially as far as metamethods are concerned (based on <bsd/sys/time.h>).
* Added a host of new methods to query the various POSIX clock sources, and made :now default to MONOTONIC.
* Removed the debug guard in __sub, as time going backwards can be a perfectly normal occurrence.
* New methods:
* Clock sources: :realtime, :monotonic, :monotonic_coarse, :realtime_coarse, :boottime
* Utility: :tonumber, :tousecs, :tomsecs, :fromnumber, :isPositive, :isZero
* UIManager:
* Ported event loop & scheduling to TimeVal, and switched to the MONOTONIC time base.
This ensures reliable and consistent scheduling, as time is ensured never to go backwards.
* Added a :getTime() method, that returns a cached TimeVal:now(), updated at the top of every UI frame.
It's used throughout the codebase to cadge a syscall in circumstances where we are guaranteed that a syscall would return a mostly identical value,
because very few time has passed.
The only code left that does live syscalls does it because it's actually necessary for accuracy,
and the only code left that does that in a REALTIME time base is code that *actually* deals with calendar time (e.g., Statistics).
* DictQuickLookup: Port delay computations to TimeVal
* FootNoteWidget: Port delay computations to TimeVal
* HTMLBoxWidget: Port delay computations to TimeVal
* Notification: Port delay computations to TimeVal
* TextBoxWidget: Port delay computations to TimeVal
* AutoSuspend: Port to TimeVal
* AutoTurn:
* Fix it so that settings are actually honored.
* Port to TimeVal
* BackgroundRunner: Port to TimeVal
* Calibre: Port benchmarking code to TimeVal
* BookInfoManager: Removed unnecessary yield in the metadata extraction subprocess now that subprocesses get scheduled properly.
* All in all, these changes reduced the CPU cost of a single tap by a factor of ten (!), and got rid of an insane amount of weird poll/wakeup cycles that must have been hell on CPU schedulers and batteries..
2021-03-30 00:57:59 +00:00
|
|
|
-- We've received touch/key event, so delay standby accordingly
|
2020-08-31 22:23:09 +00:00
|
|
|
function AutoStandby:onInputEvent()
|
2020-09-17 11:18:14 +00:00
|
|
|
logger.dbg("AutoStandby:onInputevent() instance=", tostring(self))
|
2020-08-31 22:23:09 +00:00
|
|
|
local config = self.settings.data
|
|
|
|
local t = os.time()
|
2020-09-17 11:18:14 +00:00
|
|
|
if t < AutoStandby.lastInput + config.filter then
|
2020-08-31 22:23:09 +00:00
|
|
|
-- packed too close together, ignore
|
|
|
|
logger.dbg("AutoStandby: input packed too close to previous one, ignoring")
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Nuke past timer as we'll reschedule the allow (or not)
|
2020-09-17 11:18:14 +00:00
|
|
|
UIManager:unschedule(AutoStandby.allow)
|
2020-08-31 22:23:09 +00:00
|
|
|
|
|
|
|
if PowerD:getCapacityHW() <= config.bat then
|
|
|
|
-- battery is below threshold, so allow standby aggressively
|
|
|
|
logger.dbg("AutoStandby: battery below threshold, enabling aggressive standby")
|
|
|
|
self:allow()
|
|
|
|
return
|
2020-09-17 11:18:14 +00:00
|
|
|
elseif t > AutoStandby.lastInput + config.max then
|
2020-08-31 22:23:09 +00:00
|
|
|
-- too far apart, so reset delay
|
2020-09-17 11:18:14 +00:00
|
|
|
logger.dbg("AutoStandby: input too far in future, resetting adaptive standby delay from", AutoStandby.delay, "to", config.min)
|
|
|
|
AutoStandby.delay = config.min
|
|
|
|
elseif t < AutoStandby.lastInput + AutoStandby.delay + config.win then
|
2020-08-31 22:23:09 +00:00
|
|
|
-- otherwise widen the delay - "adaptive" - with frequent inputs, but don't grow beyonnd the max
|
2020-09-17 11:18:14 +00:00
|
|
|
AutoStandby.delay = math.min((AutoStandby.delay+1) * config.mul, config.max)
|
|
|
|
logger.dbg("AutoStandby: increasing standby delay to", AutoStandby.delay)
|
2020-08-31 22:23:09 +00:00
|
|
|
end -- equilibrium: when the event arrives beyond delay + win, but still below max, we keep the delay as-is
|
|
|
|
|
2020-09-17 11:18:14 +00:00
|
|
|
AutoStandby.lastInput = t
|
2020-08-31 22:23:09 +00:00
|
|
|
|
|
|
|
if not self:isAllowedByConfig() then
|
|
|
|
-- all standbys forbidden, always prevent
|
|
|
|
self:prevent()
|
|
|
|
return
|
2020-09-17 11:18:14 +00:00
|
|
|
elseif AutoStandby.delay == 0 then
|
2020-08-31 22:23:09 +00:00
|
|
|
-- If delay is 0 now, just allow straight
|
|
|
|
self:allow()
|
|
|
|
return
|
|
|
|
end
|
|
|
|
-- otherwise prevent for a while for duration of the delay
|
|
|
|
self:prevent()
|
|
|
|
-- and schedule standby re-enable once delay expires
|
2020-09-17 11:18:14 +00:00
|
|
|
UIManager:scheduleIn(AutoStandby.delay, AutoStandby.allow, AutoStandby)
|
2020-08-31 22:23:09 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- Prevent standby (by timer)
|
|
|
|
function AutoStandby:prevent()
|
2020-09-17 11:18:14 +00:00
|
|
|
if not AutoStandby.preventing then
|
|
|
|
AutoStandby.preventing = true
|
2020-08-31 22:23:09 +00:00
|
|
|
UIManager:preventStandby()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Allow standby (by timer)
|
|
|
|
function AutoStandby:allow()
|
2020-09-17 11:18:14 +00:00
|
|
|
if AutoStandby.preventing then
|
|
|
|
AutoStandby.preventing = false
|
2020-08-31 22:23:09 +00:00
|
|
|
UIManager:allowStandby()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function AutoStandby:isAllowedByConfig()
|
|
|
|
return self.settings:isFalse("forbidden")
|
|
|
|
end
|
|
|
|
|
|
|
|
function AutoStandby:genSpinMenuItem(text, cfg, min, max)
|
|
|
|
return {
|
|
|
|
keep_menu_open = true,
|
|
|
|
text = text,
|
|
|
|
enabled_func = function() return self:isAllowedByConfig() end,
|
|
|
|
callback = function()
|
|
|
|
local spin = SpinWidget:new {
|
|
|
|
value = self.settings:readSetting(cfg),
|
|
|
|
value_min = min and min() or 0,
|
|
|
|
value_max = max and max() or 9999,
|
|
|
|
value_hold_step = 10,
|
|
|
|
ok_text = "Update",
|
|
|
|
title_text = text,
|
|
|
|
callback = function(spin) self.settings:saveSetting(cfg, spin.value):flush() end,
|
|
|
|
}
|
|
|
|
UIManager:show(spin)
|
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
-- koreader is merely waiting for user input right now.
|
|
|
|
-- UI signals us that standby is allowed at this very moment because nothing else goes on in the background.
|
|
|
|
function AutoStandby:onAllowStandby()
|
|
|
|
logger.dbg("AutoStandby: onAllowStandby()")
|
|
|
|
-- In case the OS frontend itself doesn't manage power state, we can do it on our own here.
|
|
|
|
-- One should also configure wake-up pins and perhaps wake alarm,
|
|
|
|
-- if we want to enter deeper sleep states later on from within standby.
|
|
|
|
|
|
|
|
--os.execute("echo mem > /sys/power/state")
|
|
|
|
end
|
|
|
|
|
|
|
|
return AutoStandby
|
|
|
|
|