1
0
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:
20kdc 2018-03-23 01:12:08 +00:00
parent dc0feff44d
commit 1802ec6464
3 changed files with 79 additions and 77 deletions

View File

@ -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

View File

@ -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)

View File

@ -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