/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- * * Copyright (C) 2016 Richard Hughes * * 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 #include "fu-ebitdo-device.h" #include "fu-plugin.h" #include "fu-plugin-vfuncs.h" static gboolean fu_plugin_ebitdo_device_added (FuPlugin *plugin, GUsbDevice *usb_device, GError **error) { FuEbitdoDeviceKind ebitdo_kind; const gchar *platform_id = NULL; g_autofree gchar *runtime_id = NULL; g_autoptr(AsProfile) profile = as_profile_new (); g_autoptr(AsProfileTask) ptask = NULL; g_autoptr(FuDeviceLocker) locker = NULL; g_autoptr(FuEbitdoDevice) dev = NULL; /* ignore hubs */ ptask = as_profile_start (profile, "FuPluginEbitdo:added{%04x:%04x}", g_usb_device_get_vid (usb_device), g_usb_device_get_pid (usb_device)); g_assert (ptask != NULL); /* create the device */ dev = fu_ebitdo_device_new (usb_device); if (dev == NULL) { g_set_error_literal (error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED, "invalid 8Bitdo device type detected"); return FALSE; } /* open the device */ locker = fu_device_locker_new_full (dev, (FuDeviceLockerFunc) fu_ebitdo_device_open, (FuDeviceLockerFunc) fu_ebitdo_device_close, error); if (locker == NULL) return FALSE; ebitdo_kind = fu_ebitdo_device_get_kind (dev); /* only the bootloader can do the update */ platform_id = g_usb_device_get_platform_id (usb_device); runtime_id = g_strdup_printf ("%s-runtime", platform_id); if (ebitdo_kind == FU_EBITDO_DEVICE_KIND_BOOTLOADER) { FuEbitdoDevice *dev_runtime; /* add the last seen runtime GUID too */ dev_runtime = fu_plugin_cache_lookup (plugin, runtime_id); if (dev_runtime != NULL) { const gchar *guid = fu_device_get_guid_default (FU_DEVICE (dev_runtime)); g_debug ("adding runtime GUID of %s", guid); fu_device_add_guid (FU_DEVICE (dev), guid); } } else { fu_plugin_cache_add (plugin, runtime_id, dev); } /* insert to hash */ fu_plugin_device_add (plugin, FU_DEVICE (dev)); fu_plugin_cache_add (plugin, platform_id, dev); return TRUE; } static void ebitdo_write_progress_cb (goffset current, goffset total, gpointer user_data) { FuPlugin *plugin = FU_PLUGIN (user_data); gdouble percentage = -1.f; if (total > 0) percentage = (100.f * (gdouble) current) / (gdouble) total; g_debug ("written %" G_GOFFSET_FORMAT "/%" G_GOFFSET_FORMAT " bytes [%.1f%%]", current, total, percentage); fu_plugin_set_percentage (plugin, (guint) percentage); } gboolean fu_plugin_update (FuPlugin *plugin, FuDevice *dev, GBytes *blob_fw, FwupdInstallFlags flags, GError **error) { GUsbContext *usb_ctx = fu_plugin_get_usb_context (plugin); FuEbitdoDevice *ebitdo_dev = FU_EBITDO_DEVICE (dev); g_autoptr(FuDeviceLocker) locker = NULL; g_autoptr(GUsbDevice) usb_device = NULL; g_autoptr(GUsbDevice) usb_device2 = NULL; /* get version */ usb_device = g_usb_context_find_by_platform_id (usb_ctx, fu_device_get_platform_id (dev), error); if (usb_device == NULL) return FALSE; if (fu_ebitdo_device_get_kind (ebitdo_dev) != FU_EBITDO_DEVICE_KIND_BOOTLOADER) { g_set_error_literal (error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED, "invalid 8Bitdo device type detected"); return FALSE; } /* write the firmware */ locker = fu_device_locker_new_full (ebitdo_dev, (FuDeviceLockerFunc) fu_ebitdo_device_open, (FuDeviceLockerFunc) fu_ebitdo_device_close, error); if (locker == NULL) return FALSE; fu_plugin_set_status (plugin, FWUPD_STATUS_DEVICE_WRITE); if (!fu_ebitdo_device_write_firmware (ebitdo_dev, blob_fw, ebitdo_write_progress_cb, plugin, error)) return FALSE; /* when doing a soft-reboot the device does not re-enumerate properly * so manually reboot the GUsbDevice */ fu_plugin_set_status (plugin, FWUPD_STATUS_DEVICE_RESTART); if (!g_usb_device_reset (usb_device, error)) { g_prefix_error (error, "failed to force-reset device: "); return FALSE; } g_clear_object (&locker); usb_device2 = g_usb_context_wait_for_replug (fu_plugin_get_usb_context (plugin), usb_device, 10000, error); if (usb_device2 == NULL) { g_prefix_error (error, "device did not come back: "); return FALSE; } if (!fu_ebitdo_device_set_usb_device (ebitdo_dev, usb_device2, error)) { g_prefix_error (error, "wrong device came back: "); return FALSE; } /* get the new version number */ locker = fu_device_locker_new_full (ebitdo_dev, (FuDeviceLockerFunc) fu_ebitdo_device_open, (FuDeviceLockerFunc) fu_ebitdo_device_close, error); if (locker == NULL) { g_prefix_error (error, "failed to re-open device: "); return FALSE; } /* success */ return TRUE; } static void fu_plugin_ebitdo_device_added_cb (GUsbContext *ctx, GUsbDevice *usb_device, FuPlugin *plugin) { g_autoptr(GError) error = NULL; if (!fu_plugin_ebitdo_device_added (plugin, usb_device, &error)) { if (!g_error_matches (error, FWUPD_ERROR, FWUPD_ERROR_NOT_SUPPORTED)) { g_warning ("Failed to add 8Bitdo device: %s", error->message); } } } static void fu_plugin_ebitdo_device_removed_cb (GUsbContext *ctx, GUsbDevice *usb_device, FuPlugin *plugin) { FuDevice *dev; const gchar *platform_id = NULL; /* already in database */ platform_id = g_usb_device_get_platform_id (usb_device); dev = fu_plugin_cache_lookup (plugin, platform_id); if (dev == NULL) return; fu_plugin_device_remove (plugin, dev); fu_plugin_cache_remove (plugin, platform_id); } gboolean fu_plugin_startup (FuPlugin *plugin, GError **error) { GUsbContext *usb_ctx = fu_plugin_get_usb_context (plugin); g_signal_connect (usb_ctx, "device-added", G_CALLBACK (fu_plugin_ebitdo_device_added_cb), plugin); g_signal_connect (usb_ctx, "device-removed", G_CALLBACK (fu_plugin_ebitdo_device_removed_cb), plugin); return TRUE; }