xref: /freebsd/stand/lua/config.lua (revision 60fde7ce5d7bf5d94290720ea53db5701ab406a8)
1--
2-- SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3--
4-- Copyright (c) 2015 Pedro Souza <pedrosouza@freebsd.org>
5-- Copyright (C) 2018 Kyle Evans <kevans@FreeBSD.org>
6-- All rights reserved.
7--
8-- Redistribution and use in source and binary forms, with or without
9-- modification, are permitted provided that the following conditions
10-- are met:
11-- 1. Redistributions of source code must retain the above copyright
12--    notice, this list of conditions and the following disclaimer.
13-- 2. Redistributions in binary form must reproduce the above copyright
14--    notice, this list of conditions and the following disclaimer in the
15--    documentation and/or other materials provided with the distribution.
16--
17-- THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18-- ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19-- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20-- ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21-- FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22-- DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23-- OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24-- HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25-- LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26-- OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27-- SUCH DAMAGE.
28--
29-- $FreeBSD$
30--
31
32local hook = require("hook")
33
34local config = {}
35local modules = {}
36local carousel_choices = {}
37-- Which variables we changed
38local env_changed = {}
39-- Values to restore env to (nil to unset)
40local env_restore = {}
41
42local MSG_FAILEXEC = "Failed to exec '%s'"
43local MSG_FAILSETENV = "Failed to '%s' with value: %s"
44local MSG_FAILOPENCFG = "Failed to open config: '%s'"
45local MSG_FAILREADCFG = "Failed to read config: '%s'"
46local MSG_FAILPARSECFG = "Failed to parse config: '%s'"
47local MSG_FAILEXBEF = "Failed to execute '%s' before loading '%s'"
48local MSG_FAILEXMOD = "Failed to execute '%s'"
49local MSG_FAILEXAF = "Failed to execute '%s' after loading '%s'"
50local MSG_MALFORMED = "Malformed line (%d):\n\t'%s'"
51local MSG_DEFAULTKERNFAIL = "No kernel set, failed to load from module_path"
52local MSG_KERNFAIL = "Failed to load kernel '%s'"
53local MSG_KERNLOADING = "Loading kernel..."
54local MSG_MODLOADING = "Loading configured modules..."
55local MSG_MODLOADFAIL = "Could not load one or more modules!"
56
57local function restoreEnv()
58	-- Examine changed environment variables
59	for k, v in pairs(env_changed) do
60		local restore_value = env_restore[k]
61		if restore_value == nil then
62			-- This one doesn't need restored for some reason
63			goto continue
64		end
65		local current_value = loader.getenv(k)
66		if current_value ~= v then
67			-- This was overwritten by some action taken on the menu
68			-- most likely; we'll leave it be.
69			goto continue
70		end
71		restore_value = restore_value.value
72		if restore_value ~= nil then
73			loader.setenv(k, restore_value)
74		else
75			loader.unsetenv(k)
76		end
77		::continue::
78	end
79
80	env_changed = {}
81	env_restore = {}
82end
83
84local function setEnv(key, value)
85	-- Track the original value for this if we haven't already
86	if env_restore[key] == nil then
87		env_restore[key] = {value = loader.getenv(key)}
88	end
89
90	env_changed[key] = value
91
92	return loader.setenv(key, value)
93end
94
95-- name here is one of 'name', 'type', flags', 'before', 'after', or 'error.'
96-- These are set from lines in loader.conf(5): ${key}_${name}="${value}" where
97-- ${key} is a module name.
98local function setKey(key, name, value)
99	if modules[key] == nil then
100		modules[key] = {}
101	end
102	modules[key][name] = value
103end
104
105local pattern_table = {
106	{
107		str = "^%s*(#.*)",
108		process = function(_, _)  end,
109	},
110	--  module_load="value"
111	{
112		str = "^%s*([%w_]+)_load%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
113		process = function(k, v)
114			if modules[k] == nil then
115				modules[k] = {}
116			end
117			modules[k].load = v:upper()
118		end,
119	},
120	--  module_name="value"
121	{
122		str = "^%s*([%w_]+)_name%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
123		process = function(k, v)
124			setKey(k, "name", v)
125		end,
126	},
127	--  module_type="value"
128	{
129		str = "^%s*([%w_]+)_type%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
130		process = function(k, v)
131			setKey(k, "type", v)
132		end,
133	},
134	--  module_flags="value"
135	{
136		str = "^%s*([%w_]+)_flags%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
137		process = function(k, v)
138			setKey(k, "flags", v)
139		end,
140	},
141	--  module_before="value"
142	{
143		str = "^%s*([%w_]+)_before%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
144		process = function(k, v)
145			setKey(k, "before", v)
146		end,
147	},
148	--  module_after="value"
149	{
150		str = "^%s*([%w_]+)_after%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
151		process = function(k, v)
152			setKey(k, "after", v)
153		end,
154	},
155	--  module_error="value"
156	{
157		str = "^%s*([%w_]+)_error%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
158		process = function(k, v)
159			setKey(k, "error", v)
160		end,
161	},
162	--  exec="command"
163	{
164		str = "^%s*exec%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
165		process = function(k, _)
166			if cli_execute_unparsed(k) ~= 0 then
167				print(MSG_FAILEXEC:format(k))
168			end
169		end,
170	},
171	--  env_var="value"
172	{
173		str = "^%s*([%w%p]+)%s*=%s*\"([%w%s%p]-)\"%s*(.*)",
174		process = function(k, v)
175			if setEnv(k, v) ~= 0 then
176				print(MSG_FAILSETENV:format(k, v))
177			end
178		end,
179	},
180	--  env_var=num
181	{
182		str = "^%s*([%w%p]+)%s*=%s*(%d+)%s*(.*)",
183		process = function(k, v)
184			if setEnv(k, v) ~= 0 then
185				print(MSG_FAILSETENV:format(k, tostring(v)))
186			end
187		end,
188	},
189}
190
191local function isValidComment(line)
192	if line ~= nil then
193		local s = line:match("^%s*#.*")
194		if s == nil then
195			s = line:match("^%s*$")
196		end
197		if s == nil then
198			return false
199		end
200	end
201	return true
202end
203
204local function loadModule(mod, silent)
205	local status = true
206	local pstatus
207	for k, v in pairs(mod) do
208		if v.load == "YES" then
209			local str = "load "
210			if v.flags ~= nil then
211				str = str .. v.flags .. " "
212			end
213			if v.type ~= nil then
214				str = str .. "-t " .. v.type .. " "
215			end
216			if v.name ~= nil then
217				str = str .. v.name
218			else
219				str = str .. k
220			end
221			if v.before ~= nil then
222				pstatus = cli_execute_unparsed(v.before) == 0
223				if not pstatus and not silent then
224					print(MSG_FAILEXBEF:format(v.before, k))
225				end
226				status = status and pstatus
227			end
228
229			if cli_execute_unparsed(str) ~= 0 then
230				if not silent then
231					print(MSG_FAILEXMOD:format(str))
232				end
233				if v.error ~= nil then
234					cli_execute_unparsed(v.error)
235				end
236				status = false
237			end
238
239			if v.after ~= nil then
240				pstatus = cli_execute_unparsed(v.after) == 0
241				if not pstatus and not silent then
242					print(MSG_FAILEXAF:format(v.after, k))
243				end
244				status = status and pstatus
245			end
246
247--		else
248--			if not silent then
249--				print("Skipping module '". . k .. "'")
250--			end
251		end
252	end
253
254	return status
255end
256
257
258local function readFile(name, silent)
259	local f = io.open(name)
260	if f == nil then
261		if not silent then
262			print(MSG_FAILOPENCFG:format(name))
263		end
264		return nil
265	end
266
267	local text, _ = io.read(f)
268	-- We might have read in the whole file, this won't be needed any more.
269	io.close(f)
270
271	if text == nil then
272		if not silent then
273			print(MSG_FAILREADCFG:format(name))
274		end
275		return nil
276	end
277	return text
278end
279
280local function checkNextboot()
281	local nextboot_file = loader.getenv("nextboot_file")
282	if nextboot_file == nil then
283		return
284	end
285
286	local text = readFile(nextboot_file, true)
287	if text == nil then
288		return
289	end
290
291	if text:match("^nextboot_enable=\"NO\"") ~= nil then
292		-- We're done; nextboot is not enabled
293		return
294	end
295
296	if not config.parse(text) then
297		print(MSG_FAILPARSECFG:format(nextboot_file))
298	end
299
300	-- Attempt to rewrite the first line and only the first line of the
301	-- nextboot_file. We overwrite it with nextboot_enable="NO", then
302	-- check for that on load.
303	-- It's worth noting that this won't work on every filesystem, so we
304	-- won't do anything notable if we have any errors in this process.
305	local nfile = io.open(nextboot_file, 'w')
306	if nfile ~= nil then
307		-- We need the trailing space here to account for the extra
308		-- character taken up by the string nextboot_enable="YES"
309		-- Or new end quotation mark lands on the S, and we want to
310		-- rewrite the entirety of the first line.
311		io.write(nfile, "nextboot_enable=\"NO\" ")
312		io.close(nfile)
313	end
314end
315
316-- Module exports
317config.verbose = false
318
319-- The first item in every carousel is always the default item.
320function config.getCarouselIndex(id)
321	local val = carousel_choices[id]
322	if val == nil then
323		return 1
324	end
325	return val
326end
327
328function config.setCarouselIndex(id, idx)
329	carousel_choices[id] = idx
330end
331
332-- Returns true if we processed the file successfully, false if we did not.
333-- If 'silent' is true, being unable to read the file is not considered a
334-- failure.
335function config.processFile(name, silent)
336	if silent == nil then
337		silent = false
338	end
339
340	local text = readFile(name, silent)
341	if text == nil then
342		return silent
343	end
344
345	return config.parse(text)
346end
347
348-- silent runs will not return false if we fail to open the file
349function config.parse(text)
350	local n = 1
351	local status = true
352
353	for line in text:gmatch("([^\n]+)") do
354		if line:match("^%s*$") == nil then
355			local found = false
356
357			for _, val in ipairs(pattern_table) do
358				local k, v, c = line:match(val.str)
359				if k ~= nil then
360					found = true
361
362					if isValidComment(c) then
363						val.process(k, v)
364					else
365						print(MSG_MALFORMED:format(n,
366						    line))
367						status = false
368					end
369
370					break
371				end
372			end
373
374			if not found then
375				print(MSG_MALFORMED:format(n, line))
376				status = false
377			end
378		end
379		n = n + 1
380	end
381
382	return status
383end
384
385-- other_kernel is optionally the name of a kernel to load, if not the default
386-- or autoloaded default from the module_path
387function config.loadKernel(other_kernel)
388	local flags = loader.getenv("kernel_options") or ""
389	local kernel = other_kernel or loader.getenv("kernel")
390
391	local function tryLoad(names)
392		for name in names:gmatch("([^;]+)%s*;?") do
393			local r = loader.perform("load " .. flags ..
394			    " " .. name)
395			if r == 0 then
396				return name
397			end
398		end
399		return nil
400	end
401
402	local function loadBootfile()
403		local bootfile = loader.getenv("bootfile")
404
405		-- append default kernel name
406		if bootfile == nil then
407			bootfile = "kernel"
408		else
409			bootfile = bootfile .. ";kernel"
410		end
411
412		return tryLoad(bootfile)
413	end
414
415	-- kernel not set, try load from default module_path
416	if kernel == nil then
417		local res = loadBootfile()
418
419		if res ~= nil then
420			-- Default kernel is loaded
421			config.kernel_loaded = nil
422			return true
423		else
424			print(MSG_DEFAULTKERNFAIL)
425			return false
426		end
427	else
428		-- Use our cached module_path, so we don't end up with multiple
429		-- automatically added kernel paths to our final module_path
430		local module_path = config.module_path
431		local res
432
433		if other_kernel ~= nil then
434			kernel = other_kernel
435		end
436		-- first try load kernel with module_path = /boot/${kernel}
437		-- then try load with module_path=${kernel}
438		local paths = {"/boot/" .. kernel, kernel}
439
440		for _, v in pairs(paths) do
441			loader.setenv("module_path", v)
442			res = loadBootfile()
443
444			-- succeeded, add path to module_path
445			if res ~= nil then
446				config.kernel_loaded = kernel
447				if module_path ~= nil then
448					loader.setenv("module_path", v .. ";" ..
449					    module_path)
450				end
451				return true
452			end
453		end
454
455		-- failed to load with ${kernel} as a directory
456		-- try as a file
457		res = tryLoad(kernel)
458		if res ~= nil then
459			config.kernel_loaded = kernel
460			return true
461		else
462			print(MSG_KERNFAIL:format(kernel))
463			return false
464		end
465	end
466end
467
468function config.selectKernel(kernel)
469	config.kernel_selected = kernel
470end
471
472function config.load(file)
473	if not file then
474		file = "/boot/defaults/loader.conf"
475	end
476
477	if not config.processFile(file) then
478		print(MSG_FAILPARSECFG:format(file))
479	end
480
481	local f = loader.getenv("loader_conf_files")
482	if f ~= nil then
483		for name in f:gmatch("([%w%p]+)%s*") do
484			-- These may or may not exist, and that's ok. Do a
485			-- silent parse so that we complain on parse errors but
486			-- not for them simply not existing.
487			if not config.processFile(name, true) then
488				print(MSG_FAILPARSECFG:format(name))
489			end
490		end
491	end
492
493	checkNextboot()
494
495	-- Cache the provided module_path at load time for later use
496	config.module_path = loader.getenv("module_path")
497	local verbose = loader.getenv("verbose_loading")
498	if verbose == nil then
499		verbose = "no"
500	end
501	config.verbose = verbose:lower() == "yes"
502end
503
504-- Reload configuration
505function config.reload(file)
506	modules = {}
507	restoreEnv()
508	config.load(file)
509	hook.runAll("config.reloaded")
510end
511
512function config.loadelf()
513	local kernel = config.kernel_selected or config.kernel_loaded
514	local loaded
515
516	print(MSG_KERNLOADING)
517	loaded = config.loadKernel(kernel)
518
519	if not loaded then
520		return
521	end
522
523	print(MSG_MODLOADING)
524	if not loadModule(modules, not config.verbose) then
525		print(MSG_MODLOADFAIL)
526	end
527end
528
529hook.registerType("config.reloaded")
530return config
531