mirror of
https://git.proxmox.com/git/fwupd
synced 2025-06-13 13:27:47 +00:00
489 lines
15 KiB
C
489 lines
15 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
|
|
*
|
|
* Copyright (C) 2015 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 <fwupd.h>
|
|
#include <glib-object.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gio/gfiledescriptorbased.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "fu-keyring.h"
|
|
#include "fu-pending.h"
|
|
#include "fu-provider-fake.h"
|
|
#include "fu-provider-rpi.h"
|
|
#include "fu-rom.h"
|
|
|
|
/**
|
|
* fu_test_get_filename:
|
|
**/
|
|
static gchar *
|
|
fu_test_get_filename (const gchar *filename)
|
|
{
|
|
gchar *tmp;
|
|
char full_tmp[PATH_MAX];
|
|
g_autofree gchar *path = NULL;
|
|
path = g_build_filename (TESTDATADIR, filename, NULL);
|
|
tmp = realpath (path, full_tmp);
|
|
if (tmp == NULL)
|
|
return NULL;
|
|
return g_strdup (full_tmp);
|
|
}
|
|
|
|
static void
|
|
fu_rom_func (void)
|
|
{
|
|
guint i;
|
|
struct {
|
|
FuRomKind kind;
|
|
const gchar *fn;
|
|
const gchar *ver;
|
|
const gchar *csum;
|
|
guint16 vendor;
|
|
guint16 model;
|
|
} data[] = {
|
|
{ FU_ROM_KIND_ATI,
|
|
"Asus.9800PRO.256.unknown.031114.rom",
|
|
"008.015.041.001",
|
|
"3137385685298bbf7db2c8304f60d89005c731ed",
|
|
0x1002, 0x4e48 },
|
|
{ FU_ROM_KIND_ATI, /* atombios */
|
|
"Asus.R9290X.4096.131014.rom",
|
|
"015.039.000.006.003515",
|
|
"d8e32fa09a00ab9dcc96a990266f3fe5a99eacc5",
|
|
0x1002, 0x67b0 },
|
|
{ FU_ROM_KIND_ATI, /* atombios, with serial */
|
|
"Asus.HD7970.3072.121018.rom",
|
|
"015.023.000.002.000000",
|
|
"ba8b6ce38f2499c8463fc9d983b8e0162b1121e4",
|
|
0x1002, 0x6798 },
|
|
{ FU_ROM_KIND_NVIDIA,
|
|
"Asus.GTX480.1536.100406_1.rom",
|
|
"70.00.1A.00.02",
|
|
"3fcab24e60934850246fcfc4f42eceb32540a0ad",
|
|
0x10de, 0x06c0 },
|
|
{ FU_ROM_KIND_NVIDIA, /* nvgi */
|
|
"Asus.GTX980.4096.140905.rom",
|
|
"84.04.1F.00.02",
|
|
"98f58321145bd347156455356bc04c5b04a292f5",
|
|
0x10de, 0x13c0 },
|
|
{ FU_ROM_KIND_NVIDIA, /* nvgi, with serial */
|
|
"Asus.TitanBlack.6144.140212.rom",
|
|
"80.80.4E.00.01",
|
|
"3c80f35d4e3c440ffb427957d9271384113d7721",
|
|
0x10de, 0x100c },
|
|
{ FU_ROM_KIND_UNKNOWN, NULL, NULL, NULL, 0x0000, 0x0000 }
|
|
};
|
|
|
|
for (i = 0; data[i].fn != NULL; i++) {
|
|
gboolean ret;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autofree gchar *filename = NULL;
|
|
g_autoptr(FuRom) rom = NULL;
|
|
g_autoptr(GFile) file = NULL;
|
|
rom = fu_rom_new ();
|
|
g_assert (rom != NULL);
|
|
|
|
/* load file */
|
|
filename = fu_test_get_filename (data[i].fn);
|
|
if (filename == NULL)
|
|
continue;
|
|
g_print ("\nparsing %s...", filename);
|
|
file = g_file_new_for_path (filename);
|
|
ret = fu_rom_load_file (rom, file, FU_ROM_LOAD_FLAG_BLANK_PPID, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_assert_cmpstr (fu_rom_get_version (rom), ==, data[i].ver);
|
|
g_assert_cmpstr (fu_rom_get_checksum (rom), ==, data[i].csum);
|
|
g_assert_cmpint (fu_rom_get_kind (rom), ==, data[i].kind);
|
|
g_assert_cmpint (fu_rom_get_vendor (rom), ==, data[i].vendor);
|
|
g_assert_cmpint (fu_rom_get_model (rom), ==, data[i].model);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fu_rom_all_func (void)
|
|
{
|
|
GDir *dir;
|
|
g_autofree gchar *path = NULL;
|
|
|
|
/* may or may not exist */
|
|
path = fu_test_get_filename ("roms");
|
|
if (path == NULL)
|
|
return;
|
|
g_print ("\n");
|
|
dir = g_dir_open (path, 0, NULL);
|
|
do {
|
|
const gchar *fn;
|
|
gboolean ret;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autofree gchar *filename = NULL;
|
|
g_autoptr(FuRom) rom = NULL;
|
|
g_autoptr(GFile) file = NULL;
|
|
|
|
fn = g_dir_read_name (dir);
|
|
if (fn == NULL)
|
|
break;
|
|
filename = g_build_filename (path, fn, NULL);
|
|
g_print ("\nparsing %s...", filename);
|
|
file = g_file_new_for_path (filename);
|
|
rom = fu_rom_new ();
|
|
ret = fu_rom_load_file (rom, file, FU_ROM_LOAD_FLAG_BLANK_PPID, NULL, &error);
|
|
if (!ret) {
|
|
g_print ("%s %s : %s\n",
|
|
fu_rom_kind_to_string (fu_rom_get_kind (rom)),
|
|
filename, error->message);
|
|
continue;
|
|
}
|
|
g_assert_cmpstr (fu_rom_get_version (rom), !=, NULL);
|
|
g_assert_cmpstr (fu_rom_get_version (rom), !=, "\0");
|
|
g_assert_cmpstr (fu_rom_get_checksum (rom), !=, NULL);
|
|
g_assert_cmpint (fu_rom_get_kind (rom), !=, FU_ROM_KIND_UNKNOWN);
|
|
} while (TRUE);
|
|
}
|
|
|
|
static void
|
|
_provider_status_changed_cb (FuProvider *provider, FwupdStatus status, gpointer user_data)
|
|
{
|
|
guint *cnt = (guint *) user_data;
|
|
(*cnt)++;
|
|
}
|
|
|
|
static void
|
|
_provider_device_added_cb (FuProvider *provider, FuDevice *device, gpointer user_data)
|
|
{
|
|
FuDevice **dev = (FuDevice **) user_data;
|
|
*dev = g_object_ref (device);
|
|
}
|
|
|
|
static void
|
|
fu_provider_func (void)
|
|
{
|
|
GError *error = NULL;
|
|
FuDevice *device_tmp;
|
|
FwupdResult *res;
|
|
gboolean ret;
|
|
guint cnt = 0;
|
|
g_autofree gchar *pending_cap = NULL;
|
|
g_autofree gchar *pending_db = NULL;
|
|
g_autoptr(FuDevice) device = NULL;
|
|
g_autoptr(FuPending) pending = NULL;
|
|
g_autoptr(FuProvider) provider = NULL;
|
|
g_autoptr(GBytes) blob_cab = NULL;
|
|
g_autoptr(GMappedFile) mapped_file = NULL;
|
|
|
|
/* create a fake device */
|
|
g_setenv("FWUPD_OFFLINE_DIR", "/tmp/fwupd-self-test/var/lib/fwupd", FALSE);
|
|
provider = fu_provider_fake_new ();
|
|
g_signal_connect (provider, "device-added",
|
|
G_CALLBACK (_provider_device_added_cb),
|
|
&device);
|
|
g_signal_connect (provider, "status-changed",
|
|
G_CALLBACK (_provider_status_changed_cb),
|
|
&cnt);
|
|
ret = fu_provider_coldplug (provider, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
|
|
/* check we did the right thing */
|
|
g_assert_cmpint (cnt, ==, 0);
|
|
g_assert (device != NULL);
|
|
g_assert_cmpstr (fu_device_get_id (device), ==, "FakeDevice");
|
|
g_assert_cmpstr (fu_device_get_guid (device), ==,
|
|
"00000000-0000-0000-0000-000000000000");
|
|
|
|
/* schedule an offline update */
|
|
mapped_file = g_mapped_file_new ("/etc/resolv.conf", FALSE, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (mapped_file != NULL);
|
|
blob_cab = g_mapped_file_get_bytes (mapped_file);
|
|
ret = fu_provider_update (provider, device, blob_cab, NULL, NULL,
|
|
FWUPD_INSTALL_FLAG_OFFLINE, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_assert_cmpint (cnt, ==, 1);
|
|
|
|
/* lets check the pending */
|
|
pending = fu_pending_new ();
|
|
res = fu_pending_get_device (pending, fu_device_get_id (device), &error);
|
|
g_assert_no_error (error);
|
|
g_assert (res != NULL);
|
|
g_assert_cmpint (fu_device_get_update_state (res), ==, FWUPD_UPDATE_STATE_PENDING);
|
|
g_assert_cmpstr (fu_device_get_update_error (res), ==, NULL);
|
|
g_assert_cmpstr (fu_device_get_update_filename (res), !=, NULL);
|
|
|
|
/* save this; we'll need to delete it later */
|
|
pending_cap = g_strdup (fu_device_get_update_filename (res));
|
|
g_object_unref (res);
|
|
|
|
/* lets do this online */
|
|
ret = fu_provider_update (provider, device, blob_cab, NULL, NULL,
|
|
FWUPD_INSTALL_FLAG_NONE, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_assert_cmpint (cnt, ==, 3);
|
|
|
|
/* lets check the pending */
|
|
res = fu_pending_get_device (pending, fu_device_get_id (device), &error);
|
|
g_assert_no_error (error);
|
|
g_assert (res != NULL);
|
|
g_assert_cmpint (fu_device_get_update_state (res), ==, FWUPD_UPDATE_STATE_SUCCESS);
|
|
g_assert_cmpstr (fu_device_get_update_error (res), ==, NULL);
|
|
g_object_unref (res);
|
|
|
|
/* get the status */
|
|
device_tmp = fu_device_new ();
|
|
fu_device_set_id (device_tmp, "FakeDevice");
|
|
ret = fu_provider_get_results (provider, device_tmp, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_assert_cmpint (fu_device_get_update_state (device_tmp), ==, FWUPD_UPDATE_STATE_SUCCESS);
|
|
g_assert_cmpstr (fu_device_get_update_error (device_tmp), ==, NULL);
|
|
|
|
/* clear */
|
|
ret = fu_provider_clear_results (provider, device_tmp, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
|
|
/* re-get the status */
|
|
ret = fu_provider_get_results (provider, device_tmp, &error);
|
|
g_assert_error (error, FWUPD_ERROR, FWUPD_ERROR_NOTHING_TO_DO);
|
|
g_assert (!ret);
|
|
|
|
g_object_unref (device_tmp);
|
|
g_clear_error (&error);
|
|
|
|
/* delete files */
|
|
pending_db = g_build_filename (LOCALSTATEDIR, "lib", "fwupd", "pending.db", NULL);
|
|
g_unlink (pending_db);
|
|
g_unlink (pending_cap);
|
|
}
|
|
|
|
static void
|
|
fu_provider_rpi_func (void)
|
|
{
|
|
gboolean ret;
|
|
guint cnt = 0;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autofree gchar *path = NULL;
|
|
g_autofree gchar *pending_db = NULL;
|
|
g_autofree gchar *fwfile = NULL;
|
|
g_autoptr(FuDevice) device = NULL;
|
|
g_autoptr(FuProvider) provider = NULL;
|
|
g_autoptr(GBytes) blob_fw = NULL;
|
|
g_autoptr(GMappedFile) mapped_file = NULL;
|
|
|
|
/* test location */
|
|
path = fu_test_get_filename ("rpiboot");
|
|
g_assert (path != NULL);
|
|
|
|
/* create a fake device */
|
|
provider = fu_provider_rpi_new ();
|
|
fu_provider_rpi_set_fw_dir (FU_PROVIDER_RPI (provider), path);
|
|
g_signal_connect (provider, "device-added",
|
|
G_CALLBACK (_provider_device_added_cb),
|
|
&device);
|
|
g_signal_connect (provider, "status-changed",
|
|
G_CALLBACK (_provider_status_changed_cb),
|
|
&cnt);
|
|
ret = fu_provider_coldplug (provider, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
|
|
/* check we did the right thing */
|
|
g_assert_cmpint (cnt, ==, 0);
|
|
g_assert (device != NULL);
|
|
g_assert_cmpstr (fu_device_get_id (device), ==, "raspberry-pi");
|
|
g_assert_cmpstr (fu_device_get_guid (device), ==,
|
|
"91dd7368-8640-5d72-a217-a505c034dd0b");
|
|
g_assert_cmpstr (fu_device_get_version (device), ==,
|
|
"20150803");
|
|
|
|
/* ensure clean */
|
|
g_unlink ("/tmp/rpiboot/start.elf");
|
|
|
|
/* do update */
|
|
fu_provider_rpi_set_fw_dir (FU_PROVIDER_RPI (provider), "/tmp/rpiboot");
|
|
fwfile = fu_test_get_filename ("rpiupdate/firmware.bin");
|
|
g_assert (fwfile != NULL);
|
|
mapped_file = g_mapped_file_new (fwfile, FALSE, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (mapped_file != NULL);
|
|
blob_fw = g_mapped_file_get_bytes (mapped_file);
|
|
ret = fu_provider_update (provider, device, NULL, blob_fw, NULL,
|
|
FWUPD_INSTALL_FLAG_NONE, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_assert_cmpint (cnt, ==, 3);
|
|
|
|
/* check the file was exploded to the right place */
|
|
g_assert (g_file_test ("/tmp/rpiboot/start.elf", G_FILE_TEST_EXISTS));
|
|
g_assert (g_file_test ("/tmp/rpiboot/overlays/test.dtb", G_FILE_TEST_EXISTS));
|
|
g_assert_cmpstr (fu_device_get_version (device), ==,
|
|
"20150805");
|
|
|
|
/* clean up */
|
|
pending_db = g_build_filename (LOCALSTATEDIR, "lib", "fwupd", "pending.db", NULL);
|
|
g_unlink (pending_db);
|
|
}
|
|
|
|
static void
|
|
fu_pending_func (void)
|
|
{
|
|
GError *error = NULL;
|
|
gboolean ret;
|
|
FwupdResult *res;
|
|
g_autoptr(FuPending) pending = NULL;
|
|
g_autofree gchar *dirname = NULL;
|
|
g_autofree gchar *filename = NULL;
|
|
|
|
/* create */
|
|
pending = fu_pending_new ();
|
|
g_assert (pending != NULL);
|
|
|
|
/* delete the database */
|
|
dirname = g_build_filename (LOCALSTATEDIR, "lib", "fwupd", NULL);
|
|
if (!g_file_test (dirname, G_FILE_TEST_IS_DIR))
|
|
return;
|
|
filename = g_build_filename (dirname, "pending.db", NULL);
|
|
g_unlink (filename);
|
|
|
|
/* add a device */
|
|
res = fwupd_result_new ();
|
|
fu_device_set_id (res, "self-test");
|
|
fu_device_set_update_filename (res, "/var/lib/dave.cap"),
|
|
fu_device_set_name (res, "ColorHug"),
|
|
fu_device_set_version (res, "3.0.1"),
|
|
fu_device_set_update_version (res, "3.0.2");
|
|
ret = fu_pending_add_device (pending, res, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_object_unref (res);
|
|
|
|
/* ensure database was created */
|
|
g_assert (g_file_test (filename, G_FILE_TEST_EXISTS));
|
|
|
|
/* add some extra data */
|
|
res = fwupd_result_new ();
|
|
fu_device_set_id (res, "self-test");
|
|
ret = fu_pending_set_state (pending, res, FWUPD_UPDATE_STATE_PENDING, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
ret = fu_pending_set_error_msg (pending, res, "word", &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_object_unref (res);
|
|
|
|
/* get device */
|
|
res = fu_pending_get_device (pending, "self-test", &error);
|
|
g_assert_no_error (error);
|
|
g_assert (res != NULL);
|
|
g_assert_cmpstr (fwupd_result_get_device_id (res), ==, "self-test");
|
|
g_assert_cmpstr (fwupd_result_get_update_filename (res), ==, "/var/lib/dave.cap");
|
|
g_assert_cmpstr (fwupd_result_get_device_name (res), ==, "ColorHug");
|
|
g_assert_cmpstr (fwupd_result_get_device_version (res), ==, "3.0.1");
|
|
g_assert_cmpstr (fwupd_result_get_update_version (res), ==, "3.0.2");
|
|
g_assert_cmpint (fwupd_result_get_update_state (res), ==, FWUPD_UPDATE_STATE_PENDING);
|
|
g_assert_cmpstr (fwupd_result_get_update_error (res), ==, "word");
|
|
g_object_unref (res);
|
|
|
|
/* get device that does not exist */
|
|
res = fu_pending_get_device (pending, "XXXXXXXXXXXXX", &error);
|
|
g_assert_error (error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND);
|
|
g_assert (res == NULL);
|
|
g_clear_error (&error);
|
|
|
|
/* remove device */
|
|
res = fwupd_result_new ();
|
|
fu_device_set_id (res, "self-test");
|
|
ret = fu_pending_remove_device (pending, res, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
g_object_unref (res);
|
|
|
|
/* get device that does not exist */
|
|
res = fu_pending_get_device (pending, "self-test", &error);
|
|
g_assert_error (error, FWUPD_ERROR, FWUPD_ERROR_NOT_FOUND);
|
|
g_assert (res == NULL);
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
static void
|
|
fu_keyring_func (void)
|
|
{
|
|
gboolean ret;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autofree gchar *fw_fail = NULL;
|
|
g_autofree gchar *fw_pass = NULL;
|
|
g_autofree gchar *pki_dir = NULL;
|
|
g_autoptr(FuKeyring) keyring = NULL;
|
|
const gchar *sig =
|
|
"iQEcBAABCAAGBQJVt0B4AAoJEEim2A5FOLrCFb8IAK+QTLY34Wu8xZ8nl6p3JdMu"
|
|
"HOaifXAmX7291UrsFRwdabU2m65pqxQLwcoFrqGv738KuaKtu4oIwo9LIrmmTbEh"
|
|
"IID8uszxBt0bMdcIHrvwd+ADx+MqL4hR3guXEE3YOBTLvv2RF1UBcJPInNf/7Ui1"
|
|
"3lW1c3trL8RAJyx1B5RdKqAMlyfwiuvKM5oT4SN4uRSbQf+9mt78ZSWfJVZZH/RR"
|
|
"H9q7PzR5GdmbsRPM0DgC27Trvqjo3MzoVtoLjIyEb/aWqyulUbnJUNKPYTnZgkzM"
|
|
"v2yVofWKIM3e3wX5+MOtf6EV58mWa2cHJQ4MCYmpKxbIvAIZagZ4c9A8BA6tQWg="
|
|
"=fkit";
|
|
|
|
/* add test keys to keyring */
|
|
keyring = fu_keyring_new ();
|
|
pki_dir = fu_test_get_filename ("pki");
|
|
ret = fu_keyring_add_public_keys (keyring, pki_dir, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
|
|
/* verify */
|
|
fw_pass = fu_test_get_filename ("colorhug/firmware.bin");
|
|
ret = fu_keyring_verify_file (keyring, fw_pass, sig, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
|
|
/* verify will fail */
|
|
fw_fail = fu_test_get_filename ("colorhug/colorhug-als-3.0.2.cab");
|
|
ret = fu_keyring_verify_file (keyring, fw_fail, sig, &error);
|
|
g_assert_error (error, FWUPD_ERROR, FWUPD_ERROR_SIGNATURE_INVALID);
|
|
g_assert (!ret);
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
/* only critical and error are fatal */
|
|
g_log_set_fatal_mask (NULL, G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL);
|
|
|
|
g_assert_cmpint (g_mkdir_with_parents ("/tmp/fwupd-self-test/var/lib/fwupd", 0755), ==, 0);
|
|
|
|
/* tests go here */
|
|
g_test_add_func ("/fwupd/rom", fu_rom_func);
|
|
g_test_add_func ("/fwupd/rom{all}", fu_rom_all_func);
|
|
g_test_add_func ("/fwupd/pending", fu_pending_func);
|
|
g_test_add_func ("/fwupd/provider", fu_provider_func);
|
|
g_test_add_func ("/fwupd/provider{rpi}", fu_provider_rpi_func);
|
|
g_test_add_func ("/fwupd/keyring", fu_keyring_func);
|
|
return g_test_run ();
|
|
}
|
|
|