mirror of
https://github.com/koreader/koreader.git
synced 2025-08-10 00:52:38 +00:00
The change from timeval to time completely broke reMarkable. frontend/device/remarkable/device.lua was using TimeVal:now() to manually overwrite event time values, as noted in the code comments. Input:handleTouchEv is expecting those event time values to be timevals, not integer times. So as soon as the user touches the screen, crash.
242 lines
7.5 KiB
Lua
242 lines
7.5 KiB
Lua
local Generic = require("device/generic/device") -- <= look at this file!
|
|
local PluginShare = require("pluginshare")
|
|
local logger = require("logger")
|
|
local time = require("ui/time")
|
|
local ffi = require("ffi")
|
|
local C = ffi.C
|
|
require("ffi/linux_input_h")
|
|
|
|
local function yes() return true end
|
|
local function no() return false end
|
|
|
|
-- returns isRm2, device_model
|
|
local function getModel()
|
|
local f = io.open("/sys/devices/soc0/machine")
|
|
if not f then
|
|
error("missing sysfs entry for a remarkable")
|
|
end
|
|
local model = f:read("*line")
|
|
f:close()
|
|
return model == "reMarkable 2.0", model
|
|
end
|
|
|
|
-- Resolutions from libremarkable src/framebuffer/common.rs
|
|
local screen_width = 1404 -- unscaled_size_check: ignore
|
|
local screen_height = 1872 -- unscaled_size_check: ignore
|
|
local wacom_width = 15725 -- unscaled_size_check: ignore
|
|
local wacom_height = 20967 -- unscaled_size_check: ignore
|
|
local wacom_scale_x = screen_width / wacom_width
|
|
local wacom_scale_y = screen_height / wacom_height
|
|
local isRm2, rm_model = getModel()
|
|
|
|
local Remarkable = Generic:new{
|
|
isRemarkable = yes,
|
|
model = rm_model,
|
|
hasKeys = yes,
|
|
needsScreenRefreshAfterResume = no,
|
|
hasOTAUpdates = yes,
|
|
canReboot = yes,
|
|
canPowerOff = yes,
|
|
isTouchDevice = yes,
|
|
hasFrontlight = no,
|
|
hasSystemFonts = yes,
|
|
display_dpi = 226,
|
|
-- Despite the SoC supporting it, it's finicky in practice (#6772)
|
|
canHWInvert = no,
|
|
home_dir = "/home/root",
|
|
}
|
|
|
|
local Remarkable1 = Remarkable:new{
|
|
mt_width = 767, -- unscaled_size_check: ignore
|
|
mt_height = 1023, -- unscaled_size_check: ignore
|
|
input_wacom = "/dev/input/event0",
|
|
input_ts = "/dev/input/event1",
|
|
input_buttons = "/dev/input/event2",
|
|
battery_path = "/sys/class/power_supply/bq27441-0/capacity",
|
|
status_path = "/sys/class/power_supply/bq27441-0/status",
|
|
}
|
|
|
|
function Remarkable1:adjustTouchEvent(ev, by)
|
|
if ev.type == C.EV_ABS then
|
|
-- Mirror X and Y and scale up both X & Y as touch input is different res from
|
|
-- display
|
|
if ev.code == C.ABS_MT_POSITION_X then
|
|
ev.value = (Remarkable1.mt_width - ev.value) * by.mt_scale_x
|
|
end
|
|
if ev.code == C.ABS_MT_POSITION_Y then
|
|
ev.value = (Remarkable1.mt_height - ev.value) * by.mt_scale_y
|
|
end
|
|
end
|
|
end
|
|
|
|
local Remarkable2 = Remarkable:new{
|
|
mt_width = 1403, -- unscaled_size_check: ignore
|
|
mt_height = 1871, -- unscaled_size_check: ignore
|
|
input_wacom = "/dev/input/event1",
|
|
input_ts = "/dev/input/event2",
|
|
input_buttons = "/dev/input/event0",
|
|
battery_path = "/sys/class/power_supply/max77818_battery/capacity",
|
|
status_path = "/sys/class/power_supply/max77818-charger/status",
|
|
}
|
|
|
|
function Remarkable2:adjustTouchEvent(ev, by)
|
|
if ev.type == C.EV_ABS then
|
|
-- Mirror Y and scale up both X & Y as touch input is different res from
|
|
-- display
|
|
if ev.code == C.ABS_MT_POSITION_X then
|
|
ev.value = (ev.value) * by.mt_scale_x
|
|
end
|
|
if ev.code == C.ABS_MT_POSITION_Y then
|
|
ev.value = (Remarkable2.mt_height - ev.value) * by.mt_scale_y
|
|
end
|
|
end
|
|
|
|
-- Wacom uses CLOCK_REALTIME, but the Touchscreen spits out frozen timestamps.
|
|
-- Inject CLOCK_MONOTONIC timestamps at the end of every input frame in order to have consistent gesture detection across input devices.
|
|
-- c.f., #7536
|
|
if ev.type == C.EV_SYN and ev.code == C.SYN_REPORT then
|
|
local sec, usec = time.split_s_us(time.now())
|
|
ev.time = {
|
|
sec = sec,
|
|
usec = usec
|
|
}
|
|
end
|
|
end
|
|
|
|
local adjustAbsEvt = function(self, ev)
|
|
if ev.type == C.EV_ABS then
|
|
if ev.code == C.ABS_X then
|
|
ev.code = C.ABS_Y
|
|
ev.value = (wacom_height - ev.value) * wacom_scale_y
|
|
elseif ev.code == C.ABS_Y then
|
|
ev.code = C.ABS_X
|
|
ev.value = ev.value * wacom_scale_x
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
function Remarkable:init()
|
|
local oxide_running = os.execute("systemctl is-active --quiet tarnish") == 0
|
|
logger.info(string.format("Oxide running?: %s", oxide_running))
|
|
|
|
-- experiment
|
|
-- logger.info("PPID:")
|
|
-- local parent_process = os.execute("echo $PPID")
|
|
-- os.execute("ps | grep $PPID")
|
|
-- logger.info(string.format("parent proccess is oxide?: %s", parent_process_is_oxide))
|
|
|
|
self.screen = require("ffi/framebuffer_mxcfb"):new{device = self, debug = logger.dbg}
|
|
self.powerd = require("device/remarkable/powerd"):new{
|
|
device = self,
|
|
capacity_file = self.battery_path,
|
|
status_file = self.status_path,
|
|
}
|
|
|
|
local event_map = require("device/remarkable/event_map")
|
|
-- If we are launched while Oxide is running, remove Power from the event map
|
|
if oxide_running then
|
|
event_map[116] = nil
|
|
end
|
|
|
|
self.input = require("device/input"):new{
|
|
device = self,
|
|
event_map = require("device/remarkable/event_map"),
|
|
}
|
|
|
|
self.input.open(self.input_wacom) -- Wacom
|
|
self.input.open(self.input_ts) -- Touchscreen
|
|
self.input.open(self.input_buttons) -- Buttons
|
|
|
|
local scalex = screen_width / self.mt_width
|
|
local scaley = screen_height / self.mt_height
|
|
|
|
self.input:registerEventAdjustHook(adjustAbsEvt)
|
|
self.input:registerEventAdjustHook(self.adjustTouchEvent, {mt_scale_x=scalex, mt_scale_y=scaley})
|
|
|
|
-- USB plug/unplug, battery charge/not charging are generated as fake events
|
|
self.input.open("fake_events")
|
|
|
|
local rotation_mode = self.screen.ORIENTATION_PORTRAIT
|
|
self.screen.native_rotation_mode = rotation_mode
|
|
self.screen.cur_rotation_mode = rotation_mode
|
|
|
|
if oxide_running then
|
|
-- Disable autosuspend on this device
|
|
PluginShare.pause_auto_suspend = true
|
|
end
|
|
|
|
Generic.init(self)
|
|
end
|
|
|
|
function Remarkable:supportsScreensaver() return true end
|
|
|
|
function Remarkable:initNetworkManager(NetworkMgr)
|
|
function NetworkMgr:turnOnWifi(complete_callback)
|
|
os.execute("./enable-wifi.sh")
|
|
self:reconnectOrShowNetworkMenu(function()
|
|
self:connectivityCheck(1, complete_callback)
|
|
end)
|
|
end
|
|
|
|
function NetworkMgr:turnOffWifi(complete_callback)
|
|
os.execute("./disable-wifi.sh")
|
|
if complete_callback then
|
|
complete_callback()
|
|
end
|
|
end
|
|
|
|
function NetworkMgr:getNetworkInterfaceName()
|
|
return "wlan0"
|
|
end
|
|
|
|
NetworkMgr:setWirelessBackend("wpa_supplicant", {ctrl_interface = "/var/run/wpa_supplicant/wlan0"})
|
|
|
|
NetworkMgr.isWifiOn = function()
|
|
return NetworkMgr:isConnected()
|
|
end
|
|
end
|
|
|
|
function Remarkable:setDateTime(year, month, day, hour, min, sec)
|
|
if hour == nil or min == nil then return true end
|
|
local command
|
|
if year and month and day then
|
|
command = string.format("timedatectl set-time '%d-%d-%d %d:%d:%d'", year, month, day, hour, min, sec)
|
|
else
|
|
command = string.format("timedatectl set-time '%d:%d'",hour, min)
|
|
end
|
|
return os.execute(command) == 0
|
|
end
|
|
|
|
function Remarkable:resume()
|
|
end
|
|
|
|
function Remarkable:suspend()
|
|
os.execute("./disable-wifi.sh")
|
|
os.execute("systemctl suspend")
|
|
end
|
|
|
|
function Remarkable:powerOff()
|
|
os.execute("systemctl poweroff")
|
|
end
|
|
|
|
function Remarkable:reboot()
|
|
os.execute("systemctl reboot")
|
|
end
|
|
|
|
logger.info(string.format("Starting %s", rm_model))
|
|
|
|
function Remarkable:getDefaultCoverPath()
|
|
return "/usr/share/remarkable/poweroff.png"
|
|
end
|
|
|
|
if isRm2 then
|
|
if not os.getenv("RM2FB_SHIM") then
|
|
error("reMarkable2 requires RM2FB to work (https://github.com/ddvk/remarkable2-framebuffer)")
|
|
end
|
|
return Remarkable2
|
|
else
|
|
return Remarkable1
|
|
end
|
|
|