// SPDX-License-Identifier: GPL-2.0 /* * offload.c - USB offload related functions * * Copyright (c) 2025, Google LLC. * * Author: Guan-Yu Lin */ #include #include "usb.h" /** * usb_offload_get - increment the offload_usage of a USB device * @udev: the USB device to increment its offload_usage * * Incrementing the offload_usage of a usb_device indicates that offload is * enabled on this usb_device; that is, another entity is actively handling USB * transfers. This information allows the USB driver to adjust its power * management policy based on offload activity. * * Return: 0 on success. A negative error code otherwise. */ int usb_offload_get(struct usb_device *udev) { int ret = 0; if (!usb_get_dev(udev)) return -ENODEV; if (pm_runtime_get_if_active(&udev->dev) != 1) { ret = -EBUSY; goto err_rpm; } spin_lock(&udev->offload_lock); if (udev->offload_pm_locked) { ret = -EAGAIN; goto err; } udev->offload_usage++; err: spin_unlock(&udev->offload_lock); pm_runtime_put_autosuspend(&udev->dev); err_rpm: usb_put_dev(udev); return ret; } EXPORT_SYMBOL_GPL(usb_offload_get); /** * usb_offload_put - drop the offload_usage of a USB device * @udev: the USB device to drop its offload_usage * * The inverse operation of usb_offload_get, which drops the offload_usage of * a USB device. This information allows the USB driver to adjust its power * management policy based on offload activity. * * Return: 0 on success. A negative error code otherwise. */ int usb_offload_put(struct usb_device *udev) { int ret = 0; if (!usb_get_dev(udev)) return -ENODEV; if (pm_runtime_get_if_active(&udev->dev) != 1) { ret = -EBUSY; goto err_rpm; } spin_lock(&udev->offload_lock); if (udev->offload_pm_locked) { ret = -EAGAIN; goto err; } /* Drop the count when it wasn't 0, ignore the operation otherwise. */ if (udev->offload_usage) udev->offload_usage--; err: spin_unlock(&udev->offload_lock); pm_runtime_put_autosuspend(&udev->dev); err_rpm: usb_put_dev(udev); return ret; } EXPORT_SYMBOL_GPL(usb_offload_put); /** * usb_offload_check - check offload activities on a USB device * @udev: the USB device to check its offload activity. * * Check if there are any offload activity on the USB device right now. This * information could be used for power management or other forms of resource * management. * * The caller must hold @udev's device lock. In addition, the caller should * ensure the device itself and the downstream usb devices are all marked as * "offload_pm_locked" to ensure the correctness of the return value. * * Returns true on any offload activity, false otherwise. */ bool usb_offload_check(struct usb_device *udev) __must_hold(&udev->dev->mutex) { struct usb_device *child; bool active = false; int port1; if (udev->offload_usage) return true; usb_hub_for_each_child(udev, port1, child) { usb_lock_device(child); active = usb_offload_check(child); usb_unlock_device(child); if (active) break; } return active; } EXPORT_SYMBOL_GPL(usb_offload_check); /** * usb_offload_set_pm_locked - set the PM lock state of a USB device * @udev: the USB device to modify * @locked: the new lock state * * Setting @locked to true prevents offload_usage from being modified. This * ensures that offload activities cannot be started or stopped during critical * power management transitions, maintaining a stable state for the duration * of the transition. */ void usb_offload_set_pm_locked(struct usb_device *udev, bool locked) { spin_lock(&udev->offload_lock); udev->offload_pm_locked = locked; spin_unlock(&udev->offload_lock); } EXPORT_SYMBOL_GPL(usb_offload_set_pm_locked);