diff options
author | Michal Sojka <sojka@merica.cz> | 2014-09-24 16:43:21 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2014-09-25 11:05:12 -0400 |
commit | 0cfbd328d60f85b0dcf66df61a3615e9a8e5d4e4 (patch) | |
tree | bae04eca569eecee711f27fd2e1d08e6f837ebbe | |
parent | aa923ef1aa39473b7d1f413c73a3e1d19ebde65d (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/Kconfig | 10 | ||||
-rw-r--r-- | drivers/usb/common/Makefile | 1 | ||||
-rw-r--r-- | drivers/usb/common/led.c | 57 | ||||
-rw-r--r-- | drivers/usb/core/hcd.c | 2 | ||||
-rw-r--r-- | drivers/usb/gadget/udc/udc-core.c | 4 | ||||
-rw-r--r-- | include/linux/usb.h | 12 |
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 | ||
150 | source "drivers/usb/gadget/Kconfig" | 150 | source "drivers/usb/gadget/Kconfig" |
151 | 151 | ||
152 | config 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 | |||
152 | endif # USB_SUPPORT | 162 | endif # 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 | ||
5 | obj-$(CONFIG_USB_COMMON) += usb-common.o | 5 | obj-$(CONFIG_USB_COMMON) += usb-common.o |
6 | usb-common-y += common.o | 6 | usb-common-y += common.o |
7 | usb-common-$(CONFIG_USB_LED_TRIG) += led.o | ||
7 | 8 | ||
8 | obj-$(CONFIG_USB_OTG_FSM) += usb-otg-fsm.o | 9 | obj-$(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 | |||
20 | static unsigned long usb_blink_delay = BLINK_DELAY; | ||
21 | |||
22 | DEFINE_LED_TRIGGER(ledtrig_usb_gadget); | ||
23 | DEFINE_LED_TRIGGER(ledtrig_usb_host); | ||
24 | |||
25 | void 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 | } | ||
40 | EXPORT_SYMBOL_GPL(usb_led_activity); | ||
41 | |||
42 | |||
43 | static 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 | |||
50 | static 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 | |||
56 | module_init(ledtrig_usb_init); | ||
57 | module_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); | |||
116 | void usb_gadget_giveback_request(struct usb_ep *ep, | 117 | void 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 | } |
121 | EXPORT_SYMBOL_GPL(usb_gadget_giveback_request); | 125 | EXPORT_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 */ |
1863 | extern struct dentry *usb_debug_root; | 1863 | extern struct dentry *usb_debug_root; |
1864 | 1864 | ||
1865 | /* LED triggers */ | ||
1866 | enum usb_led_event { | ||
1867 | USB_LED_EVENT_HOST = 0, | ||
1868 | USB_LED_EVENT_GADGET = 1, | ||
1869 | }; | ||
1870 | |||
1871 | #ifdef CONFIG_USB_LED_TRIG | ||
1872 | extern void usb_led_activity(enum usb_led_event ev); | ||
1873 | #else | ||
1874 | static 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 |