Move prepare-root karg helpers into otcore, add unit tests

Add long overdue unit testing coverage for this, which
at least slightly closes out the android boot CI gap.

Actually, this *copies* the karg parsing code into otcore because
it now uses glib, which we're not yet using in the static
prepare-root.  It's pretty tempting to drop support for the
static prepare root entirely.  But for now we'll live with some
code duplication.
This commit is contained in:
Colin Walters 2023-08-23 16:06:23 -04:00
parent 4451949213
commit 20b8cb174c
6 changed files with 170 additions and 51 deletions

View File

@ -18,6 +18,7 @@ noinst_LTLIBRARIES += libotcore.la
libotcore_la_SOURCES = \ libotcore_la_SOURCES = \
src/libotcore/otcore.h \ src/libotcore/otcore.h \
src/libotcore/otcore-ed25519-verify.c \ src/libotcore/otcore-ed25519-verify.c \
src/libotcore/otcore-prepare-root.c \
$(NULL) $(NULL)
libotcore_la_CFLAGS = $(AM_CFLAGS) -I$(srcdir)/libglnx -I$(srcdir)/src/libotutil -DLOCALEDIR=\"$(datadir)/locale\" $(OT_INTERNAL_GIO_UNIX_CFLAGS) $(OT_INTERNAL_GPGME_CFLAGS) $(OT_DEP_CRYPTO_LIBS) $(LIBSYSTEMD_CFLAGS) libotcore_la_CFLAGS = $(AM_CFLAGS) -I$(srcdir)/libglnx -I$(srcdir)/src/libotutil -DLOCALEDIR=\"$(datadir)/locale\" $(OT_INTERNAL_GIO_UNIX_CFLAGS) $(OT_INTERNAL_GPGME_CFLAGS) $(OT_DEP_CRYPTO_LIBS) $(LIBSYSTEMD_CFLAGS)

View File

@ -34,7 +34,7 @@
#include "ostree-core-private.h" #include "ostree-core-private.h"
#include "ostree-mount-util.h" #include "ostree-mount-util.h"
#include "ostree-sysroot-private.h" #include "ostree-sysroot-private.h"
#include "ostree.h" #include "otcore.h"
#ifdef HAVE_LIBMOUNT #ifdef HAVE_LIBMOUNT
typedef FILE OtLibMountFile; typedef FILE OtLibMountFile;
@ -260,7 +260,7 @@ _ostree_impl_system_generator (const char *normal_dir, const char *early_dir, co
* exit so that we don't error, but at the same time work where switchroot * exit so that we don't error, but at the same time work where switchroot
* is PID 1 (and so hasn't created /run/ostree-booted). * is PID 1 (and so hasn't created /run/ostree-booted).
*/ */
g_autofree char *ostree_cmdline = find_proc_cmdline_key (cmdline, "ostree"); g_autofree char *ostree_cmdline = otcore_find_proc_cmdline_key (cmdline, "ostree");
if (!ostree_cmdline) if (!ostree_cmdline)
return TRUE; return TRUE;

View File

@ -0,0 +1,108 @@
/*
* SPDX-License-Identifier: LGPL-2.0+
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#include "otcore.h"
static bool
proc_cmdline_has_key_starting_with (const char *cmdline, const char *key)
{
for (const char *iter = cmdline; iter;)
{
if (g_str_has_prefix (iter, key))
return true;
iter = strchr (iter, ' ');
if (iter == NULL)
return false;
iter += strspn (iter, " ");
}
return false;
}
// Parse a kernel cmdline to find the provided key.
// TODO: Deduplicate this with the kernel argument code from libostree.so
char *
otcore_find_proc_cmdline_key (const char *cmdline, const char *key)
{
const size_t key_len = strlen (key);
for (const char *iter = cmdline; iter;)
{
const char *next = strchr (iter, ' ');
if (strncmp (iter, key, key_len) == 0 && iter[key_len] == '=')
{
const char *start = iter + key_len + 1;
if (next)
return strndup (start, next - start);
return strdup (start);
}
if (next)
next += strspn (next, " ");
iter = next;
}
return NULL;
}
// Find the target OSTree root filesystem from parsing the provided kernel commandline.
// If none is found, @out_target will be set to NULL, and the function will return successfully.
//
// If invalid data is found, @error will be set.
gboolean
otcore_get_ostree_target (const char *cmdline, char **out_target, GError **error)
{
g_assert (cmdline);
g_assert (out_target && *out_target == NULL);
static const char slot_a[] = "/ostree/root.a";
static const char slot_b[] = "/ostree/root.b";
// First, handle the Android boot case
g_autofree char *slot_suffix = otcore_find_proc_cmdline_key (cmdline, "androidboot.slot_suffix");
if (slot_suffix)
{
if (strcmp (slot_suffix, "_a") == 0)
{
*out_target = strdup (slot_a);
return TRUE;
}
else if (strcmp (slot_suffix, "_b") == 0)
{
*out_target = strdup (slot_b);
return TRUE;
}
return glnx_throw (error, "androidboot.slot_suffix invalid: %s", slot_suffix);
}
/* Non-A/B androidboot:
* https://source.android.com/docs/core/ota/nonab
*/
if (proc_cmdline_has_key_starting_with (cmdline, "androidboot."))
{
*out_target = strdup (slot_a);
return TRUE;
}
// Otherwise, fall back to the default `ostree=` kernel cmdline
*out_target = otcore_find_proc_cmdline_key (cmdline, "ostree");
return TRUE;
}

