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(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(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(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
;
104 ehci
= hcd_to_ehci(dev_get_drvdata(dev
));
105 if (kstrtouint(buf
, 0, &uframe_periodic_max
) < 0)
108 if (uframe_periodic_max
< 100 || uframe_periodic_max
>= 125) {
109 ehci_info(ehci
, "rejecting invalid request for "
110 "uframe_periodic_max=%u\n", uframe_periodic_max
);
117 * lock, so that our checking does not race with possible periodic
118 * bandwidth allocation through submitting new urbs.
120 spin_lock_irqsave (&ehci
->lock
, flags
);
123 * for request to decrease max periodic bandwidth, we have to check
124 * to see whether the decrease is possible.
126 if (uframe_periodic_max
< ehci
->uframe_periodic_max
) {
127 u8 allocated_max
= 0;
129 for (uframe
= 0; uframe
< EHCI_BANDWIDTH_SIZE
; ++uframe
)
130 allocated_max
= max(allocated_max
,
131 ehci
->bandwidth
[uframe
]);
133 if (allocated_max
> uframe_periodic_max
) {
135 "cannot decrease uframe_periodic_max becase "
136 "periodic bandwidth is already allocated "
138 allocated_max
, uframe_periodic_max
);
143 /* increasing is always ok */
145 ehci_info(ehci
, "setting max periodic bandwidth to %u%% "
146 "(== %u usec/uframe)\n",
147 100*uframe_periodic_max
/125, uframe_periodic_max
);
149 if (uframe_periodic_max
!= 100)
150 ehci_warn(ehci
, "max periodic bandwidth set is non-standard\n");
152 ehci
->uframe_periodic_max
= uframe_periodic_max
;
156 spin_unlock_irqrestore (&ehci
->lock
, flags
);
159 static DEVICE_ATTR(uframe_periodic_max
, 0644, show_uframe_periodic_max
, store_uframe_periodic_max
);
162 static inline int create_sysfs_files(struct ehci_hcd
*ehci
)
164 struct device
*controller
= ehci_to_hcd(ehci
)->self
.controller
;
167 /* with integrated TT there is no companion! */
168 if (!ehci_is_TDI(ehci
))
169 i
= device_create_file(controller
, &dev_attr_companion
);
173 i
= device_create_file(controller
, &dev_attr_uframe_periodic_max
);
178 static inline void remove_sysfs_files(struct ehci_hcd
*ehci
)
180 struct device
*controller
= ehci_to_hcd(ehci
)->self
.controller
;
182 /* with integrated TT there is no companion! */
183 if (!ehci_is_TDI(ehci
))
184 device_remove_file(controller
, &dev_attr_companion
);
186 device_remove_file(controller
, &dev_attr_uframe_periodic_max
);