mirror of
https://git.proxmox.com/git/fwupd
synced 2025-05-25 07:10:50 +00:00

CSR is short for Cambridge Silicon Radio, which is a the OEM that makes most of the bluetooth audio chips in vendor hardware. The hardware vendor can enable or disable features on the CSR microcontroller depending on licensing options. The hardware vendor can also use a custom USB descriptor, or just set a custom PID. In the latter case we need to set the vendor and model to reality using quirks. This commit allows the user to update the firmware in the AIAIAI H05 wireless headphones.
388 lines
11 KiB
C
388 lines
11 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 <locale.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "dfu-firmware.h"
|
|
|
|
#include "fu-progressbar.h"
|
|
#include "fu-csr-device.h"
|
|
|
|
typedef struct {
|
|
FuQuirks *quirks;
|
|
GPtrArray *cmd_array;
|
|
FuProgressbar *progressbar;
|
|
} FuCsrToolPrivate;
|
|
|
|
static void
|
|
fu_csr_tool_private_free (FuCsrToolPrivate *priv)
|
|
{
|
|
if (priv == NULL)
|
|
return;
|
|
g_object_unref (priv->quirks);
|
|
g_object_unref (priv->progressbar);
|
|
if (priv->cmd_array != NULL)
|
|
g_ptr_array_unref (priv->cmd_array);
|
|
g_free (priv);
|
|
}
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(FuCsrToolPrivate, fu_csr_tool_private_free)
|
|
|
|
typedef gboolean (*FuCsrToolPrivateCb) (FuCsrToolPrivate *util,
|
|
gchar **values,
|
|
GError **error);
|
|
|
|
typedef struct {
|
|
gchar *name;
|
|
gchar *arguments;
|
|
gchar *description;
|
|
FuCsrToolPrivateCb callback;
|
|
} FuCsrToolItem;
|
|
|
|
static void
|
|
fu_csr_tool_item_free (FuCsrToolItem *item)
|
|
{
|
|
g_free (item->name);
|
|
g_free (item->arguments);
|
|
g_free (item->description);
|
|
g_free (item);
|
|
}
|
|
|
|
static gint
|
|
fu_csr_tool_sort_command_name_cb (FuCsrToolItem **item1, FuCsrToolItem **item2)
|
|
{
|
|
return g_strcmp0 ((*item1)->name, (*item2)->name);
|
|
}
|
|
|
|
static void
|
|
fu_csr_tool_add (GPtrArray *array,
|
|
const gchar *name,
|
|
const gchar *arguments,
|
|
const gchar *description,
|
|
FuCsrToolPrivateCb callback)
|
|
{
|
|
g_auto(GStrv) names = NULL;
|
|
|
|
g_return_if_fail (name != NULL);
|
|
g_return_if_fail (description != NULL);
|
|
g_return_if_fail (callback != NULL);
|
|
|
|
/* add each one */
|
|
names = g_strsplit (name, ",", -1);
|
|
for (guint i = 0; names[i] != NULL; i++) {
|
|
FuCsrToolItem *item = g_new0 (FuCsrToolItem, 1);
|
|
item->name = g_strdup (names[i]);
|
|
if (i == 0) {
|
|
item->description = g_strdup (description);
|
|
} else {
|
|
item->description = g_strdup_printf ("Alias to %s", names[0]);
|
|
}
|
|
item->arguments = g_strdup (arguments);
|
|
item->callback = callback;
|
|
g_ptr_array_add (array, item);
|
|
}
|
|
}
|
|
|
|
static gchar *
|
|
fu_csr_tool_get_descriptions (GPtrArray *array)
|
|
{
|
|
const gsize max_len = 31;
|
|
GString *str;
|
|
|
|
/* print each command */
|
|
str = g_string_new ("");
|
|
for (guint i = 0; i < array->len; i++) {
|
|
FuCsrToolItem *item = g_ptr_array_index (array, i);
|
|
gsize len;
|
|
g_string_append (str, " ");
|
|
g_string_append (str, item->name);
|
|
len = strlen (item->name) + 2;
|
|
if (item->arguments != NULL) {
|
|
g_string_append (str, " ");
|
|
g_string_append (str, item->arguments);
|
|
len += strlen (item->arguments) + 1;
|
|
}
|
|
if (len < max_len) {
|
|
for (guint j = len; j < max_len + 1; j++)
|
|
g_string_append_c (str, ' ');
|
|
g_string_append (str, item->description);
|
|
g_string_append_c (str, '\n');
|
|
} else {
|
|
g_string_append_c (str, '\n');
|
|
for (guint j = 0; j < max_len + 1; j++)
|
|
g_string_append_c (str, ' ');
|
|
g_string_append (str, item->description);
|
|
g_string_append_c (str, '\n');
|
|
}
|
|
}
|
|
|
|
/* remove trailing newline */
|
|
if (str->len > 0)
|
|
g_string_set_size (str, str->len - 1);
|
|
|
|
return g_string_free (str, FALSE);
|
|
}
|
|
|
|
static gboolean
|
|
fu_csr_tool_run (FuCsrToolPrivate *priv, const gchar *command, gchar **values, GError **error)
|
|
{
|
|
/* find command */
|
|
for (guint i = 0; i < priv->cmd_array->len; i++) {
|
|
FuCsrToolItem *item = g_ptr_array_index (priv->cmd_array, i);
|
|
if (g_strcmp0 (item->name, command) == 0)
|
|
return item->callback (priv, values, error);
|
|
}
|
|
|
|
/* not found */
|
|
g_set_error_literal (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_FOUND,
|
|
"Command not found");
|
|
return FALSE;
|
|
}
|
|
|
|
static FuCsrDevice *
|
|
fu_csr_get_default_device (FuCsrToolPrivate *priv, GError **error)
|
|
{
|
|
g_autoptr(GUsbContext) usb_context = NULL;
|
|
g_autoptr(GPtrArray) devices = NULL;
|
|
|
|
/* get all the DFU devices */
|
|
usb_context = g_usb_context_new (error);
|
|
if (usb_context == NULL)
|
|
return NULL;
|
|
devices = g_usb_context_get_devices (usb_context);
|
|
for (guint i = 0; i < devices->len; i++) {
|
|
GUsbDevice *usb_device = g_ptr_array_index (devices, i);
|
|
g_autoptr(FuCsrDevice) device = fu_csr_device_new (usb_device);
|
|
fu_device_set_quirks (FU_DEVICE (device), priv->quirks);
|
|
if (fu_usb_device_probe (FU_USB_DEVICE (device), NULL))
|
|
return g_steal_pointer (&device);
|
|
}
|
|
|
|
/* unsupported */
|
|
g_set_error_literal (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_SUPPORTED,
|
|
"no supported devices found");
|
|
return NULL;
|
|
}
|
|
|
|
static gboolean
|
|
fu_csr_tool_info (FuCsrToolPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FuCsrDevice) device = fu_csr_get_default_device (priv, error);
|
|
g_autofree gchar *str = NULL;
|
|
if (device == NULL)
|
|
return FALSE;
|
|
if (!fu_usb_device_open (FU_USB_DEVICE (device), error))
|
|
return FALSE;
|
|
str = fu_device_to_string (FU_DEVICE (device));
|
|
g_print ("%s", str);
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
fu_csr_tool_progress_cb (FuDevice *device, GParamSpec *pspec, FuCsrToolPrivate *priv)
|
|
{
|
|
fu_progressbar_update (priv->progressbar,
|
|
fu_device_get_status (device),
|
|
fu_device_get_progress (device));
|
|
}
|
|
|
|
static gboolean
|
|
fu_csr_tool_dump (FuCsrToolPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
GUsbDevice *usb_device;
|
|
g_autoptr(DfuElement) dfu_element = dfu_element_new ();
|
|
g_autoptr(DfuFirmware) dfu_firmware = dfu_firmware_new ();
|
|
g_autoptr(DfuImage) dfu_image = dfu_image_new ();
|
|
g_autoptr(FuCsrDevice) device = NULL;
|
|
g_autoptr(GBytes) blob = NULL;
|
|
g_autoptr(GFile) file = NULL;
|
|
|
|
/* check args */
|
|
if (g_strv_length (values) != 1) {
|
|
g_set_error_literal (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"Invalid arguments, expected FILENAME"
|
|
" -- e.g. `firmware.hex`");
|
|
return FALSE;
|
|
}
|
|
|
|
/* upload from the device */
|
|
device = fu_csr_get_default_device (priv, error);
|
|
if (device == NULL)
|
|
return FALSE;
|
|
if (!fu_usb_device_open (FU_USB_DEVICE (device), error))
|
|
return FALSE;
|
|
g_signal_connect (device, "notify::status",
|
|
G_CALLBACK (fu_csr_tool_progress_cb), priv);
|
|
g_signal_connect (device, "notify::progress",
|
|
G_CALLBACK (fu_csr_tool_progress_cb), priv);
|
|
blob = fu_csr_device_upload (device, error);
|
|
if (blob == NULL)
|
|
return FALSE;
|
|
|
|
/* create DFU file */
|
|
usb_device = fu_usb_device_get_dev (FU_USB_DEVICE (device));
|
|
dfu_element_set_contents (dfu_element, blob);
|
|
dfu_image_add_element (dfu_image, dfu_element);
|
|
dfu_firmware_add_image (dfu_firmware, dfu_image);
|
|
dfu_firmware_set_format (dfu_firmware, DFU_FIRMWARE_FORMAT_DFU);
|
|
dfu_firmware_set_vid (dfu_firmware, g_usb_device_get_vid (usb_device));
|
|
dfu_firmware_set_pid (dfu_firmware, g_usb_device_get_pid (usb_device));
|
|
|
|
/* save file */
|
|
file = g_file_new_for_path (values[0]);
|
|
return dfu_firmware_write_file (dfu_firmware, file, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_csr_tool_write (FuCsrToolPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FuCsrDevice) device = NULL;
|
|
g_autoptr(GBytes) blob = NULL;
|
|
|
|
/* check args */
|
|
if (g_strv_length (values) != 1) {
|
|
g_set_error_literal (error,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
"Invalid arguments, expected "
|
|
"FILENAME -- e.g. `firmware.hex`");
|
|
return FALSE;
|
|
}
|
|
|
|
/* get device */
|
|
device = fu_csr_get_default_device (priv, error);
|
|
if (device == NULL)
|
|
return FALSE;
|
|
|
|
/* load firmware file */
|
|
blob = fu_common_get_contents_bytes (values[0], error);
|
|
if (blob == NULL)
|
|
return FALSE;
|
|
|
|
/* write new firmware */
|
|
if (!fu_usb_device_open (FU_USB_DEVICE (device), error))
|
|
return FALSE;
|
|
g_signal_connect (device, "notify::status",
|
|
G_CALLBACK (fu_csr_tool_progress_cb), priv);
|
|
g_signal_connect (device, "notify::progress",
|
|
G_CALLBACK (fu_csr_tool_progress_cb), priv);
|
|
return fu_csr_device_download (device, blob, error);
|
|
}
|
|
|
|
static gboolean
|
|
fu_csr_tool_attach (FuCsrToolPrivate *priv, gchar **values, GError **error)
|
|
{
|
|
g_autoptr(FuCsrDevice) device = NULL;
|
|
if (device == NULL)
|
|
return FALSE;
|
|
if (!fu_usb_device_open (FU_USB_DEVICE (device), error))
|
|
return FALSE;
|
|
return fu_csr_device_attach (device, error);
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
gboolean verbose = FALSE;
|
|
g_autofree gchar *cmd_descriptions = NULL;
|
|
g_autoptr(GError) error = NULL;
|
|
g_autoptr(GOptionContext) context = NULL;
|
|
g_autoptr(FuCsrToolPrivate) priv = g_new0 (FuCsrToolPrivate, 1);
|
|
const GOptionEntry options[] = {
|
|
{ "verbose", 'v', 0, G_OPTION_ARG_NONE, &verbose,
|
|
"Print verbose debug statements", NULL },
|
|
{ NULL}
|
|
};
|
|
setlocale (LC_ALL, "");
|
|
|
|
priv->progressbar = fu_progressbar_new ();
|
|
fu_progressbar_set_length_percentage (priv->progressbar, 50);
|
|
fu_progressbar_set_length_status (priv->progressbar, 20);
|
|
|
|
/* add commands */
|
|
priv->cmd_array = g_ptr_array_new_with_free_func ((GDestroyNotify) fu_csr_tool_item_free);
|
|
fu_csr_tool_add (priv->cmd_array,
|
|
"info", NULL,
|
|
"Show information about the device",
|
|
fu_csr_tool_info);
|
|
fu_csr_tool_add (priv->cmd_array,
|
|
"write", "FILENAME",
|
|
"Update the firmware",
|
|
fu_csr_tool_write);
|
|
fu_csr_tool_add (priv->cmd_array,
|
|
"dump", "FILENAME",
|
|
"Dump the firmware",
|
|
fu_csr_tool_dump);
|
|
fu_csr_tool_add (priv->cmd_array,
|
|
"attach", NULL,
|
|
"Attach to firmware mode",
|
|
fu_csr_tool_attach);
|
|
|
|
/* sort by command name */
|
|
g_ptr_array_sort (priv->cmd_array,
|
|
(GCompareFunc) fu_csr_tool_sort_command_name_cb);
|
|
|
|
/* get a list of the commands */
|
|
context = g_option_context_new (NULL);
|
|
cmd_descriptions = fu_csr_tool_get_descriptions (priv->cmd_array);
|
|
g_option_context_set_summary (context, cmd_descriptions);
|
|
g_set_application_name ("CSR Debug Tool");
|
|
g_option_context_add_main_entries (context, options, NULL);
|
|
if (!g_option_context_parse (context, &argc, &argv, &error)) {
|
|
g_print ("%s: %s\n", "Failed to parse arguments", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* set verbose? */
|
|
if (verbose)
|
|
g_setenv ("G_MESSAGES_DEBUG", "all", FALSE);
|
|
|
|
/* use quirks */
|
|
priv->quirks = fu_quirks_new ();
|
|
if (!fu_quirks_load (priv->quirks, &error)) {
|
|
g_print ("Failed to load quirks: %s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* run the specified command */
|
|
if (!fu_csr_tool_run (priv, argv[1], (gchar**) &argv[2], &error)) {
|
|
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND)) {
|
|
g_autofree gchar *tmp = NULL;
|
|
tmp = g_option_context_get_help (context, TRUE, NULL);
|
|
g_print ("%s\n\n%s", error->message, tmp);
|
|
} else {
|
|
g_print ("%s\n", error->message);
|
|
}
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
return 0;
|
|
}
|