View File

@ -43,6 +43,9 @@ bool otcore_ed25519_init (void);
gboolean otcore_validate_ed25519_signature (GBytes *data, GBytes *pubkey, GBytes *signature, gboolean otcore_validate_ed25519_signature (GBytes *data, GBytes *pubkey, GBytes *signature,
bool *out_valid, GError **error); bool *out_valid, GError **error);
char *otcore_find_proc_cmdline_key (const char *cmdline, const char *key);
gboolean otcore_get_ostree_target (const char *cmdline, char **out_target, GError **error);
// Our directory with transient state (eventually /run/ostree-booted should be a link to // Our directory with transient state (eventually /run/ostree-booted should be a link to
// /run/ostree/booted) // /run/ostree/booted)
#define OTCORE_RUN_OSTREE "/run/ostree" #define OTCORE_RUN_OSTREE "/run/ostree"

View File

@ -150,53 +150,6 @@ sysroot_is_configured_ro (const char *sysroot)
return g_key_file_get_boolean (repo_config, SYSROOT_KEY, READONLY_KEY, NULL); return g_key_file_get_boolean (repo_config, SYSROOT_KEY, READONLY_KEY, NULL);
} }
static inline char *
get_aboot_root_slot (const char *slot_suffix)
{
if (strcmp (slot_suffix, "_a") == 0)
return strdup ("/ostree/root.a");
else if (strcmp (slot_suffix, "_b") == 0)
return strdup ("/ostree/root.b");
errx (EXIT_FAILURE, "androidboot.slot_suffix invalid: %s", slot_suffix);
return NULL;
}
static bool
proc_cmdline_has_key_starting_with (const char *cmdline, const char *key)
{
for (const char *iter = cmdline; iter;)
{
if (g_str_has_prefix (iter, key))
return true;
iter = strchr (iter, ' ');
if (iter == NULL)
return false;
iter += strspn (iter, " ");
}
return false;
}
static inline char *
get_ostree_target (const char *cmdline)
{
autofree char *slot_suffix = find_proc_cmdline_key (cmdline, "androidboot.slot_suffix");
if (slot_suffix)
return get_aboot_root_slot (slot_suffix);
/* Non-A/B androidboot:
* https://source.android.com/docs/core/ota/nonab
*/
if (proc_cmdline_has_key_starting_with (cmdline, "androidboot."))
return strdup ("/ostree/root.a");
return find_proc_cmdline_key (cmdline, "ostree");
}
static char * static char *
resolve_deploy_path (const char *root_mountpoint) resolve_deploy_path (const char *root_mountpoint)
{ {
@ -207,9 +160,12 @@ resolve_deploy_path (const char *root_mountpoint)
if (!kernel_cmdline) if (!kernel_cmdline)
errx (EXIT_FAILURE, "Failed to read kernel cmdline"); errx (EXIT_FAILURE, "Failed to read kernel cmdline");
g_autofree char *ostree_target = get_ostree_target (kernel_cmdline); g_autoptr (GError) error = NULL;
g_autofree char *ostree_target = NULL;
if (!otcore_get_ostree_target (kernel_cmdline, &ostree_target, &error))
errx (EXIT_FAILURE, "Failed to determine ostree target: %s", error->message);
if (!ostree_target) if (!ostree_target)
errx (EXIT_FAILURE, "No ostree target"); errx (EXIT_FAILURE, "No ostree target found");
if (snprintf (destpath, sizeof (destpath), "%s/%s", root_mountpoint, ostree_target) < 0) if (snprintf (destpath, sizeof (destpath), "%s/%s", root_mountpoint, ostree_target) < 0)
err (EXIT_FAILURE, "failed to assemble ostree target path"); err (EXIT_FAILURE, "failed to assemble ostree target path");

View File

@ -31,11 +31,62 @@ test_ed25519 (void)
g_clear_error (&error); g_clear_error (&error);
} }
static void
test_prepare_root_cmdline (void)
{
g_autoptr (GError) error = NULL;
g_autofree char *target = NULL;
static const char *notfound_cases[]
= { "", "foo", "foo=bar baz sometest", "xostree foo", "xostree=blah bar", NULL };
for (const char **iter = notfound_cases; iter && *iter; iter++)
{
const char *tcase = *iter;
g_assert (otcore_get_ostree_target (tcase, &target, &error));
g_assert_no_error (error);
g_assert (target == NULL);
}
// Test the default ostree=
g_assert (otcore_get_ostree_target ("blah baz=blah ostree=/foo/bar somearg", &target, &error));
g_assert_no_error (error);
g_assert_cmpstr (target, ==, "/foo/bar");
free (g_steal_pointer (&target));
// Test android boot
g_assert (otcore_get_ostree_target ("blah baz=blah androidboot.slot_suffix=_b somearg", &target,
&error));
g_assert_no_error (error);
g_assert_cmpstr (target, ==, "/ostree/root.b");
free (g_steal_pointer (&target));
g_assert (otcore_get_ostree_target ("blah baz=blah androidboot.slot_suffix=_a somearg", &target,
&error));
g_assert_no_error (error);
g_assert_cmpstr (target, ==, "/ostree/root.a");
free (g_steal_pointer (&target));
// And an expected failure to parse a "c" suffix
g_assert (!otcore_get_ostree_target ("blah baz=blah androidboot.slot_suffix=_c somearg", &target,
&error));
g_assert (error);
g_assert (target == NULL);
g_clear_error (&error);
// And non-A/B androidboot
g_assert (otcore_get_ostree_target ("blah baz=blah androidboot.somethingelse somearg", &target,
&error));
g_assert_no_error (error);
g_assert_cmpstr (target, ==, "/ostree/root.a");
free (g_steal_pointer (&target));
}
int int
main (int argc, char **argv) main (int argc, char **argv)
{ {
g_test_init (&argc, &argv, NULL); g_test_init (&argc, &argv, NULL);
otcore_ed25519_init (); otcore_ed25519_init ();
g_test_add_func ("/ed25519", test_ed25519); g_test_add_func ("/ed25519", test_ed25519);
g_test_add_func ("/prepare-root-cmdline", test_prepare_root_cmdline);
return g_test_run (); return g_test_run ();
} }