2 * Copyright (C) 2007 by Alan Stern
4 * This program is free software; you can redistribute it and/or modify it
5 * under the terms of the GNU General Public License as published by the
6 * Free Software Foundation; either version 2 of the License, or (at your
7 * option) any later version.
9 * This program is distributed in the hope that it will be useful, but
10 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
11 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software Foundation,
16 * Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19 /* this file is part of ehci-hcd.c */
22 /* Display the ports dedicated to the companion controller */
23 static ssize_t
show_companion(struct device
*dev
,
24 struct device_attribute
*attr
,
27 struct ehci_hcd
*ehci
;
29 int count
= PAGE_SIZE
;
32 ehci
= hcd_to_ehci(bus_to_hcd(dev_get_drvdata(dev
)));
33 nports
= HCS_N_PORTS(ehci
->hcs_params
);
35 for (index
= 0; index
< nports
; ++index
) {
36 if (test_bit(index
, &ehci
->companion_ports
)) {
37 n
= scnprintf(ptr
, count
, "%d\n", index
+ 1);
46 * Dedicate or undedicate a port to the companion controller.
47 * Syntax is "[-]portnum", where a leading '-' sign means
48 * return control of the port to the EHCI controller.
50 static ssize_t
store_companion(struct device
*dev
,
51 struct device_attribute
*attr
,
52 const char *buf
, size_t count
)
54 struct ehci_hcd
*ehci
;
55 int portnum
, new_owner
;
57 ehci
= hcd_to_ehci(bus_to_hcd(dev_get_drvdata(dev
)));
58 new_owner
= PORT_OWNER
; /* Owned by companion */
59 if (sscanf(buf
, "%d", &portnum
) != 1)
63 new_owner
= 0; /* Owned by EHCI */
65 if (portnum
<= 0 || portnum
> HCS_N_PORTS(ehci
->hcs_params
))
69 set_bit(portnum
, &ehci
->companion_ports
);
71 clear_bit(portnum
, &ehci
->companion_ports
);
72 set_owner(ehci
, portnum
, new_owner
);
75 static DEVICE_ATTR(companion
, 0644, show_companion
, store_companion
);
79 * Display / Set uframe_periodic_max
81 static ssize_t
show_uframe_periodic_max(struct device
*dev
,
82 struct device_attribute
*attr
,
85 struct ehci_hcd
*ehci
;
88 ehci
= hcd_to_ehci(bus_to_hcd(dev_get_drvdata(dev
)));
89 n
= scnprintf(buf
, PAGE_SIZE
, "%d\n", ehci
->uframe_periodic_max
);
94 static ssize_t
store_uframe_periodic_max(struct device
*dev
,
95 struct device_attribute
*attr
,
96 const char *buf
, size_t count
)
98 struct ehci_hcd
*ehci
;
99 unsigned uframe_periodic_max
;
100 unsigned frame
, uframe
;
101 unsigned short allocated_max
;
105 ehci
= hcd_to_ehci(bus_to_hcd(dev_get_drvdata(dev
)));
106 if (kstrtouint(buf
, 0, &uframe_periodic_max
) < 0)
109 if (uframe_periodic_max
< 100 || uframe_periodic_max
>= 125) {
110 ehci_info(ehci
, "rejecting invalid request for "
111 "uframe_periodic_max=%u\n", uframe_periodic_max
);
118 * lock, so that our checking does not race with possible periodic
119 * bandwidth allocation through submitting new urbs.
121 spin_lock_irqsave (&ehci
->lock
, flags
);
124 * for request to decrease max periodic bandwidth, we have to check
125 * every microframe in the schedule to see whether the decrease is
128 if (uframe_periodic_max
< ehci
->uframe_periodic_max
) {
131 for (frame
= 0; frame
< ehci
->periodic_size
; ++frame
)
132 for (uframe
= 0; uframe
< 7; ++uframe
)
133 allocated_max
= max(allocated_max
,
134 periodic_usecs (ehci
, frame
, uframe
));
136 if (allocated_max
> uframe_periodic_max
) {
138 "cannot decrease uframe_periodic_max becase "
139 "periodic bandwidth is already allocated "
141 allocated_max
, uframe_periodic_max
);
146 /* increasing is always ok */
148 ehci_info(ehci
, "setting max periodic bandwidth to %u%% "
149 "(== %u usec/uframe)\n",
150 100*uframe_periodic_max
/125, uframe_periodic_max
);
152 if (uframe_periodic_max
!= 100)
153 ehci_warn(ehci
, "max periodic bandwidth set is non-standard\n");
155 ehci
->uframe_periodic_max
= uframe_periodic_max
;
159 spin_unlock_irqrestore (&ehci
->lock
, flags
);
162 static DEVICE_ATTR(uframe_periodic_max
, 0644, show_uframe_periodic_max
, store_uframe_periodic_max
);
165 static inline int create_sysfs_files(struct ehci_hcd
*ehci
)
167 struct device
*controller
= ehci_to_hcd(ehci
)->self
.controller
;
170 /* with integrated TT there is no companion! */
171 if (!ehci_is_TDI(ehci
))
172 i
= device_create_file(controller
, &dev_attr_companion
);
176 i
= device_create_file(controller
, &dev_attr_uframe_periodic_max
);
181 static inline void remove_sysfs_files(struct ehci_hcd
*ehci
)
183 struct device
*controller
= ehci_to_hcd(ehci
)->self
.controller
;
185 /* with integrated TT there is no companion! */
186 if (!ehci_is_TDI(ehci
))
187 device_remove_file(controller
, &dev_attr_companion
);
189 device_remove_file(controller
, &dev_attr_uframe_periodic_max
);