mirror of
https://github.com/systemd/systemd.git
synced 2025-01-10 05:18:17 +03:00
bootctl: update to new kernel-install semantics
This makes a bunch of closely related changes: 1. The "entry-token" concept already introduced in kernel-install is now made use of. i.e. specifically there's a new option --entry-token= that can be used to explicitly select by which ID to identify boot loader entries: the machine ID, or some OS ID (ID= or IMAGE_ID= from /etc/os-release, or even some completely different string. The selected string is then persisted to /etc/kernel/entry-token, so that kernel-install can find it there. 2. The --make-machine-id-directory= switch is renamed to --make-entry-directory= since after all it's not necessarily the machine ID the dir is named after, but can be any other string as selected by the entry token. 3. This drops all code to make automatic changes to /etc/machine-info. Specifically, the KERNEL_INSTALL_MACHINE_ID= field is now more generically implemented in /etc/kernel/entry-token described above, hence no need to place it at two locations. And the KERNEL_INSTALL_LAYOUT= field is not configurable by user switch or similar anyway in bootctl, but only read from /etc/kernel/install.conf, and hence copying it from one configuration file to another appears unnecessary, the second copy is fully redundant. Note that this just drops writing these fields, they'll still be honoured when already set.
This commit is contained in:
parent
fe81e346c8
commit
f337f903df
@ -258,23 +258,60 @@
|
|||||||
</varlistentry>
|
</varlistentry>
|
||||||
|
|
||||||
<varlistentry>
|
<varlistentry>
|
||||||
<term><option>--make-machine-id-directory=yes|no</option></term>
|
<term><option>--make-entry-directory=yes|no</option></term>
|
||||||
<listitem><para>Control creation and deletion of the top-level machine ID directory on the file
|
<listitem><para>Controls creation and deletion of the <ulink
|
||||||
system containing boot loader entries (i.e. beneath the file system returned by the
|
url="https://systemd.io/BOOT_LOADER_SPECIFICATION">Boot Loader Specification</ulink> Type #1 entry
|
||||||
<option>--print-boot-path</option> option, see above) during <option>install</option> and
|
directory on the file system containing resources such as kernel images and initial RAM disk images
|
||||||
<option>remove</option>, respectively. Defaults to <literal>no</literal>. See
|
during <option>install</option> and <option>remove</option>, respectively. The directory is named
|
||||||
|
after the entry token, as specified with <option>--entry-token=</option> parameter described below,
|
||||||
|
and is placed immediately below the <varname>$BOOT</varname> root directory (i.e. beneath the file
|
||||||
|
system returned by the <option>--print-boot-path</option> option, see above). Defaults to
|
||||||
|
<literal>no</literal>.</para></listitem>
|
||||||
|
</varlistentry>
|
||||||
|
|
||||||
|
<varlistentry>
|
||||||
|
<term><option>--entry-token=</option></term>
|
||||||
|
|
||||||
|
<listitem><para>Controls how to name and identify boot loader entries for this OS
|
||||||
|
installation. Accepted during <option>install</option>, and takes one of <literal>auto</literal>,
|
||||||
|
<literal>machine-id</literal>, <literal>os-id</literal>, <literal>os-image-id</literal> or an
|
||||||
|
arbitrary string prefixed by <literal>literal:</literal> as argument.</para>
|
||||||
|
|
||||||
|
<para>If set to <option>machine-id</option> the entries are named after the machine ID of the running
|
||||||
|
system (e.g. <literal>b0e793a9baf14b5fa13ecbe84ff637ac</literal>). See
|
||||||
<citerefentry><refentrytitle>machine-id</refentrytitle><manvolnum>5</manvolnum></citerefentry> for
|
<citerefentry><refentrytitle>machine-id</refentrytitle><manvolnum>5</manvolnum></citerefentry> for
|
||||||
details about the machine ID concept and file.</para>
|
details about the machine ID concept and file.</para>
|
||||||
|
|
||||||
<para>Overriding this may be desirable to hide the machine ID from the (unencrypted) ESP, configure a
|
<para>If set to <option>os-id</option> the entries are named after the OS ID of the running system,
|
||||||
<citerefentry><refentrytitle>kernel-install</refentrytitle><manvolnum>8</manvolnum></citerefentry>
|
i.e. the <varname>ID=</varname> field of
|
||||||
script, or, conversely, commit a transient machine ID.</para>
|
<citerefentry><refentrytitle>os-release</refentrytitle><manvolnum>5</manvolnum></citerefentry>
|
||||||
|
(e.g. <literal>fedora</literal>). Similar, if set to <option>os-image-id</option> the entries are
|
||||||
|
named after the OS image ID of the running system, i.e. the <varname>IMAGE_ID=</varname> field of
|
||||||
|
<filename>os-release</filename> (e.g. <literal>vendorx-cashier-system</literal>).</para>
|
||||||
|
|
||||||
<para>The top-level machine ID directory is useful to allow smooth multi-boot installations: each
|
<para>If set to <option>auto</option> (the default), the <filename>/etc/kernel/entry-token</filename>
|
||||||
installed OS instance will have a different machine ID and thus a separate directory to place its
|
file will be read if it exists, and the stored value used. Otherwise if the local machine ID is
|
||||||
boot-time resources in. If this feature is turned off with this option, care needs to be taken that
|
initialized it is used. Otherwise <varname>IMAGE_ID=</varname> from <filename>os-release</filename>
|
||||||
multiple OS instances do not place conflicting files on the shared ESP and Extended Boot Loader
|
will be used, if set. Otherwise, <varname>ID=</varname> from <filename>os-release</filename> will be
|
||||||
Partitions, or that multiple OS instances are not possible.</para></listitem>
|
used, if set.</para>
|
||||||
|
|
||||||
|
<para>Unless set to <literal>machine-id</literal>, or when
|
||||||
|
<option>--make-entry-directory=yes</option> is used the selected token string is written to a file
|
||||||
|
<filename>/etc/kernel/entry-token</filename>, to ensure it will be used for future entries. This file
|
||||||
|
is also read by
|
||||||
|
<citerefentry><refentrytitle>kernel-install</refentrytitle><manvolnum>8</manvolnum></citerefentry>,
|
||||||
|
in order to identify under which name to generate boot loader entries for newly installed kernels, or
|
||||||
|
to determine the entry names for removing old ones.</para>
|
||||||
|
|
||||||
|
<para>Using the machine ID for naming the entries is generally preferable, however there are cases
|
||||||
|
where using the other identifiers is a good option. Specifically: if the identification data that the
|
||||||
|
machine ID entails shall not be stored on the (unencrypted) <varname>$BOOT</varname> partition, or if
|
||||||
|
the ID shall be generated on first boot and is not known when the entries are prepared. Note that
|
||||||
|
using the machine ID has the benefit that multiple parallel installations of the same OS can coexist
|
||||||
|
on the same medium, and they can update their boot loader entries independently. When using another
|
||||||
|
identifier (such as the OS ID or the OS image ID), parallel installations of the same OS would try to
|
||||||
|
use the same entry name. To support parallel installations, the installer must use a different entry
|
||||||
|
token when adding a second installation.</para></listitem>
|
||||||
</varlistentry>
|
</varlistentry>
|
||||||
|
|
||||||
<xi:include href="standard-options.xml" xpointer="no-pager"/>
|
<xi:include href="standard-options.xml" xpointer="no-pager"/>
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
#include "glyph-util.h"
|
#include "glyph-util.h"
|
||||||
#include "main-func.h"
|
#include "main-func.h"
|
||||||
#include "mkdir.h"
|
#include "mkdir.h"
|
||||||
|
#include "os-util.h"
|
||||||
#include "pager.h"
|
#include "pager.h"
|
||||||
#include "parse-argument.h"
|
#include "parse-argument.h"
|
||||||
#include "parse-util.h"
|
#include "parse-util.h"
|
||||||
@ -56,13 +57,22 @@ static bool arg_print_dollar_boot_path = false;
|
|||||||
static bool arg_touch_variables = true;
|
static bool arg_touch_variables = true;
|
||||||
static PagerFlags arg_pager_flags = 0;
|
static PagerFlags arg_pager_flags = 0;
|
||||||
static bool arg_graceful = false;
|
static bool arg_graceful = false;
|
||||||
static int arg_make_machine_id_directory = 0;
|
static int arg_make_entry_directory = false; /* tri-state: < 0 for automatic logic */
|
||||||
static sd_id128_t arg_machine_id = SD_ID128_NULL;
|
static sd_id128_t arg_machine_id = SD_ID128_NULL;
|
||||||
static char *arg_install_layout = NULL;
|
static char *arg_install_layout = NULL;
|
||||||
|
static enum {
|
||||||
|
ARG_ENTRY_TOKEN_MACHINE_ID,
|
||||||
|
ARG_ENTRY_TOKEN_OS_IMAGE_ID,
|
||||||
|
ARG_ENTRY_TOKEN_OS_ID,
|
||||||
|
ARG_ENTRY_TOKEN_LITERAL,
|
||||||
|
ARG_ENTRY_TOKEN_AUTO,
|
||||||
|
} arg_entry_token_type = ARG_ENTRY_TOKEN_AUTO;
|
||||||
|
static char *arg_entry_token = NULL;
|
||||||
|
|
||||||
STATIC_DESTRUCTOR_REGISTER(arg_esp_path, freep);
|
STATIC_DESTRUCTOR_REGISTER(arg_esp_path, freep);
|
||||||
STATIC_DESTRUCTOR_REGISTER(arg_xbootldr_path, freep);
|
STATIC_DESTRUCTOR_REGISTER(arg_xbootldr_path, freep);
|
||||||
STATIC_DESTRUCTOR_REGISTER(arg_install_layout, freep);
|
STATIC_DESTRUCTOR_REGISTER(arg_install_layout, freep);
|
||||||
|
STATIC_DESTRUCTOR_REGISTER(arg_entry_token, freep);
|
||||||
|
|
||||||
static const char *arg_dollar_boot_path(void) {
|
static const char *arg_dollar_boot_path(void) {
|
||||||
/* $BOOT shall be the XBOOTLDR partition if it exists, and otherwise the ESP */
|
/* $BOOT shall be the XBOOTLDR partition if it exists, and otherwise the ESP */
|
||||||
@ -170,32 +180,133 @@ static int load_install_machine_id_and_layout(void) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int settle_install_machine_id(void) {
|
static int settle_entry_token(void) {
|
||||||
|
int r;
|
||||||
|
|
||||||
|
switch (arg_entry_token_type) {
|
||||||
|
|
||||||
|
case ARG_ENTRY_TOKEN_AUTO: {
|
||||||
|
_cleanup_free_ char *buf = NULL;
|
||||||
|
r = read_one_line_file("/etc/kernel/entry-token", &buf);
|
||||||
|
if (r < 0 && r != -ENOENT)
|
||||||
|
return log_error_errno(r, "Failed to read /etc/kernel/entry-token: %m");
|
||||||
|
|
||||||
|
if (!isempty(buf)) {
|
||||||
|
free_and_replace(arg_entry_token, buf);
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_LITERAL;
|
||||||
|
} else if (sd_id128_is_null(arg_machine_id)) {
|
||||||
|
_cleanup_free_ char *id = NULL, *image_id = NULL;
|
||||||
|
|
||||||
|
r = parse_os_release(NULL,
|
||||||
|
"IMAGE_ID", &image_id,
|
||||||
|
"ID", &id);
|
||||||
|
if (r < 0)
|
||||||
|
return log_error_errno(r, "Failed to load /etc/os-release: %m");
|
||||||
|
|
||||||
|
if (!isempty(image_id)) {
|
||||||
|
free_and_replace(arg_entry_token, image_id);
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_OS_IMAGE_ID;
|
||||||
|
} else if (!isempty(id)) {
|
||||||
|
free_and_replace(arg_entry_token, id);
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_OS_ID;
|
||||||
|
} else
|
||||||
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "No machine ID set, and /etc/os-release carries no ID=/IMAGE_ID= fields.");
|
||||||
|
} else {
|
||||||
|
r = free_and_strdup_warn(&arg_entry_token, SD_ID128_TO_STRING(arg_machine_id));
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_MACHINE_ID;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ARG_ENTRY_TOKEN_MACHINE_ID:
|
||||||
|
if (sd_id128_is_null(arg_machine_id))
|
||||||
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "No machine ID set.");
|
||||||
|
|
||||||
|
r = free_and_strdup_warn(&arg_entry_token, SD_ID128_TO_STRING(arg_machine_id));
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ARG_ENTRY_TOKEN_OS_IMAGE_ID: {
|
||||||
|
_cleanup_free_ char *buf = NULL;
|
||||||
|
|
||||||
|
r = parse_os_release(NULL, "IMAGE_ID", &buf);
|
||||||
|
if (r < 0)
|
||||||
|
return log_error_errno(r, "Failed to load /etc/os-release: %m");
|
||||||
|
|
||||||
|
if (isempty(buf))
|
||||||
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "IMAGE_ID= field not set in /etc/os-release.");
|
||||||
|
|
||||||
|
free_and_replace(arg_entry_token, buf);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ARG_ENTRY_TOKEN_OS_ID: {
|
||||||
|
_cleanup_free_ char *buf = NULL;
|
||||||
|
|
||||||
|
r = parse_os_release(NULL, "ID", &buf);
|
||||||
|
if (r < 0)
|
||||||
|
return log_error_errno(r, "Failed to load /etc/os-release: %m");
|
||||||
|
|
||||||
|
if (isempty(buf))
|
||||||
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "ID= field not set in /etc/os-release.");
|
||||||
|
|
||||||
|
free_and_replace(arg_entry_token, buf);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ARG_ENTRY_TOKEN_LITERAL:
|
||||||
|
assert(!isempty(arg_entry_token)); /* already filled in by command line parser */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isempty(arg_entry_token) || !string_is_safe(arg_entry_token))
|
||||||
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "Selected entry token not valid: %s", arg_entry_token);
|
||||||
|
|
||||||
|
log_debug("Using entry token: %s", arg_entry_token);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool use_boot_loader_spec_type1(void) {
|
||||||
|
/* If the layout is not specified, or if it is set explicitly to "bls" we assume Boot Loader
|
||||||
|
* Specification Type #1 is the chosen format for our boot loader entries */
|
||||||
|
return !arg_install_layout || streq(arg_install_layout, "bls");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int settle_make_entry_directory(void) {
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
r = load_install_machine_id_and_layout();
|
r = load_install_machine_id_and_layout();
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
bool layout_non_bls = arg_install_layout && !streq(arg_install_layout, "bls");
|
r = settle_entry_token();
|
||||||
if (arg_make_machine_id_directory < 0) {
|
if (r < 0)
|
||||||
if (layout_non_bls || sd_id128_is_null(arg_machine_id))
|
return r;
|
||||||
arg_make_machine_id_directory = 0;
|
|
||||||
else {
|
bool layout_type1 = use_boot_loader_spec_type1();
|
||||||
r = path_is_temporary_fs("/etc/machine-id");
|
if (arg_make_entry_directory < 0) { /* Automatic mode */
|
||||||
if (r < 0)
|
if (layout_type1) {
|
||||||
return log_debug_errno(r, "Couldn't determine whether /etc/machine-id is on a temporary file system: %m");
|
if (arg_entry_token == ARG_ENTRY_TOKEN_MACHINE_ID) {
|
||||||
arg_make_machine_id_directory = r == 0;
|
r = path_is_temporary_fs("/etc/machine-id");
|
||||||
}
|
if (r < 0)
|
||||||
|
return log_debug_errno(r, "Couldn't determine whether /etc/machine-id is on a temporary file system: %m");
|
||||||
|
|
||||||
|
arg_make_entry_directory = r == 0;
|
||||||
|
} else
|
||||||
|
arg_make_entry_directory = true;
|
||||||
|
} else
|
||||||
|
arg_make_entry_directory = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (arg_make_machine_id_directory > 0 && sd_id128_is_null(arg_machine_id))
|
if (arg_make_entry_directory > 0 && !layout_type1)
|
||||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
|
||||||
"Machine ID not found, but bls directory creation was requested.");
|
"KERNEL_INSTALL_LAYOUT=%s is configured, but Boot Loader Specification Type #1 entry directory creation was requested.",
|
||||||
|
|
||||||
if (arg_make_machine_id_directory > 0 && layout_non_bls)
|
|
||||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
|
|
||||||
"KERNEL_INSTALL_LAYOUT=%s is configured, but bls directory creation was requested.",
|
|
||||||
arg_install_layout);
|
arg_install_layout);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -1039,14 +1150,14 @@ static int remove_subdirs(const char *root, const char *const *subdirs) {
|
|||||||
return r < 0 ? r : q;
|
return r < 0 ? r : q;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int remove_machine_id_directory(const char *root) {
|
static int remove_entry_directory(const char *root) {
|
||||||
assert(root);
|
assert(root);
|
||||||
assert(arg_make_machine_id_directory >= 0);
|
assert(arg_make_entry_directory >= 0);
|
||||||
|
|
||||||
if (!arg_make_machine_id_directory || sd_id128_is_null(arg_machine_id))
|
if (!arg_make_entry_directory || !arg_entry_token)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
return rmdir_one(root, SD_ID128_TO_STRING(arg_machine_id));
|
return rmdir_one(root, arg_entry_token);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int remove_binaries(const char *esp_path) {
|
static int remove_binaries(const char *esp_path) {
|
||||||
@ -1138,7 +1249,7 @@ static int install_loader_config(const char *esp_path) {
|
|||||||
const char *p;
|
const char *p;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
assert(arg_make_machine_id_directory >= 0);
|
assert(arg_make_entry_directory >= 0);
|
||||||
|
|
||||||
p = prefix_roota(esp_path, "/loader/loader.conf");
|
p = prefix_roota(esp_path, "/loader/loader.conf");
|
||||||
if (access(p, F_OK) >= 0) /* Silently skip creation if the file already exists (early check) */
|
if (access(p, F_OK) >= 0) /* Silently skip creation if the file already exists (early check) */
|
||||||
@ -1155,9 +1266,9 @@ static int install_loader_config(const char *esp_path) {
|
|||||||
fprintf(f, "#timeout 3\n"
|
fprintf(f, "#timeout 3\n"
|
||||||
"#console-mode keep\n");
|
"#console-mode keep\n");
|
||||||
|
|
||||||
if (arg_make_machine_id_directory) {
|
if (arg_make_entry_directory) {
|
||||||
assert(!sd_id128_is_null(arg_machine_id));
|
assert(arg_entry_token);
|
||||||
fprintf(f, "default %s-*\n", SD_ID128_TO_STRING(arg_machine_id));
|
fprintf(f, "default %s-*\n", arg_entry_token);
|
||||||
}
|
}
|
||||||
|
|
||||||
r = fflush_sync_and_check(f);
|
r = fflush_sync_and_check(f);
|
||||||
@ -1174,100 +1285,33 @@ static int install_loader_config(const char *esp_path) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int install_machine_id_directory(const char *root) {
|
static int install_entry_directory(const char *root) {
|
||||||
assert(root);
|
assert(root);
|
||||||
assert(arg_make_machine_id_directory >= 0);
|
assert(arg_make_entry_directory >= 0);
|
||||||
|
|
||||||
if (!arg_make_machine_id_directory)
|
if (!arg_make_entry_directory)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
assert(!sd_id128_is_null(arg_machine_id));
|
assert(arg_entry_token);
|
||||||
return mkdir_one(root, SD_ID128_TO_STRING(arg_machine_id));
|
return mkdir_one(root, arg_entry_token);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int install_machine_info_config(void) {
|
static int install_entry_token(void) {
|
||||||
_cleanup_free_ char *contents = NULL;
|
|
||||||
size_t length;
|
|
||||||
bool need_install_layout = true, need_machine_id;
|
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
assert(arg_make_machine_id_directory >= 0);
|
assert(arg_make_entry_directory >= 0);
|
||||||
|
assert(arg_entry_token);
|
||||||
|
|
||||||
/* We only want to save the machine-id if we created any directories using it. */
|
/* Let's save the used entry token in /etc/kernel/entry-token if we used it to create the entry
|
||||||
need_machine_id = arg_make_machine_id_directory;
|
* directory, or if anything else but the machine ID */
|
||||||
|
|
||||||
_cleanup_fclose_ FILE *orig = fopen("/etc/machine-info", "re");
|
if (!arg_make_entry_directory && arg_entry_token_type == ARG_ENTRY_TOKEN_MACHINE_ID)
|
||||||
if (!orig && errno != ENOENT)
|
return 0;
|
||||||
return log_error_errno(errno, "Failed to open /etc/machine-info: %m");
|
|
||||||
|
|
||||||
if (orig) {
|
r = write_string_file("/etc/kernel/entry-token", arg_entry_token, WRITE_STRING_FILE_CREATE|WRITE_STRING_FILE_ATOMIC|WRITE_STRING_FILE_MKDIR_0755);
|
||||||
_cleanup_free_ char *install_layout = NULL, *machine_id = NULL;
|
|
||||||
|
|
||||||
r = parse_env_file(orig, "/etc/machine-info",
|
|
||||||
"KERNEL_INSTALL_LAYOUT", &install_layout,
|
|
||||||
"KERNEL_INSTALL_MACHINE_ID", &machine_id);
|
|
||||||
if (r < 0)
|
|
||||||
return log_error_errno(r, "Failed to parse /etc/machine-info: %m");
|
|
||||||
|
|
||||||
rewind(orig);
|
|
||||||
|
|
||||||
if (!isempty(install_layout))
|
|
||||||
need_install_layout = false;
|
|
||||||
|
|
||||||
if (!isempty(machine_id))
|
|
||||||
need_machine_id = false;
|
|
||||||
|
|
||||||
if (!need_install_layout && !need_machine_id) {
|
|
||||||
log_debug("/etc/machine-info already has KERNEL_INSTALL_MACHINE_ID=%s and KERNEL_INSTALL_LAYOUT=%s.",
|
|
||||||
machine_id, install_layout);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
r = read_full_stream(orig, &contents, &length);
|
|
||||||
if (r < 0)
|
|
||||||
return log_error_errno(r, "Failed to read /etc/machine-info: %m");
|
|
||||||
}
|
|
||||||
|
|
||||||
_cleanup_(unlink_and_freep) char *dst_tmp = NULL;
|
|
||||||
_cleanup_fclose_ FILE *dst = NULL;
|
|
||||||
r = fopen_temporary_label("/etc/machine-info", /* The path for which to the look up the label */
|
|
||||||
"/etc/machine-info", /* Where we want the file actually to end up */
|
|
||||||
&dst, /* The temporary file we write to */
|
|
||||||
&dst_tmp);
|
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return log_debug_errno(r, "Failed to open temporary copy of /etc/machine-info: %m");
|
return log_error_errno(r, "Failed to write entry token '%s' to /etc/kernel/entry-token", arg_entry_token);
|
||||||
|
|
||||||
if (contents)
|
|
||||||
fwrite_unlocked(contents, 1, length, dst);
|
|
||||||
|
|
||||||
bool no_newline = !contents || contents[length - 1] == '\n';
|
|
||||||
|
|
||||||
if (need_install_layout) {
|
|
||||||
const char *line = "\nKERNEL_INSTALL_LAYOUT=bls\n" + no_newline;
|
|
||||||
fwrite_unlocked(line, 1, strlen(line), dst);
|
|
||||||
no_newline = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *mid_string = SD_ID128_TO_STRING(arg_machine_id);
|
|
||||||
if (need_machine_id)
|
|
||||||
fprintf(dst, "%sKERNEL_INSTALL_MACHINE_ID=%s\n",
|
|
||||||
no_newline ? "" : "\n",
|
|
||||||
mid_string);
|
|
||||||
|
|
||||||
r = fflush_and_check(dst);
|
|
||||||
if (r < 0)
|
|
||||||
return log_error_errno(r, "Failed to write temporary copy of /etc/machine-info: %m");
|
|
||||||
if (fchmod(fileno(dst), 0644) < 0)
|
|
||||||
return log_debug_errno(errno, "Failed to fchmod %s: %m", dst_tmp);
|
|
||||||
|
|
||||||
if (rename(dst_tmp, "/etc/machine-info") < 0)
|
|
||||||
return log_error_errno(errno, "Failed to replace /etc/machine-info: %m");
|
|
||||||
|
|
||||||
log_info("%s /etc/machine-info with%s%s%s",
|
|
||||||
orig ? "Updated" : "Created",
|
|
||||||
need_install_layout ? " KERNEL_INSTALL_LAYOUT=bls" : "",
|
|
||||||
need_machine_id ? " KERNEL_INSTALL_MACHINE_ID=" : "",
|
|
||||||
need_machine_id ? mid_string : "");
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1311,8 +1355,10 @@ static int help(int argc, char *argv[], void *userdata) {
|
|||||||
" --no-pager Do not pipe output into a pager\n"
|
" --no-pager Do not pipe output into a pager\n"
|
||||||
" --graceful Don't fail when the ESP cannot be found or EFI\n"
|
" --graceful Don't fail when the ESP cannot be found or EFI\n"
|
||||||
" variables cannot be written\n"
|
" variables cannot be written\n"
|
||||||
" --make-machine-id-directory=yes|no|auto\n"
|
" --make-entry-directory=yes|no|auto\n"
|
||||||
" Create $BOOT/$MACHINE_ID\n"
|
" Create $BOOT/ENTRY-TOKEN/ directory\n"
|
||||||
|
" --entry-token=machine-id|os-id|os-image-id|auto|literal:…\n"
|
||||||
|
" Entry token to use for this installation\n"
|
||||||
"\nSee the %2$s for details.\n",
|
"\nSee the %2$s for details.\n",
|
||||||
program_invocation_short_name,
|
program_invocation_short_name,
|
||||||
link,
|
link,
|
||||||
@ -1332,7 +1378,8 @@ static int parse_argv(int argc, char *argv[]) {
|
|||||||
ARG_NO_VARIABLES,
|
ARG_NO_VARIABLES,
|
||||||
ARG_NO_PAGER,
|
ARG_NO_PAGER,
|
||||||
ARG_GRACEFUL,
|
ARG_GRACEFUL,
|
||||||
ARG_MAKE_MACHINE_ID_DIRECTORY,
|
ARG_MAKE_ENTRY_DIRECTORY,
|
||||||
|
ARG_ENTRY_TOKEN,
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct option options[] = {
|
static const struct option options[] = {
|
||||||
@ -1347,7 +1394,9 @@ static int parse_argv(int argc, char *argv[]) {
|
|||||||
{ "no-variables", no_argument, NULL, ARG_NO_VARIABLES },
|
{ "no-variables", no_argument, NULL, ARG_NO_VARIABLES },
|
||||||
{ "no-pager", no_argument, NULL, ARG_NO_PAGER },
|
{ "no-pager", no_argument, NULL, ARG_NO_PAGER },
|
||||||
{ "graceful", no_argument, NULL, ARG_GRACEFUL },
|
{ "graceful", no_argument, NULL, ARG_GRACEFUL },
|
||||||
{ "make-machine-id-directory", required_argument, NULL, ARG_MAKE_MACHINE_ID_DIRECTORY },
|
{ "make-entry-directory", required_argument, NULL, ARG_MAKE_ENTRY_DIRECTORY },
|
||||||
|
{ "make-machine-id-directory", required_argument, NULL, ARG_MAKE_ENTRY_DIRECTORY }, /* Compatibility alias */
|
||||||
|
{ "entry-token", required_argument, NULL, ARG_ENTRY_TOKEN },
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1405,14 +1454,40 @@ static int parse_argv(int argc, char *argv[]) {
|
|||||||
arg_graceful = true;
|
arg_graceful = true;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ARG_MAKE_MACHINE_ID_DIRECTORY:
|
case ARG_ENTRY_TOKEN: {
|
||||||
if (streq(optarg, "auto")) /* retained for backwards compatibility */
|
const char *e;
|
||||||
arg_make_machine_id_directory = -1; /* yes if machine-id is permanent */
|
|
||||||
else {
|
if (streq(optarg, "machine-id")) {
|
||||||
r = parse_boolean_argument("--make-machine-id-directory=", optarg, &b);
|
arg_entry_token_type = ARG_ENTRY_TOKEN_MACHINE_ID;
|
||||||
|
arg_entry_token = mfree(arg_entry_token);
|
||||||
|
} else if (streq(optarg, "os-image-id")) {
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_OS_IMAGE_ID;
|
||||||
|
arg_entry_token = mfree(arg_entry_token);
|
||||||
|
} else if (streq(optarg, "os-id")) {
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_OS_ID;
|
||||||
|
arg_entry_token = mfree(arg_entry_token);
|
||||||
|
} else if ((e = startswith(optarg, "literal:"))) {
|
||||||
|
arg_entry_token_type = ARG_ENTRY_TOKEN_LITERAL;
|
||||||
|
|
||||||
|
r = free_and_strdup_warn(&arg_entry_token, e);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
arg_make_machine_id_directory = b;
|
} else
|
||||||
|
return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
|
||||||
|
"Unexpected parameter for --entry-token=: %s", optarg);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ARG_MAKE_ENTRY_DIRECTORY:
|
||||||
|
if (streq(optarg, "auto")) /* retained for backwards compatibility */
|
||||||
|
arg_make_entry_directory = -1; /* yes if machine-id is permanent */
|
||||||
|
else {
|
||||||
|
r = parse_boolean_argument("--make-entry-directory=", optarg, &b);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
arg_make_entry_directory = b;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1881,7 +1956,7 @@ static int verb_install(int argc, char *argv[], void *userdata) {
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = settle_install_machine_id();
|
r = settle_make_entry_directory();
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -1908,11 +1983,11 @@ static int verb_install(int argc, char *argv[], void *userdata) {
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = install_machine_id_directory(arg_dollar_boot_path());
|
r = install_entry_directory(arg_dollar_boot_path());
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = install_machine_info_config();
|
r = install_entry_token();
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -1945,7 +2020,7 @@ static int verb_remove(int argc, char *argv[], void *userdata) {
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = settle_install_machine_id();
|
r = settle_make_entry_directory();
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -1967,7 +2042,7 @@ static int verb_remove(int argc, char *argv[], void *userdata) {
|
|||||||
if (q < 0 && r >= 0)
|
if (q < 0 && r >= 0)
|
||||||
r = q;
|
r = q;
|
||||||
|
|
||||||
q = remove_machine_id_directory(arg_esp_path);
|
q = remove_entry_directory(arg_esp_path);
|
||||||
if (q < 0 && r >= 0)
|
if (q < 0 && r >= 0)
|
||||||
r = 1;
|
r = 1;
|
||||||
|
|
||||||
@ -1977,7 +2052,7 @@ static int verb_remove(int argc, char *argv[], void *userdata) {
|
|||||||
if (q < 0 && r >= 0)
|
if (q < 0 && r >= 0)
|
||||||
r = q;
|
r = q;
|
||||||
|
|
||||||
q = remove_machine_id_directory(arg_xbootldr_path);
|
q = remove_entry_directory(arg_xbootldr_path);
|
||||||
if (q < 0 && r >= 0)
|
if (q < 0 && r >= 0)
|
||||||
r = q;
|
r = q;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user