mirror of
https://github.com/koreader/koreader.git
synced 2025-08-10 00:52:38 +00:00
452 lines
15 KiB
Lua
452 lines
15 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
|
|
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 rm_model = getModel()
|
|
local isRm2 = rm_model == "reMarkable 2.0"
|
|
local isRmPaperPro = rm_model == "reMarkable Ferrari"
|
|
|
|
if isRmPaperPro then
|
|
screen_width = 1620 -- unscaled_size_check: ignore
|
|
screen_height = 2160 -- unscaled_size_check: ignore
|
|
wacom_width = 11180 -- unscaled_size_check: ignore
|
|
wacom_height = 15340 -- unscaled_size_check: ignore
|
|
end
|
|
|
|
local wacom_scale_x = screen_width / wacom_width
|
|
local wacom_scale_y = screen_height / wacom_height
|
|
|
|
local Remarkable = Generic:extend{
|
|
isRemarkable = yes,
|
|
model = rm_model,
|
|
ota_model = "remarkable",
|
|
hasKeys = yes,
|
|
needsScreenRefreshAfterResume = no,
|
|
hasOTAUpdates = yes,
|
|
hasFastWifiStatusQuery = yes,
|
|
hasWifiManager = yes,
|
|
canReboot = yes,
|
|
canPowerOff = yes,
|
|
canSuspend = 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",
|
|
input_hall = nil,
|
|
}
|
|
|
|
local Remarkable1 = Remarkable:extend{
|
|
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:extend{
|
|
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 RemarkablePaperPro = Remarkable:extend{
|
|
mt_width = 2064, -- unscaled_size_check: ignore
|
|
mt_height = 2832, -- unscaled_size_check: ignore
|
|
display_dpi = 229,
|
|
input_wacom = "/dev/input/event2",
|
|
input_ts = "/dev/input/event3",
|
|
input_buttons = "/dev/input/event0",
|
|
input_hall = "/dev/input/event1",
|
|
battery_path = "/sys/class/power_supply/max1726x_battery/capacity",
|
|
status_path = "/sys/class/power_supply/max1726x_battery/status",
|
|
canSuspend = no, -- Suspend and Standby should be handled by xochitl with KO_DONT_GRAB_INPUT=1 set, otherwise bad things will happen
|
|
canStandby = no,
|
|
hasFrontlight = yes,
|
|
canTurnFrontlightOff = yes,
|
|
hasColorScreen = yes,
|
|
frontlight_settings = {
|
|
frontlight_white = "/sys/class/backlight/rm_frontlight",
|
|
}
|
|
}
|
|
|
|
function RemarkablePaperPro: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 = ev.value * by.mt_scale_x
|
|
end
|
|
if ev.code == C.ABS_MT_POSITION_Y then
|
|
ev.value = ev.value * by.mt_scale_y
|
|
end
|
|
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
|
|
|
|
if isRmPaperPro then
|
|
adjustAbsEvt = function(self, ev)
|
|
if ev.type == C.EV_ABS then
|
|
if ev.code == C.ABS_X then
|
|
ev.value = ev.value * wacom_scale_x
|
|
elseif ev.code == C.ABS_Y then
|
|
ev.value = ev.value * wacom_scale_y
|
|
end
|
|
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 process 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,
|
|
hall_file = isRmPaperPro and "/sys/class/input/input1/inhibited" or nil,
|
|
}
|
|
|
|
local event_map = dofile("frontend/device/remarkable/event_map.lua")
|
|
-- If we are launched while Oxide is running, remove Power from the event map
|
|
if oxide_running then
|
|
event_map[116] = nil
|
|
event_map[143] = nil
|
|
end
|
|
|
|
self.input = require("device/input"):new{
|
|
device = self,
|
|
event_map = dofile("frontend/device/remarkable/event_map.lua"),
|
|
event_map_adapter = {
|
|
SleepCover = function(ev)
|
|
if ev.value == 1 then
|
|
return "Suspend"
|
|
else
|
|
return "Resume"
|
|
end
|
|
end,
|
|
},
|
|
wacom_protocol = true,
|
|
}
|
|
|
|
-- Assume input stuff is saner on mainline kernels...
|
|
-- (c.f., https://github.com/koreader/koreader/issues/10012)
|
|
local is_mainline = false
|
|
--- @fixme: Find a better way to discriminate mainline from stock...
|
|
local std_out = io.popen("uname -r", "r")
|
|
if std_out then
|
|
local release = std_out:read("*line")
|
|
std_out:close()
|
|
release = release:match("^(%d+%.%d+)%.%d+.*$")
|
|
release = tonumber(release)
|
|
if release and release >= 6.2 and not isRmPaperPro then -- seems like it triggers on rMPP 3.19+ so just disable it on rMPP
|
|
is_mainline = true
|
|
end
|
|
end
|
|
|
|
if is_mainline then
|
|
self.input_wacom = "/dev/input/by-path/platform-30a20000.i2c-event-mouse"
|
|
self.input_buttons = "/dev/input/by-path/platform-30370000.snvs:snvs-powerkey-event"
|
|
self.input_ts = "/dev/input/touchscreen0"
|
|
end
|
|
|
|
self.input:open(self.input_wacom) -- Wacom (it's not Wacom on Paper Pro but it should work)
|
|
self.input:open(self.input_ts) -- Touchscreen
|
|
self.input:open(self.input_buttons) -- Buttons
|
|
|
|
if self.input_hall ~= nil then
|
|
self.input:open(self.input_hall) -- Hall sensor
|
|
local hallSensorMangling = function(this, ev)
|
|
if ev.type == C.EV_SW then
|
|
if ev.code == 0 then
|
|
ev.type = C.EV_KEY
|
|
ev.code = 20001
|
|
end
|
|
end
|
|
end
|
|
self.input:registerEventAdjustHook(hallSensorMangling)
|
|
end
|
|
|
|
local scalex = screen_width / self.mt_width
|
|
local scaley = screen_height / self.mt_height
|
|
|
|
if is_mainline then
|
|
-- NOTE: The panel sends *both* ABS_MT_ & ABS_ coordinates, while the pen only sends ABS_ coordinates.
|
|
-- Since we have to apply *different* mangling to each of them,
|
|
-- we use a custom input handler that'll ignore ABS_ coordinates from the panel...
|
|
self.input.handleTouchEv = self.input.handleMixedTouchEv
|
|
local mt_height = self.mt_height
|
|
local mainlineInputMangling = function(this, ev)
|
|
if ev.type == C.EV_ABS then
|
|
-- Mirror Y for the touch panel
|
|
if ev.code == C.ABS_MT_POSITION_Y then
|
|
ev.value = mt_height - ev.value
|
|
-- Handle the Wacom pen
|
|
elseif 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
|
|
self.input:registerEventAdjustHook(mainlineInputMangling)
|
|
else
|
|
self.input:registerEventAdjustHook(adjustAbsEvt)
|
|
self.input:registerEventAdjustHook(self.adjustTouchEvent, {mt_scale_x=scalex, mt_scale_y=scaley})
|
|
end
|
|
|
|
-- USB plug/unplug, battery charge/not charging are generated as fake events
|
|
self.input:open("fake_events")
|
|
|
|
local rotation_mode = self.screen.DEVICE_ROTATED_UPRIGHT
|
|
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
|
|
|
|
if isRmPaperPro then
|
|
-- disable autosuspend of xochitl
|
|
os.execute("cp -a ~/.config/remarkable/xochitl.conf ~/.config/remarkable/xochitl.conf.bak")
|
|
os.execute("sed -ri 's/IdleSuspendDelay=[0-9]+/IdleSuspendDelay=0/' ~/.config/remarkable/xochitl.conf")
|
|
end
|
|
|
|
if self.powerd:hasHallSensor() then
|
|
if G_reader_settings:has("remarkable_hall_effect_sensor_enabled") then
|
|
self.powerd:onToggleHallSensor(G_reader_settings:readSetting("remarkable_hall_effect_sensor_enabled"))
|
|
end
|
|
end
|
|
|
|
Generic.init(self)
|
|
end
|
|
|
|
function Remarkable:supportsScreensaver() return true end
|
|
|
|
function Remarkable:initNetworkManager(NetworkMgr)
|
|
function NetworkMgr:turnOnWifi(complete_callback, interactive)
|
|
if isRmPaperPro then
|
|
os.execute("/usr/bin/csl wifi -p on")
|
|
else
|
|
os.execute("./enable-wifi.sh")
|
|
end
|
|
return self:reconnectOrShowNetworkMenu(complete_callback, interactive)
|
|
end
|
|
|
|
function NetworkMgr:turnOffWifi(complete_callback)
|
|
if isRmPaperPro then
|
|
os.execute("/usr/bin/csl wifi -p off")
|
|
else
|
|
os.execute("./disable-wifi.sh")
|
|
end
|
|
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 = NetworkMgr.sysfsWifiOn
|
|
NetworkMgr.isConnected = NetworkMgr.ifHasAnAddress
|
|
end
|
|
|
|
function Remarkable:exit()
|
|
if isRmPaperPro then
|
|
os.execute("mv -f ~/.config/remarkable/xochitl.conf.bak ~/.config/remarkable/xochitl.conf")
|
|
if os.getenv("KO_DONT_GRAB_INPUT") == "1" then
|
|
os.execute("~/xovi/start")
|
|
end
|
|
end
|
|
Generic.exit(self)
|
|
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:saveSettings()
|
|
self.powerd:saveSettings()
|
|
end
|
|
|
|
function Remarkable:resume()
|
|
if isRmPaperPro then
|
|
os.execute("csl wifi -p on")
|
|
else
|
|
os.execute("./enable-wifi.sh")
|
|
end
|
|
end
|
|
|
|
function Remarkable:suspend()
|
|
if isRmPaperPro then
|
|
os.execute("csl wifi -p off")
|
|
else
|
|
os.execute("./disable-wifi.sh")
|
|
end
|
|
os.execute("systemctl suspend")
|
|
end
|
|
|
|
function Remarkable:powerOff()
|
|
if isRmPaperPro then
|
|
os.execute("mv -f ~/.config/remarkable/xochitl.conf.bak ~/.config/remarkable/xochitl.conf")
|
|
end
|
|
os.execute("systemctl poweroff")
|
|
end
|
|
|
|
function Remarkable:reboot()
|
|
if isRmPaperPro then
|
|
os.execute("mv -f ~/.config/remarkable/xochitl.conf.bak ~/.config/remarkable/xochitl.conf")
|
|
end
|
|
os.execute("systemctl reboot")
|
|
end
|
|
|
|
logger.info(string.format("Starting %s", rm_model))
|
|
|
|
function Remarkable:getDefaultCoverPath()
|
|
return "/usr/share/remarkable/poweroff.png"
|
|
end
|
|
|
|
function Remarkable:setEventHandlers(UIManager)
|
|
UIManager.event_handlers.Suspend = function()
|
|
self:onPowerEvent("Suspend")
|
|
end
|
|
UIManager.event_handlers.Resume = function()
|
|
self:onPowerEvent("Resume")
|
|
end
|
|
UIManager.event_handlers.PowerPress = function()
|
|
UIManager:scheduleIn(2, UIManager.poweroff_action)
|
|
end
|
|
UIManager.event_handlers.PowerRelease = function()
|
|
if not UIManager._entered_poweroff_stage then
|
|
UIManager:unschedule(UIManager.poweroff_action)
|
|
-- resume if we were suspended
|
|
if self.screen_saver_mode then
|
|
if self.screen_saver_lock then
|
|
UIManager.event_handlers.Suspend()
|
|
else
|
|
UIManager.event_handlers.Resume()
|
|
end
|
|
else
|
|
UIManager.event_handlers.Suspend()
|
|
end
|
|
end
|
|
end
|
|
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
|
|
elseif isRmPaperPro then
|
|
if not os.getenv("LD_PRELOAD") then
|
|
error("reMarkable Paper Pro requires qtfb and qtfb-rmpp-shim to work")
|
|
end
|
|
if os.getenv("QTFB_SHIM_INPUT") ~= "false" or os.getenv("QTFB_SHIM_MODEL") ~= "false" then
|
|
error("You must set both QTFB_SHIM_INPUT and QTFB_SHIM_MODEL to false")
|
|
end
|
|
if os.getenv("QTFB_SHIM_MODE") ~= "RGB565" then
|
|
error("You must set QTFB_SHIM_MODE to RGB565")
|
|
end
|
|
return RemarkablePaperPro
|
|
else
|
|
return Remarkable1
|
|
end
|