/* * Copyright (C) 2013 Colin Walters * * This program 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 licence 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 . */ #include "config.h" #include "ostree-bootconfig-parser.h" #include "otutil.h" struct _OstreeBootconfigParser { GObject parent_instance; gboolean parsed; const char *separators; GHashTable *options; /* Additional initrds; the primary initrd is in options. */ char **overlay_initrds; }; typedef GObjectClass OstreeBootconfigParserClass; G_DEFINE_TYPE (OstreeBootconfigParser, ostree_bootconfig_parser, G_TYPE_OBJECT) /** * ostree_bootconfig_parser_clone: * @self: Bootconfig to clone * * Returns: (transfer full): Copy of @self */ OstreeBootconfigParser * ostree_bootconfig_parser_clone (OstreeBootconfigParser *self) { OstreeBootconfigParser *parser = ostree_bootconfig_parser_new (); GLNX_HASH_TABLE_FOREACH_KV (self->options, const char*, k, const char*, v) g_hash_table_replace (parser->options, g_strdup (k), g_strdup (v)); parser->overlay_initrds = g_strdupv (self->overlay_initrds); return parser; } /** * ostree_bootconfig_parser_parse_at: * @self: Parser * @dfd: Directory fd * @path: File path * @cancellable: Cancellable * @error: Error * * Initialize a bootconfig from the given file. */ gboolean ostree_bootconfig_parser_parse_at (OstreeBootconfigParser *self, int dfd, const char *path, GCancellable *cancellable, GError **error) { g_return_val_if_fail (!self->parsed, FALSE); g_autofree char *contents = glnx_file_get_contents_utf8_at (dfd, path, NULL, cancellable, error); if (!contents) return FALSE; g_autoptr(GPtrArray) overlay_initrds = NULL; g_auto(GStrv) lines = g_strsplit (contents, "\n", -1); for (char **iter = lines; *iter; iter++) { const char *line = *iter; if (g_ascii_isalpha (*line)) { char **items = NULL; items = g_strsplit_set (line, self->separators, 2); if (g_strv_length (items) == 2 && items[0][0] != '\0') { if (g_str_equal (items[0], "initrd") && g_hash_table_contains (self->options, "initrd")) { if (!overlay_initrds) overlay_initrds = g_ptr_array_new_with_free_func (g_free); g_ptr_array_add (overlay_initrds, items[1]); g_free (items[0]); } else { g_hash_table_insert (self->options, items[0], items[1]); } g_free (items); /* Free container; we stole the elements */ } else { g_strfreev (items); } } } if (overlay_initrds) { g_ptr_array_add (overlay_initrds, NULL); self->overlay_initrds = (char**)g_ptr_array_free (g_steal_pointer (&overlay_initrds), FALSE); } self->parsed = TRUE; return TRUE; } gboolean ostree_bootconfig_parser_parse (OstreeBootconfigParser *self, GFile *path, GCancellable *cancellable, GError **error) { return ostree_bootconfig_parser_parse_at (self, AT_FDCWD, gs_file_get_path_cached (path), cancellable, error); } void ostree_bootconfig_parser_set (OstreeBootconfigParser *self, const char *key, const char *value) { g_hash_table_replace (self->options, g_strdup (key), g_strdup (value)); } const char * ostree_bootconfig_parser_get (OstreeBootconfigParser *self, const char *key) { return g_hash_table_lookup (self->options, key); } /** * ostree_bootconfig_parser_set_overlay_initrds: * @self: Parser * @initrds: (array zero-terminated=1) (transfer none) (allow-none): Array of overlay * initrds or %NULL to unset. * * These are rendered as additional `initrd` keys in the final bootloader configs. The * base initrd is part of the primary keys. * * Since: 2020.7 */ void ostree_bootconfig_parser_set_overlay_initrds (OstreeBootconfigParser *self, char **initrds) { g_assert (g_hash_table_contains (self->options, "initrd")); g_strfreev (self->overlay_initrds); self->overlay_initrds = g_strdupv (initrds); } /** * ostree_bootconfig_parser_get_overlay_initrds: * @self: Parser * * Returns: (array zero-terminated=1) (transfer none) (nullable): Array of initrds or %NULL * if none are set. * * Since: 2020.7 */ char** ostree_bootconfig_parser_get_overlay_initrds (OstreeBootconfigParser *self) { return self->overlay_initrds; } static void write_key (OstreeBootconfigParser *self, GString *buf, const char *key, const char *value) { g_string_append (buf, key); g_string_append_c (buf, self->separators[0]); g_string_append (buf, value); g_string_append_c (buf, '\n'); } gboolean ostree_bootconfig_parser_write_at (OstreeBootconfigParser *self, int dfd, const char *path, GCancellable *cancellable, GError **error) { /* Write the fields in a deterministic order, following what is used * in the bootconfig example of the BootLoaderspec document: * https://systemd.io/BOOT_LOADER_SPECIFICATION */ const char *fields[] = { "title", "version", "options", "devicetree", "linux", "initrd" }; g_autoptr(GHashTable) keys_written = g_hash_table_new (g_str_hash, g_str_equal); g_autoptr(GString) buf = g_string_new (""); for (guint i = 0; i < G_N_ELEMENTS (fields); i++) { const char *key = fields[i]; const char *value = g_hash_table_lookup (self->options, key); if (value != NULL) { write_key (self, buf, key, value); g_hash_table_add (keys_written, (gpointer)key); } } /* Write overlay initrds */ if (self->overlay_initrds && (g_strv_length (self->overlay_initrds) > 0)) { /* we should've written the primary initrd already */ g_assert (g_hash_table_contains (keys_written, "initrd")); for (char **it = self->overlay_initrds; it && *it; it++) write_key (self, buf, "initrd", *it); } /* Write unknown fields */ GLNX_HASH_TABLE_FOREACH_KV (self->options, const char*, k, const char*, v) { if (g_hash_table_lookup (keys_written, k)) continue; write_key (self, buf, k, v); } if (!glnx_file_replace_contents_at (dfd, path, (guint8*)buf->str, buf->len, GLNX_FILE_REPLACE_NODATASYNC, cancellable, error)) return FALSE; return TRUE; } gboolean ostree_bootconfig_parser_write (OstreeBootconfigParser *self, GFile *output, GCancellable *cancellable, GError **error) { return ostree_bootconfig_parser_write_at (self, AT_FDCWD, gs_file_get_path_cached (output), cancellable, error); } static void ostree_bootconfig_parser_finalize (GObject *object) { OstreeBootconfigParser *self = OSTREE_BOOTCONFIG_PARSER (object); g_strfreev (self->overlay_initrds); g_hash_table_unref (self->options); G_OBJECT_CLASS (ostree_bootconfig_parser_parent_class)->finalize (object); } static void ostree_bootconfig_parser_init (OstreeBootconfigParser *self) { self->options = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free); } void ostree_bootconfig_parser_class_init (OstreeBootconfigParserClass *class) { GObjectClass *object_class = G_OBJECT_CLASS (class); object_class->finalize = ostree_bootconfig_parser_finalize; } OstreeBootconfigParser * ostree_bootconfig_parser_new (void) { OstreeBootconfigParser *self = NULL; self = g_object_new (OSTREE_TYPE_BOOTCONFIG_PARSER, NULL); self->separators = " \t"; return self; }