mirror of
https://github.com/opnsense/src.git
synced 2026-06-12 18:20:49 -04:00
While these should be moved to the new format, it wasn't my intention to force them over immediately. Downstreams may embed their own brands in drawer.lua, and we shouldn't break them for something like this. Move adapt_fb_shim() up and use it for preloaded definitions to avoid forcing the matter for now. Perhaps in the future we'll start writing out warnings for those that do need adapted. Reported by: 0x1eef on IRC
620 lines
15 KiB
Lua
620 lines
15 KiB
Lua
--
|
|
-- SPDX-License-Identifier: BSD-2-Clause
|
|
--
|
|
-- Copyright (c) 2015 Pedro Souza <pedrosouza@freebsd.org>
|
|
-- Copyright (c) 2018 Kyle Evans <kevans@FreeBSD.org>
|
|
-- All rights reserved.
|
|
--
|
|
-- Redistribution and use in source and binary forms, with or without
|
|
-- modification, are permitted provided that the following conditions
|
|
-- are met:
|
|
-- 1. Redistributions of source code must retain the above copyright
|
|
-- notice, this list of conditions and the following disclaimer.
|
|
-- 2. Redistributions in binary form must reproduce the above copyright
|
|
-- notice, this list of conditions and the following disclaimer in the
|
|
-- documentation and/or other materials provided with the distribution.
|
|
--
|
|
-- THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
-- ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
-- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
-- ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
-- FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
-- DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
-- OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
-- HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
-- LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
-- OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
-- SUCH DAMAGE.
|
|
--
|
|
|
|
local color = require("color")
|
|
local config = require("config")
|
|
local core = require("core")
|
|
local screen = require("screen")
|
|
|
|
local drawer = {}
|
|
|
|
local fbsd_brand
|
|
local none
|
|
|
|
local menu_name_handlers
|
|
local branddefs
|
|
local logodefs
|
|
local brand_position
|
|
local logo_position
|
|
local menu_position
|
|
local frame_size
|
|
local default_shift
|
|
local shift
|
|
|
|
-- Make this code compatible with older loader binaries. We moved the term_*
|
|
-- functions from loader to the gfx. if we're running on an older loader that
|
|
-- has these functions, create aliases for them in gfx. The loader binary might
|
|
-- be so old as to not have them, but in that case, we want to copy the nil
|
|
-- values. The new loader will provide loader.* versions of all the gfx.*
|
|
-- functions for backwards compatibility, so we only define the functions we use
|
|
-- here.
|
|
if gfx == nil then
|
|
gfx = {}
|
|
gfx.term_drawrect = loader.term_drawrect
|
|
gfx.term_putimage = loader.term_putimage
|
|
end
|
|
|
|
local function menuEntryName(drawing_menu, entry)
|
|
local name_handler = menu_name_handlers[entry.entry_type]
|
|
|
|
if name_handler ~= nil then
|
|
return name_handler(drawing_menu, entry)
|
|
end
|
|
if type(entry.name) == "function" then
|
|
return entry.name()
|
|
end
|
|
return entry.name
|
|
end
|
|
|
|
local function processFile(gfxname)
|
|
if gfxname == nil then
|
|
return false, "Missing filename"
|
|
end
|
|
|
|
local ret = try_include('gfx-' .. gfxname)
|
|
if ret == nil then
|
|
return false, "Failed to include gfx-" .. gfxname
|
|
end
|
|
|
|
-- Legacy format
|
|
if type(ret) ~= "table" then
|
|
return true
|
|
end
|
|
|
|
for gfxtype, def in pairs(ret) do
|
|
if gfxtype == "brand" then
|
|
drawer.addBrand(gfxname, def)
|
|
elseif gfxtype == "logo" then
|
|
drawer.addLogo(gfxname, def)
|
|
else
|
|
return false, "Unknown graphics type '" .. gfxtype ..
|
|
"'"
|
|
end
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
-- Backwards compatibility shims for previous FreeBSD versions, please document
|
|
-- new additions
|
|
local function adapt_fb_shim(def)
|
|
-- In FreeBSD 14.x+, we have improved framebuffer support in the loader
|
|
-- and some graphics may have images that we can actually draw on the
|
|
-- screen. Those graphics may come with shifts that are distinct from
|
|
-- the ASCII version, so we move both ascii and image versions into
|
|
-- their own tables.
|
|
if not def.ascii then
|
|
def.ascii = {
|
|
image = def.graphic,
|
|
requires_color = def.requires_color,
|
|
shift = def.shift,
|
|
}
|
|
end
|
|
if def.image then
|
|
assert(not def.fb,
|
|
"Unrecognized graphic definition format")
|
|
|
|
-- Legacy images may have adapted a shift from the ASCII
|
|
-- version, or perhaps we just didn't care enough to adjust it.
|
|
-- Steal the shift.
|
|
def.fb = {
|
|
image = def.image,
|
|
width = def.image_rl,
|
|
shift = def.shift,
|
|
}
|
|
end
|
|
|
|
def.adapted = true
|
|
return def
|
|
end
|
|
|
|
local function getBranddef(brand)
|
|
if brand == nil then
|
|
return nil
|
|
end
|
|
-- Look it up
|
|
local branddef = branddefs[brand]
|
|
|
|
-- Try to pull it in
|
|
if branddef == nil then
|
|
local res, err = processFile(brand)
|
|
if not res then
|
|
-- This fallback should go away after FreeBSD 13.
|
|
try_include('brand-' .. brand)
|
|
-- If the fallback also failed, print whatever error
|
|
-- we encountered in the original processing.
|
|
if branddefs[brand] == nil then
|
|
print(err)
|
|
return nil
|
|
end
|
|
end
|
|
|
|
branddef = branddefs[brand]
|
|
elseif not branddef.adapted then
|
|
adapt_fb_shim(branddef)
|
|
end
|
|
|
|
return branddef
|
|
end
|
|
|
|
local function getLogodef(logo)
|
|
if logo == nil then
|
|
return nil
|
|
end
|
|
-- Look it up
|
|
local logodef = logodefs[logo]
|
|
|
|
-- Try to pull it in
|
|
if logodef == nil then
|
|
local res, err = processFile(logo)
|
|
if not res then
|
|
-- This fallback should go away after FreeBSD 13.
|
|
try_include('logo-' .. logo)
|
|
-- If the fallback also failed, print whatever error
|
|
-- we encountered in the original processing.
|
|
if logodefs[logo] == nil then
|
|
print(err)
|
|
return nil
|
|
end
|
|
end
|
|
|
|
logodef = logodefs[logo]
|
|
elseif not logodef.adapted then
|
|
adapt_fb_shim(logodef)
|
|
end
|
|
|
|
return logodef
|
|
end
|
|
|
|
local function draw(x, y, logo)
|
|
for i = 1, #logo do
|
|
screen.setcursor(x, y + i - 1)
|
|
printc(logo[i])
|
|
end
|
|
end
|
|
|
|
local function drawmenu(menudef)
|
|
local x = menu_position.x
|
|
local y = menu_position.y
|
|
|
|
if string.lower(loader.getenv("loader_menu") or "") == "none" then
|
|
return
|
|
end
|
|
|
|
x = x + shift.x
|
|
y = y + shift.y
|
|
|
|
-- print the menu and build the alias table
|
|
local alias_table = {}
|
|
local entry_num = 0
|
|
local menu_entries = menudef.entries
|
|
local effective_line_num = 0
|
|
if type(menu_entries) == "function" then
|
|
menu_entries = menu_entries()
|
|
end
|
|
for _, e in ipairs(menu_entries) do
|
|
-- Allow menu items to be conditionally visible by specifying
|
|
-- a visible function.
|
|
if e.visible ~= nil and not e.visible() then
|
|
goto continue
|
|
end
|
|
effective_line_num = effective_line_num + 1
|
|
if e.entry_type ~= core.MENU_SEPARATOR then
|
|
entry_num = entry_num + 1
|
|
screen.setcursor(x, y + effective_line_num)
|
|
|
|
printc(entry_num .. ". " .. menuEntryName(menudef, e))
|
|
|
|
-- fill the alias table
|
|
alias_table[tostring(entry_num)] = e
|
|
if e.alias ~= nil then
|
|
for _, a in ipairs(e.alias) do
|
|
alias_table[a] = e
|
|
end
|
|
end
|
|
else
|
|
screen.setcursor(x, y + effective_line_num)
|
|
printc(menuEntryName(menudef, e))
|
|
end
|
|
::continue::
|
|
end
|
|
return alias_table
|
|
end
|
|
|
|
local function defaultframe()
|
|
if core.isSerialConsole() then
|
|
return "ascii"
|
|
end
|
|
return "double"
|
|
end
|
|
|
|
local function gfxenabled()
|
|
return (loader.getenv("loader_gfx") or "yes"):lower() ~= "no"
|
|
end
|
|
local function gfxcapable()
|
|
return core.isFramebufferConsole() and gfx.term_putimage
|
|
end
|
|
|
|
local function drawframe()
|
|
local x = menu_position.x - 3
|
|
local y = menu_position.y - 1
|
|
local w = frame_size.w
|
|
local h = frame_size.h
|
|
|
|
local framestyle = loader.getenv("loader_menu_frame") or defaultframe()
|
|
local framespec = drawer.frame_styles[framestyle]
|
|
-- If we don't have a framespec for the current frame style, just don't
|
|
-- draw a box.
|
|
if framespec == nil then
|
|
return false
|
|
end
|
|
|
|
local hl = framespec.horizontal
|
|
local vl = framespec.vertical
|
|
|
|
local tl = framespec.top_left
|
|
local bl = framespec.bottom_left
|
|
local tr = framespec.top_right
|
|
local br = framespec.bottom_right
|
|
|
|
x = x + shift.x
|
|
y = y + shift.y
|
|
|
|
if gfxenabled() and gfxcapable() then
|
|
gfx.term_drawrect(x, y, x + w, y + h)
|
|
return true
|
|
end
|
|
|
|
screen.setcursor(x, y); printc(tl)
|
|
screen.setcursor(x, y + h); printc(bl)
|
|
screen.setcursor(x + w, y); printc(tr)
|
|
screen.setcursor(x + w, y + h); printc(br)
|
|
|
|
screen.setcursor(x + 1, y)
|
|
for _ = 1, w - 1 do
|
|
printc(hl)
|
|
end
|
|
|
|
screen.setcursor(x + 1, y + h)
|
|
for _ = 1, w - 1 do
|
|
printc(hl)
|
|
end
|
|
|
|
for i = 1, h - 1 do
|
|
screen.setcursor(x, y + i)
|
|
printc(vl)
|
|
screen.setcursor(x + w, y + i)
|
|
printc(vl)
|
|
end
|
|
return true
|
|
end
|
|
|
|
local function drawbox()
|
|
local x = menu_position.x - 3
|
|
local y = menu_position.y - 1
|
|
local w = frame_size.w
|
|
local menu_header = loader.getenv("loader_menu_title") or
|
|
"Welcome to FreeBSD"
|
|
local menu_header_align = loader.getenv("loader_menu_title_align")
|
|
local menu_header_x
|
|
|
|
if string.lower(loader.getenv("loader_menu") or "") == "none" then
|
|
return
|
|
end
|
|
|
|
x = x + shift.x
|
|
y = y + shift.y
|
|
|
|
if drawframe(x, y, w) == false then
|
|
return
|
|
end
|
|
|
|
if menu_header_align ~= nil then
|
|
menu_header_align = menu_header_align:lower()
|
|
if menu_header_align == "left" then
|
|
-- Just inside the left border on top
|
|
menu_header_x = x + 1
|
|
elseif menu_header_align == "right" then
|
|
-- Just inside the right border on top
|
|
menu_header_x = x + w - #menu_header
|
|
end
|
|
end
|
|
if menu_header_x == nil then
|
|
menu_header_x = x + (w // 2) - (#menu_header // 2)
|
|
end
|
|
screen.setcursor(menu_header_x - 1, y)
|
|
if menu_header ~= "" then
|
|
printc(" " .. menu_header .. " ")
|
|
end
|
|
|
|
end
|
|
|
|
local function drawbrand()
|
|
local x = tonumber(loader.getenv("loader_brand_x")) or
|
|
brand_position.x
|
|
local y = tonumber(loader.getenv("loader_brand_y")) or
|
|
brand_position.y
|
|
|
|
local branddef = getBranddef(loader.getenv("loader_brand"))
|
|
|
|
if branddef == nil then
|
|
branddef = getBranddef(drawer.default_brand)
|
|
end
|
|
|
|
local graphic = branddef.ascii.image
|
|
|
|
x = x + shift.x
|
|
y = y + shift.y
|
|
|
|
local gfx_requested = branddef.fb and gfxenabled()
|
|
if gfx_requested and gfxcapable() then
|
|
if branddef.fb.shift then
|
|
x = x + (branddef.fb.shift.x or 0)
|
|
y = y + (branddef.fb.shift.y or 0)
|
|
end
|
|
if gfx.term_putimage(branddef.fb.image, x, y, 0, 7, 0) then
|
|
return true
|
|
end
|
|
elseif branddef.ascii.shift then
|
|
x = x + (branddef.ascii.shift.x or 0)
|
|
y = y + (branddef.ascii.shift.y or 0)
|
|
end
|
|
draw(x, y, graphic)
|
|
end
|
|
|
|
local function drawlogo()
|
|
local x = tonumber(loader.getenv("loader_logo_x")) or
|
|
logo_position.x
|
|
local y = tonumber(loader.getenv("loader_logo_y")) or
|
|
logo_position.y
|
|
|
|
local logo = loader.getenv("loader_logo")
|
|
local colored = color.isEnabled()
|
|
|
|
local logodef = getLogodef(logo)
|
|
|
|
if logodef == nil or logodef.ascii == nil or
|
|
(not colored and logodef.ascii.requires_color) then
|
|
-- Choose a sensible default
|
|
if colored then
|
|
logodef = getLogodef(drawer.default_color_logodef)
|
|
else
|
|
logodef = getLogodef(drawer.default_bw_logodef)
|
|
end
|
|
|
|
-- Something has gone terribly wrong.
|
|
if logodef == nil then
|
|
logodef = getLogodef(drawer.default_fallback_logodef)
|
|
end
|
|
end
|
|
|
|
-- This is a special little hack for the "none" logo to re-align the
|
|
-- menu and the brand to avoid having a lot of extraneous whitespace on
|
|
-- the right side.
|
|
if logodef and logodef.ascii.image == none then
|
|
shift = logodef.shift
|
|
else
|
|
shift = default_shift
|
|
end
|
|
|
|
x = x + shift.x
|
|
y = y + shift.y
|
|
|
|
local gfx_requested = logodef.fb and gfxenabled()
|
|
if gfx_requested and gfxcapable() then
|
|
local y1 = logodef.fb.width or 15
|
|
|
|
if logodef.fb.shift then
|
|
x = x + (logodef.fb.shift.x or 0)
|
|
y = y + (logodef.fb.shift.y or 0)
|
|
end
|
|
if gfx.term_putimage(logodef.fb.image, x, y, 0, y + y1, 0) then
|
|
return true
|
|
end
|
|
elseif logodef.ascii.shift then
|
|
x = x + (logodef.ascii.shift.x or 0)
|
|
y = y + (logodef.ascii.shift.y or 0)
|
|
end
|
|
|
|
draw(x, y, logodef.ascii.image)
|
|
end
|
|
|
|
local function drawitem(func)
|
|
local console = loader.getenv("console")
|
|
|
|
for c in string.gmatch(console, "%w+") do
|
|
loader.setenv("console", c)
|
|
func()
|
|
end
|
|
loader.setenv("console", console)
|
|
end
|
|
|
|
fbsd_brand = {
|
|
" ______ ____ _____ _____ ",
|
|
" | ____| | _ \\ / ____| __ \\ ",
|
|
" | |___ _ __ ___ ___ | |_) | (___ | | | |",
|
|
" | ___| '__/ _ \\/ _ \\| _ < \\___ \\| | | |",
|
|
" | | | | | __/ __/| |_) |____) | |__| |",
|
|
" | | | | | | || | | |",
|
|
" |_| |_| \\___|\\___||____/|_____/|_____/ "
|
|
}
|
|
none = {""}
|
|
|
|
menu_name_handlers = {
|
|
-- Menu name handlers should take the menu being drawn and entry being
|
|
-- drawn as parameters, and return the name of the item.
|
|
-- This is designed so that everything, including menu separators, may
|
|
-- have their names derived differently. The default action for entry
|
|
-- types not specified here is to use entry.name directly.
|
|
[core.MENU_SEPARATOR] = function(_, entry)
|
|
if entry.name ~= nil then
|
|
if type(entry.name) == "function" then
|
|
return entry.name()
|
|
end
|
|
return entry.name
|
|
end
|
|
return ""
|
|
end,
|
|
[core.MENU_CAROUSEL_ENTRY] = function(_, entry)
|
|
local carid = entry.carousel_id
|
|
local caridx = config.getCarouselIndex(carid)
|
|
local choices = entry.items
|
|
if type(choices) == "function" then
|
|
choices = choices()
|
|
end
|
|
if #choices < caridx then
|
|
caridx = 1
|
|
end
|
|
return entry.name(caridx, choices[caridx], choices)
|
|
end,
|
|
}
|
|
|
|
branddefs = {
|
|
-- Indexed by valid values for loader_brand in loader.conf(5). Valid
|
|
-- keys are: graphic (table depicting graphic)
|
|
["fbsd"] = {
|
|
ascii = {
|
|
image = fbsd_brand,
|
|
},
|
|
fb = {
|
|
image = "/boot/images/freebsd-brand-rev.png",
|
|
},
|
|
},
|
|
["none"] = {
|
|
ascii = { image = none },
|
|
},
|
|
}
|
|
|
|
logodefs = {
|
|
-- Indexed by valid values for loader_logo in loader.conf(5). Valid keys
|
|
-- are: requires_color (boolean), graphic (table depicting graphic), and
|
|
-- shift (table containing x and y).
|
|
["tribute"] = {
|
|
ascii = {
|
|
image = fbsd_brand,
|
|
},
|
|
},
|
|
["tributebw"] = {
|
|
ascii = {
|
|
image = fbsd_brand,
|
|
},
|
|
},
|
|
["none"] = {
|
|
ascii = {
|
|
image = none,
|
|
},
|
|
shift = {x = 17, y = 0},
|
|
},
|
|
}
|
|
|
|
brand_position = {x = 2, y = 1}
|
|
logo_position = {x = 40, y = 10}
|
|
menu_position = {x = 5, y = 10}
|
|
frame_size = {w = 39, h = 14}
|
|
default_shift = {x = 0, y = 0}
|
|
shift = default_shift
|
|
|
|
-- Module exports
|
|
drawer.default_brand = 'fbsd'
|
|
drawer.default_color_logodef = 'orb'
|
|
drawer.default_bw_logodef = 'orbbw'
|
|
-- For when things go terribly wrong; this def should be present here in the
|
|
-- drawer module in case it's a filesystem issue.
|
|
drawer.default_fallback_logodef = 'none'
|
|
|
|
function drawer.addBrand(name, def)
|
|
branddefs[name] = adapt_fb_shim(def)
|
|
end
|
|
|
|
function drawer.addLogo(name, def)
|
|
logodefs[name] = adapt_fb_shim(def)
|
|
end
|
|
|
|
drawer.frame_styles = {
|
|
-- Indexed by valid values for loader_menu_frame in loader.conf(5).
|
|
-- All of the keys appearing below must be set for any menu frame style
|
|
-- added to drawer.frame_styles.
|
|
["ascii"] = {
|
|
horizontal = "-",
|
|
vertical = "|",
|
|
top_left = "+",
|
|
bottom_left = "+",
|
|
top_right = "+",
|
|
bottom_right = "+",
|
|
},
|
|
}
|
|
|
|
if core.hasUnicode() then
|
|
-- unicode based framing characters
|
|
drawer.frame_styles["single"] = {
|
|
horizontal = "\xE2\x94\x80",
|
|
vertical = "\xE2\x94\x82",
|
|
top_left = "\xE2\x94\x8C",
|
|
bottom_left = "\xE2\x94\x94",
|
|
top_right = "\xE2\x94\x90",
|
|
bottom_right = "\xE2\x94\x98",
|
|
}
|
|
drawer.frame_styles["double"] = {
|
|
horizontal = "\xE2\x95\x90",
|
|
vertical = "\xE2\x95\x91",
|
|
top_left = "\xE2\x95\x94",
|
|
bottom_left = "\xE2\x95\x9A",
|
|
top_right = "\xE2\x95\x97",
|
|
bottom_right = "\xE2\x95\x9D",
|
|
}
|
|
else
|
|
-- non-unicode cons25-style framing characters
|
|
drawer.frame_styles["single"] = {
|
|
horizontal = "\xC4",
|
|
vertical = "\xB3",
|
|
top_left = "\xDA",
|
|
bottom_left = "\xC0",
|
|
top_right = "\xBF",
|
|
bottom_right = "\xD9",
|
|
}
|
|
drawer.frame_styles["double"] = {
|
|
horizontal = "\xCD",
|
|
vertical = "\xBA",
|
|
top_left = "\xC9",
|
|
bottom_left = "\xC8",
|
|
top_right = "\xBB",
|
|
bottom_right = "\xBC",
|
|
}
|
|
end
|
|
|
|
function drawer.drawscreen(menudef)
|
|
-- drawlogo() must go first.
|
|
-- it determines the positions of other elements
|
|
drawitem(drawlogo)
|
|
drawitem(drawbrand)
|
|
drawitem(drawbox)
|
|
return drawmenu(menudef)
|
|
end
|
|
|
|
return drawer
|