mirror of
https://github.com/20kdc/OC-KittenOS.git
synced 2025-04-05 04:08:39 +11:00
memopt & bugfixes: Clean up session system while keeping init flexible
Everest now lives and dies like the other side of the moon to sys-init, in order to get rid of the session nonsense that nobody was using anyway. While this means applications not using neoux will act slightly differently to applications that do, I don't think this will be a problem.
This commit is contained in:
parent
dc0feff44d
commit
1802ec6464
@ -57,14 +57,14 @@ monitors[0] = {nil, nil, 160, 50}
|
|||||||
-- line y
|
-- line y
|
||||||
local surfaces = {}
|
local surfaces = {}
|
||||||
|
|
||||||
|
-- Stops the main loop
|
||||||
|
local shuttingDown = false
|
||||||
|
|
||||||
local savingThrow = neo.requestAccess("x.neo.sys.manage")
|
local savingThrow = neo.requestAccess("x.neo.sys.manage")
|
||||||
if savingThrow then
|
if savingThrow then
|
||||||
savingThrow.registerForShutdownEvent()
|
savingThrow.registerForShutdownEvent()
|
||||||
savingThrow.registerSavingThrow(function ()
|
savingThrow.registerSavingThrow(function ()
|
||||||
if #monitors > 0 then
|
neo.executeAsync("sys-init", monitors[1][2])
|
||||||
neo.executeAsync("sys-init", monitors[1][2])
|
|
||||||
end
|
|
||||||
neo.executeAsync("sys-everest")
|
|
||||||
-- In this case the surfaces are leaked and hold references here. They have to be removed manually.
|
-- In this case the surfaces are leaked and hold references here. They have to be removed manually.
|
||||||
-- Do this via a "primary event" (k.deregistration) and "deathtrap events"
|
-- Do this via a "primary event" (k.deregistration) and "deathtrap events"
|
||||||
-- If a process evades the deathtrap then it clearly has reason to stay alive regardless of Everest status.
|
-- If a process evades the deathtrap then it clearly has reason to stay alive regardless of Everest status.
|
||||||
@ -77,9 +77,6 @@ if savingThrow then
|
|||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Grab all available monitors when they become available
|
|
||||||
local inSession = false
|
|
||||||
|
|
||||||
local function renderingAllowed()
|
local function renderingAllowed()
|
||||||
-- This is a safety feature to prevent implosion due to missing monitors.
|
-- This is a safety feature to prevent implosion due to missing monitors.
|
||||||
return #monitors > 0
|
return #monitors > 0
|
||||||
@ -370,7 +367,7 @@ everestProvider(function (pkg, pid, sendSig)
|
|||||||
local m = 0
|
local m = 0
|
||||||
if renderingAllowed() then m = 1 end
|
if renderingAllowed() then m = 1 end
|
||||||
if surfaces[1] then m = surfaces[1][1] end
|
if surfaces[1] then m = surfaces[1][1] end
|
||||||
local surf = {m, 1, 1, w, h}
|
local surf = {m, 1, 2, w, h}
|
||||||
local focusState = false
|
local focusState = false
|
||||||
local llid = lid
|
local llid = lid
|
||||||
lid = lid + 1
|
lid = lid + 1
|
||||||
@ -423,8 +420,6 @@ everestProvider(function (pkg, pid, sendSig)
|
|||||||
handleSpan(surf, 1, 1, vtitle, bg, fg)
|
handleSpan(surf, 1, 1, vtitle, bg, fg)
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
-- WCHAX : Wide-char-cleanup has to be done left-to-right, so this handles the important part of that.
|
|
||||||
handleSpan(surf, surf[4], a, " ", 0, 0)
|
|
||||||
a = a - 1
|
a = a - 1
|
||||||
end
|
end
|
||||||
sendSig(llid, ev, a, b, c, d, e)
|
sendSig(llid, ev, a, b, c, d, e)
|
||||||
@ -476,26 +471,11 @@ end)
|
|||||||
-- THE EVEREST USER API ENDS (now for the session API, which just does boring stuff)
|
-- THE EVEREST USER API ENDS (now for the session API, which just does boring stuff)
|
||||||
everestSessionProvider(function (pkg, pid, sendSig)
|
everestSessionProvider(function (pkg, pid, sendSig)
|
||||||
return {
|
return {
|
||||||
startSession = function ()
|
endSession = function (gotoBristol)
|
||||||
inSession = true
|
shuttingDown = true
|
||||||
end,
|
if gotoBristol then
|
||||||
endSession = function (startBristol)
|
-- Notably, savingThrow only triggers for error-death...
|
||||||
if not inSession then return end
|
neo.executeAsync("sys-init", (monitors[1] or {})[2])
|
||||||
local m = nil
|
|
||||||
if monitors[1] then
|
|
||||||
m = monitors[1][2]
|
|
||||||
end
|
|
||||||
inSession = false
|
|
||||||
for k = 1, #monitors do
|
|
||||||
screens.disclaim(monitors[k][2])
|
|
||||||
monitors[k] = nil
|
|
||||||
end
|
|
||||||
if startBristol then
|
|
||||||
neo.executeAsync("sys-init", m)
|
|
||||||
end
|
|
||||||
reconcileAll()
|
|
||||||
if not startBristol then
|
|
||||||
return m
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
@ -562,7 +542,24 @@ local function key(ka, kc, down)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
while true do
|
-- take all displays!
|
||||||
|
local function performClaim(s3)
|
||||||
|
local gpu = screens.claim(s3)
|
||||||
|
local gpucb = gpu and (gpu())
|
||||||
|
if gpucb then
|
||||||
|
local w, h = gpucb.getResolution()
|
||||||
|
table.insert(monitors, {gpu, s3, w, h, -1, -1})
|
||||||
|
-- This is required to ensure windows are moved off of the null monitor.
|
||||||
|
-- Luckily, there's an obvious sign if they aren't - everest will promptly crash.
|
||||||
|
reconcileAll()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
for _, v in ipairs(screens.getClaimable()) do
|
||||||
|
performClaim(v)
|
||||||
|
end
|
||||||
|
|
||||||
|
while not shuttingDown do
|
||||||
local s = {coroutine.yield()}
|
local s = {coroutine.yield()}
|
||||||
if renderingAllowed() then
|
if renderingAllowed() then
|
||||||
if s[1] == "h.key_down" then
|
if s[1] == "h.key_down" then
|
||||||
@ -634,17 +631,7 @@ while true do
|
|||||||
end
|
end
|
||||||
if s[1] == "x.neo.sys.screens" then
|
if s[1] == "x.neo.sys.screens" then
|
||||||
if s[2] == "available" then
|
if s[2] == "available" then
|
||||||
if inSession then
|
performClaim(s[3])
|
||||||
local gpu = screens.claim(s[3])
|
|
||||||
local gpucb = gpu and (gpu())
|
|
||||||
if gpucb then
|
|
||||||
local w, h = gpucb.getResolution()
|
|
||||||
table.insert(monitors, {gpu, s[3], w, h, -1, -1})
|
|
||||||
-- This is required to ensure windows are moved off of the null monitor.
|
|
||||||
-- Luckily, there's an obvious sign if they aren't - everest will promptly crash.
|
|
||||||
reconcileAll()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
if s[2] == "lost" then
|
if s[2] == "lost" then
|
||||||
for k, v in ipairs(monitors) do
|
for k, v in ipairs(monitors) do
|
||||||
|
@ -39,7 +39,7 @@ local settings = {
|
|||||||
-- The list of settings is here:
|
-- The list of settings is here:
|
||||||
-- password
|
-- password
|
||||||
password = "",
|
password = "",
|
||||||
["run.sys-everest"] = "yes",
|
["sys-init.shell"] = "sys-everest",
|
||||||
["run.sys-icecap"] = "yes",
|
["run.sys-icecap"] = "yes",
|
||||||
-- scr.w/h/d.<uuid>
|
-- scr.w/h/d.<uuid>
|
||||||
}
|
}
|
||||||
@ -78,6 +78,8 @@ local monitorPool = {}
|
|||||||
local monitorClaims = {}
|
local monitorClaims = {}
|
||||||
-- [gpuAddr] = monitorAddr
|
-- [gpuAddr] = monitorAddr
|
||||||
local currentGPUBinding = {}
|
local currentGPUBinding = {}
|
||||||
|
-- [gpuAddr] = userCount
|
||||||
|
local currentGPUUsers = {}
|
||||||
|
|
||||||
local function announceFreeMonitor(address, except)
|
local function announceFreeMonitor(address, except)
|
||||||
for k, v in pairs(targsRD) do
|
for k, v in pairs(targsRD) do
|
||||||
@ -92,10 +94,18 @@ local function getGPU(monitor)
|
|||||||
local bestD = 0
|
local bestD = 0
|
||||||
for v in gpus.list() do
|
for v in gpus.list() do
|
||||||
v.bind(monitor.address, false)
|
v.bind(monitor.address, false)
|
||||||
|
currentGPUBinding[v.address] = monitor.address
|
||||||
local d = v.maxDepth()
|
local d = v.maxDepth()
|
||||||
if d > bestD then
|
if d > bestD then
|
||||||
bestG = v
|
bestG = v
|
||||||
bestD = d
|
bestD = d
|
||||||
|
bestU = currentGPUUsers[v.address] or 0
|
||||||
|
elseif d == bestD then
|
||||||
|
if (currentGPUUsers[v.address] or 0) < bestU then
|
||||||
|
bestG = v
|
||||||
|
bestD = d
|
||||||
|
bestU = currentGPUUsers[v.address] or 0
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return bestG
|
return bestG
|
||||||
@ -124,6 +134,7 @@ local function getMonitorSettings(a)
|
|||||||
end
|
end
|
||||||
local function setupMonitor(gpu, monitor)
|
local function setupMonitor(gpu, monitor)
|
||||||
gpu.bind(monitor.address, false)
|
gpu.bind(monitor.address, false)
|
||||||
|
currentGPUBinding[gpu.address] = monitor.address
|
||||||
local maxW, maxH = gpu.maxResolution()
|
local maxW, maxH = gpu.maxResolution()
|
||||||
local maxD = gpu.maxDepth()
|
local maxD = gpu.maxDepth()
|
||||||
local w, h, d = getMonitorSettings(monitor.address)
|
local w, h, d = getMonitorSettings(monitor.address)
|
||||||
@ -208,8 +219,7 @@ donkonitRDProvider(function (pkg, pid, sendSig)
|
|||||||
targsRD[pid] = {sendSig, function ()
|
targsRD[pid] = {sendSig, function ()
|
||||||
for k, v in pairs(claimed) do
|
for k, v in pairs(claimed) do
|
||||||
-- Nothing to really do here
|
-- Nothing to really do here
|
||||||
monitorClaims[k] = nil
|
v(false)
|
||||||
announceFreeMonitor(k, pid)
|
|
||||||
end
|
end
|
||||||
end}
|
end}
|
||||||
return {
|
return {
|
||||||
@ -231,11 +241,13 @@ donkonitRDProvider(function (pkg, pid, sendSig)
|
|||||||
setupMonitor(gpu, v)
|
setupMonitor(gpu, v)
|
||||||
gpu = gpu.address
|
gpu = gpu.address
|
||||||
currentGPUBinding[gpu] = address
|
currentGPUBinding[gpu] = address
|
||||||
|
currentGPUUsers[gpu] = (currentGPUUsers[gpu] or 0) + 1
|
||||||
local disclaimer = function (wasDevLoss)
|
local disclaimer = function (wasDevLoss)
|
||||||
-- we lost it
|
-- we lost it
|
||||||
monitorClaims[address] = nil
|
monitorClaims[address] = nil
|
||||||
claimed[address] = nil
|
claimed[address] = nil
|
||||||
if not wasDevLoss then
|
if not wasDevLoss then
|
||||||
|
currentGPUUsers[gpu] = currentGPUUsers[gpu] - 1
|
||||||
table.insert(monitorPool, v)
|
table.insert(monitorPool, v)
|
||||||
announceFreeMonitor(address, pid)
|
announceFreeMonitor(address, pid)
|
||||||
else
|
else
|
||||||
@ -249,10 +261,7 @@ donkonitRDProvider(function (pkg, pid, sendSig)
|
|||||||
for v in gpus.list() do
|
for v in gpus.list() do
|
||||||
if v.address == gpu then
|
if v.address == gpu then
|
||||||
if currentGPUBinding[gpu] ~= address then
|
if currentGPUBinding[gpu] ~= address then
|
||||||
local _, v2 = v.bind(address, false)
|
v.bind(address, false)
|
||||||
if v2 then
|
|
||||||
return
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
currentGPUBinding[gpu] = address
|
currentGPUBinding[gpu] = address
|
||||||
return v
|
return v
|
||||||
@ -277,6 +286,7 @@ loadSettings()
|
|||||||
local function rescanDevs()
|
local function rescanDevs()
|
||||||
monitorPool = {}
|
monitorPool = {}
|
||||||
currentGPUBinding = {}
|
currentGPUBinding = {}
|
||||||
|
currentGPUUsers = {}
|
||||||
local hasGPU = gpus.list()()
|
local hasGPU = gpus.list()()
|
||||||
for k, v in pairs(monitorClaims) do
|
for k, v in pairs(monitorClaims) do
|
||||||
v[2](true)
|
v[2](true)
|
||||||
|
@ -58,7 +58,8 @@ local function retrieveNssMonitor(nss)
|
|||||||
local subpool = {}
|
local subpool = {}
|
||||||
while not gpuG do
|
while not gpuG do
|
||||||
if performDisclaim then
|
if performDisclaim then
|
||||||
performDisclaim()
|
performDisclaim(true)
|
||||||
|
performDisclaim = nil
|
||||||
end
|
end
|
||||||
-- nss available - this means the monitor pool is now ready.
|
-- nss available - this means the monitor pool is now ready.
|
||||||
-- If no monitors are available, shut down now.
|
-- If no monitors are available, shut down now.
|
||||||
@ -98,7 +99,9 @@ local function retrieveNssMonitor(nss)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if not subpool[1] then error("Unable to claim any monitor.") end
|
if not subpool[1] then
|
||||||
|
error("None of the GPUs we got were actually usable")
|
||||||
|
end
|
||||||
gpuG = subpool[1][1]
|
gpuG = subpool[1][1]
|
||||||
screen = subpool[1][2]
|
screen = subpool[1][2]
|
||||||
end
|
end
|
||||||
@ -107,9 +110,12 @@ local function retrieveNssMonitor(nss)
|
|||||||
scrW, scrH = gpu.getResolution()
|
scrW, scrH = gpu.getResolution()
|
||||||
rstfbDraw(gpu)
|
rstfbDraw(gpu)
|
||||||
gpu.fill(1, 1, scrW, scrH, " ")
|
gpu.fill(1, 1, scrW, scrH, " ")
|
||||||
performDisclaim = function ()
|
performDisclaim = function (full)
|
||||||
for _, v in ipairs(subpool) do
|
nss.disclaim(subpool[1][2])
|
||||||
nss.disclaim(v[2])
|
if full then
|
||||||
|
for _, v in ipairs(subpool) do
|
||||||
|
nss.disclaim(v[2])
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return gpu
|
return gpu
|
||||||
@ -137,6 +143,8 @@ local function finalPrompt()
|
|||||||
local nss = neo.requestAccess("x.neo.sys.screens")
|
local nss = neo.requestAccess("x.neo.sys.screens")
|
||||||
if nss then
|
if nss then
|
||||||
retrieveNssMonitor(nss)
|
retrieveNssMonitor(nss)
|
||||||
|
else
|
||||||
|
error("no glacier to provide GPU for the prompt")
|
||||||
end
|
end
|
||||||
-- This is nsm's final chance to make itself available and thus allow the password to be set
|
-- This is nsm's final chance to make itself available and thus allow the password to be set
|
||||||
local nsm = neo.requestAccess("x.neo.sys.manage")
|
local nsm = neo.requestAccess("x.neo.sys.manage")
|
||||||
@ -266,27 +274,28 @@ local function finalPrompt()
|
|||||||
end
|
end
|
||||||
local function postPrompt()
|
local function postPrompt()
|
||||||
local gpu = gpuG()
|
local gpu = gpuG()
|
||||||
|
local nsm = neo.requestAccess("x.neo.sys.manage")
|
||||||
|
local sh = "sys-everest"
|
||||||
|
warnings = {"Unable to get sys-init.shell due to no NSM, using sys-everest"}
|
||||||
|
if nsm then
|
||||||
|
sh = nsm.getSetting("sys-init.shell") or sh
|
||||||
|
warnings = {"Starting " .. sh}
|
||||||
|
end
|
||||||
rstfbDraw(gpu)
|
rstfbDraw(gpu)
|
||||||
-- Begin to finish login, or fail
|
advDraw(gpu)
|
||||||
local everests = neo.requestAccess("x.neo.sys.session")
|
performDisclaim()
|
||||||
if everests then
|
neo.executeAsync(sh)
|
||||||
local s, e = pcall(everests.startSession)
|
sleep(0.5)
|
||||||
if not s then
|
for i = 1, 9 do
|
||||||
table.insert(warnings, "Everest failed to create a session")
|
local v = neo.requestAccess("x.neo.sys.session")
|
||||||
table.insert(warnings, tostring(e))
|
sleep(0.5) -- Important timing - allows it to take the monitor
|
||||||
else
|
if v then
|
||||||
warnings = {"Transferring to Everest..."}
|
|
||||||
advDraw(gpu)
|
|
||||||
if performDisclaim then
|
|
||||||
performDisclaim()
|
|
||||||
-- Give Everest time (this isn't perceptible, and is really just a safety measure)
|
|
||||||
sleep(1)
|
|
||||||
end
|
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
else
|
|
||||||
table.insert(warnings, "Couldn't communicate with Everest...")
|
|
||||||
end
|
end
|
||||||
|
-- ...oh. hope this works then?
|
||||||
|
warnings = {"That wasn't a shell. Try Safe Mode."}
|
||||||
|
rstfbDraw(gpu)
|
||||||
advDraw(gpu)
|
advDraw(gpu)
|
||||||
sleep(1)
|
sleep(1)
|
||||||
shutdown(true)
|
shutdown(true)
|
||||||
@ -305,17 +314,13 @@ local function initializeSystem()
|
|||||||
for s in screenAc.list() do
|
for s in screenAc.list() do
|
||||||
for g in gpuAc.list() do
|
for g in gpuAc.list() do
|
||||||
g.bind(s.address, false)
|
g.bind(s.address, false)
|
||||||
local w, h = g.maxResolution()
|
local whd = g.maxDepth()
|
||||||
local whd = w * h * g.maxDepth()
|
|
||||||
if whd > scrBestWHD then
|
if whd > scrBestWHD then
|
||||||
screen = s
|
screen = s
|
||||||
gpu = g
|
gpu = g
|
||||||
scrBestWHD = whd
|
scrBestWHD = whd
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if screen then
|
|
||||||
break
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if gpu then
|
if gpu then
|
||||||
@ -413,7 +418,7 @@ end
|
|||||||
|
|
||||||
if callerPkg ~= nil then
|
if callerPkg ~= nil then
|
||||||
-- Everest can call into this to force a login screen
|
-- Everest can call into this to force a login screen
|
||||||
-- In this case it locks Everest, then starts Bristol.
|
-- In this case Everest dies, then starts Bristol.
|
||||||
--
|
--
|
||||||
if callerPkg ~= "sys-everest" then
|
if callerPkg ~= "sys-everest" then
|
||||||
return
|
return
|
||||||
|
Loading…
Reference in New Issue
Block a user