fwupd/plugins/unifying/lu-tool.c
2017-09-28 09:23:52 +01:00

550 lines
15 KiB
C

/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-
*
* Copyright (C) 2016 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 <string.h>
#include <stdlib.h>
#include "lu-context.h"
#include "lu-device-bootloader.h"
#include "lu-hidpp.h"
typedef struct {
GCancellable *cancellable;
GPtrArray *cmd_array;
LuContext *ctx;
LuDeviceKind emulation_kind;
} FuLuToolPrivate;
static void
lu_tool_private_free (FuLuToolPrivate *priv)
{
if (priv == NULL)
return;
if (priv->ctx != NULL)
g_object_unref (priv->ctx);
g_object_unref (priv->cancellable);
if (priv->cmd_array != NULL)
g_ptr_array_unref (priv->cmd_array);
g_free (priv);
}
G_DEFINE_AUTOPTR_CLEANUP_FUNC(FuLuToolPrivate, lu_tool_private_free)
typedef gboolean (*FuLuToolPrivateCb) (FuLuToolPrivate *util,
gchar **values,
GError **error);
typedef struct {
gchar *name;
gchar *arguments;
gchar *description;
FuLuToolPrivateCb callback;
} FuLuToolItem;
static void
lu_tool_item_free (FuLuToolItem *item)
{
g_free (item->name);
g_free (item->arguments);
g_free (item->description);
g_free (item);
}
static gint
lu_tool_sort_command_name_cb (FuLuToolItem **item1, FuLuToolItem **item2)
{
return g_strcmp0 ((*item1)->name, (*item2)->name);
}
static void
lu_tool_add (GPtrArray *array,
const gchar *name,
const gchar *arguments,
const gchar *description,
FuLuToolPrivateCb 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++) {
FuLuToolItem *item = g_new0 (FuLuToolItem, 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 *
lu_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++) {
FuLuToolItem *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
lu_tool_run (FuLuToolPrivate *priv,
const gchar *command,
gchar **values,
GError **error)
{
/* find command */
for (guint i = 0; i < priv->cmd_array->len; i++) {
FuLuToolItem *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 LuDevice *
lu_get_default_device (FuLuToolPrivate *priv, GError **error)
{
GPtrArray *devices = NULL;
LuDevice *device = NULL;
devices = lu_context_get_devices (priv->ctx);
for (guint i = 0; i < devices->len; i++) {
LuDevice *device_tmp = g_ptr_array_index (devices, i);
g_debug ("got %s", lu_device_kind_to_string (lu_device_get_kind (device_tmp)));
if (lu_device_get_kind (device_tmp) != LU_DEVICE_KIND_PERIPHERAL) {
device = g_object_ref (device_tmp);
break;
}
}
/* nothing supported */
if (device == NULL) {
g_set_error_literal (error,
G_IO_ERROR,
G_IO_ERROR_FAILED,
"No supported device plugged in");
return NULL;
}
return device;
}
static gboolean
lu_tool_info_device (LuDevice *device)
{
g_autofree gchar *str = lu_device_to_string (device);
g_print ("%s", str);
return TRUE;
}
static gboolean
lu_tool_info (FuLuToolPrivate *priv, gchar **values, GError **error)
{
GPtrArray *devices = NULL;
g_autoptr(LuContext) ctx = NULL;
/* emulated */
if (priv->emulation_kind != LU_DEVICE_KIND_UNKNOWN) {
g_autoptr(LuDevice) device = NULL;
device = lu_device_fake_new (priv->emulation_kind);
lu_tool_info_device (device);
}
/* get the devices */
ctx = lu_context_new (error);
if (ctx == NULL) {
g_prefix_error (error, "Failed to create context: ");
return FALSE;
}
devices = lu_context_get_devices (ctx);
for (guint i = 0; i < devices->len; i++) {
LuDevice *device = g_ptr_array_index (devices, i);
lu_tool_info_device (device);
if (i != devices->len - 1)
g_print ("\n");
}
return TRUE;
}
static void
lu_write_progress_cb (goffset current, goffset total, gpointer user_data)
{
FuLuToolPrivate *priv = (FuLuToolPrivate *) user_data;
gdouble percentage = -1.f;
if (priv->emulation_kind != LU_DEVICE_KIND_UNKNOWN)
return;
if (total > 0)
percentage = (100.f * (gdouble) current) / (gdouble) total;
g_print ("Written %" G_GOFFSET_FORMAT "/%" G_GOFFSET_FORMAT " bytes [%.1f%%]\n",
current, total, percentage);
}
static gboolean
lu_tool_dump (FuLuToolPrivate *priv, gchar **values, GError **error)
{
g_autoptr(GPtrArray) reqs = NULL;
g_autoptr(LuDevice) device = NULL;
g_autoptr(GBytes) fw = NULL;
g_autofree gchar *data = NULL;
gsize len = 0;
/* 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;
}
/* fake a huge device */
device = g_object_new (LU_TYPE_DEVICE_BOOTLOADER, NULL);
lu_device_bootloader_set_addr_lo (device, 0x0000);
lu_device_bootloader_set_addr_hi (device, 0xffff);
/* load file and display */
if (!g_file_get_contents (values[0], &data, &len, error))
return FALSE;
fw = g_bytes_new_static (data, len);
reqs = lu_device_bootloader_parse_requests (device, fw, error);
if (reqs == NULL)
return FALSE;
for (guint i = 0; i < reqs->len; i++) {
LuDeviceBootloaderRequest *req = g_ptr_array_index (reqs, i);
g_print ("0x%04x [0x%02x]", req->addr, req->len);
for (guint j = 0; j < req->len; j++)
g_print (" %02x", req->data[j]);
g_print ("\n");
}
return TRUE;
}
static gboolean
lu_tool_write (FuLuToolPrivate *priv, gchar **values, GError **error)
{
gsize len;
g_autofree guint8 *data = NULL;
g_autoptr(GBytes) fw = NULL;
g_autoptr(LuDevice) device = 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 [PLATFORM-ID]"
" -- e.g. `firmware.hex`");
return FALSE;
}
/* open device */
if (g_strv_length (values) == 2) {
device = lu_context_find_by_platform_id (priv->ctx,
values[1],
error);
} else if (priv->emulation_kind == LU_DEVICE_KIND_UNKNOWN) {
device = lu_get_default_device (priv, error);
} else {
device = lu_device_fake_new (priv->emulation_kind);
}
if (device == NULL)
return FALSE;
/* do we need to go into bootloader mode */
if (lu_device_has_flag (device, LU_DEVICE_FLAG_REQUIRES_DETACH)) {
if (!lu_device_detach (device, error))
return FALSE;
if (lu_device_has_flag (device, LU_DEVICE_FLAG_DETACH_WILL_REPLUG)) {
if (!lu_context_wait_for_replug (priv->ctx,
device,
FU_DEVICE_TIMEOUT_REPLUG,
error))
return FALSE;
g_usleep (G_USEC_PER_SEC);
g_clear_object (&device);
if (g_strv_length (values) == 2) {
device = lu_context_find_by_platform_id (priv->ctx,
values[1],
error);
} else if (priv->emulation_kind == LU_DEVICE_KIND_UNKNOWN) {
device = lu_get_default_device (priv, error);
} else {
device = lu_device_fake_new (priv->emulation_kind);
}
if (device == NULL)
return FALSE;
if (!lu_device_open (device, error)) {
g_prefix_error (error, "failed to reclaim device: ");
return FALSE;
}
}
}
/* load firmware file */
if (!g_file_get_contents (values[0], (gchar **) &data, &len, error)) {
g_prefix_error (error, "Failed to load %s: ", values[0]);
return FALSE;
}
/* update with data blob */
fw = g_bytes_new (data, len);
if (!lu_device_write_firmware (device, fw,
lu_write_progress_cb, priv,
error))
return FALSE;
/* detach back into runtime */
if (!lu_device_attach (device, error))
return FALSE;
return TRUE;
}
static gboolean
lu_tool_attach (FuLuToolPrivate *priv, gchar **values, GError **error)
{
g_autoptr(LuDevice) device = NULL;
if (g_strv_length (values) == 1) {
device = lu_context_find_by_platform_id (priv->ctx,
values[0],
error);
if (device == NULL)
return FALSE;
} else {
GPtrArray *devices = NULL;
devices = lu_context_get_devices (priv->ctx);
for (guint i = 0; i < devices->len; i++) {
LuDevice *device_tmp = g_ptr_array_index (devices, i);
g_debug ("got %s", lu_device_kind_to_string (lu_device_get_kind (device_tmp)));
if (lu_device_has_flag (device_tmp, LU_DEVICE_FLAG_REQUIRES_ATTACH)) {
device = g_object_ref (device_tmp);
break;
}
}
if (device == NULL) {
g_set_error_literal (error,
G_IO_ERROR,
G_IO_ERROR_FAILED,
"No attachable device plugged in");
return FALSE;
}
}
if (!lu_device_attach (device, error))
return FALSE;
return TRUE;
}
static void
lu_tool_device_added_cb (LuContext* ctx, LuDevice *device, FuLuToolPrivate *priv)
{
g_print ("ADDED\tLogitech Unifying device %s {%p} [%s]\n",
lu_device_kind_to_string (lu_device_get_kind (device)),
device, lu_device_get_platform_id (device));
lu_tool_info_device (device);
}
static void
lu_tool_device_removed_cb (LuContext* ctx, LuDevice *device, FuLuToolPrivate *priv)
{
g_print ("REMOVED\tLogitech Unifying device %s {%p} [%s]\n",
lu_device_kind_to_string (lu_device_get_kind (device)),
device, lu_device_get_platform_id (device));
}
static gboolean
lu_tool_watch (FuLuToolPrivate *priv, gchar **values, GError **error)
{
g_autoptr(GMainLoop) loop = g_main_loop_new (NULL, FALSE);
g_signal_connect (priv->ctx, "added", G_CALLBACK (lu_tool_device_added_cb), priv);
g_signal_connect (priv->ctx, "removed", G_CALLBACK (lu_tool_device_removed_cb), priv);
lu_context_coldplug (priv->ctx);
lu_context_set_poll_interval (priv->ctx, 2000);
g_main_loop_run (loop);
return TRUE;
}
static gboolean
lu_tool_detach (FuLuToolPrivate *priv, gchar **values, GError **error)
{
g_autoptr(LuDevice) device = NULL;
if (g_strv_length (values) == 1) {
device = lu_context_find_by_platform_id (priv->ctx,
values[0],
error);
} else {
device = lu_get_default_device (priv, error);
}
if (device == NULL)
return FALSE;
if (!lu_device_detach (device, error))
return FALSE;
return TRUE;
}
static void
lu_tool_log_handler_cb (const gchar *log_domain,
GLogLevelFlags log_level,
const gchar *message,
gpointer user_data)
{
g_print ("%s\t%s\n", log_domain, message);
}
int
main (int argc, char **argv)
{
gboolean verbose = FALSE;
g_autofree gchar *cmd_descriptions = NULL;
g_autofree gchar *emulation_kind = NULL;
g_autoptr(GError) error = NULL;
g_autoptr(GOptionContext) context = NULL;
g_autoptr(FuLuToolPrivate) priv = g_new0 (FuLuToolPrivate, 1);
const GOptionEntry options[] = {
{ "verbose", 'v', 0, G_OPTION_ARG_NONE, &verbose,
"Print verbose debug statements", NULL },
{ "emulate", 'e', 0, G_OPTION_ARG_STRING, &emulation_kind,
"Emulate a device type", NULL },
{ NULL}
};
/* FIXME: do stuff on ctrl+c */
priv->cancellable = g_cancellable_new ();
/* add commands */
priv->cmd_array = g_ptr_array_new_with_free_func ((GDestroyNotify) lu_tool_item_free);
lu_tool_add (priv->cmd_array,
"info", NULL,
"Show information about the device",
lu_tool_info);
lu_tool_add (priv->cmd_array,
"write", "FILENAME",
"Update the firmware",
lu_tool_write);
lu_tool_add (priv->cmd_array,
"dump", "FILENAME",
"Dump the firmware",
lu_tool_dump);
lu_tool_add (priv->cmd_array,
"attach", NULL,
"Attach to firmware mode",
lu_tool_attach);
lu_tool_add (priv->cmd_array,
"watch", NULL,
"Watch for hardare changes",
lu_tool_watch);
lu_tool_add (priv->cmd_array,
"detach", NULL,
"Detach to bootloader mode",
lu_tool_detach);
/* sort by command name */
g_ptr_array_sort (priv->cmd_array,
(GCompareFunc) lu_tool_sort_command_name_cb);
/* get a list of the commands */
context = g_option_context_new (NULL);
cmd_descriptions = lu_tool_get_descriptions (priv->cmd_array);
g_option_context_set_summary (context, cmd_descriptions);
g_set_application_name ("Logitech Lu 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;
}
/* emulate */
priv->emulation_kind = lu_device_kind_from_string (emulation_kind);
if (priv->emulation_kind != LU_DEVICE_KIND_UNKNOWN)
g_log_set_default_handler (lu_tool_log_handler_cb, priv);
/* get the device */
priv->ctx = lu_context_new (&error);
if (priv->ctx == NULL) {
g_print ("Failed to open USB devices: %s\n", error->message);
return EXIT_FAILURE;
}
/* set verbose? */
if (verbose)
g_setenv ("G_MESSAGES_DEBUG", "all", FALSE);
/* run the specified command */
if (!lu_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;
}