mirror of
https://git.proxmox.com/git/fwupd
synced 2026-03-29 01:20:55 +00:00
530 lines
14 KiB
C
530 lines
14 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
|
|
*
|
|
* Copyright (C) 2017 Richard Hughes <richard@hughsie.com>
|
|
*
|
|
* Licensed under the GNU General Public License Version 2
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <gio/gunixinputstream.h>
|
|
#include <glib/gstdio.h>
|
|
|
|
#include <archive_entry.h>
|
|
#include <archive.h>
|
|
#include <errno.h>
|
|
|
|
#include "fwupd-error.h"
|
|
|
|
#include "fu-common.h"
|
|
|
|
/**
|
|
* fu_common_rmtree:
|
|
* @directory: a directory name
|
|
* @error: A #GError or %NULL
|
|
*
|
|
* Recursively removes a directory.
|
|
*
|
|
* Returns: %TRUE for success, %FALSE otherwise
|
|
**/
|
|
gboolean
|
|
fu_common_rmtree (const gchar *directory, GError **error)
|
|
{
|
|
const gchar *filename;
|
|
g_autoptr(GDir) dir = NULL;
|
|
|
|
/* try to open */
|
|
g_debug ("removing %s", directory);
|
|
dir = g_dir_open (directory, 0, error);
|
|
if (dir == NULL)
|
|
return FALSE;
|
|
|
|
/* find each */
|
|
while ((filename = g_dir_read_name (dir))) {
|
|
g_autofree gchar *src = NULL;
|
|
src = g_build_filename (directory, filename, NULL);
|
|
if (g_file_test (src, G_FILE_TEST_IS_DIR)) {
|
|
if (!fu_common_rmtree (src, error))
|
|
return FALSE;
|
|
} else {
|
|
if (g_unlink (src) != 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"Failed to delete: %s", src);
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
if (g_remove (directory) != 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"Failed to delete: %s", directory);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* fu_common_mkdir_parent:
|
|
* @filename: A full pathname
|
|
* @error: A #GError, or %NULL
|
|
*
|
|
* Creates any required directories, including any parent directories.
|
|
*
|
|
* Returns: %TRUE for success
|
|
**/
|
|
gboolean
|
|
fu_common_mkdir_parent (const gchar *filename, GError **error)
|
|
{
|
|
g_autofree gchar *parent = NULL;
|
|
|
|
parent = g_path_get_dirname (filename);
|
|
g_debug ("creating path %s", parent);
|
|
if (g_mkdir_with_parents (parent, 0755) == -1) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"Failed to create '%s': %s",
|
|
parent, g_strerror (errno));
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* fu_common_set_contents_bytes:
|
|
* @filename: A filename
|
|
* @bytes: The data to write
|
|
* @error: A #GError, or %NULL
|
|
*
|
|
* Writes a blob of data to a filename, creating the parent directories as
|
|
* required.
|
|
*
|
|
* Returns: %TRUE for success
|
|
**/
|
|
gboolean
|
|
fu_common_set_contents_bytes (const gchar *filename, GBytes *bytes, GError **error)
|
|
{
|
|
const gchar *data;
|
|
gsize size;
|
|
g_autoptr(GFile) file = NULL;
|
|
g_autoptr(GFile) file_parent = NULL;
|
|
|
|
file = g_file_new_for_path (filename);
|
|
file_parent = g_file_get_parent (file);
|
|
if (!g_file_query_exists (file_parent, NULL)) {
|
|
if (!g_file_make_directory_with_parents (file_parent, NULL, error))
|
|
return FALSE;
|
|
}
|
|
data = g_bytes_get_data (bytes, &size);
|
|
g_debug ("writing %s with %" G_GSIZE_FORMAT " bytes", filename, size);
|
|
return g_file_set_contents (filename, data, size, error);
|
|
}
|
|
|
|
/**
|
|
* fu_common_get_contents_bytes:
|
|
* @filename: A filename
|
|
* @error: A #GError, or %NULL
|
|
*
|
|
* Reads a blob of data from a file.
|
|
*
|
|
* Returns: a #GBytes, or %NULL for failure
|
|
**/
|
|
GBytes *
|
|
fu_common_get_contents_bytes (const gchar *filename, GError **error)
|
|
{
|
|
gchar *data = NULL;
|
|
gsize len = 0;
|
|
if (!g_file_get_contents (filename, &data, &len, error))
|
|
return NULL;
|
|
g_debug ("reading %s with %" G_GSIZE_FORMAT " bytes", filename, len);
|
|
return g_bytes_new_take (data, len);
|
|
}
|
|
|
|
/**
|
|
* fu_common_get_contents_fd:
|
|
* @fd: A file descriptor
|
|
* @count: The maximum number of bytes to read
|
|
* @error: A #GError, or %NULL
|
|
*
|
|
* Reads a blob from a specific file descriptor.
|
|
*
|
|
* Note: this will close the fd when done
|
|
*
|
|
* Returns: (transfer container): a #GBytes, or %NULL
|
|
**/
|
|
GBytes *
|
|
fu_common_get_contents_fd (gint fd, gsize count, GError **error)
|
|
{
|
|
g_autoptr(GBytes) blob = NULL;
|
|
g_autoptr(GError) error_local = NULL;
|
|
g_autoptr(GInputStream) stream = NULL;
|
|
|
|
g_return_val_if_fail (fd > 0, NULL);
|
|
g_return_val_if_fail (count > 0, NULL);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
|
|
|
|
/* read the entire fd to a data blob */
|
|
stream = g_unix_input_stream_new (fd, TRUE);
|
|
blob = g_input_stream_read_bytes (stream, count, NULL, &error_local);
|
|
if (blob == NULL) {
|
|
g_set_error_literal (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INVALID_FILE,
|
|
error_local->message);
|
|
return NULL;
|
|
}
|
|
return g_steal_pointer (&blob);
|
|
}
|
|
|
|
static gboolean
|
|
fu_common_extract_archive_entry (struct archive_entry *entry, const gchar *dir)
|
|
{
|
|
const gchar *tmp;
|
|
g_autofree gchar *buf = NULL;
|
|
|
|
/* no output file */
|
|
if (archive_entry_pathname (entry) == NULL)
|
|
return FALSE;
|
|
|
|
/* update output path */
|
|
tmp = archive_entry_pathname (entry);
|
|
buf = g_build_filename (dir, tmp, NULL);
|
|
archive_entry_update_pathname_utf8 (entry, buf);
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* fu_common_extract_archive:
|
|
* @blob: a #GBytes archive as a blob
|
|
* @directory: a directory name to extract to
|
|
* @error: A #GError, or %NULL
|
|
*
|
|
* Extracts an achive to a directory.
|
|
*
|
|
* Returns: %TRUE for success
|
|
**/
|
|
gboolean
|
|
fu_common_extract_archive (GBytes *blob, const gchar *dir, GError **error)
|
|
{
|
|
gboolean ret = TRUE;
|
|
int r;
|
|
struct archive *arch = NULL;
|
|
struct archive_entry *entry;
|
|
|
|
/* decompress anything matching either glob */
|
|
g_debug ("decompressing into %s", dir);
|
|
arch = archive_read_new ();
|
|
archive_read_support_format_all (arch);
|
|
archive_read_support_filter_all (arch);
|
|
r = archive_read_open_memory (arch,
|
|
(void *) g_bytes_get_data (blob, NULL),
|
|
(size_t) g_bytes_get_size (blob));
|
|
if (r != 0) {
|
|
ret = FALSE;
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"Cannot open: %s",
|
|
archive_error_string (arch));
|
|
goto out;
|
|
}
|
|
for (;;) {
|
|
gboolean valid;
|
|
g_autofree gchar *path = NULL;
|
|
r = archive_read_next_header (arch, &entry);
|
|
if (r == ARCHIVE_EOF)
|
|
break;
|
|
if (r != ARCHIVE_OK) {
|
|
ret = FALSE;
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"Cannot read header: %s",
|
|
archive_error_string (arch));
|
|
goto out;
|
|
}
|
|
|
|
/* only extract if valid */
|
|
valid = fu_common_extract_archive_entry (entry, dir);
|
|
if (!valid)
|
|
continue;
|
|
r = archive_read_extract (arch, entry, 0);
|
|
if (r != ARCHIVE_OK) {
|
|
ret = FALSE;
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"Cannot extract: %s",
|
|
archive_error_string (arch));
|
|
goto out;
|
|
}
|
|
}
|
|
out:
|
|
if (arch != NULL) {
|
|
archive_read_close (arch);
|
|
archive_read_free (arch);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
fu_common_add_argv (GPtrArray *argv, const gchar *fmt, ...)
|
|
{
|
|
va_list args;
|
|
g_autofree gchar *tmp = NULL;
|
|
g_auto(GStrv) split = NULL;
|
|
|
|
va_start (args, fmt);
|
|
tmp = g_strdup_vprintf (fmt, args);
|
|
va_end (args);
|
|
|
|
split = g_strsplit (tmp, " ", -1);
|
|
for (guint i = 0; split[i] != NULL; i++)
|
|
g_ptr_array_add (argv, g_strdup (split[i]));
|
|
}
|
|
|
|
/**
|
|
* fu_common_firmware_builder:
|
|
* @bytes: The data to use
|
|
* @script_fn: Name of the script to run in the tarball, e.g. "startup.sh"
|
|
* @output_fn: Name of the generated firmware, e.g. "firmware.bin"
|
|
* @error: A #GError, or %NULL
|
|
*
|
|
* Builds a firmware file using tools from the host session in a bubblewrap
|
|
* jail. Several things happen during build:
|
|
*
|
|
* 1. The @bytes data is untarred to a temporary location
|
|
* 2. A bubblewrap container is set up
|
|
* 3. The startup.sh script is run inside the container
|
|
* 4. The firmware.bin is extracted from the container
|
|
* 5. The temporary location is deleted
|
|
*
|
|
* Returns: a new #GBytes, or %NULL for error
|
|
**/
|
|
GBytes *
|
|
fu_common_firmware_builder (GBytes *bytes,
|
|
const gchar *script_fn,
|
|
const gchar *output_fn,
|
|
GError **error)
|
|
{
|
|
gint rc = 0;
|
|
g_autofree gchar *argv_str = NULL;
|
|
g_autofree gchar *localstatedir = NULL;
|
|
g_autofree gchar *output2_fn = NULL;
|
|
g_autofree gchar *standard_error = NULL;
|
|
g_autofree gchar *standard_output = NULL;
|
|
g_autofree gchar *tmpdir = NULL;
|
|
g_autoptr(GBytes) firmware_blob = NULL;
|
|
g_autoptr(GPtrArray) argv = g_ptr_array_new_with_free_func (g_free);
|
|
|
|
g_return_val_if_fail (bytes != NULL, NULL);
|
|
g_return_val_if_fail (script_fn != NULL, NULL);
|
|
g_return_val_if_fail (output_fn != NULL, NULL);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
|
|
|
|
/* untar file to temp location */
|
|
tmpdir = g_dir_make_tmp ("fwupd-gen-XXXXXX", error);
|
|
if (tmpdir == NULL)
|
|
return NULL;
|
|
if (!fu_common_extract_archive (bytes, tmpdir, error))
|
|
return NULL;
|
|
|
|
/* this is shared with the plugins */
|
|
localstatedir = g_build_filename (LOCALSTATEDIR, "lib", "fwupd", "builder", NULL);
|
|
|
|
/* launch bubblewrap and generate firmware */
|
|
g_ptr_array_add (argv, g_strdup ("bwrap"));
|
|
fu_common_add_argv (argv, "--die-with-parent");
|
|
fu_common_add_argv (argv, "--ro-bind /usr /usr");
|
|
fu_common_add_argv (argv, "--dir /tmp");
|
|
fu_common_add_argv (argv, "--dir /var");
|
|
fu_common_add_argv (argv, "--bind %s /tmp", tmpdir);
|
|
if (g_file_test (localstatedir, G_FILE_TEST_EXISTS))
|
|
fu_common_add_argv (argv, "--ro-bind %s /boot", localstatedir);
|
|
fu_common_add_argv (argv, "--dev /dev");
|
|
fu_common_add_argv (argv, "--symlink usr/lib /lib");
|
|
fu_common_add_argv (argv, "--symlink usr/lib64 /lib64");
|
|
fu_common_add_argv (argv, "--symlink usr/bin /bin");
|
|
fu_common_add_argv (argv, "--symlink usr/sbin /sbin");
|
|
fu_common_add_argv (argv, "--chdir /tmp");
|
|
fu_common_add_argv (argv, "--unshare-all");
|
|
fu_common_add_argv (argv, "/tmp/%s", script_fn);
|
|
g_ptr_array_add (argv, NULL);
|
|
argv_str = g_strjoinv (" ", (gchar **) argv->pdata);
|
|
g_debug ("running '%s' in %s", argv_str, tmpdir);
|
|
if (!g_spawn_sync ("/tmp",
|
|
(gchar **) argv->pdata,
|
|
NULL,
|
|
G_SPAWN_SEARCH_PATH,
|
|
NULL, NULL, /* child_setup */
|
|
&standard_output,
|
|
&standard_error,
|
|
&rc,
|
|
error)) {
|
|
g_prefix_error (error, "failed to run '%s': ", argv_str);
|
|
return NULL;
|
|
}
|
|
if (standard_output != NULL && standard_output[0] != '\0')
|
|
g_debug ("console output was: %s", standard_output);
|
|
if (rc != 0) {
|
|
g_set_error (error,
|
|
FWUPD_ERROR,
|
|
FWUPD_ERROR_INTERNAL,
|
|
"failed to build firmware: %s",
|
|
standard_error);
|
|
return NULL;
|
|
}
|
|
|
|
/* get generated file */
|
|
output2_fn = g_build_filename (tmpdir, output_fn, NULL);
|
|
firmware_blob = fu_common_get_contents_bytes (output2_fn, error);
|
|
if (firmware_blob == NULL)
|
|
return NULL;
|
|
|
|
/* cleanup temp directory */
|
|
if (!fu_common_rmtree (tmpdir, error))
|
|
return NULL;
|
|
|
|
/* success */
|
|
return g_steal_pointer (&firmware_blob);
|
|
}
|
|
|
|
typedef struct {
|
|
FuOutputHandler handler_cb;
|
|
gpointer handler_user_data;
|
|
GMainLoop *loop;
|
|
GSource *source;
|
|
GInputStream *stream;
|
|
GCancellable *cancellable;
|
|
} FuCommonSpawnHelper;
|
|
|
|
static void fu_common_spawn_create_pollable_source (FuCommonSpawnHelper *helper);
|
|
|
|
static gboolean
|
|
fu_common_spawn_source_pollable_cb (GObject *stream, gpointer user_data)
|
|
{
|
|
FuCommonSpawnHelper *helper = (FuCommonSpawnHelper *) user_data;
|
|
gchar buffer[1024];
|
|
gssize sz;
|
|
g_auto(GStrv) split = NULL;
|
|
g_autoptr(GError) error = NULL;
|
|
|
|
/* read from stream */
|
|
sz = g_pollable_input_stream_read_nonblocking (G_POLLABLE_INPUT_STREAM (stream),
|
|
buffer,
|
|
sizeof(buffer) - 1,
|
|
NULL,
|
|
&error);
|
|
if (sz < 0) {
|
|
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_WOULD_BLOCK)) {
|
|
g_warning ("failed to get read from nonblocking fd: %s",
|
|
error->message);
|
|
}
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
/* no read possible */
|
|
if (sz == 0)
|
|
g_main_loop_quit (helper->loop);
|
|
|
|
/* emit lines */
|
|
if (helper->handler_cb != NULL) {
|
|
buffer[sz] = '\0';
|
|
split = g_strsplit (buffer, "\n", -1);
|
|
for (guint i = 0; split[i] != NULL; i++) {
|
|
if (split[i][0] == '\0')
|
|
continue;
|
|
helper->handler_cb (split[i], helper->handler_user_data);
|
|
}
|
|
}
|
|
|
|
/* set up the source for the next read */
|
|
fu_common_spawn_create_pollable_source (helper);
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
static void
|
|
fu_common_spawn_create_pollable_source (FuCommonSpawnHelper *helper)
|
|
{
|
|
if (helper->source != NULL)
|
|
g_source_destroy (helper->source);
|
|
helper->source = g_pollable_input_stream_create_source (G_POLLABLE_INPUT_STREAM (helper->stream),
|
|
helper->cancellable);
|
|
g_source_attach (helper->source, NULL);
|
|
g_source_set_callback (helper->source, (GSourceFunc) fu_common_spawn_source_pollable_cb, helper, NULL);
|
|
}
|
|
|
|
static void
|
|
fu_common_spawn_helper_free (FuCommonSpawnHelper *helper)
|
|
{
|
|
if (helper->stream != NULL)
|
|
g_object_unref (helper->stream);
|
|
if (helper->source != NULL)
|
|
g_source_destroy (helper->source);
|
|
if (helper->loop != NULL)
|
|
g_main_loop_unref (helper->loop);
|
|
g_free (helper);
|
|
}
|
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(FuCommonSpawnHelper, fu_common_spawn_helper_free)
|
|
|
|
/**
|
|
* fu_common_spawn_sync:
|
|
* @argv: The argument list to run
|
|
* @handler_cb: A #FuOutputHandler or %NULL
|
|
* @handler_user_data: the user data to pass to @handler
|
|
* @cancellable: a #GCancellable, or %NULL
|
|
* @error: A #GError or %NULL
|
|
*
|
|
* Runs a subprocess and waits for it to exit. Any output on standard out or
|
|
* standard error will be forwarded to @handler_cb as whole lines.
|
|
*
|
|
* Returns: %TRUE for success
|
|
**/
|
|
gboolean
|
|
fu_common_spawn_sync (const gchar * const * argv,
|
|
FuOutputHandler handler_cb,
|
|
gpointer handler_user_data,
|
|
GCancellable *cancellable, GError **error)
|
|
{
|
|
g_autoptr(FuCommonSpawnHelper) helper = NULL;
|
|
g_autoptr(GSubprocess) subprocess = NULL;
|
|
g_autofree gchar *argv_str = NULL;
|
|
|
|
/* create subprocess */
|
|
argv_str = g_strjoinv (" ", (gchar **) argv);
|
|
g_debug ("running '%s'", argv_str);
|
|
subprocess = g_subprocess_newv (argv, G_SUBPROCESS_FLAGS_STDOUT_PIPE |
|
|
G_SUBPROCESS_FLAGS_STDERR_MERGE, error);
|
|
if (subprocess == NULL)
|
|
return FALSE;
|
|
|
|
/* watch for process to exit */
|
|
helper = g_new0 (FuCommonSpawnHelper, 1);
|
|
helper->handler_cb = handler_cb;
|
|
helper->handler_user_data = handler_user_data;
|
|
helper->loop = g_main_loop_new (NULL, FALSE);
|
|
helper->stream = g_subprocess_get_stdout_pipe (subprocess);
|
|
helper->cancellable = cancellable;
|
|
fu_common_spawn_create_pollable_source (helper);
|
|
g_main_loop_run (helper->loop);
|
|
return g_subprocess_wait_check (subprocess, cancellable, error);
|
|
}
|