summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorWilliam Manley <will@stb-tester.com>2020-07-15 15:40:14 +0100
committerWilliam Manley <will@stb-tester.com>2020-07-15 20:37:49 +0100
commit0ced9fde7649271d9458ca424aa8c41908634b02 (patch)
treec9e3d8d1d7652f3c17faed8556eda4264b3b4b10
parent4deb426835d234eaa55046175f7dffaea7864fdb (diff)
downloadostree-0ced9fde7649271d9458ca424aa8c41908634b02.tar.gz
sysroot: Support /boot on root or as seperate filesystem for syslinux and u-boot
We use a similar trick to having a `sysroot -> .` symlink on the real root here to support both /boot on root as well as on a separate filesystem. No matter how it's mounted `/boot/xyz` will always refer to the file you'd expect. This is nicer than my previous attempts at this because there's no configuration nor auto-detection required.
-rw-r--r--src/libostree/ostree-bootloader-syslinux.c15
-rw-r--r--src/libostree/ostree-bootloader-uboot.c8
-rw-r--r--src/libostree/ostree-sysroot-deploy.c6
-rwxr-xr-xtests/bootloader-entries-crosscheck.py27
4 files changed, 41 insertions, 15 deletions
diff --git a/src/libostree/ostree-bootloader-syslinux.c b/src/libostree/ostree-bootloader-syslinux.c
index 5fb8a1db..0055896b 100644
--- a/src/libostree/ostree-bootloader-syslinux.c
+++ b/src/libostree/ostree-bootloader-syslinux.c
@@ -89,15 +89,15 @@ append_config_from_loader_entries (OstreeBootloaderSyslinux *self,
val = ostree_bootconfig_parser_get (config, "linux");
if (!val)
return glnx_throw (error, "No \"linux\" key in bootloader config");
- g_ptr_array_add (new_lines, g_strdup_printf ("\tKERNEL %s", val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("\tKERNEL /boot%s", val));
val = ostree_bootconfig_parser_get (config, "initrd");
if (val)
- g_ptr_array_add (new_lines, g_strdup_printf ("\tINITRD %s", val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("\tINITRD /boot%s", val));
val = ostree_bootconfig_parser_get (config, "devicetree");
if (val)
- g_ptr_array_add (new_lines, g_strdup_printf ("\tDEVICETREE %s", val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("\tDEVICETREE /boot%s", val));
val = ostree_bootconfig_parser_get (config, "options");
if (val)
@@ -150,10 +150,13 @@ _ostree_bootloader_syslinux_write_config (OstreeBootloader *bootloader,
if (kernel_arg == NULL)
return glnx_throw (error, "No KERNEL argument found after LABEL");
- /* If this is a non-ostree kernel, just emit the lines
- * we saw.
+ /* If this is a non-ostree kernel, just emit the lines we saw.
+ *
+ * We check for /ostree (without /boot prefix) as well to support
+ * upgrading ostree from <v2020.4.
*/
- if (!g_str_has_prefix (kernel_arg, "/ostree/"))
+ if (!g_str_has_prefix (kernel_arg, "/ostree/") &&
+ !g_str_has_prefix (kernel_arg, "/boot/ostree/"))
{
for (guint i = 0; i < tmp_lines->len; i++)
{
diff --git a/src/libostree/ostree-bootloader-uboot.c b/src/libostree/ostree-bootloader-uboot.c
index 1e1f0371..7e23001e 100644
--- a/src/libostree/ostree-bootloader-uboot.c
+++ b/src/libostree/ostree-bootloader-uboot.c
@@ -134,19 +134,19 @@ create_config_from_boot_loader_entries (OstreeBootloaderUboot *self,
"No \"linux\" key in bootloader config");
return FALSE;
}
- g_ptr_array_add (new_lines, g_strdup_printf ("kernel_image%s=%s", index_suffix, val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("kernel_image%s=/boot%s", index_suffix, val));
val = ostree_bootconfig_parser_get (config, "initrd");
if (val)
- g_ptr_array_add (new_lines, g_strdup_printf ("ramdisk_image%s=%s", index_suffix, val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("ramdisk_image%s=/boot%s", index_suffix, val));
val = ostree_bootconfig_parser_get (config, "devicetree");
if (val)
- g_ptr_array_add (new_lines, g_strdup_printf ("fdt_file%s=%s", index_suffix, val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("fdt_file%s=/boot%s", index_suffix, val));
val = ostree_bootconfig_parser_get (config, "fdtdir");
if (val)
- g_ptr_array_add (new_lines, g_strdup_printf ("fdtdir%s=%s", index_suffix, val));
+ g_ptr_array_add (new_lines, g_strdup_printf ("fdtdir%s=/boot%s", index_suffix, val));
val = ostree_bootconfig_parser_get (config, "options");
if (val)
diff --git a/src/libostree/ostree-sysroot-deploy.c b/src/libostree/ostree-sysroot-deploy.c
index cb593020..ec51ad9a 100644
--- a/src/libostree/ostree-sysroot-deploy.c
+++ b/src/libostree/ostree-sysroot-deploy.c
@@ -1998,6 +1998,12 @@ prepare_new_bootloader_link (OstreeSysroot *sysroot,
g_assert ((current_bootversion == 0 && new_bootversion == 1) ||
(current_bootversion == 1 && new_bootversion == 0));
+ /* This allows us to support both /boot on a seperate filesystem to / as well
+ * as on the same filesystem. */
+ if (TEMP_FAILURE_RETRY (symlinkat (".", sysroot->sysroot_fd, "boot/boot")) < 0)
+ if (errno != EEXIST)
+ return glnx_throw_errno_prefix (error, "symlinkat");
+
g_autofree char *new_target = g_strdup_printf ("loader.%d", new_bootversion);
/* We shouldn't actually need to replace but it's easier to reuse
diff --git a/tests/bootloader-entries-crosscheck.py b/tests/bootloader-entries-crosscheck.py
index b4cdcbf5..605bd080 100755
--- a/tests/bootloader-entries-crosscheck.py
+++ b/tests/bootloader-entries-crosscheck.py
@@ -97,15 +97,32 @@ with open(syslinuxpath) as f:
if len(entries) != len(syslinux_entries):
fatal("Found {0} loader entries, but {1} SYSLINUX entries\n".format(len(entries), len(syslinux_entries)))
-def assert_matches_key(a, b, key):
+
+def assert_eq(a, b):
+ assert a == b, "%r == %r" % (a, b)
+
+
+def assert_key_same_file(a, b, key):
aval = a[key]
bval = b[key]
- if aval != bval:
- fatal("Mismatch on {0}: {1} != {2}".format(key, aval, bval))
+ sys.stderr.write("aval: %r\nbval: %r\n" % (aval, bval))
+
+ # Paths in entries are always relative to /boot
+ entry = os.stat(sysroot + "/boot" + aval)
+
+ # Syslinux entries can be relative to /boot (if it's on another filesystem)
+ # or relative to / if /boot is on /.
+ s1 = os.stat(sysroot + bval)
+ s2 = os.stat(sysroot + "/boot" + bval)
+
+ # A symlink ensures that no matter what they point at the same file
+ assert_eq(entry, s1)
+ assert_eq(entry, s2)
+
for i,(entry,syslinuxentry) in enumerate(zip(entries, syslinux_entries)):
- assert_matches_key(entry, syslinuxentry, 'linux')
- assert_matches_key(entry, syslinuxentry, 'initrd')
+ assert_key_same_file(entry, syslinuxentry, 'linux')
+ assert_key_same_file(entry, syslinuxentry, 'initrd')
entry_ostree = get_ostree_option(entry['options'])
syslinux_ostree = get_ostree_option(syslinuxentry['options'])
if entry_ostree != syslinux_ostree: