1088b4f5fSWarner Losh-- 2088b4f5fSWarner Losh-- Copyright (c) 2015 Pedro Souza <pedrosouza@freebsd.org> 3088b4f5fSWarner Losh-- All rights reserved. 4088b4f5fSWarner Losh-- 5088b4f5fSWarner Losh-- Redistribution and use in source and binary forms, with or without 6088b4f5fSWarner Losh-- modification, are permitted provided that the following conditions 7088b4f5fSWarner Losh-- are met: 8088b4f5fSWarner Losh-- 1. Redistributions of source code must retain the above copyright 9088b4f5fSWarner Losh-- notice, this list of conditions and the following disclaimer. 10088b4f5fSWarner Losh-- 2. Redistributions in binary form must reproduce the above copyright 11088b4f5fSWarner Losh-- notice, this list of conditions and the following disclaimer in the 12088b4f5fSWarner Losh-- documentation and/or other materials provided with the distribution. 13088b4f5fSWarner Losh-- 14088b4f5fSWarner Losh-- THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15088b4f5fSWarner Losh-- ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16088b4f5fSWarner Losh-- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17088b4f5fSWarner Losh-- ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18088b4f5fSWarner Losh-- FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19088b4f5fSWarner Losh-- DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20088b4f5fSWarner Losh-- OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21088b4f5fSWarner Losh-- HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22088b4f5fSWarner Losh-- LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23088b4f5fSWarner Losh-- OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24088b4f5fSWarner Losh-- SUCH DAMAGE. 25088b4f5fSWarner Losh-- 26088b4f5fSWarner Losh-- $FreeBSD$ 27088b4f5fSWarner Losh-- 28088b4f5fSWarner Losh 29088b4f5fSWarner Losh 30088b4f5fSWarner Loshlocal menu = {}; 31088b4f5fSWarner Losh 32088b4f5fSWarner Loshlocal core = require("core"); 33088b4f5fSWarner Loshlocal color = require("color"); 34088b4f5fSWarner Loshlocal config = require("config"); 35088b4f5fSWarner Loshlocal screen = require("screen"); 36088b4f5fSWarner Loshlocal drawer = require("drawer"); 37088b4f5fSWarner Losh 38088b4f5fSWarner Loshlocal OnOff; 39088b4f5fSWarner Loshlocal skip; 40088b4f5fSWarner Loshlocal run; 41088b4f5fSWarner Loshlocal autoboot; 42ada26c4aSKyle Evanslocal carousel_choices = {}; 43088b4f5fSWarner Losh 44280e990bSKyle Evans-- loader menu tree is rooted at menu.welcome 45088b4f5fSWarner Losh 46088b4f5fSWarner Loshmenu.boot_options = { 47d8757746SKyle Evans entries = { 48088b4f5fSWarner Losh -- return to welcome menu 49088b4f5fSWarner Losh { 50a7cf0562SKyle Evans entry_type = core.MENU_RETURN, 51088b4f5fSWarner Losh name = function() 5257099121SKyle Evans return "Back to main menu" .. 5357099121SKyle Evans color.highlight(" [Backspace]"); 541666dfc0SKyle Evans end 55088b4f5fSWarner Losh }, 56088b4f5fSWarner Losh 57088b4f5fSWarner Losh -- load defaults 58088b4f5fSWarner Losh { 59a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 60088b4f5fSWarner Losh name = function() 6157099121SKyle Evans return "Load System " .. color.highlight("D") .. 6257099121SKyle Evans "efaults"; 63088b4f5fSWarner Losh end, 64088b4f5fSWarner Losh func = function() 6524a1bd54SKyle Evans core.setDefaults(); 66088b4f5fSWarner Losh end, 67088b4f5fSWarner Losh alias = {"d", "D"} 68088b4f5fSWarner Losh }, 69088b4f5fSWarner Losh 70088b4f5fSWarner Losh { 71a7cf0562SKyle Evans entry_type = core.MENU_SEPARATOR, 72088b4f5fSWarner Losh name = function() 73088b4f5fSWarner Losh return ""; 74088b4f5fSWarner Losh end 75088b4f5fSWarner Losh }, 76088b4f5fSWarner Losh 77088b4f5fSWarner Losh { 78a7cf0562SKyle Evans entry_type = core.MENU_SEPARATOR, 79088b4f5fSWarner Losh name = function() 80088b4f5fSWarner Losh return "Boot Options:"; 81088b4f5fSWarner Losh end 82088b4f5fSWarner Losh }, 83088b4f5fSWarner Losh 84088b4f5fSWarner Losh -- acpi 85088b4f5fSWarner Losh { 86a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 87088b4f5fSWarner Losh name = function() 8857099121SKyle Evans return OnOff(color.highlight("A") .. "CPI :", 8957099121SKyle Evans core.acpi); 90088b4f5fSWarner Losh end, 91088b4f5fSWarner Losh func = function() 92088b4f5fSWarner Losh core.setACPI(); 93088b4f5fSWarner Losh end, 94088b4f5fSWarner Losh alias = {"a", "A"} 95088b4f5fSWarner Losh }, 96088b4f5fSWarner Losh -- safe mode 97088b4f5fSWarner Losh { 98a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 99088b4f5fSWarner Losh name = function() 10057099121SKyle Evans return OnOff("Safe " .. color.highlight("M") .. 10157099121SKyle Evans "ode :", core.sm); 102088b4f5fSWarner Losh end, 103088b4f5fSWarner Losh func = function() 104088b4f5fSWarner Losh core.setSafeMode(); 105088b4f5fSWarner Losh end, 106088b4f5fSWarner Losh alias = {"m", "M"} 107088b4f5fSWarner Losh }, 108088b4f5fSWarner Losh -- single user 109088b4f5fSWarner Losh { 110a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 111088b4f5fSWarner Losh name = function() 11257099121SKyle Evans return OnOff(color.highlight("S") .. "ingle user:", 11357099121SKyle Evans core.su); 114088b4f5fSWarner Losh end, 115088b4f5fSWarner Losh func = function() 116088b4f5fSWarner Losh core.setSingleUser(); 117088b4f5fSWarner Losh end, 118088b4f5fSWarner Losh alias = {"s", "S"} 119088b4f5fSWarner Losh }, 120088b4f5fSWarner Losh -- verbose boot 121088b4f5fSWarner Losh { 122a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 123088b4f5fSWarner Losh name = function() 12457099121SKyle Evans return OnOff(color.highlight("V") .. "erbose :", 12557099121SKyle Evans core.verbose); 126088b4f5fSWarner Losh end, 127088b4f5fSWarner Losh func = function() 128088b4f5fSWarner Losh core.setVerbose(); 129088b4f5fSWarner Losh end, 130088b4f5fSWarner Losh alias = {"v", "V"} 131088b4f5fSWarner Losh }, 132d8757746SKyle Evans }, 133088b4f5fSWarner Losh}; 134088b4f5fSWarner Losh 135088b4f5fSWarner Loshmenu.welcome = { 136*303253e5SKyle Evans entries = function() 137*303253e5SKyle Evans local menu_entries = menu.welcome.all_entries; 138*303253e5SKyle Evans -- Swap the first two menu items on single user boot 139*303253e5SKyle Evans if (core.isSingleUserBoot()) then 140*303253e5SKyle Evans local multiuser = menu_entries[1]; 141*303253e5SKyle Evans local singleuser = menu_entries[2]; 142*303253e5SKyle Evans 143*303253e5SKyle Evans menu_entries[2] = multiuser; 144*303253e5SKyle Evans menu_entries[1] = singleuser; 145*303253e5SKyle Evans end 146*303253e5SKyle Evans return menu_entries; 147*303253e5SKyle Evans end, 148*303253e5SKyle Evans all_entries = { 149088b4f5fSWarner Losh -- boot multi user 150088b4f5fSWarner Losh { 151a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 152088b4f5fSWarner Losh name = function() 15357099121SKyle Evans return color.highlight("B") .. "oot Multi user " .. 15457099121SKyle Evans color.highlight("[Enter]"); 155088b4f5fSWarner Losh end, 156088b4f5fSWarner Losh func = function() 157088b4f5fSWarner Losh core.setSingleUser(false); 158088b4f5fSWarner Losh core.boot(); 159088b4f5fSWarner Losh end, 160b458bf0dSKyle Evans alias = {"b", "B"} 161088b4f5fSWarner Losh }, 162088b4f5fSWarner Losh 163088b4f5fSWarner Losh -- boot single user 164088b4f5fSWarner Losh { 165a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 166088b4f5fSWarner Losh name = function() 167088b4f5fSWarner Losh return "Boot " .. color.highlight("S") .. "ingle user"; 168088b4f5fSWarner Losh end, 169088b4f5fSWarner Losh func = function() 170088b4f5fSWarner Losh core.setSingleUser(true); 171088b4f5fSWarner Losh core.boot(); 172088b4f5fSWarner Losh end, 173088b4f5fSWarner Losh alias = {"s", "S"} 174088b4f5fSWarner Losh }, 175088b4f5fSWarner Losh 176088b4f5fSWarner Losh -- escape to interpreter 177088b4f5fSWarner Losh { 178a7cf0562SKyle Evans entry_type = core.MENU_RETURN, 179088b4f5fSWarner Losh name = function() 180e9084012SKyle Evans return color.highlight("Esc") .. "ape to loader prompt"; 181088b4f5fSWarner Losh end, 182ef625845SKyle Evans func = function() 18324a1bd54SKyle Evans loader.setenv("autoboot_delay", "NO"); 184ef625845SKyle Evans end, 18539006570SKyle Evans alias = {core.KEYSTR_ESCAPE} 186088b4f5fSWarner Losh }, 187088b4f5fSWarner Losh 188088b4f5fSWarner Losh -- reboot 189088b4f5fSWarner Losh { 190a7cf0562SKyle Evans entry_type = core.MENU_ENTRY, 191088b4f5fSWarner Losh name = function() 192088b4f5fSWarner Losh return color.highlight("R") .. "eboot"; 193088b4f5fSWarner Losh end, 194088b4f5fSWarner Losh func = function() 195088b4f5fSWarner Losh loader.perform("reboot"); 196088b4f5fSWarner Losh end, 197088b4f5fSWarner Losh alias = {"r", "R"} 198088b4f5fSWarner Losh }, 199088b4f5fSWarner Losh 200088b4f5fSWarner Losh 201088b4f5fSWarner Losh { 202a7cf0562SKyle Evans entry_type = core.MENU_SEPARATOR, 203088b4f5fSWarner Losh name = function() 204088b4f5fSWarner Losh return ""; 205088b4f5fSWarner Losh end 206088b4f5fSWarner Losh }, 207088b4f5fSWarner Losh 208088b4f5fSWarner Losh { 209a7cf0562SKyle Evans entry_type = core.MENU_SEPARATOR, 210088b4f5fSWarner Losh name = function() 211088b4f5fSWarner Losh return "Options:"; 212088b4f5fSWarner Losh end 213088b4f5fSWarner Losh }, 214088b4f5fSWarner Losh 215088b4f5fSWarner Losh -- kernel options 216088b4f5fSWarner Losh { 217a7cf0562SKyle Evans entry_type = core.MENU_CAROUSEL_ENTRY, 218ada26c4aSKyle Evans carousel_id = "kernel", 219ada26c4aSKyle Evans items = core.kernelList, 220ada26c4aSKyle Evans name = function(idx, choice, all_choices) 22124a1bd54SKyle Evans if (#all_choices == 0) then 222b1b1f2b8SKyle Evans return "Kernel: "; 223088b4f5fSWarner Losh end 224b1b1f2b8SKyle Evans 225bcf48a15SKyle Evans local is_default = (idx == 1); 226bcf48a15SKyle Evans local kernel_name = ""; 227bcf48a15SKyle Evans local name_color; 22824a1bd54SKyle Evans if (is_default) then 229bcf48a15SKyle Evans name_color = color.escapef(color.GREEN); 230bcf48a15SKyle Evans kernel_name = "default/"; 231bcf48a15SKyle Evans else 232bcf48a15SKyle Evans name_color = color.escapef(color.BLUE); 233b1b1f2b8SKyle Evans end 234bcf48a15SKyle Evans kernel_name = kernel_name .. name_color .. choice .. 235bcf48a15SKyle Evans color.default(); 236b1b1f2b8SKyle Evans return color.highlight("K").."ernel: " .. kernel_name .. 237ada26c4aSKyle Evans " (" .. idx .. 238ada26c4aSKyle Evans " of " .. #all_choices .. ")"; 239088b4f5fSWarner Losh end, 2405d1e2f83SKyle Evans func = function(idx, choice, all_choices) 241fa4a2394SKyle Evans config.selectkernel(choice); 242088b4f5fSWarner Losh end, 243088b4f5fSWarner Losh alias = {"k", "K"} 244088b4f5fSWarner Losh }, 245088b4f5fSWarner Losh 246088b4f5fSWarner Losh -- boot options 247088b4f5fSWarner Losh { 248a7cf0562SKyle Evans entry_type = core.MENU_SUBMENU, 249088b4f5fSWarner Losh name = function() 250088b4f5fSWarner Losh return "Boot " .. color.highlight("O") .. "ptions"; 251088b4f5fSWarner Losh end, 252088b4f5fSWarner Losh submenu = function() 253088b4f5fSWarner Losh return menu.boot_options; 254088b4f5fSWarner Losh end, 255088b4f5fSWarner Losh alias = {"o", "O"} 256d8757746SKyle Evans }, 257d8757746SKyle Evans }, 258088b4f5fSWarner Losh}; 259088b4f5fSWarner Losh 260ada26c4aSKyle Evans-- The first item in every carousel is always the default item. 261ada26c4aSKyle Evansfunction menu.getCarouselIndex(id) 262ada26c4aSKyle Evans local val = carousel_choices[id]; 263ada26c4aSKyle Evans if (val == nil) then 264ada26c4aSKyle Evans return 1; 265ada26c4aSKyle Evans end 266ada26c4aSKyle Evans return val; 267ada26c4aSKyle Evansend 268ada26c4aSKyle Evans 269ada26c4aSKyle Evansfunction menu.setCarouselIndex(id, idx) 270ada26c4aSKyle Evans carousel_choices[id] = idx; 271ada26c4aSKyle Evansend 272ada26c4aSKyle Evans 273088b4f5fSWarner Loshfunction menu.run(m) 274088b4f5fSWarner Losh 275088b4f5fSWarner Losh if (menu.skip()) then 276088b4f5fSWarner Losh core.autoboot(); 277088b4f5fSWarner Losh return false; 278088b4f5fSWarner Losh end 279088b4f5fSWarner Losh 280088b4f5fSWarner Losh if (m == nil) then 281088b4f5fSWarner Losh m = menu.welcome; 282088b4f5fSWarner Losh end 283088b4f5fSWarner Losh 284088b4f5fSWarner Losh -- redraw screen 285088b4f5fSWarner Losh screen.clear(); 286088b4f5fSWarner Losh screen.defcursor(); 287088b4f5fSWarner Losh local alias_table = drawer.drawscreen(m); 288088b4f5fSWarner Losh 2893a0a07d0SKyle Evans menu.autoboot(); 290088b4f5fSWarner Losh 291088b4f5fSWarner Losh cont = true; 29224a1bd54SKyle Evans while (cont) do 293abc4f7e7SKyle Evans local key = io.getchar(); 294088b4f5fSWarner Losh 295b458bf0dSKyle Evans -- Special key behaviors 2961504bce3SKyle Evans if ((key == core.KEY_BACKSPACE) or (key == core.KEY_DELETE)) and 2971504bce3SKyle Evans (m ~= menu.welcome) then 298abc4f7e7SKyle Evans break 299fe672a15SKyle Evans elseif (key == core.KEY_ENTER) then 300b458bf0dSKyle Evans core.boot(); 301b458bf0dSKyle Evans -- Should not return 302abc4f7e7SKyle Evans end 303abc4f7e7SKyle Evans 304abc4f7e7SKyle Evans key = string.char(key) 305088b4f5fSWarner Losh -- check to see if key is an alias 306088b4f5fSWarner Losh local sel_entry = nil; 307088b4f5fSWarner Losh for k, v in pairs(alias_table) do 308088b4f5fSWarner Losh if (key == k) then 309088b4f5fSWarner Losh sel_entry = v; 310088b4f5fSWarner Losh end 311088b4f5fSWarner Losh end 312088b4f5fSWarner Losh 313088b4f5fSWarner Losh -- if we have an alias do the assigned action: 314088b4f5fSWarner Losh if (sel_entry ~= nil) then 315a7cf0562SKyle Evans if (sel_entry.entry_type == core.MENU_ENTRY) then 316088b4f5fSWarner Losh -- run function 317088b4f5fSWarner Losh sel_entry.func(); 318a7cf0562SKyle Evans elseif (sel_entry.entry_type == core.MENU_CAROUSEL_ENTRY) then 319ada26c4aSKyle Evans -- carousel (rotating) functionality 320ada26c4aSKyle Evans local carid = sel_entry.carousel_id; 321ada26c4aSKyle Evans local caridx = menu.getCarouselIndex(carid); 322ada26c4aSKyle Evans local choices = sel_entry.items(); 323ada26c4aSKyle Evans 324aefcaa7eSKyle Evans if (#choices > 0) then 325ada26c4aSKyle Evans caridx = (caridx % #choices) + 1; 326ada26c4aSKyle Evans menu.setCarouselIndex(carid, caridx); 3275d1e2f83SKyle Evans sel_entry.func(caridx, choices[caridx], 32884f82e46SKyle Evans choices); 329aefcaa7eSKyle Evans end 330a7cf0562SKyle Evans elseif (sel_entry.entry_type == core.MENU_SUBMENU) then 331088b4f5fSWarner Losh -- recurse 332088b4f5fSWarner Losh cont = menu.run(sel_entry.submenu()); 333a7cf0562SKyle Evans elseif (sel_entry.entry_type == core.MENU_RETURN) then 334ef625845SKyle Evans -- allow entry to have a function/side effect 335ef625845SKyle Evans if (sel_entry.func ~= nil) then 336ef625845SKyle Evans sel_entry.func(); 337ef625845SKyle Evans end 338088b4f5fSWarner Losh -- break recurse 339088b4f5fSWarner Losh cont = false; 340088b4f5fSWarner Losh end 341088b4f5fSWarner Losh -- if we got an alias key the screen is out of date: 342088b4f5fSWarner Losh screen.clear(); 343088b4f5fSWarner Losh screen.defcursor(); 344088b4f5fSWarner Losh alias_table = drawer.drawscreen(m); 345088b4f5fSWarner Losh end 346088b4f5fSWarner Losh end 347088b4f5fSWarner Losh 348088b4f5fSWarner Losh if (m == menu.welcome) then 349088b4f5fSWarner Losh screen.defcursor(); 350088b4f5fSWarner Losh print("Exiting menu!"); 351fa4a2394SKyle Evans config.loadelf(); 352088b4f5fSWarner Losh return false; 353088b4f5fSWarner Losh end 354088b4f5fSWarner Losh 355088b4f5fSWarner Losh return true; 356088b4f5fSWarner Loshend 357088b4f5fSWarner Losh 358088b4f5fSWarner Loshfunction menu.skip() 359b140d14bSKyle Evans if (core.isSerialBoot()) then 360088b4f5fSWarner Losh return true; 361088b4f5fSWarner Losh end 362088b4f5fSWarner Losh local c = string.lower(loader.getenv("console") or ""); 36324a1bd54SKyle Evans if ((c:match("^efi[ ;]") or c:match("[ ;]efi[ ;]")) ~= nil) then 364088b4f5fSWarner Losh return true; 365088b4f5fSWarner Losh end 366088b4f5fSWarner Losh 367088b4f5fSWarner Losh c = string.lower(loader.getenv("beastie_disable") or ""); 368088b4f5fSWarner Losh print("beastie_disable", c); 369088b4f5fSWarner Losh return c == "yes"; 370088b4f5fSWarner Loshend 371088b4f5fSWarner Losh 372088b4f5fSWarner Loshfunction menu.autoboot() 37324a1bd54SKyle Evans if (menu.already_autoboot == true) then 374088b4f5fSWarner Losh return; 375088b4f5fSWarner Losh end 376088b4f5fSWarner Losh menu.already_autoboot = true; 377088b4f5fSWarner Losh 378088b4f5fSWarner Losh local ab = loader.getenv("autoboot_delay"); 379702b460dSKyle Evans if (ab ~= nil) and (ab:lower() == "no") then 380702b460dSKyle Evans return; 381702b460dSKyle Evans elseif (tonumber(ab) == -1) then 382088b4f5fSWarner Losh core.boot(); 383088b4f5fSWarner Losh end 384088b4f5fSWarner Losh ab = tonumber(ab) or 10; 385088b4f5fSWarner Losh 386088b4f5fSWarner Losh local x = loader.getenv("loader_menu_timeout_x") or 5; 387088b4f5fSWarner Losh local y = loader.getenv("loader_menu_timeout_y") or 22; 388088b4f5fSWarner Losh 389088b4f5fSWarner Losh local endtime = loader.time() + ab; 390088b4f5fSWarner Losh local time; 391088b4f5fSWarner Losh 392088b4f5fSWarner Losh repeat 393088b4f5fSWarner Losh time = endtime - loader.time(); 394088b4f5fSWarner Losh screen.setcursor(x, y); 39557099121SKyle Evans print("Autoboot in " .. time .. 39657099121SKyle Evans " seconds, hit [Enter] to boot" .. 39757099121SKyle Evans " or any other key to stop "); 398088b4f5fSWarner Losh screen.defcursor(); 39924a1bd54SKyle Evans if (io.ischar()) then 400088b4f5fSWarner Losh local ch = io.getchar(); 40124a1bd54SKyle Evans if (ch == core.KEY_ENTER) then 402088b4f5fSWarner Losh break; 403088b4f5fSWarner Losh else 404088b4f5fSWarner Losh -- erase autoboot msg 405088b4f5fSWarner Losh screen.setcursor(0, y); 406088b4f5fSWarner Losh print(" " 407088b4f5fSWarner Losh .. " "); 408088b4f5fSWarner Losh screen.defcursor(); 409088b4f5fSWarner Losh return; 410088b4f5fSWarner Losh end 411088b4f5fSWarner Losh end 412088b4f5fSWarner Losh 413088b4f5fSWarner Losh loader.delay(50000); 414088b4f5fSWarner Losh until time <= 0 415088b4f5fSWarner Losh core.boot(); 416088b4f5fSWarner Losh 417088b4f5fSWarner Loshend 418088b4f5fSWarner Losh 419088b4f5fSWarner Loshfunction OnOff(str, b) 420088b4f5fSWarner Losh if (b) then 42157099121SKyle Evans return str .. color.escapef(color.GREEN) .. "On" .. 42257099121SKyle Evans color.escapef(color.WHITE); 423088b4f5fSWarner Losh else 42457099121SKyle Evans return str .. color.escapef(color.RED) .. "off" .. 42557099121SKyle Evans color.escapef(color.WHITE); 426088b4f5fSWarner Losh end 427088b4f5fSWarner Loshend 428088b4f5fSWarner Losh 42924a1bd54SKyle Evansreturn menu; 430