aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichal Sojka <sojka@merica.cz>2014-09-24 16:43:21 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2014-09-25 11:05:12 -0400
commit0cfbd328d60f85b0dcf66df61a3615e9a8e5d4e4 (patch)
treebae04eca569eecee711f27fd2e1d08e6f837ebbe
parentaa923ef1aa39473b7d1f413c73a3e1d19ebde65d (diff)
usb: Add LED triggers for USB activity
With this patch, USB activity can be signaled by blinking a LED. There are two triggers, one for activity on USB host and one for USB gadget. Both triggers should work with all host/device controllers. Tested only with musb. Performace: I measured performance overheads on ARM Cortex-A8 (TI AM335x) running on 600 MHz. Duration of usb_led_activity(): - with no LED attached to the trigger: 2 ± 1 µs - with one GPIO LED attached to the trigger: 2 ± 1 µs or 8 ± 2 µs (two peaks in histogram) Duration of functions calling usb_led_activity() (with this patch applied and no LED attached to the trigger): - __usb_hcd_giveback_urb(): 10 - 25 µs - usb_gadget_giveback_request(): 2 - 6 µs Signed-off-by: Michal Sojka <sojka@merica.cz> Acked-by: Felipe Balbi <balbi@ti.com> Tested-by: Felipe Balbi <balbi@ti.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--drivers/usb/Kconfig10
-rw-r--r--drivers/usb/common/Makefile1
-rw-r--r--drivers/usb/common/led.c57
-rw-r--r--drivers/usb/core/hcd.c2
-rw-r--r--drivers/usb/gadget/udc/udc-core.c4
-rw-r--r--include/linux/usb.h12
6 files changed, 86 insertions, 0 deletions
diff --git a/drivers/usb/Kconfig b/drivers/usb/Kconfig
index cf1b19bca306..ae481c37a208 100644
--- a/drivers/usb/Kconfig
+++ b/drivers/usb/Kconfig
@@ -149,4 +149,14 @@ source "drivers/usb/phy/Kconfig"
149 149
150source "drivers/usb/gadget/Kconfig" 150source "drivers/usb/gadget/Kconfig"
151 151
152config USB_LED_TRIG
153 bool "USB LED Triggers"
154 depends on LEDS_CLASS && USB_COMMON && LEDS_TRIGGERS
155 help
156 This option adds LED triggers for USB host and/or gadget activity.
157
158 Say Y here if you are working on a system with led-class supported
159 LEDs and you want to use them as activity indicators for USB host or
160 gadget.
161
152endif # USB_SUPPORT 162endif # USB_SUPPORT
diff --git a/drivers/usb/common/Makefile b/drivers/usb/common/Makefile
index 052c12069c24..ca2f8bd0e431 100644
--- a/drivers/usb/common/Makefile
+++ b/drivers/usb/common/Makefile
@@ -4,5 +4,6 @@
4 4
5obj-$(CONFIG_USB_COMMON) += usb-common.o 5obj-$(CONFIG_USB_COMMON) += usb-common.o
6usb-common-y += common.o 6usb-common-y += common.o
7usb-common-$(CONFIG_USB_LED_TRIG) += led.o
7 8
8obj-$(CONFIG_USB_OTG_FSM) += usb-otg-fsm.o 9obj-$(CONFIG_USB_OTG_FSM) += usb-otg-fsm.o
diff --git a/drivers/usb/common/led.c b/drivers/usb/common/led.c
new file mode 100644
index 000000000000..df23da00a901
--- /dev/null
+++ b/drivers/usb/common/led.c
@@ -0,0 +1,57 @@
1/*
2 * LED Triggers for USB Activity
3 *
4 * Copyright 2014 Michal Sojka <sojka@merica.cz>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 */
11
12#include <linux/module.h>
13#include <linux/kernel.h>
14#include <linux/init.h>
15#include <linux/leds.h>
16#include <linux/usb.h>
17
18#define BLINK_DELAY 30
19
20static unsigned long usb_blink_delay = BLINK_DELAY;
21
22DEFINE_LED_TRIGGER(ledtrig_usb_gadget);
23DEFINE_LED_TRIGGER(ledtrig_usb_host);
24
25void usb_led_activity(enum usb_led_event ev)
26{
27 struct led_trigger *trig = NULL;
28
29 switch (ev) {
30 case USB_LED_EVENT_GADGET:
31 trig = ledtrig_usb_gadget;
32 break;
33 case USB_LED_EVENT_HOST:
34 trig = ledtrig_usb_host;
35 break;
36 }
37 /* led_trigger_blink_oneshot() handles trig == NULL gracefully */
38 led_trigger_blink_oneshot(trig, &usb_blink_delay, &usb_blink_delay, 0);
39}
40EXPORT_SYMBOL_GPL(usb_led_activity);
41
42
43static int __init ledtrig_usb_init(void)
44{
45 led_trigger_register_simple("usb-gadget", &ledtrig_usb_gadget);
46 led_trigger_register_simple("usb-host", &ledtrig_usb_host);
47 return 0;
48}
49
50static void __exit ledtrig_usb_exit(void)
51{
52 led_trigger_unregister_simple(ledtrig_usb_gadget);
53 led_trigger_unregister_simple(ledtrig_usb_host);
54}
55
56module_init(ledtrig_usb_init);
57module_exit(ledtrig_usb_exit);
diff --git a/drivers/usb/core/hcd.c b/drivers/usb/core/hcd.c
index d3fe161bec05..bcb96ff207ba 100644
--- a/drivers/usb/core/hcd.c
+++ b/drivers/usb/core/hcd.c
@@ -1664,6 +1664,8 @@ static void __usb_hcd_giveback_urb(struct urb *urb)
1664 usbmon_urb_complete(&hcd->self, urb, status); 1664 usbmon_urb_complete(&hcd->self, urb, status);
1665 usb_anchor_suspend_wakeups(anchor); 1665 usb_anchor_suspend_wakeups(anchor);
1666 usb_unanchor_urb(urb); 1666 usb_unanchor_urb(urb);
1667 if (likely(status == 0))
1668 usb_led_activity(USB_LED_EVENT_HOST);
1667 1669
1668 /* pass ownership to the completion handler */ 1670 /* pass ownership to the completion handler */
1669 urb->status = status; 1671 urb->status = status;
diff --git a/drivers/usb/gadget/udc/udc-core.c b/drivers/usb/gadget/udc/udc-core.c
index 16d3f6fedd1c..f107bb60a5ab 100644
--- a/drivers/usb/gadget/udc/udc-core.c
+++ b/drivers/usb/gadget/udc/udc-core.c
@@ -27,6 +27,7 @@
27 27
28#include <linux/usb/ch9.h> 28#include <linux/usb/ch9.h>
29#include <linux/usb/gadget.h> 29#include <linux/usb/gadget.h>
30#include <linux/usb.h>
30 31
31/** 32/**
32 * struct usb_udc - describes one usb device controller 33 * struct usb_udc - describes one usb device controller
@@ -116,6 +117,9 @@ EXPORT_SYMBOL_GPL(usb_gadget_unmap_request);
116void usb_gadget_giveback_request(struct usb_ep *ep, 117void usb_gadget_giveback_request(struct usb_ep *ep,
117 struct usb_request *req) 118 struct usb_request *req)
118{ 119{
120 if (likely(req->status == 0))
121 usb_led_activity(USB_LED_EVENT_GADGET);
122
119 req->complete(ep, req); 123 req->complete(ep, req);
120} 124}
121EXPORT_SYMBOL_GPL(usb_gadget_giveback_request); 125EXPORT_SYMBOL_GPL(usb_gadget_giveback_request);
diff --git a/include/linux/usb.h b/include/linux/usb.h
index d2465bc0e73c..447a7e2fc19b 100644
--- a/include/linux/usb.h
+++ b/include/linux/usb.h
@@ -1862,6 +1862,18 @@ extern void usb_unregister_notify(struct notifier_block *nb);
1862/* debugfs stuff */ 1862/* debugfs stuff */
1863extern struct dentry *usb_debug_root; 1863extern struct dentry *usb_debug_root;
1864 1864
1865/* LED triggers */
1866enum usb_led_event {
1867 USB_LED_EVENT_HOST = 0,
1868 USB_LED_EVENT_GADGET = 1,
1869};
1870
1871#ifdef CONFIG_USB_LED_TRIG
1872extern void usb_led_activity(enum usb_led_event ev);
1873#else
1874static inline void usb_led_activity(enum usb_led_event ev) {}
1875#endif
1876
1865#endif /* __KERNEL__ */ 1877#endif /* __KERNEL__ */
1866 1878
1867#endif 1879#endif