Commit b6cf160c authored by Linus Torvalds's avatar Linus Torvalds

Merge git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/usb-2.6

* git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/usb-2.6: (128 commits)
  USB: fix codingstyle issues in drivers/usb/core/*.c
  USB: fix codingstyle issues in drivers/usb/core/message.c
  USB: fix codingstyle issues in drivers/usb/core/hcd-pci.c
  USB: fix codingstyle issues in drivers/usb/core/devio.c
  USB: fix codingstyle issues in drivers/usb/core/devices.c
  USB: fix codingstyle issues in drivers/usb/core/*.h
  USB: fix codingstyle issues in include/linux/usb/
  USB: fix codingstyle issues in include/linux/usb.h
  USB: mark USB drivers as being GPL only
  USB: use a real vendor and product id for root hubs
  USB: mount options: fix usbfs
  USB: Fix usb_serial_driver structure for Kobil cardreader driver.
  usb: ehci should use u16 for isochronous intervals
  usb: ehci, remove false clear-reset path
  USB: Use menuconfig objects
  usb: ohci-sm501 driver
  usb: dma bounce buffer support
  USB: last abuses of intfdata in close for usb-serial drivers
  USB: kl5kusb105 don't flush to logically disconnected devices
  USB: oti6858: cleanup
  ...
parents ed50d6cb 2c044a48
...@@ -52,3 +52,36 @@ Description: ...@@ -52,3 +52,36 @@ Description:
facility is inherently dangerous, it is disabled by default facility is inherently dangerous, it is disabled by default
for all devices except hubs. For more information, see for all devices except hubs. For more information, see
Documentation/usb/persist.txt. Documentation/usb/persist.txt.
What: /sys/bus/usb/device/.../power/connected_duration
Date: January 2008
KernelVersion: 2.6.25
Contact: Sarah Sharp <sarah.a.sharp@intel.com>
Description:
If CONFIG_PM and CONFIG_USB_SUSPEND are enabled, then this file
is present. When read, it returns the total time (in msec)
that the USB device has been connected to the machine. This
file is read-only.
Users:
PowerTOP <power@bughost.org>
http://www.lesswatts.org/projects/powertop/
What: /sys/bus/usb/device/.../power/active_duration
Date: January 2008
KernelVersion: 2.6.25
Contact: Sarah Sharp <sarah.a.sharp@intel.com>
Description:
If CONFIG_PM and CONFIG_USB_SUSPEND are enabled, then this file
is present. When read, it returns the total time (in msec)
that the USB device has been active, i.e. not in a suspended
state. This file is read-only.
Tools can use this file and the connected_duration file to
compute the percentage of time that a device has been active.
For example,
echo $((100 * `cat active_duration` / `cat connected_duration`))
will give an integer percentage. Note that this does not
account for counter wrap.
Users:
PowerTOP <power@bughost.org>
http://www.lesswatts.org/projects/powertop/
...@@ -156,22 +156,6 @@ Who: Arjan van de Ven <arjan@linux.intel.com> ...@@ -156,22 +156,6 @@ Who: Arjan van de Ven <arjan@linux.intel.com>
--------------------------- ---------------------------
What: USB driver API moves to EXPORT_SYMBOL_GPL
When: February 2008
Files: include/linux/usb.h, drivers/usb/core/driver.c
Why: The USB subsystem has changed a lot over time, and it has been
possible to create userspace USB drivers using usbfs/libusb/gadgetfs
that operate as fast as the USB bus allows. Because of this, the USB
subsystem will not be allowing closed source kernel drivers to
register with it, after this grace period is over. If anyone needs
any help in converting their closed source drivers over to use the
userspace filesystems, please contact the
linux-usb-devel@lists.sourceforge.net mailing list, and the developers
there will be glad to help you out.
Who: Greg Kroah-Hartman <gregkh@suse.de>
---------------------------
What: vm_ops.nopage What: vm_ops.nopage
When: Soon, provided in-kernel callers have been converted When: Soon, provided in-kernel callers have been converted
Why: This interface is replaced by vm_ops.fault, but it has been around Why: This interface is replaced by vm_ops.fault, but it has been around
......
This diff is collapsed.
Infinity Usb Unlimited Readme
-----------------------------
Hi all,
This module provide a serial interface to use your
IUU unit in phoenix mode. Loading this module will
bring a ttyUSB[0-x] interface. This driver must be
used by your favorite application to pilot the IUU
This driver is still in beta stage, so bugs can
occur and your system may freeze. As far I now,
I never had any problem with it, but I'm not a real
guru, so don't blame me if your system is unstable
You can plug more than one IUU. Every unit will
have his own device file(/dev/ttyUSB0,/dev/ttyUSB1,...)
How to tune the reader speed ?
A few parameters can be used at load time
To use parameters, just unload the module if it is
already loaded and use modprobe iuu_phoenix param=value.
In case of prebuilt module, use the command
insmod iuu_phoenix param=value.
Example:
modprobe iuu_phoenix clockmode=3
The parameters are:
parm: clockmode:1=3Mhz579,2=3Mhz680,3=6Mhz (int)
parm: boost:overclock boost percent 100 to 500 (int)
parm: cdmode:Card detect mode 0=none, 1=CD, 2=!CD, 3=DSR, 4=!DSR, 5=CTS, 6=!CTS, 7=RING, 8=!RING (int)
parm: xmas:xmas color enabled or not (bool)
parm: debug:Debug enabled or not (bool)
- clockmode will provide 3 different base settings commonly adopted by
different software:
1. 3Mhz579
2. 3Mhz680
3. 6Mhz
- boost provide a way to overclock the reader ( my favorite :-) )
For example to have best performance than a simple clockmode=3, try this:
modprobe boost=195
This will put the reader in a base of 3Mhz579 but boosted a 195 % !
the real clock will be now : 6979050 Hz ( 6Mhz979 ) and will increase
the speed to a score 10 to 20% better than the simple clockmode=3 !!!
- cdmode permit to setup the signal used to inform the userland ( ioctl answer )
if the card is present or not. Eight signals are possible.
- xmas is completely useless except for your eyes. This is one of my friend who was
so sad to have a nice device like the iuu without seeing all color range available.
So I have added this option to permit him to see a lot of color ( each activity change the color
and the frequency randomly )
- debug will produce a lot of debugging messages...
Last notes:
Don't worry about the serial settings, the serial emulation
is an abstraction, so use any speed or parity setting will
work. ( This will not change anything ).Later I will perhaps
use this settings to deduce de boost but is that feature
really necessary ?
The autodetect feature used is the serial CD. If that doesn't
work for your software, disable detection mechanism in it.
Have fun !
Alain Degreffe
eczema(at)ecze.com
...@@ -157,15 +157,10 @@ static void tosa_udc_command(int cmd) ...@@ -157,15 +157,10 @@ static void tosa_udc_command(int cmd)
} }
} }
static int tosa_udc_is_connected(void)
{
return ((GPLR(TOSA_GPIO_USB_IN) & GPIO_bit(TOSA_GPIO_USB_IN)) == 0);
}
static struct pxa2xx_udc_mach_info udc_info __initdata = { static struct pxa2xx_udc_mach_info udc_info __initdata = {
.udc_command = tosa_udc_command, .udc_command = tosa_udc_command,
.udc_is_connected = tosa_udc_is_connected, .gpio_vbus = TOSA_GPIO_USB_IN,
.gpio_vbus_inverted = 1,
}; };
/* /*
......
...@@ -922,11 +922,6 @@ static int ub_scsi_cmd_start(struct ub_dev *sc, struct ub_scsi_cmd *cmd) ...@@ -922,11 +922,6 @@ static int ub_scsi_cmd_start(struct ub_dev *sc, struct ub_scsi_cmd *cmd)
usb_fill_bulk_urb(&sc->work_urb, sc->dev, sc->send_bulk_pipe, usb_fill_bulk_urb(&sc->work_urb, sc->dev, sc->send_bulk_pipe,
bcb, US_BULK_CB_WRAP_LEN, ub_urb_complete, sc); bcb, US_BULK_CB_WRAP_LEN, ub_urb_complete, sc);
/* Fill what we shouldn't be filling, because usb-storage did so. */
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) { if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) {
/* XXX Clear stalls */ /* XXX Clear stalls */
ub_complete(&sc->work_done); ub_complete(&sc->work_done);
...@@ -1313,9 +1308,6 @@ static void ub_data_start(struct ub_dev *sc, struct ub_scsi_cmd *cmd) ...@@ -1313,9 +1308,6 @@ static void ub_data_start(struct ub_dev *sc, struct ub_scsi_cmd *cmd)
sc->last_pipe = pipe; sc->last_pipe = pipe;
usb_fill_bulk_urb(&sc->work_urb, sc->dev, pipe, sg_virt(sg), usb_fill_bulk_urb(&sc->work_urb, sc->dev, pipe, sg_virt(sg),
sg->length, ub_urb_complete, sc); sg->length, ub_urb_complete, sc);
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) { if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) {
/* XXX Clear stalls */ /* XXX Clear stalls */
...@@ -1356,9 +1348,6 @@ static int __ub_state_stat(struct ub_dev *sc, struct ub_scsi_cmd *cmd) ...@@ -1356,9 +1348,6 @@ static int __ub_state_stat(struct ub_dev *sc, struct ub_scsi_cmd *cmd)
sc->last_pipe = sc->recv_bulk_pipe; sc->last_pipe = sc->recv_bulk_pipe;
usb_fill_bulk_urb(&sc->work_urb, sc->dev, sc->recv_bulk_pipe, usb_fill_bulk_urb(&sc->work_urb, sc->dev, sc->recv_bulk_pipe,
&sc->work_bcs, US_BULK_CS_WRAP_LEN, ub_urb_complete, sc); &sc->work_bcs, US_BULK_CS_WRAP_LEN, ub_urb_complete, sc);
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) { if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) {
/* XXX Clear stalls */ /* XXX Clear stalls */
...@@ -1473,9 +1462,6 @@ static int ub_submit_clear_stall(struct ub_dev *sc, struct ub_scsi_cmd *cmd, ...@@ -1473,9 +1462,6 @@ static int ub_submit_clear_stall(struct ub_dev *sc, struct ub_scsi_cmd *cmd,
usb_fill_control_urb(&sc->work_urb, sc->dev, sc->send_ctrl_pipe, usb_fill_control_urb(&sc->work_urb, sc->dev, sc->send_ctrl_pipe,
(unsigned char*) cr, NULL, 0, ub_urb_complete, sc); (unsigned char*) cr, NULL, 0, ub_urb_complete, sc);
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) { if ((rc = usb_submit_urb(&sc->work_urb, GFP_ATOMIC)) != 0) {
ub_complete(&sc->work_done); ub_complete(&sc->work_done);
...@@ -1953,9 +1939,6 @@ static int ub_sync_reset(struct ub_dev *sc) ...@@ -1953,9 +1939,6 @@ static int ub_sync_reset(struct ub_dev *sc)
usb_fill_control_urb(&sc->work_urb, sc->dev, sc->send_ctrl_pipe, usb_fill_control_urb(&sc->work_urb, sc->dev, sc->send_ctrl_pipe,
(unsigned char*) cr, NULL, 0, ub_probe_urb_complete, &compl); (unsigned char*) cr, NULL, 0, ub_probe_urb_complete, &compl);
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_KERNEL)) != 0) { if ((rc = usb_submit_urb(&sc->work_urb, GFP_KERNEL)) != 0) {
printk(KERN_WARNING printk(KERN_WARNING
...@@ -2007,9 +1990,6 @@ static int ub_sync_getmaxlun(struct ub_dev *sc) ...@@ -2007,9 +1990,6 @@ static int ub_sync_getmaxlun(struct ub_dev *sc)
usb_fill_control_urb(&sc->work_urb, sc->dev, sc->recv_ctrl_pipe, usb_fill_control_urb(&sc->work_urb, sc->dev, sc->recv_ctrl_pipe,
(unsigned char*) cr, p, 1, ub_probe_urb_complete, &compl); (unsigned char*) cr, p, 1, ub_probe_urb_complete, &compl);
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_KERNEL)) != 0) if ((rc = usb_submit_urb(&sc->work_urb, GFP_KERNEL)) != 0)
goto err_submit; goto err_submit;
...@@ -2077,9 +2057,6 @@ static int ub_probe_clear_stall(struct ub_dev *sc, int stalled_pipe) ...@@ -2077,9 +2057,6 @@ static int ub_probe_clear_stall(struct ub_dev *sc, int stalled_pipe)
usb_fill_control_urb(&sc->work_urb, sc->dev, sc->send_ctrl_pipe, usb_fill_control_urb(&sc->work_urb, sc->dev, sc->send_ctrl_pipe,
(unsigned char*) cr, NULL, 0, ub_probe_urb_complete, &compl); (unsigned char*) cr, NULL, 0, ub_probe_urb_complete, &compl);
sc->work_urb.actual_length = 0;
sc->work_urb.error_count = 0;
sc->work_urb.status = 0;
if ((rc = usb_submit_urb(&sc->work_urb, GFP_KERNEL)) != 0) { if ((rc = usb_submit_urb(&sc->work_urb, GFP_KERNEL)) != 0) {
printk(KERN_WARNING printk(KERN_WARNING
......
...@@ -42,6 +42,10 @@ config USB_ARCH_HAS_OHCI ...@@ -42,6 +42,10 @@ config USB_ARCH_HAS_OHCI
default y if PPC_MPC52xx default y if PPC_MPC52xx
# MIPS: # MIPS:
default y if SOC_AU1X00 default y if SOC_AU1X00
# SH:
default y if CPU_SUBTYPE_SH7720
default y if CPU_SUBTYPE_SH7721
default y if CPU_SUBTYPE_SH7763
# more: # more:
default PCI default PCI
...@@ -50,6 +54,7 @@ config USB_ARCH_HAS_EHCI ...@@ -50,6 +54,7 @@ config USB_ARCH_HAS_EHCI
boolean boolean
default y if PPC_83xx default y if PPC_83xx
default y if SOC_AU1200 default y if SOC_AU1200
default y if ARCH_IXP4XX
default PCI default PCI
# ARM SA1111 chips have a non-PCI based "OHCI-compatible" USB host interface. # ARM SA1111 chips have a non-PCI based "OHCI-compatible" USB host interface.
......
...@@ -2,10 +2,7 @@ ...@@ -2,10 +2,7 @@
# USB/ATM DSL configuration # USB/ATM DSL configuration
# #
menu "USB DSL modem support" menuconfig USB_ATM
depends on USB
config USB_ATM
tristate "USB DSL modem support" tristate "USB DSL modem support"
depends on USB && ATM depends on USB && ATM
select CRC32 select CRC32
...@@ -18,6 +15,8 @@ config USB_ATM ...@@ -18,6 +15,8 @@ config USB_ATM
To compile this driver as a module, choose M here: the To compile this driver as a module, choose M here: the
module will be called usbatm. module will be called usbatm.
if USB_ATM
config USB_SPEEDTOUCH config USB_SPEEDTOUCH
tristate "Speedtouch USB support" tristate "Speedtouch USB support"
depends on USB_ATM depends on USB_ATM
...@@ -70,4 +69,4 @@ config USB_XUSBATM ...@@ -70,4 +69,4 @@ config USB_XUSBATM
To compile this driver as a module, choose M here: the To compile this driver as a module, choose M here: the
module will be called xusbatm. module will be called xusbatm.
endmenu endif # USB_ATM
...@@ -999,7 +999,7 @@ static void __uea_load_page_e4(struct uea_softc *sc, u8 pageno, int boot) ...@@ -999,7 +999,7 @@ static void __uea_load_page_e4(struct uea_softc *sc, u8 pageno, int boot)
bi.dwAddress = swab32(blockidx->PageAddress); bi.dwAddress = swab32(blockidx->PageAddress);
uea_dbg(INS_TO_USBDEV(sc), uea_dbg(INS_TO_USBDEV(sc),
"sending block %u for DSP page %u size %u adress %x\n", "sending block %u for DSP page %u size %u address %x\n",
blockno, pageno, blocksize, le32_to_cpu(blockidx->PageAddress)); blockno, pageno, blocksize, le32_to_cpu(blockidx->PageAddress));
/* send block info through the IDMA pipe */ /* send block info through the IDMA pipe */
...@@ -1990,7 +1990,7 @@ static void uea_dispatch_cmv_e1(struct uea_softc *sc, struct intr_pkt *intr) ...@@ -1990,7 +1990,7 @@ static void uea_dispatch_cmv_e1(struct uea_softc *sc, struct intr_pkt *intr)
return; return;
bad2: bad2:
uea_err(INS_TO_USBDEV(sc), "unexpected cmv received," uea_err(INS_TO_USBDEV(sc), "unexpected cmv received, "
"Function : %d, Subfunction : %d\n", "Function : %d, Subfunction : %d\n",
E1_FUNCTION_TYPE(cmv->bFunction), E1_FUNCTION_TYPE(cmv->bFunction),
E1_FUNCTION_SUBTYPE(cmv->bFunction)); E1_FUNCTION_SUBTYPE(cmv->bFunction));
...@@ -2038,7 +2038,7 @@ static void uea_dispatch_cmv_e4(struct uea_softc *sc, struct intr_pkt *intr) ...@@ -2038,7 +2038,7 @@ static void uea_dispatch_cmv_e4(struct uea_softc *sc, struct intr_pkt *intr)
return; return;
bad2: bad2:
uea_err(INS_TO_USBDEV(sc), "unexpected cmv received," uea_err(INS_TO_USBDEV(sc), "unexpected cmv received, "
"Function : %d, Subfunction : %d\n", "Function : %d, Subfunction : %d\n",
E4_FUNCTION_TYPE(cmv->wFunction), E4_FUNCTION_TYPE(cmv->wFunction),
E4_FUNCTION_SUBTYPE(cmv->wFunction)); E4_FUNCTION_SUBTYPE(cmv->wFunction));
......
...@@ -496,10 +496,19 @@ static int acm_tty_open(struct tty_struct *tty, struct file *filp) ...@@ -496,10 +496,19 @@ static int acm_tty_open(struct tty_struct *tty, struct file *filp)
otherwise it is scheduled, and with high data rates data can get lost. */ otherwise it is scheduled, and with high data rates data can get lost. */
tty->low_latency = 1; tty->low_latency = 1;
if (usb_autopm_get_interface(acm->control)) {
mutex_unlock(&open_mutex);
return -EIO;
}
mutex_lock(&acm->mutex);
mutex_unlock(&open_mutex);
if (acm->used++) { if (acm->used++) {
usb_autopm_put_interface(acm->control);
goto done; goto done;
} }
acm->ctrlurb->dev = acm->dev; acm->ctrlurb->dev = acm->dev;
if (usb_submit_urb(acm->ctrlurb, GFP_KERNEL)) { if (usb_submit_urb(acm->ctrlurb, GFP_KERNEL)) {
dbg("usb_submit_urb(ctrl irq) failed"); dbg("usb_submit_urb(ctrl irq) failed");
...@@ -526,14 +535,15 @@ static int acm_tty_open(struct tty_struct *tty, struct file *filp) ...@@ -526,14 +535,15 @@ static int acm_tty_open(struct tty_struct *tty, struct file *filp)
done: done:
err_out: err_out:
mutex_unlock(&open_mutex); mutex_unlock(&acm->mutex);
return rv; return rv;
full_bailout: full_bailout:
usb_kill_urb(acm->ctrlurb); usb_kill_urb(acm->ctrlurb);
bail_out: bail_out:
usb_autopm_put_interface(acm->control);
acm->used--; acm->used--;
mutex_unlock(&open_mutex); mutex_unlock(&acm->mutex);
return -EIO; return -EIO;
} }
...@@ -570,6 +580,7 @@ static void acm_tty_close(struct tty_struct *tty, struct file *filp) ...@@ -570,6 +580,7 @@ static void acm_tty_close(struct tty_struct *tty, struct file *filp)
usb_kill_urb(acm->writeurb); usb_kill_urb(acm->writeurb);
for (i = 0; i < nr; i++) for (i = 0; i < nr; i++)
usb_kill_urb(acm->ru[i].urb); usb_kill_urb(acm->ru[i].urb);
usb_autopm_put_interface(acm->control);
} else } else
acm_tty_unregister(acm); acm_tty_unregister(acm);
} }
...@@ -904,7 +915,7 @@ static int acm_probe (struct usb_interface *intf, ...@@ -904,7 +915,7 @@ static int acm_probe (struct usb_interface *intf,
} }
if (data_interface_num != call_interface_num) if (data_interface_num != call_interface_num)
dev_dbg(&intf->dev,"Seperate call control interface. That is not fully supported.\n"); dev_dbg(&intf->dev,"Separate call control interface. That is not fully supported.\n");
skip_normal_probe: skip_normal_probe:
...@@ -980,6 +991,7 @@ static int acm_probe (struct usb_interface *intf, ...@@ -980,6 +991,7 @@ static int acm_probe (struct usb_interface *intf,
spin_lock_init(&acm->throttle_lock); spin_lock_init(&acm->throttle_lock);
spin_lock_init(&acm->write_lock); spin_lock_init(&acm->write_lock);
spin_lock_init(&acm->read_lock); spin_lock_init(&acm->read_lock);
mutex_init(&acm->mutex);
acm->write_ready = 1; acm->write_ready = 1;
acm->rx_endpoint = usb_rcvbulkpipe(usb_dev, epread->bEndpointAddress); acm->rx_endpoint = usb_rcvbulkpipe(usb_dev, epread->bEndpointAddress);
...@@ -1096,6 +1108,25 @@ static int acm_probe (struct usb_interface *intf, ...@@ -1096,6 +1108,25 @@ static int acm_probe (struct usb_interface *intf,
return -ENOMEM; return -ENOMEM;
} }
static void stop_data_traffic(struct acm *acm)
{
int i;
tasklet_disable(&acm->urb_task);
usb_kill_urb(acm->ctrlurb);
usb_kill_urb(acm->writeurb);
for (i = 0; i < acm->rx_buflimit; i++)
usb_kill_urb(acm->ru[i].urb);
INIT_LIST_HEAD(&acm->filled_read_bufs);
INIT_LIST_HEAD(&acm->spare_read_bufs);
tasklet_enable(&acm->urb_task);
cancel_work_sync(&acm->work);
}
static void acm_disconnect(struct usb_interface *intf) static void acm_disconnect(struct usb_interface *intf)
{ {
struct acm *acm = usb_get_intfdata(intf); struct acm *acm = usb_get_intfdata(intf);
...@@ -1123,19 +1154,7 @@ static void acm_disconnect(struct usb_interface *intf) ...@@ -1123,19 +1154,7 @@ static void acm_disconnect(struct usb_interface *intf)
usb_set_intfdata(acm->control, NULL); usb_set_intfdata(acm->control, NULL);
usb_set_intfdata(acm->data, NULL); usb_set_intfdata(acm->data, NULL);
tasklet_disable(&acm->urb_task); stop_data_traffic(acm);
usb_kill_urb(acm->ctrlurb);
usb_kill_urb(acm->writeurb);
for (i = 0; i < acm->rx_buflimit; i++)
usb_kill_urb(acm->ru[i].urb);
INIT_LIST_HEAD(&acm->filled_read_bufs);
INIT_LIST_HEAD(&acm->spare_read_bufs);
tasklet_enable(&acm->urb_task);
flush_scheduled_work(); /* wait for acm_softint */
acm_write_buffers_free(acm); acm_write_buffers_free(acm);
usb_buffer_free(usb_dev, acm->ctrlsize, acm->ctrl_buffer, acm->ctrl_dma); usb_buffer_free(usb_dev, acm->ctrlsize, acm->ctrl_buffer, acm->ctrl_dma);
...@@ -1156,6 +1175,46 @@ static void acm_disconnect(struct usb_interface *intf) ...@@ -1156,6 +1175,46 @@ static void acm_disconnect(struct usb_interface *intf)
tty_hangup(acm->tty); tty_hangup(acm->tty);
} }
static int acm_suspend(struct usb_interface *intf, pm_message_t message)
{
struct acm *acm = usb_get_intfdata(intf);
if (acm->susp_count++)
return 0;
/*
we treat opened interfaces differently,
we must guard against open
*/
mutex_lock(&acm->mutex);
if (acm->used)
stop_data_traffic(acm);
mutex_unlock(&acm->mutex);
return 0;
}
static int acm_resume(struct usb_interface *intf)
{
struct acm *acm = usb_get_intfdata(intf);
int rv = 0;
if (--acm->susp_count)
return 0;
mutex_lock(&acm->mutex);
if (acm->used) {
rv = usb_submit_urb(acm->ctrlurb, GFP_NOIO);
if (rv < 0)
goto err_out;
tasklet_schedule(&acm->urb_task);
}
err_out:
mutex_unlock(&acm->mutex);
return rv;
}
/* /*
* USB driver structure. * USB driver structure.
*/ */
...@@ -1208,7 +1267,10 @@ static struct usb_driver acm_driver = { ...@@ -1208,7 +1267,10 @@ static struct usb_driver acm_driver = {
.name = "cdc_acm", .name = "cdc_acm",
.probe = acm_probe, .probe = acm_probe,
.disconnect = acm_disconnect, .disconnect = acm_disconnect,
.suspend = acm_suspend,
.resume = acm_resume,
.id_table = acm_ids, .id_table = acm_ids,
.supports_autosuspend = 1,
}; };
/* /*
......
...@@ -107,6 +107,7 @@ struct acm { ...@@ -107,6 +107,7 @@ struct acm {
int write_used; /* number of non-empty write buffers */ int write_used; /* number of non-empty write buffers */
int write_ready; /* write urb is not running */ int write_ready; /* write urb is not running */
spinlock_t write_lock; spinlock_t write_lock;
struct mutex mutex;
struct usb_cdc_line_coding line; /* bits, stop, parity */ struct usb_cdc_line_coding line; /* bits, stop, parity */
struct work_struct work; /* work queue entry for line discipline waking up */ struct work_struct work; /* work queue entry for line discipline waking up */
struct tasklet_struct urb_task; /* rx processing */ struct tasklet_struct urb_task; /* rx processing */
...@@ -120,6 +121,7 @@ struct acm { ...@@ -120,6 +121,7 @@ struct acm {
unsigned char throttle; /* throttled by tty layer */ unsigned char throttle; /* throttled by tty layer */
unsigned char clocal; /* termios CLOCAL */ unsigned char clocal; /* termios CLOCAL */
unsigned int ctrl_caps; /* control capabilities from the class specific header */ unsigned int ctrl_caps; /* control capabilities from the class specific header */
unsigned int susp_count; /* number of suspended interfaces */
}; };
#define CDC_DATA_INTERFACE_TYPE 0x0a #define CDC_DATA_INTERFACE_TYPE 0x0a
......
...@@ -9,6 +9,21 @@ config USB_DEBUG ...@@ -9,6 +9,21 @@ config USB_DEBUG
of debug messages to the system log. Select this if you are having a of debug messages to the system log. Select this if you are having a
problem with USB support and want to see more of what is going on. problem with USB support and want to see more of what is going on.
config USB_ANNOUNCE_NEW_DEVICES
bool "USB announce new devices"
depends on USB
default N
help
Say Y here if you want the USB core to always announce the
idVendor, idProduct, Manufacturer, Product, and SerialNumber
strings for every new USB device to the syslog. This option is
usually used by distro vendors to help with debugging and to
let users know what specific device was added to the machine
in what location.
If you do not want this kind of information sent to the system
log, or have any doubts about this, say N here.
comment "Miscellaneous USB options" comment "Miscellaneous USB options"
depends on USB depends on USB
......
...@@ -53,11 +53,13 @@ int hcd_buffer_create(struct usb_hcd *hcd) ...@@ -53,11 +53,13 @@ int hcd_buffer_create(struct usb_hcd *hcd)
char name[16]; char name[16];
int i, size; int i, size;
if (!hcd->self.controller->dma_mask) if (!hcd->self.controller->dma_mask &&
!(hcd->driver->flags & HCD_LOCAL_MEM))
return 0; return 0;
for (i = 0; i < HCD_BUFFER_POOLS; i++) { for (i = 0; i < HCD_BUFFER_POOLS; i++) {
if (!(size = pool_max [i])) size = pool_max[i];
if (!size)
continue; continue;
snprintf(name, sizeof name, "buffer-%d", size); snprintf(name, sizeof name, "buffer-%d", size);
hcd->pool[i] = dma_pool_create(name, hcd->self.controller, hcd->pool[i] = dma_pool_create(name, hcd->self.controller,
...@@ -80,10 +82,10 @@ int hcd_buffer_create(struct usb_hcd *hcd) ...@@ -80,10 +82,10 @@ int hcd_buffer_create(struct usb_hcd *hcd)
*/ */
void hcd_buffer_destroy(struct usb_hcd *hcd) void hcd_buffer_destroy(struct usb_hcd *hcd)
{ {
int i; int i;
for (i = 0; i < HCD_BUFFER_POOLS; i++) { for (i = 0; i < HCD_BUFFER_POOLS; i++) {
struct dma_pool *pool = hcd->pool[i]; struct dma_pool *pool = hcd->pool[i];
if (pool) { if (pool) {
dma_pool_destroy(pool); dma_pool_destroy(pool);
hcd->pool[i] = NULL; hcd->pool[i] = NULL;
...@@ -107,7 +109,8 @@ void *hcd_buffer_alloc( ...@@ -107,7 +109,8 @@ void *hcd_buffer_alloc(
int i; int i;
/* some USB hosts just use PIO */ /* some USB hosts just use PIO */
if (!bus->controller->dma_mask) { if (!bus->controller->dma_mask &&
!(hcd->driver->flags & HCD_LOCAL_MEM)) {
*dma = ~(dma_addr_t) 0; *dma = ~(dma_addr_t) 0;
return kmalloc(size, mem_flags); return kmalloc(size, mem_flags);
} }
...@@ -132,7 +135,8 @@ void hcd_buffer_free( ...@@ -132,7 +135,8 @@ void hcd_buffer_free(
if (!addr) if (!addr)
return; return;
if (!bus->controller->dma_mask) { if (!bus->controller->dma_mask &&
!(hcd->driver->flags & HCD_LOCAL_MEM)) {
kfree(addr); kfree(addr);
return; return;
} }
......
...@@ -238,7 +238,7 @@ static int usb_parse_interface(struct device *ddev, int cfgno, ...@@ -238,7 +238,7 @@ static int usb_parse_interface(struct device *ddev, int cfgno,
/* Allocate space for the right(?) number of endpoints */ /* Allocate space for the right(?) number of endpoints */
num_ep = num_ep_orig = alt->desc.bNumEndpoints; num_ep = num_ep_orig = alt->desc.bNumEndpoints;
alt->desc.bNumEndpoints = 0; // Use as a counter alt->desc.bNumEndpoints = 0; /* Use as a counter */
if (num_ep > USB_MAXENDPOINTS) { if (num_ep > USB_MAXENDPOINTS) {
dev_warn(ddev, "too many endpoints for config %d interface %d " dev_warn(ddev, "too many endpoints for config %d interface %d "
"altsetting %d: %d, using maximum allowed: %d\n", "altsetting %d: %d, using maximum allowed: %d\n",
...@@ -246,7 +246,8 @@ static int usb_parse_interface(struct device *ddev, int cfgno, ...@@ -246,7 +246,8 @@ static int usb_parse_interface(struct device *ddev, int cfgno,
num_ep = USB_MAXENDPOINTS; num_ep = USB_MAXENDPOINTS;
} }
if (num_ep > 0) { /* Can't allocate 0 bytes */ if (num_ep > 0) {
/* Can't allocate 0 bytes */
len = sizeof(struct usb_host_endpoint) * num_ep; len = sizeof(struct usb_host_endpoint) * num_ep;
alt->endpoint = kzalloc(len, GFP_KERNEL); alt->endpoint = kzalloc(len, GFP_KERNEL);
if (!alt->endpoint) if (!alt->endpoint)
...@@ -475,8 +476,9 @@ static int usb_parse_configuration(struct device *ddev, int cfgidx, ...@@ -475,8 +476,9 @@ static int usb_parse_configuration(struct device *ddev, int cfgidx,
return 0; return 0;
} }
// hub-only!! ... and only exported for reset/reinit path. /* hub-only!! ... and only exported for reset/reinit path.
// otherwise used internally on disconnect/destroy path * otherwise used internally on disconnect/destroy path
*/
void usb_destroy_configuration(struct usb_device *dev) void usb_destroy_configuration(struct usb_device *dev)
{ {
int c, i; int c, i;
...@@ -498,7 +500,7 @@ void usb_destroy_configuration(struct usb_device *dev) ...@@ -498,7 +500,7 @@ void usb_destroy_configuration(struct usb_device *dev)
kfree(cf->string); kfree(cf->string);
for (i = 0; i < cf->desc.bNumInterfaces; i++) { for (i = 0; i < cf->desc.bNumInterfaces; i++) {
if (cf->intf_cache[i]) if (cf->intf_cache[i])
kref_put(&cf->intf_cache[i]->ref, kref_put(&cf->intf_cache[i]->ref,
usb_release_interface_cache); usb_release_interface_cache);
} }
} }
...@@ -525,7 +527,7 @@ int usb_get_configuration(struct usb_device *dev) ...@@ -525,7 +527,7 @@ int usb_get_configuration(struct usb_device *dev)
unsigned int cfgno, length; unsigned int cfgno, length;
unsigned char *buffer; unsigned char *buffer;
unsigned char *bigbuffer; unsigned char *bigbuffer;
struct usb_config_descriptor *desc; struct usb_config_descriptor *desc;
cfgno = 0; cfgno = 0;
if (dev->authorized == 0) /* Not really an error */ if (dev->authorized == 0) /* Not really an error */
......
This diff is collapsed.
This diff is collapsed.
...@@ -202,10 +202,10 @@ static int usb_probe_interface(struct device *dev) ...@@ -202,10 +202,10 @@ static int usb_probe_interface(struct device *dev)
intf = to_usb_interface(dev); intf = to_usb_interface(dev);
udev = interface_to_usbdev(intf); udev = interface_to_usbdev(intf);
if (udev->authorized == 0) { if (udev->authorized == 0) {
dev_err(&intf->dev, "Device is not authorized for usage\n"); dev_err(&intf->dev, "Device is not authorized for usage\n");
return -ENODEV; return -ENODEV;
} }
id = usb_match_id(intf, driver->id_table); id = usb_match_id(intf, driver->id_table);
if (!id) if (!id)
...@@ -299,7 +299,7 @@ static int usb_unbind_interface(struct device *dev) ...@@ -299,7 +299,7 @@ static int usb_unbind_interface(struct device *dev)
* lock. * lock.
*/ */
int usb_driver_claim_interface(struct usb_driver *driver, int usb_driver_claim_interface(struct usb_driver *driver,
struct usb_interface *iface, void* priv) struct usb_interface *iface, void *priv)
{ {
struct device *dev = &iface->dev; struct device *dev = &iface->dev;
struct usb_device *udev = interface_to_usbdev(iface); struct usb_device *udev = interface_to_usbdev(iface);
...@@ -325,7 +325,7 @@ int usb_driver_claim_interface(struct usb_driver *driver, ...@@ -325,7 +325,7 @@ int usb_driver_claim_interface(struct usb_driver *driver,
return retval; return retval;
} }
EXPORT_SYMBOL(usb_driver_claim_interface); EXPORT_SYMBOL_GPL(usb_driver_claim_interface);
/** /**
* usb_driver_release_interface - unbind a driver from an interface * usb_driver_release_interface - unbind a driver from an interface
...@@ -370,7 +370,7 @@ void usb_driver_release_interface(struct usb_driver *driver, ...@@ -370,7 +370,7 @@ void usb_driver_release_interface(struct usb_driver *driver,
iface->needs_remote_wakeup = 0; iface->needs_remote_wakeup = 0;
usb_pm_unlock(udev); usb_pm_unlock(udev);
} }
EXPORT_SYMBOL(usb_driver_release_interface); EXPORT_SYMBOL_GPL(usb_driver_release_interface);
/* returns 0 if no match, 1 if match */ /* returns 0 if no match, 1 if match */
int usb_match_device(struct usb_device *dev, const struct usb_device_id *id) int usb_match_device(struct usb_device *dev, const struct usb_device_id *id)
...@@ -398,7 +398,7 @@ int usb_match_device(struct usb_device *dev, const struct usb_device_id *id) ...@@ -398,7 +398,7 @@ int usb_match_device(struct usb_device *dev, const struct usb_device_id *id)
return 0; return 0;
if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_SUBCLASS) && if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_SUBCLASS) &&
(id->bDeviceSubClass!= dev->descriptor.bDeviceSubClass)) (id->bDeviceSubClass != dev->descriptor.bDeviceSubClass))
return 0; return 0;
if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_PROTOCOL) && if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_PROTOCOL) &&
...@@ -534,15 +534,15 @@ const struct usb_device_id *usb_match_id(struct usb_interface *interface, ...@@ -534,15 +534,15 @@ const struct usb_device_id *usb_match_id(struct usb_interface *interface,
id->driver_info is the way to create an entry that id->driver_info is the way to create an entry that
indicates that the driver want to examine every indicates that the driver want to examine every
device and interface. */ device and interface. */
for (; id->idVendor || id->bDeviceClass || id->bInterfaceClass || for (; id->idVendor || id->idProduct || id->bDeviceClass ||
id->driver_info; id++) { id->bInterfaceClass || id->driver_info; id++) {
if (usb_match_one_id(interface, id)) if (usb_match_one_id(interface, id))
return id; return id;
} }
return NULL; return NULL;
} }
EXPORT_SYMBOL_GPL_FUTURE(usb_match_id); EXPORT_SYMBOL_GPL(usb_match_id);
static int usb_device_match(struct device *dev, struct device_driver *drv) static int usb_device_match(struct device *dev, struct device_driver *drv)
{ {
...@@ -586,7 +586,7 @@ static int usb_uevent(struct device *dev, struct kobj_uevent_env *env) ...@@ -586,7 +586,7 @@ static int usb_uevent(struct device *dev, struct kobj_uevent_env *env)
struct usb_device *usb_dev; struct usb_device *usb_dev;
/* driver is often null here; dev_dbg() would oops */ /* driver is often null here; dev_dbg() would oops */
pr_debug ("usb %s: uevent\n", dev->bus_id); pr_debug("usb %s: uevent\n", dev->bus_id);
if (is_usb_device(dev)) if (is_usb_device(dev))
usb_dev = to_usb_device(dev); usb_dev = to_usb_device(dev);
...@@ -596,11 +596,11 @@ static int usb_uevent(struct device *dev, struct kobj_uevent_env *env) ...@@ -596,11 +596,11 @@ static int usb_uevent(struct device *dev, struct kobj_uevent_env *env)
} }
if (usb_dev->devnum < 0) { if (usb_dev->devnum < 0) {
pr_debug ("usb %s: already deleted?\n", dev->bus_id); pr_debug("usb %s: already deleted?\n", dev->bus_id);
return -ENODEV; return -ENODEV;
} }
if (!usb_dev->bus) { if (!usb_dev->bus) {
pr_debug ("usb %s: bus removed?\n", dev->bus_id); pr_debug("usb %s: bus removed?\n", dev->bus_id);
return -ENODEV; return -ENODEV;
} }
...@@ -745,7 +745,7 @@ int usb_register_driver(struct usb_driver *new_driver, struct module *owner, ...@@ -745,7 +745,7 @@ int usb_register_driver(struct usb_driver *new_driver, struct module *owner,
return retval; return retval;
} }
EXPORT_SYMBOL_GPL_FUTURE(usb_register_driver); EXPORT_SYMBOL_GPL(usb_register_driver);
/** /**
* usb_deregister - unregister a USB interface driver * usb_deregister - unregister a USB interface driver
...@@ -769,7 +769,7 @@ void usb_deregister(struct usb_driver *driver) ...@@ -769,7 +769,7 @@ void usb_deregister(struct usb_driver *driver)
usbfs_update_special(); usbfs_update_special();
} }
EXPORT_SYMBOL_GPL_FUTURE(usb_deregister); EXPORT_SYMBOL_GPL(usb_deregister);
#ifdef CONFIG_PM #ifdef CONFIG_PM
...@@ -854,8 +854,10 @@ static int usb_suspend_interface(struct usb_interface *intf, pm_message_t msg) ...@@ -854,8 +854,10 @@ static int usb_suspend_interface(struct usb_interface *intf, pm_message_t msg)
dev_err(&intf->dev, "%s error %d\n", dev_err(&intf->dev, "%s error %d\n",
"suspend", status); "suspend", status);
} else { } else {
// FIXME else if there's no suspend method, disconnect... /*
// Not possible if auto_pm is set... * FIXME else if there's no suspend method, disconnect...
* Not possible if auto_pm is set...
*/
dev_warn(&intf->dev, "no suspend for driver %s?\n", dev_warn(&intf->dev, "no suspend for driver %s?\n",
driver->name); driver->name);
mark_quiesced(intf); mark_quiesced(intf);
...@@ -894,7 +896,7 @@ static int usb_resume_interface(struct usb_interface *intf, int reset_resume) ...@@ -894,7 +896,7 @@ static int usb_resume_interface(struct usb_interface *intf, int reset_resume)
dev_err(&intf->dev, "%s error %d\n", dev_err(&intf->dev, "%s error %d\n",
"reset_resume", status); "reset_resume", status);
} else { } else {
// status = -EOPNOTSUPP; /* status = -EOPNOTSUPP; */
dev_warn(&intf->dev, "no %s for driver %s?\n", dev_warn(&intf->dev, "no %s for driver %s?\n",
"reset_resume", driver->name); "reset_resume", driver->name);
} }
...@@ -905,7 +907,7 @@ static int usb_resume_interface(struct usb_interface *intf, int reset_resume) ...@@ -905,7 +907,7 @@ static int usb_resume_interface(struct usb_interface *intf, int reset_resume)
dev_err(&intf->dev, "%s error %d\n", dev_err(&intf->dev, "%s error %d\n",
"resume", status); "resume", status);
} else { } else {
// status = -EOPNOTSUPP; /* status = -EOPNOTSUPP; */
dev_warn(&intf->dev, "no %s for driver %s?\n", dev_warn(&intf->dev, "no %s for driver %s?\n",
"resume", driver->name); "resume", driver->name);
} }
...@@ -1175,7 +1177,7 @@ static int usb_resume_both(struct usb_device *udev) ...@@ -1175,7 +1177,7 @@ static int usb_resume_both(struct usb_device *udev)
* so if a root hub's controller is suspended * so if a root hub's controller is suspended
* then we're stuck. */ * then we're stuck. */
status = usb_resume_device(udev); status = usb_resume_device(udev);
} }
} else { } else {
/* Needed for setting udev->dev.power.power_state.event, /* Needed for setting udev->dev.power.power_state.event,
......
...@@ -204,7 +204,7 @@ int usb_register_dev(struct usb_interface *intf, ...@@ -204,7 +204,7 @@ int usb_register_dev(struct usb_interface *intf,
exit: exit:
return retval; return retval;
} }
EXPORT_SYMBOL(usb_register_dev); EXPORT_SYMBOL_GPL(usb_register_dev);
/** /**
* usb_deregister_dev - deregister a USB device's dynamic minor. * usb_deregister_dev - deregister a USB device's dynamic minor.
...@@ -245,4 +245,4 @@ void usb_deregister_dev(struct usb_interface *intf, ...@@ -245,4 +245,4 @@ void usb_deregister_dev(struct usb_interface *intf,
intf->minor = -1; intf->minor = -1;
destroy_usb_class(); destroy_usb_class();
} }
EXPORT_SYMBOL(usb_deregister_dev); EXPORT_SYMBOL_GPL(usb_deregister_dev);
This diff is collapsed.
This diff is collapsed.
...@@ -125,7 +125,7 @@ struct usb_hcd { ...@@ -125,7 +125,7 @@ struct usb_hcd {
/* more shared queuing code would be good; it should support /* more shared queuing code would be good; it should support
* smarter scheduling, handle transaction translators, etc; * smarter scheduling, handle transaction translators, etc;
* input size of periodic table to an interrupt scheduler. * input size of periodic table to an interrupt scheduler.
* (ohci 32, uhci 1024, ehci 256/512/1024). * (ohci 32, uhci 1024, ehci 256/512/1024).
*/ */
...@@ -133,16 +133,16 @@ struct usb_hcd { ...@@ -133,16 +133,16 @@ struct usb_hcd {
* this structure. * this structure.
*/ */
unsigned long hcd_priv[0] unsigned long hcd_priv[0]
__attribute__ ((aligned (sizeof(unsigned long)))); __attribute__ ((aligned(sizeof(unsigned long))));
}; };
/* 2.4 does this a bit differently ... */ /* 2.4 does this a bit differently ... */
static inline struct usb_bus *hcd_to_bus (struct usb_hcd *hcd) static inline struct usb_bus *hcd_to_bus(struct usb_hcd *hcd)
{ {
return &hcd->self; return &hcd->self;
} }
static inline struct usb_hcd *bus_to_hcd (struct usb_bus *bus) static inline struct usb_hcd *bus_to_hcd(struct usb_bus *bus)
{ {
return container_of(bus, struct usb_hcd, self); return container_of(bus, struct usb_hcd, self);
} }
...@@ -165,6 +165,7 @@ struct hc_driver { ...@@ -165,6 +165,7 @@ struct hc_driver {
int flags; int flags;
#define HCD_MEMORY 0x0001 /* HC regs use memory (else I/O) */ #define HCD_MEMORY 0x0001 /* HC regs use memory (else I/O) */
#define HCD_LOCAL_MEM 0x0002 /* HC needs local memory */
#define HCD_USB11 0x0010 /* USB 1.1 */ #define HCD_USB11 0x0010 /* USB 1.1 */
#define HCD_USB2 0x0020 /* USB 2.0 */ #define HCD_USB2 0x0020 /* USB 2.0 */
...@@ -201,15 +202,18 @@ struct hc_driver { ...@@ -201,15 +202,18 @@ struct hc_driver {
struct usb_host_endpoint *ep); struct usb_host_endpoint *ep);
/* root hub support */ /* root hub support */
int (*hub_status_data) (struct usb_hcd *hcd, char *buf); int (*hub_status_data) (struct usb_hcd *hcd, char *buf);
int (*hub_control) (struct usb_hcd *hcd, int (*hub_control) (struct usb_hcd *hcd,
u16 typeReq, u16 wValue, u16 wIndex, u16 typeReq, u16 wValue, u16 wIndex,
char *buf, u16 wLength); char *buf, u16 wLength);
int (*bus_suspend)(struct usb_hcd *); int (*bus_suspend)(struct usb_hcd *);
int (*bus_resume)(struct usb_hcd *); int (*bus_resume)(struct usb_hcd *);
int (*start_port_reset)(struct usb_hcd *, unsigned port_num); int (*start_port_reset)(struct usb_hcd *, unsigned port_num);
void (*hub_irq_enable)(struct usb_hcd *); void (*hub_irq_enable)(struct usb_hcd *);
/* Needed only if port-change IRQs are level-triggered */ /* Needed only if port-change IRQs are level-triggered */
/* force handover of high-speed port to full-speed companion */
void (*relinquish_port)(struct usb_hcd *, int);
}; };
extern int usb_hcd_link_urb_to_ep(struct usb_hcd *hcd, struct urb *urb); extern int usb_hcd_link_urb_to_ep(struct usb_hcd *hcd, struct urb *urb);
...@@ -217,56 +221,56 @@ extern int usb_hcd_check_unlink_urb(struct usb_hcd *hcd, struct urb *urb, ...@@ -217,56 +221,56 @@ extern int usb_hcd_check_unlink_urb(struct usb_hcd *hcd, struct urb *urb,
int status); int status);
extern void usb_hcd_unlink_urb_from_ep(struct usb_hcd *hcd, struct urb *urb); extern void usb_hcd_unlink_urb_from_ep(struct usb_hcd *hcd, struct urb *urb);
extern int usb_hcd_submit_urb (struct urb *urb, gfp_t mem_flags); extern int usb_hcd_submit_urb(struct urb *urb, gfp_t mem_flags);
extern int usb_hcd_unlink_urb (struct urb *urb, int status); extern int usb_hcd_unlink_urb(struct urb *urb, int status);
extern void usb_hcd_giveback_urb(struct usb_hcd *hcd, struct urb *urb, extern void usb_hcd_giveback_urb(struct usb_hcd *hcd, struct urb *urb,
int status); int status);
extern void usb_hcd_flush_endpoint(struct usb_device *udev, extern void usb_hcd_flush_endpoint(struct usb_device *udev,
struct usb_host_endpoint *ep); struct usb_host_endpoint *ep);
extern void usb_hcd_disable_endpoint(struct usb_device *udev, extern void usb_hcd_disable_endpoint(struct usb_device *udev,
struct usb_host_endpoint *ep); struct usb_host_endpoint *ep);
extern int usb_hcd_get_frame_number (struct usb_device *udev); extern int usb_hcd_get_frame_number(struct usb_device *udev);
extern struct usb_hcd *usb_create_hcd (const struct hc_driver *driver, extern struct usb_hcd *usb_create_hcd(const struct hc_driver *driver,
struct device *dev, char *bus_name); struct device *dev, char *bus_name);
extern struct usb_hcd *usb_get_hcd (struct usb_hcd *hcd); extern struct usb_hcd *usb_get_hcd(struct usb_hcd *hcd);
extern void usb_put_hcd (struct usb_hcd *hcd); extern void usb_put_hcd(struct usb_hcd *hcd);
extern int usb_add_hcd(struct usb_hcd *hcd, extern int usb_add_hcd(struct usb_hcd *hcd,
unsigned int irqnum, unsigned long irqflags); unsigned int irqnum, unsigned long irqflags);
extern void usb_remove_hcd(struct usb_hcd *hcd); extern void usb_remove_hcd(struct usb_hcd *hcd);
struct platform_device; struct platform_device;
extern void usb_hcd_platform_shutdown(struct platform_device* dev); extern void usb_hcd_platform_shutdown(struct platform_device *dev);
#ifdef CONFIG_PCI #ifdef CONFIG_PCI
struct pci_dev; struct pci_dev;
struct pci_device_id; struct pci_device_id;
extern int usb_hcd_pci_probe (struct pci_dev *dev, extern int usb_hcd_pci_probe(struct pci_dev *dev,
const struct pci_device_id *id); const struct pci_device_id *id);
extern void usb_hcd_pci_remove (struct pci_dev *dev); extern void usb_hcd_pci_remove(struct pci_dev *dev);
#ifdef CONFIG_PM #ifdef CONFIG_PM
extern int usb_hcd_pci_suspend (struct pci_dev *dev, pm_message_t state); extern int usb_hcd_pci_suspend(struct pci_dev *dev, pm_message_t state);
extern int usb_hcd_pci_resume (struct pci_dev *dev); extern int usb_hcd_pci_resume(struct pci_dev *dev);
#endif /* CONFIG_PM */ #endif /* CONFIG_PM */
extern void usb_hcd_pci_shutdown (struct pci_dev *dev); extern void usb_hcd_pci_shutdown(struct pci_dev *dev);
#endif /* CONFIG_PCI */ #endif /* CONFIG_PCI */
/* pci-ish (pdev null is ok) buffer alloc/mapping support */ /* pci-ish (pdev null is ok) buffer alloc/mapping support */
int hcd_buffer_create (struct usb_hcd *hcd); int hcd_buffer_create(struct usb_hcd *hcd);
void hcd_buffer_destroy (struct usb_hcd *hcd); void hcd_buffer_destroy(struct usb_hcd *hcd);
void *hcd_buffer_alloc (struct usb_bus *bus, size_t size, void *hcd_buffer_alloc(struct usb_bus *bus, size_t size,
gfp_t mem_flags, dma_addr_t *dma); gfp_t mem_flags, dma_addr_t *dma);
void hcd_buffer_free (struct usb_bus *bus, size_t size, void hcd_buffer_free(struct usb_bus *bus, size_t size,
void *addr, dma_addr_t dma); void *addr, dma_addr_t dma);
/* generic bus glue, needed for host controllers that don't use PCI */ /* generic bus glue, needed for host controllers that don't use PCI */
extern irqreturn_t usb_hcd_irq (int irq, void *__hcd); extern irqreturn_t usb_hcd_irq(int irq, void *__hcd);
extern void usb_hc_died (struct usb_hcd *hcd); extern void usb_hc_died(struct usb_hcd *hcd);
extern void usb_hcd_poll_rh_status(struct usb_hcd *hcd); extern void usb_hcd_poll_rh_status(struct usb_hcd *hcd);
/* -------------------------------------------------------------------------- */ /* -------------------------------------------------------------------------- */
...@@ -319,9 +323,9 @@ extern void usb_destroy_configuration(struct usb_device *dev); ...@@ -319,9 +323,9 @@ extern void usb_destroy_configuration(struct usb_device *dev);
* Generic bandwidth allocation constants/support * Generic bandwidth allocation constants/support
*/ */
#define FRAME_TIME_USECS 1000L #define FRAME_TIME_USECS 1000L
#define BitTime(bytecount) (7 * 8 * bytecount / 6) /* with integer truncation */ #define BitTime(bytecount) (7 * 8 * bytecount / 6) /* with integer truncation */
/* Trying not to use worst-case bit-stuffing /* Trying not to use worst-case bit-stuffing
of (7/6 * 8 * bytecount) = 9.33 * bytecount */ * of (7/6 * 8 * bytecount) = 9.33 * bytecount */
/* bytecount = data payload byte count */ /* bytecount = data payload byte count */
#define NS_TO_US(ns) ((ns + 500L) / 1000L) #define NS_TO_US(ns) ((ns + 500L) / 1000L)
...@@ -333,9 +337,9 @@ extern void usb_destroy_configuration(struct usb_device *dev); ...@@ -333,9 +337,9 @@ extern void usb_destroy_configuration(struct usb_device *dev);
*/ */
#define BW_HOST_DELAY 1000L /* nanoseconds */ #define BW_HOST_DELAY 1000L /* nanoseconds */
#define BW_HUB_LS_SETUP 333L /* nanoseconds */ #define BW_HUB_LS_SETUP 333L /* nanoseconds */
/* 4 full-speed bit times (est.) */ /* 4 full-speed bit times (est.) */
#define FRAME_TIME_BITS 12000L /* frame = 1 millisecond */ #define FRAME_TIME_BITS 12000L /* frame = 1 millisecond */
#define FRAME_TIME_MAX_BITS_ALLOC (90L * FRAME_TIME_BITS / 100L) #define FRAME_TIME_MAX_BITS_ALLOC (90L * FRAME_TIME_BITS / 100L)
#define FRAME_TIME_MAX_USECS_ALLOC (90L * FRAME_TIME_USECS / 100L) #define FRAME_TIME_MAX_USECS_ALLOC (90L * FRAME_TIME_USECS / 100L)
...@@ -345,16 +349,16 @@ extern void usb_destroy_configuration(struct usb_device *dev); ...@@ -345,16 +349,16 @@ extern void usb_destroy_configuration(struct usb_device *dev);
* to preallocate bandwidth) * to preallocate bandwidth)
*/ */
#define USB2_HOST_DELAY 5 /* nsec, guess */ #define USB2_HOST_DELAY 5 /* nsec, guess */
#define HS_NSECS(bytes) ( ((55 * 8 * 2083) \ #define HS_NSECS(bytes) (((55 * 8 * 2083) \
+ (2083UL * (3 + BitTime(bytes))))/1000 \ + (2083UL * (3 + BitTime(bytes))))/1000 \
+ USB2_HOST_DELAY) + USB2_HOST_DELAY)
#define HS_NSECS_ISO(bytes) ( ((38 * 8 * 2083) \ #define HS_NSECS_ISO(bytes) (((38 * 8 * 2083) \
+ (2083UL * (3 + BitTime(bytes))))/1000 \ + (2083UL * (3 + BitTime(bytes))))/1000 \
+ USB2_HOST_DELAY) + USB2_HOST_DELAY)
#define HS_USECS(bytes) NS_TO_US (HS_NSECS(bytes)) #define HS_USECS(bytes) NS_TO_US (HS_NSECS(bytes))
#define HS_USECS_ISO(bytes) NS_TO_US (HS_NSECS_ISO(bytes)) #define HS_USECS_ISO(bytes) NS_TO_US (HS_NSECS_ISO(bytes))
extern long usb_calc_bus_time (int speed, int is_input, extern long usb_calc_bus_time(int speed, int is_input,
int isoc, int bytecount); int isoc, int bytecount);
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
...@@ -370,16 +374,16 @@ extern struct list_head usb_bus_list; ...@@ -370,16 +374,16 @@ extern struct list_head usb_bus_list;
extern struct mutex usb_bus_list_lock; extern struct mutex usb_bus_list_lock;
extern wait_queue_head_t usb_kill_urb_queue; extern wait_queue_head_t usb_kill_urb_queue;
extern void usb_enable_root_hub_irq (struct usb_bus *bus); extern void usb_enable_root_hub_irq(struct usb_bus *bus);
extern int usb_find_interface_driver (struct usb_device *dev, extern int usb_find_interface_driver(struct usb_device *dev,
struct usb_interface *interface); struct usb_interface *interface);
#define usb_endpoint_out(ep_dir) (!((ep_dir) & USB_DIR_IN)) #define usb_endpoint_out(ep_dir) (!((ep_dir) & USB_DIR_IN))
#ifdef CONFIG_PM #ifdef CONFIG_PM
extern void usb_hcd_resume_root_hub (struct usb_hcd *hcd); extern void usb_hcd_resume_root_hub(struct usb_hcd *hcd);
extern void usb_root_hub_lost_power (struct usb_device *rhdev); extern void usb_root_hub_lost_power(struct usb_device *rhdev);
extern int hcd_bus_suspend(struct usb_device *rhdev); extern int hcd_bus_suspend(struct usb_device *rhdev);
extern int hcd_bus_resume(struct usb_device *rhdev); extern int hcd_bus_resume(struct usb_device *rhdev);
#else #else
...@@ -399,13 +403,13 @@ static inline void usb_hcd_resume_root_hub(struct usb_hcd *hcd) ...@@ -399,13 +403,13 @@ static inline void usb_hcd_resume_root_hub(struct usb_hcd *hcd)
* these are expected to be called from the USB core/hub thread * these are expected to be called from the USB core/hub thread
* with the kernel lock held * with the kernel lock held
*/ */
extern void usbfs_update_special (void); extern void usbfs_update_special(void);
extern int usbfs_init(void); extern int usbfs_init(void);
extern void usbfs_cleanup(void); extern void usbfs_cleanup(void);
#else /* CONFIG_USB_DEVICEFS */ #else /* CONFIG_USB_DEVICEFS */
static inline void usbfs_update_special (void) {} static inline void usbfs_update_special(void) {}
static inline int usbfs_init(void) { return 0; } static inline int usbfs_init(void) { return 0; }
static inline void usbfs_cleanup(void) { } static inline void usbfs_cleanup(void) { }
...@@ -460,7 +464,7 @@ static inline void usbmon_urb_complete(struct usb_bus *bus, struct urb *urb, ...@@ -460,7 +464,7 @@ static inline void usbmon_urb_complete(struct usb_bus *bus, struct urb *urb,
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
/* hub.h ... DeviceRemovable in 2.4.2-ac11, gone in 2.4.10 */ /* hub.h ... DeviceRemovable in 2.4.2-ac11, gone in 2.4.10 */
// bleech -- resurfaced in 2.4.11 or 2.4.12 /* bleech -- resurfaced in 2.4.11 or 2.4.12 */
#define bitmap DeviceRemovable #define bitmap DeviceRemovable
...@@ -468,8 +472,8 @@ static inline void usbmon_urb_complete(struct usb_bus *bus, struct urb *urb, ...@@ -468,8 +472,8 @@ static inline void usbmon_urb_complete(struct usb_bus *bus, struct urb *urb,
/* random stuff */ /* random stuff */
#define RUN_CONTEXT (in_irq () ? "in_irq" \ #define RUN_CONTEXT (in_irq() ? "in_irq" \
: (in_interrupt () ? "in_interrupt" : "can sleep")) : (in_interrupt() ? "in_interrupt" : "can sleep"))
/* This rwsem is for use only by the hub driver and ehci-hcd. /* This rwsem is for use only by the hub driver and ehci-hcd.
......
...@@ -37,6 +37,13 @@ ...@@ -37,6 +37,13 @@
#define USB_PERSIST 0 #define USB_PERSIST 0
#endif #endif
/* if we are in debug mode, always announce new devices */
#ifdef DEBUG
#ifndef CONFIG_USB_ANNOUNCE_NEW_DEVICES
#define CONFIG_USB_ANNOUNCE_NEW_DEVICES
#endif
#endif
struct usb_hub { struct usb_hub {
struct device *intfdev; /* the "interface" device */ struct device *intfdev; /* the "interface" device */
struct usb_device *hdev; struct usb_device *hdev;
...@@ -487,6 +494,7 @@ void usb_hub_tt_clear_buffer (struct usb_device *udev, int pipe) ...@@ -487,6 +494,7 @@ void usb_hub_tt_clear_buffer (struct usb_device *udev, int pipe)
schedule_work (&tt->kevent); schedule_work (&tt->kevent);
spin_unlock_irqrestore (&tt->lock, flags); spin_unlock_irqrestore (&tt->lock, flags);
} }
EXPORT_SYMBOL_GPL(usb_hub_tt_clear_buffer);
static void hub_power_on(struct usb_hub *hub) static void hub_power_on(struct usb_hub *hub)
{ {
...@@ -1027,8 +1035,10 @@ static void recursively_mark_NOTATTACHED(struct usb_device *udev) ...@@ -1027,8 +1035,10 @@ static void recursively_mark_NOTATTACHED(struct usb_device *udev)
if (udev->children[i]) if (udev->children[i])
recursively_mark_NOTATTACHED(udev->children[i]); recursively_mark_NOTATTACHED(udev->children[i]);
} }
if (udev->state == USB_STATE_SUSPENDED) if (udev->state == USB_STATE_SUSPENDED) {
udev->discon_suspended = 1; udev->discon_suspended = 1;
udev->active_duration -= jiffies;
}
udev->state = USB_STATE_NOTATTACHED; udev->state = USB_STATE_NOTATTACHED;
} }
...@@ -1077,6 +1087,12 @@ void usb_set_device_state(struct usb_device *udev, ...@@ -1077,6 +1087,12 @@ void usb_set_device_state(struct usb_device *udev,
else else
device_init_wakeup(&udev->dev, 0); device_init_wakeup(&udev->dev, 0);
} }
if (udev->state == USB_STATE_SUSPENDED &&
new_state != USB_STATE_SUSPENDED)
udev->active_duration -= jiffies;
else if (new_state == USB_STATE_SUSPENDED &&
udev->state != USB_STATE_SUSPENDED)
udev->active_duration += jiffies;
udev->state = new_state; udev->state = new_state;
} else } else
recursively_mark_NOTATTACHED(udev); recursively_mark_NOTATTACHED(udev);
...@@ -1207,7 +1223,7 @@ void usb_disconnect(struct usb_device **pdev) ...@@ -1207,7 +1223,7 @@ void usb_disconnect(struct usb_device **pdev)
put_device(&udev->dev); put_device(&udev->dev);
} }
#ifdef DEBUG #ifdef CONFIG_USB_ANNOUNCE_NEW_DEVICES
static void show_string(struct usb_device *udev, char *id, char *string) static void show_string(struct usb_device *udev, char *id, char *string)
{ {
if (!string) if (!string)
...@@ -1215,12 +1231,24 @@ static void show_string(struct usb_device *udev, char *id, char *string) ...@@ -1215,12 +1231,24 @@ static void show_string(struct usb_device *udev, char *id, char *string)
dev_printk(KERN_INFO, &udev->dev, "%s: %s\n", id, string); dev_printk(KERN_INFO, &udev->dev, "%s: %s\n", id, string);
} }
static void announce_device(struct usb_device *udev)
{
dev_info(&udev->dev, "New USB device found, idVendor=%04x, idProduct=%04x\n",
le16_to_cpu(udev->descriptor.idVendor),
le16_to_cpu(udev->descriptor.idProduct));
dev_info(&udev->dev, "New USB device strings: Mfr=%d, Product=%d, "
"SerialNumber=%d\n",
udev->descriptor.iManufacturer,
udev->descriptor.iProduct,
udev->descriptor.iSerialNumber);
show_string(udev, "Product", udev->product);
show_string(udev, "Manufacturer", udev->manufacturer);
show_string(udev, "SerialNumber", udev->serial);
}
#else #else
static inline void show_string(struct usb_device *udev, char *id, char *string) static inline void announce_device(struct usb_device *udev) { }
{}
#endif #endif
#ifdef CONFIG_USB_OTG #ifdef CONFIG_USB_OTG
#include "otg_whitelist.h" #include "otg_whitelist.h"
#endif #endif
...@@ -1390,14 +1418,7 @@ int usb_new_device(struct usb_device *udev) ...@@ -1390,14 +1418,7 @@ int usb_new_device(struct usb_device *udev)
} }
/* Tell the world! */ /* Tell the world! */
dev_dbg(&udev->dev, "new device strings: Mfr=%d, Product=%d, " announce_device(udev);
"SerialNumber=%d\n",
udev->descriptor.iManufacturer,
udev->descriptor.iProduct,
udev->descriptor.iSerialNumber);
show_string(udev, "Product", udev->product);
show_string(udev, "Manufacturer", udev->manufacturer);
show_string(udev, "SerialNumber", udev->serial);
return err; return err;
fail: fail:
...@@ -2482,6 +2503,7 @@ static void hub_port_connect_change(struct usb_hub *hub, int port1, ...@@ -2482,6 +2503,7 @@ static void hub_port_connect_change(struct usb_hub *hub, int port1,
{ {
struct usb_device *hdev = hub->hdev; struct usb_device *hdev = hub->hdev;
struct device *hub_dev = hub->intfdev; struct device *hub_dev = hub->intfdev;
struct usb_hcd *hcd = bus_to_hcd(hdev->bus);
u16 wHubCharacteristics = le16_to_cpu(hub->descriptor->wHubCharacteristics); u16 wHubCharacteristics = le16_to_cpu(hub->descriptor->wHubCharacteristics);
int status, i; int status, i;
...@@ -2645,6 +2667,8 @@ static void hub_port_connect_change(struct usb_hub *hub, int port1, ...@@ -2645,6 +2667,8 @@ static void hub_port_connect_change(struct usb_hub *hub, int port1,
done: done:
hub_port_disable(hub, port1, 1); hub_port_disable(hub, port1, 1);
if (hcd->driver->relinquish_port && !hub->hdev->parent)
hcd->driver->relinquish_port(hcd, port1);
} }
static void hub_events(void) static void hub_events(void)
...@@ -2946,7 +2970,7 @@ static int config_descriptors_changed(struct usb_device *udev) ...@@ -2946,7 +2970,7 @@ static int config_descriptors_changed(struct usb_device *udev)
if (len < le16_to_cpu(udev->config[index].desc.wTotalLength)) if (len < le16_to_cpu(udev->config[index].desc.wTotalLength))
len = le16_to_cpu(udev->config[index].desc.wTotalLength); len = le16_to_cpu(udev->config[index].desc.wTotalLength);
} }
buf = kmalloc (len, GFP_KERNEL); buf = kmalloc(len, GFP_NOIO);
if (buf == NULL) { if (buf == NULL) {
dev_err(&udev->dev, "no mem to re-read configs after reset\n"); dev_err(&udev->dev, "no mem to re-read configs after reset\n");
/* assume the worst */ /* assume the worst */
...@@ -3093,7 +3117,7 @@ int usb_reset_device(struct usb_device *udev) ...@@ -3093,7 +3117,7 @@ int usb_reset_device(struct usb_device *udev)
hub_port_logical_disconnect(parent_hub, port1); hub_port_logical_disconnect(parent_hub, port1);
return -ENODEV; return -ENODEV;
} }
EXPORT_SYMBOL(usb_reset_device); EXPORT_SYMBOL_GPL(usb_reset_device);
/** /**
* usb_reset_composite_device - warn interface drivers and perform a USB port reset * usb_reset_composite_device - warn interface drivers and perform a USB port reset
...@@ -3110,16 +3134,12 @@ EXPORT_SYMBOL(usb_reset_device); ...@@ -3110,16 +3134,12 @@ EXPORT_SYMBOL(usb_reset_device);
* this from a driver probe() routine after downloading new firmware. * this from a driver probe() routine after downloading new firmware.
* For calls that might not occur during probe(), drivers should lock * For calls that might not occur during probe(), drivers should lock
* the device using usb_lock_device_for_reset(). * the device using usb_lock_device_for_reset().
*
* The interface locks are acquired during the pre_reset stage and released
* during the post_reset stage. However if iface is not NULL and is
* currently being probed, we assume that the caller already owns its
* lock.
*/ */
int usb_reset_composite_device(struct usb_device *udev, int usb_reset_composite_device(struct usb_device *udev,
struct usb_interface *iface) struct usb_interface *iface)
{ {
int ret; int ret;
int i;
struct usb_host_config *config = udev->actconfig; struct usb_host_config *config = udev->actconfig;
if (udev->state == USB_STATE_NOTATTACHED || if (udev->state == USB_STATE_NOTATTACHED ||
...@@ -3136,16 +3156,11 @@ int usb_reset_composite_device(struct usb_device *udev, ...@@ -3136,16 +3156,11 @@ int usb_reset_composite_device(struct usb_device *udev,
iface = NULL; iface = NULL;
if (config) { if (config) {
int i;
struct usb_interface *cintf;
struct usb_driver *drv;
for (i = 0; i < config->desc.bNumInterfaces; ++i) { for (i = 0; i < config->desc.bNumInterfaces; ++i) {
cintf = config->interface[i]; struct usb_interface *cintf = config->interface[i];
if (cintf != iface) struct usb_driver *drv;
down(&cintf->dev.sem);
if (device_is_registered(&cintf->dev) && if (cintf->dev.driver) {
cintf->dev.driver) {
drv = to_usb_driver(cintf->dev.driver); drv = to_usb_driver(cintf->dev.driver);
if (drv->pre_reset) if (drv->pre_reset)
(drv->pre_reset)(cintf); (drv->pre_reset)(cintf);
...@@ -3157,25 +3172,20 @@ int usb_reset_composite_device(struct usb_device *udev, ...@@ -3157,25 +3172,20 @@ int usb_reset_composite_device(struct usb_device *udev,
ret = usb_reset_device(udev); ret = usb_reset_device(udev);
if (config) { if (config) {
int i;
struct usb_interface *cintf;
struct usb_driver *drv;
for (i = config->desc.bNumInterfaces - 1; i >= 0; --i) { for (i = config->desc.bNumInterfaces - 1; i >= 0; --i) {
cintf = config->interface[i]; struct usb_interface *cintf = config->interface[i];
if (device_is_registered(&cintf->dev) && struct usb_driver *drv;
cintf->dev.driver) {
if (cintf->dev.driver) {
drv = to_usb_driver(cintf->dev.driver); drv = to_usb_driver(cintf->dev.driver);
if (drv->post_reset) if (drv->post_reset)
(drv->post_reset)(cintf); (drv->post_reset)(cintf);
/* FIXME: Unbind if post_reset returns an error or isn't defined */ /* FIXME: Unbind if post_reset returns an error or isn't defined */
} }
if (cintf != iface)
up(&cintf->dev.sem);
} }
} }
usb_autosuspend_device(udev); usb_autosuspend_device(udev);
return ret; return ret;
} }
EXPORT_SYMBOL(usb_reset_composite_device); EXPORT_SYMBOL_GPL(usb_reset_composite_device);
...@@ -55,16 +55,16 @@ ...@@ -55,16 +55,16 @@
#define USB_PORT_FEAT_TEST 21 #define USB_PORT_FEAT_TEST 21
#define USB_PORT_FEAT_INDICATOR 22 #define USB_PORT_FEAT_INDICATOR 22
/* /*
* Hub Status and Hub Change results * Hub Status and Hub Change results
* See USB 2.0 spec Table 11-19 and Table 11-20 * See USB 2.0 spec Table 11-19 and Table 11-20
*/ */
struct usb_port_status { struct usb_port_status {
__le16 wPortStatus; __le16 wPortStatus;
__le16 wPortChange; __le16 wPortChange;
} __attribute__ ((packed)); } __attribute__ ((packed));
/* /*
* wPortStatus bit field * wPortStatus bit field
* See USB 2.0 spec Table 11-21 * See USB 2.0 spec Table 11-21
*/ */
...@@ -81,7 +81,7 @@ struct usb_port_status { ...@@ -81,7 +81,7 @@ struct usb_port_status {
#define USB_PORT_STAT_INDICATOR 0x1000 #define USB_PORT_STAT_INDICATOR 0x1000
/* bits 13 to 15 are reserved */ /* bits 13 to 15 are reserved */
/* /*
* wPortChange bit field * wPortChange bit field
* See USB 2.0 spec Table 11-22 * See USB 2.0 spec Table 11-22
* Bits 0 to 4 shown, bits 5 to 15 are reserved * Bits 0 to 4 shown, bits 5 to 15 are reserved
...@@ -93,7 +93,7 @@ struct usb_port_status { ...@@ -93,7 +93,7 @@ struct usb_port_status {
#define USB_PORT_STAT_C_RESET 0x0010 #define USB_PORT_STAT_C_RESET 0x0010
/* /*
* wHubCharacteristics (masks) * wHubCharacteristics (masks)
* See USB 2.0 spec Table 11-13, offset 3 * See USB 2.0 spec Table 11-13, offset 3
*/ */
#define HUB_CHAR_LPSM 0x0003 /* D1 .. D0 */ #define HUB_CHAR_LPSM 0x0003 /* D1 .. D0 */
...@@ -119,8 +119,8 @@ struct usb_hub_status { ...@@ -119,8 +119,8 @@ struct usb_hub_status {
#define HUB_CHANGE_OVERCURRENT 0x0002 #define HUB_CHANGE_OVERCURRENT 0x0002
/* /*
* Hub descriptor * Hub descriptor
* See USB 2.0 spec Table 11-13 * See USB 2.0 spec Table 11-13
*/ */
...@@ -134,7 +134,7 @@ struct usb_hub_descriptor { ...@@ -134,7 +134,7 @@ struct usb_hub_descriptor {
__le16 wHubCharacteristics; __le16 wHubCharacteristics;
__u8 bPwrOn2PwrGood; __u8 bPwrOn2PwrGood;
__u8 bHubContrCurrent; __u8 bHubContrCurrent;
/* add 1 bit for hub status change; round to bytes */ /* add 1 bit for hub status change; round to bytes */
__u8 DeviceRemovable[(USB_MAXCHILDREN + 1 + 7) / 8]; __u8 DeviceRemovable[(USB_MAXCHILDREN + 1 + 7) / 8];
__u8 PortPwrCtrlMask[(USB_MAXCHILDREN + 1 + 7) / 8]; __u8 PortPwrCtrlMask[(USB_MAXCHILDREN + 1 + 7) / 8];
} __attribute__ ((packed)); } __attribute__ ((packed));
...@@ -190,6 +190,6 @@ struct usb_tt_clear { ...@@ -190,6 +190,6 @@ struct usb_tt_clear {
u16 devinfo; u16 devinfo;
}; };
extern void usb_hub_tt_clear_buffer (struct usb_device *dev, int pipe); extern void usb_hub_tt_clear_buffer(struct usb_device *dev, int pipe);
#endif /* __LINUX_HUB_H */ #endif /* __LINUX_HUB_H */
...@@ -38,10 +38,15 @@ ...@@ -38,10 +38,15 @@
#include <linux/usbdevice_fs.h> #include <linux/usbdevice_fs.h>
#include <linux/parser.h> #include <linux/parser.h>
#include <linux/notifier.h> #include <linux/notifier.h>
#include <linux/seq_file.h>
#include <asm/byteorder.h> #include <asm/byteorder.h>
#include "usb.h" #include "usb.h"
#include "hcd.h" #include "hcd.h"
#define USBFS_DEFAULT_DEVMODE (S_IWUSR | S_IRUGO)
#define USBFS_DEFAULT_BUSMODE (S_IXUGO | S_IRUGO)
#define USBFS_DEFAULT_LISTMODE S_IRUGO
static struct super_operations usbfs_ops; static struct super_operations usbfs_ops;
static const struct file_operations default_file_operations; static const struct file_operations default_file_operations;
static struct vfsmount *usbfs_mount; static struct vfsmount *usbfs_mount;
...@@ -57,9 +62,33 @@ static uid_t listuid; /* = 0 */ ...@@ -57,9 +62,33 @@ static uid_t listuid; /* = 0 */
static gid_t devgid; /* = 0 */ static gid_t devgid; /* = 0 */
static gid_t busgid; /* = 0 */ static gid_t busgid; /* = 0 */
static gid_t listgid; /* = 0 */ static gid_t listgid; /* = 0 */
static umode_t devmode = S_IWUSR | S_IRUGO; static umode_t devmode = USBFS_DEFAULT_DEVMODE;
static umode_t busmode = S_IXUGO | S_IRUGO; static umode_t busmode = USBFS_DEFAULT_BUSMODE;
static umode_t listmode = S_IRUGO; static umode_t listmode = USBFS_DEFAULT_LISTMODE;
static int usbfs_show_options(struct seq_file *seq, struct vfsmount *mnt)
{
if (devuid != 0)
seq_printf(seq, ",devuid=%u", devuid);
if (devgid != 0)
seq_printf(seq, ",devgid=%u", devgid);
if (devmode != USBFS_DEFAULT_DEVMODE)
seq_printf(seq, ",devmode=%o", devmode);
if (busuid != 0)
seq_printf(seq, ",busuid=%u", busuid);
if (busgid != 0)
seq_printf(seq, ",busgid=%u", busgid);
if (busmode != USBFS_DEFAULT_BUSMODE)
seq_printf(seq, ",busmode=%o", busmode);
if (listuid != 0)
seq_printf(seq, ",listuid=%u", listuid);
if (listgid != 0)
seq_printf(seq, ",listgid=%u", listgid);
if (listmode != USBFS_DEFAULT_LISTMODE)
seq_printf(seq, ",listmode=%o", listmode);
return 0;
}
enum { enum {
Opt_devuid, Opt_devgid, Opt_devmode, Opt_devuid, Opt_devgid, Opt_devmode,
...@@ -93,9 +122,9 @@ static int parse_options(struct super_block *s, char *data) ...@@ -93,9 +122,9 @@ static int parse_options(struct super_block *s, char *data)
devgid = 0; devgid = 0;
busgid = 0; busgid = 0;
listgid = 0; listgid = 0;
devmode = S_IWUSR | S_IRUGO; devmode = USBFS_DEFAULT_DEVMODE;
busmode = S_IXUGO | S_IRUGO; busmode = USBFS_DEFAULT_BUSMODE;
listmode = S_IRUGO; listmode = USBFS_DEFAULT_LISTMODE;
while ((p = strsep(&data, ",")) != NULL) { while ((p = strsep(&data, ",")) != NULL) {
substring_t args[MAX_OPT_ARGS]; substring_t args[MAX_OPT_ARGS];
...@@ -418,6 +447,7 @@ static struct super_operations usbfs_ops = { ...@@ -418,6 +447,7 @@ static struct super_operations usbfs_ops = {
.statfs = simple_statfs, .statfs = simple_statfs,
.drop_inode = generic_delete_inode, .drop_inode = generic_delete_inode,
.remount_fs = remount, .remount_fs = remount,
.show_options = usbfs_show_options,
}; };
static int usbfs_fill_super(struct super_block *sb, void *data, int silent) static int usbfs_fill_super(struct super_block *sb, void *data, int silent)
......
This diff is collapsed.
...@@ -33,7 +33,7 @@ EXPORT_SYMBOL_GPL(usb_register_notify); ...@@ -33,7 +33,7 @@ EXPORT_SYMBOL_GPL(usb_register_notify);
* usb_unregister_notify - unregister a notifier callback * usb_unregister_notify - unregister a notifier callback
* @nb: pointer to the notifier block for the callback events. * @nb: pointer to the notifier block for the callback events.
* *
* usb_register_notifier() must have been previously called for this function * usb_register_notify() must have been previously called for this function
* to work properly. * to work properly.
*/ */
void usb_unregister_notify(struct notifier_block *nb) void usb_unregister_notify(struct notifier_block *nb)
......
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
* mostly use of USB_DEVICE() or USB_DEVICE_VER() entries.. * mostly use of USB_DEVICE() or USB_DEVICE_VER() entries..
* *
* YOU _SHOULD_ CHANGE THIS LIST TO MATCH YOUR PRODUCT AND ITS TESTING! * YOU _SHOULD_ CHANGE THIS LIST TO MATCH YOUR PRODUCT AND ITS TESTING!
*/ */
static struct usb_device_id whitelist_table [] = { static struct usb_device_id whitelist_table [] = {
...@@ -55,7 +55,7 @@ static int is_targeted(struct usb_device *dev) ...@@ -55,7 +55,7 @@ static int is_targeted(struct usb_device *dev)
return 1; return 1;
/* HNP test device is _never_ targeted (see OTG spec 6.6.6) */ /* HNP test device is _never_ targeted (see OTG spec 6.6.6) */
if ((le16_to_cpu(dev->descriptor.idVendor) == 0x1a0a && if ((le16_to_cpu(dev->descriptor.idVendor) == 0x1a0a &&
le16_to_cpu(dev->descriptor.idProduct) == 0xbadd)) le16_to_cpu(dev->descriptor.idProduct) == 0xbadd))
return 0; return 0;
...@@ -86,7 +86,7 @@ static int is_targeted(struct usb_device *dev) ...@@ -86,7 +86,7 @@ static int is_targeted(struct usb_device *dev)
continue; continue;
if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_SUBCLASS) && if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_SUBCLASS) &&
(id->bDeviceSubClass!= dev->descriptor.bDeviceSubClass)) (id->bDeviceSubClass != dev->descriptor.bDeviceSubClass))
continue; continue;
if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_PROTOCOL) && if ((id->match_flags & USB_DEVICE_ID_MATCH_DEV_PROTOCOL) &&
......
...@@ -72,7 +72,7 @@ set_bConfigurationValue(struct device *dev, struct device_attribute *attr, ...@@ -72,7 +72,7 @@ set_bConfigurationValue(struct device *dev, struct device_attribute *attr,
return (value < 0) ? value : count; return (value < 0) ? value : count;
} }
static DEVICE_ATTR(bConfigurationValue, S_IRUGO | S_IWUSR, static DEVICE_ATTR(bConfigurationValue, S_IRUGO | S_IWUSR,
show_bConfigurationValue, set_bConfigurationValue); show_bConfigurationValue, set_bConfigurationValue);
/* String fields */ /* String fields */
...@@ -248,6 +248,41 @@ static void remove_persist_attributes(struct device *dev) ...@@ -248,6 +248,41 @@ static void remove_persist_attributes(struct device *dev)
#ifdef CONFIG_USB_SUSPEND #ifdef CONFIG_USB_SUSPEND
static ssize_t
show_connected_duration(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct usb_device *udev = to_usb_device(dev);
return sprintf(buf, "%u\n",
jiffies_to_msecs(jiffies - udev->connect_time));
}
static DEVICE_ATTR(connected_duration, S_IRUGO, show_connected_duration, NULL);
/*
* If the device is resumed, the last time the device was suspended has
* been pre-subtracted from active_duration. We add the current time to
* get the duration that the device was actually active.
*
* If the device is suspended, the active_duration is up-to-date.
*/
static ssize_t
show_active_duration(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct usb_device *udev = to_usb_device(dev);
int duration;
if (udev->state != USB_STATE_SUSPENDED)
duration = jiffies_to_msecs(jiffies + udev->active_duration);
else
duration = jiffies_to_msecs(udev->active_duration);
return sprintf(buf, "%u\n", duration);
}
static DEVICE_ATTR(active_duration, S_IRUGO, show_active_duration, NULL);
static ssize_t static ssize_t
show_autosuspend(struct device *dev, struct device_attribute *attr, char *buf) show_autosuspend(struct device *dev, struct device_attribute *attr, char *buf)
{ {
...@@ -365,12 +400,26 @@ static int add_power_attributes(struct device *dev) ...@@ -365,12 +400,26 @@ static int add_power_attributes(struct device *dev)
rc = sysfs_add_file_to_group(&dev->kobj, rc = sysfs_add_file_to_group(&dev->kobj,
&dev_attr_level.attr, &dev_attr_level.attr,
power_group); power_group);
if (rc == 0)
rc = sysfs_add_file_to_group(&dev->kobj,
&dev_attr_connected_duration.attr,
power_group);
if (rc == 0)
rc = sysfs_add_file_to_group(&dev->kobj,
&dev_attr_active_duration.attr,
power_group);
} }
return rc; return rc;
} }
static void remove_power_attributes(struct device *dev) static void remove_power_attributes(struct device *dev)
{ {
sysfs_remove_file_from_group(&dev->kobj,
&dev_attr_active_duration.attr,
power_group);
sysfs_remove_file_from_group(&dev->kobj,
&dev_attr_connected_duration.attr,
power_group);
sysfs_remove_file_from_group(&dev->kobj, sysfs_remove_file_from_group(&dev->kobj,
&dev_attr_level.attr, &dev_attr_level.attr,
power_group); power_group);
...@@ -601,21 +650,21 @@ void usb_remove_sysfs_dev_files(struct usb_device *udev) ...@@ -601,21 +650,21 @@ void usb_remove_sysfs_dev_files(struct usb_device *udev)
/* Interface Accociation Descriptor fields */ /* Interface Accociation Descriptor fields */
#define usb_intf_assoc_attr(field, format_string) \ #define usb_intf_assoc_attr(field, format_string) \
static ssize_t \ static ssize_t \
show_iad_##field (struct device *dev, struct device_attribute *attr, \ show_iad_##field(struct device *dev, struct device_attribute *attr, \
char *buf) \ char *buf) \
{ \ { \
struct usb_interface *intf = to_usb_interface (dev); \ struct usb_interface *intf = to_usb_interface(dev); \
\ \
return sprintf (buf, format_string, \ return sprintf(buf, format_string, \
intf->intf_assoc->field); \ intf->intf_assoc->field); \
} \ } \
static DEVICE_ATTR(iad_##field, S_IRUGO, show_iad_##field, NULL); static DEVICE_ATTR(iad_##field, S_IRUGO, show_iad_##field, NULL);
usb_intf_assoc_attr (bFirstInterface, "%02x\n") usb_intf_assoc_attr(bFirstInterface, "%02x\n")
usb_intf_assoc_attr (bInterfaceCount, "%02d\n") usb_intf_assoc_attr(bInterfaceCount, "%02d\n")
usb_intf_assoc_attr (bFunctionClass, "%02x\n") usb_intf_assoc_attr(bFunctionClass, "%02x\n")
usb_intf_assoc_attr (bFunctionSubClass, "%02x\n") usb_intf_assoc_attr(bFunctionSubClass, "%02x\n")
usb_intf_assoc_attr (bFunctionProtocol, "%02x\n") usb_intf_assoc_attr(bFunctionProtocol, "%02x\n")
/* Interface fields */ /* Interface fields */
#define usb_intf_attr(field, format_string) \ #define usb_intf_attr(field, format_string) \
......
...@@ -42,6 +42,7 @@ void usb_init_urb(struct urb *urb) ...@@ -42,6 +42,7 @@ void usb_init_urb(struct urb *urb)
INIT_LIST_HEAD(&urb->anchor_list); INIT_LIST_HEAD(&urb->anchor_list);
} }
} }
EXPORT_SYMBOL_GPL(usb_init_urb);
/** /**
* usb_alloc_urb - creates a new urb for a USB driver to use * usb_alloc_urb - creates a new urb for a USB driver to use
...@@ -73,6 +74,7 @@ struct urb *usb_alloc_urb(int iso_packets, gfp_t mem_flags) ...@@ -73,6 +74,7 @@ struct urb *usb_alloc_urb(int iso_packets, gfp_t mem_flags)
usb_init_urb(urb); usb_init_urb(urb);
return urb; return urb;
} }
EXPORT_SYMBOL_GPL(usb_alloc_urb);
/** /**
* usb_free_urb - frees the memory used by a urb when all users of it are finished * usb_free_urb - frees the memory used by a urb when all users of it are finished
...@@ -89,6 +91,7 @@ void usb_free_urb(struct urb *urb) ...@@ -89,6 +91,7 @@ void usb_free_urb(struct urb *urb)
if (urb) if (urb)
kref_put(&urb->kref, urb_destroy); kref_put(&urb->kref, urb_destroy);
} }
EXPORT_SYMBOL_GPL(usb_free_urb);
/** /**
* usb_get_urb - increments the reference count of the urb * usb_get_urb - increments the reference count of the urb
...@@ -100,12 +103,13 @@ void usb_free_urb(struct urb *urb) ...@@ -100,12 +103,13 @@ void usb_free_urb(struct urb *urb)
* *
* A pointer to the urb with the incremented reference counter is returned. * A pointer to the urb with the incremented reference counter is returned.
*/ */
struct urb * usb_get_urb(struct urb *urb) struct urb *usb_get_urb(struct urb *urb)
{ {
if (urb) if (urb)
kref_get(&urb->kref); kref_get(&urb->kref);
return urb; return urb;
} }
EXPORT_SYMBOL_GPL(usb_get_urb);
/** /**
* usb_anchor_urb - anchors an URB while it is processed * usb_anchor_urb - anchors an URB while it is processed
...@@ -172,7 +176,7 @@ EXPORT_SYMBOL_GPL(usb_unanchor_urb); ...@@ -172,7 +176,7 @@ EXPORT_SYMBOL_GPL(usb_unanchor_urb);
* describing that request to the USB subsystem. Request completion will * describing that request to the USB subsystem. Request completion will
* be indicated later, asynchronously, by calling the completion handler. * be indicated later, asynchronously, by calling the completion handler.
* The three types of completion are success, error, and unlink * The three types of completion are success, error, and unlink
* (a software-induced fault, also called "request cancellation"). * (a software-induced fault, also called "request cancellation").
* *
* URBs may be submitted in interrupt context. * URBs may be submitted in interrupt context.
* *
...@@ -255,7 +259,7 @@ EXPORT_SYMBOL_GPL(usb_unanchor_urb); ...@@ -255,7 +259,7 @@ EXPORT_SYMBOL_GPL(usb_unanchor_urb);
* semaphores), or * semaphores), or
* (c) current->state != TASK_RUNNING, this is the case only after * (c) current->state != TASK_RUNNING, this is the case only after
* you've changed it. * you've changed it.
* *
* GFP_NOIO is used in the block io path and error handling of storage * GFP_NOIO is used in the block io path and error handling of storage
* devices. * devices.
* *
...@@ -284,7 +288,8 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags) ...@@ -284,7 +288,8 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags)
if (!urb || urb->hcpriv || !urb->complete) if (!urb || urb->hcpriv || !urb->complete)
return -EINVAL; return -EINVAL;
if (!(dev = urb->dev) || dev->state < USB_STATE_DEFAULT) dev = urb->dev;
if ((!dev) || (dev->state < USB_STATE_DEFAULT))
return -ENODEV; return -ENODEV;
/* For now, get the endpoint from the pipe. Eventually drivers /* For now, get the endpoint from the pipe. Eventually drivers
...@@ -347,11 +352,11 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags) ...@@ -347,11 +352,11 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags)
max *= mult; max *= mult;
} }
if (urb->number_of_packets <= 0) if (urb->number_of_packets <= 0)
return -EINVAL; return -EINVAL;
for (n = 0; n < urb->number_of_packets; n++) { for (n = 0; n < urb->number_of_packets; n++) {
len = urb->iso_frame_desc[n].length; len = urb->iso_frame_desc[n].length;
if (len < 0 || len > max) if (len < 0 || len > max)
return -EMSGSIZE; return -EMSGSIZE;
urb->iso_frame_desc[n].status = -EXDEV; urb->iso_frame_desc[n].status = -EXDEV;
urb->iso_frame_desc[n].actual_length = 0; urb->iso_frame_desc[n].actual_length = 0;
...@@ -416,7 +421,7 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags) ...@@ -416,7 +421,7 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags)
/* too big? */ /* too big? */
switch (dev->speed) { switch (dev->speed) {
case USB_SPEED_HIGH: /* units are microframes */ case USB_SPEED_HIGH: /* units are microframes */
// NOTE usb handles 2^15 /* NOTE usb handles 2^15 */
if (urb->interval > (1024 * 8)) if (urb->interval > (1024 * 8))
urb->interval = 1024 * 8; urb->interval = 1024 * 8;
max = 1024 * 8; max = 1024 * 8;
...@@ -426,12 +431,12 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags) ...@@ -426,12 +431,12 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags)
if (xfertype == USB_ENDPOINT_XFER_INT) { if (xfertype == USB_ENDPOINT_XFER_INT) {
if (urb->interval > 255) if (urb->interval > 255)
return -EINVAL; return -EINVAL;
// NOTE ohci only handles up to 32 /* NOTE ohci only handles up to 32 */
max = 128; max = 128;
} else { } else {
if (urb->interval > 1024) if (urb->interval > 1024)
urb->interval = 1024; urb->interval = 1024;
// NOTE usb and ohci handle up to 2^15 /* NOTE usb and ohci handle up to 2^15 */
max = 1024; max = 1024;
} }
break; break;
...@@ -444,6 +449,7 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags) ...@@ -444,6 +449,7 @@ int usb_submit_urb(struct urb *urb, gfp_t mem_flags)
return usb_hcd_submit_urb(urb, mem_flags); return usb_hcd_submit_urb(urb, mem_flags);
} }
EXPORT_SYMBOL_GPL(usb_submit_urb);
/*-------------------------------------------------------------------*/ /*-------------------------------------------------------------------*/
...@@ -514,6 +520,7 @@ int usb_unlink_urb(struct urb *urb) ...@@ -514,6 +520,7 @@ int usb_unlink_urb(struct urb *urb)
return -EIDRM; return -EIDRM;
return usb_hcd_unlink_urb(urb, -ECONNRESET); return usb_hcd_unlink_urb(urb, -ECONNRESET);
} }
EXPORT_SYMBOL_GPL(usb_unlink_urb);
/** /**
* usb_kill_urb - cancel a transfer request and wait for it to finish * usb_kill_urb - cancel a transfer request and wait for it to finish
...@@ -553,6 +560,7 @@ void usb_kill_urb(struct urb *urb) ...@@ -553,6 +560,7 @@ void usb_kill_urb(struct urb *urb)
--urb->reject; --urb->reject;
mutex_unlock(&reject_mutex); mutex_unlock(&reject_mutex);
} }
EXPORT_SYMBOL_GPL(usb_kill_urb);
/** /**
* usb_kill_anchored_urbs - cancel transfer requests en masse * usb_kill_anchored_urbs - cancel transfer requests en masse
...@@ -567,7 +575,8 @@ void usb_kill_anchored_urbs(struct usb_anchor *anchor) ...@@ -567,7 +575,8 @@ void usb_kill_anchored_urbs(struct usb_anchor *anchor)
spin_lock_irq(&anchor->lock); spin_lock_irq(&anchor->lock);
while (!list_empty(&anchor->urb_list)) { while (!list_empty(&anchor->urb_list)) {
victim = list_entry(anchor->urb_list.prev, struct urb, anchor_list); victim = list_entry(anchor->urb_list.prev, struct urb,
anchor_list);
/* we must make sure the URB isn't freed before we kill it*/ /* we must make sure the URB isn't freed before we kill it*/
usb_get_urb(victim); usb_get_urb(victim);
spin_unlock_irq(&anchor->lock); spin_unlock_irq(&anchor->lock);
...@@ -595,11 +604,3 @@ int usb_wait_anchor_empty_timeout(struct usb_anchor *anchor, ...@@ -595,11 +604,3 @@ int usb_wait_anchor_empty_timeout(struct usb_anchor *anchor,
msecs_to_jiffies(timeout)); msecs_to_jiffies(timeout));
} }
EXPORT_SYMBOL_GPL(usb_wait_anchor_empty_timeout); EXPORT_SYMBOL_GPL(usb_wait_anchor_empty_timeout);
EXPORT_SYMBOL(usb_init_urb);
EXPORT_SYMBOL(usb_alloc_urb);
EXPORT_SYMBOL(usb_free_urb);
EXPORT_SYMBOL(usb_get_urb);
EXPORT_SYMBOL(usb_submit_urb);
EXPORT_SYMBOL(usb_unlink_urb);
EXPORT_SYMBOL(usb_kill_urb);
This diff is collapsed.
/* Functions local to drivers/usb/core/ */ /* Functions local to drivers/usb/core/ */
extern int usb_create_sysfs_dev_files (struct usb_device *dev); extern int usb_create_sysfs_dev_files(struct usb_device *dev);
extern void usb_remove_sysfs_dev_files (struct usb_device *dev); extern void usb_remove_sysfs_dev_files(struct usb_device *dev);
extern int usb_create_sysfs_intf_files (struct usb_interface *intf); extern int usb_create_sysfs_intf_files(struct usb_interface *intf);
extern void usb_remove_sysfs_intf_files (struct usb_interface *intf); extern void usb_remove_sysfs_intf_files(struct usb_interface *intf);
extern int usb_create_ep_files(struct device *parent, struct usb_host_endpoint *endpoint, extern int usb_create_ep_files(struct device *parent,
struct usb_host_endpoint *endpoint,
struct usb_device *udev); struct usb_device *udev);
extern void usb_remove_ep_files(struct usb_host_endpoint *endpoint); extern void usb_remove_ep_files(struct usb_host_endpoint *endpoint);
extern void usb_enable_endpoint(struct usb_device *dev, extern void usb_enable_endpoint(struct usb_device *dev,
struct usb_host_endpoint *ep); struct usb_host_endpoint *ep);
extern void usb_disable_endpoint (struct usb_device *dev, unsigned int epaddr); extern void usb_disable_endpoint(struct usb_device *dev, unsigned int epaddr);
extern void usb_disable_interface (struct usb_device *dev, extern void usb_disable_interface(struct usb_device *dev,
struct usb_interface *intf); struct usb_interface *intf);
extern void usb_release_interface_cache(struct kref *ref); extern void usb_release_interface_cache(struct kref *ref);
extern void usb_disable_device (struct usb_device *dev, int skip_ep0); extern void usb_disable_device(struct usb_device *dev, int skip_ep0);
extern int usb_deauthorize_device (struct usb_device *); extern int usb_deauthorize_device(struct usb_device *);
extern int usb_authorize_device (struct usb_device *); extern int usb_authorize_device(struct usb_device *);
extern void usb_detect_quirks(struct usb_device *udev); extern void usb_detect_quirks(struct usb_device *udev);
extern int usb_get_device_descriptor(struct usb_device *dev, extern int usb_get_device_descriptor(struct usb_device *dev,
......
...@@ -12,10 +12,9 @@ ...@@ -12,10 +12,9 @@
# With help from a special transceiver and a "Mini-AB" jack, systems with # With help from a special transceiver and a "Mini-AB" jack, systems with
# both kinds of controller can also support "USB On-the-Go" (CONFIG_USB_OTG). # both kinds of controller can also support "USB On-the-Go" (CONFIG_USB_OTG).
# #
menu "USB Gadget Support"
config USB_GADGET menuconfig USB_GADGET
tristate "Support for USB Gadgets" tristate "USB Gadget Support"
help help
USB is a master/slave protocol, organized with one master USB is a master/slave protocol, organized with one master
host (such as a PC) controlling up to 127 peripheral devices. host (such as a PC) controlling up to 127 peripheral devices.
...@@ -42,6 +41,8 @@ config USB_GADGET ...@@ -42,6 +41,8 @@ config USB_GADGET
For more information, see <http://www.linux-usb.org/gadget> and For more information, see <http://www.linux-usb.org/gadget> and
the kernel DocBook documentation for this API. the kernel DocBook documentation for this API.
if USB_GADGET
config USB_GADGET_DEBUG config USB_GADGET_DEBUG
boolean "Debugging messages" boolean "Debugging messages"
depends on USB_GADGET && DEBUG_KERNEL && EXPERIMENTAL depends on USB_GADGET && DEBUG_KERNEL && EXPERIMENTAL
...@@ -220,6 +221,16 @@ config USB_M66592 ...@@ -220,6 +221,16 @@ config USB_M66592
default USB_GADGET default USB_GADGET
select USB_GADGET_SELECTED select USB_GADGET_SELECTED
config SUPERH_BUILT_IN_M66592
boolean "Enable SuperH built-in USB like the M66592"
depends on USB_GADGET_M66592 && CPU_SUBTYPE_SH7722
help
SH7722 has USB like the M66592.
The transfer rate is very slow when use "Ethernet Gadget".
However, this problem is improved if change a value of
NET_IP_ALIGN to 4.
config USB_GADGET_GOKU config USB_GADGET_GOKU
boolean "Toshiba TC86C001 'Goku-S'" boolean "Toshiba TC86C001 'Goku-S'"
depends on PCI depends on PCI
...@@ -538,6 +549,20 @@ config USB_MIDI_GADGET ...@@ -538,6 +549,20 @@ config USB_MIDI_GADGET
Say "y" to link the driver statically, or "m" to build a Say "y" to link the driver statically, or "m" to build a
dynamically linked module called "g_midi". dynamically linked module called "g_midi".
config USB_G_PRINTER
tristate "Printer Gadget"
help
The Printer Gadget channels data between the USB host and a
userspace program driving the print engine. The user space
program reads and writes the device file /dev/g_printer to
receive or send printer data. It can use ioctl calls to
the device file to get or set printer status.
Say "y" to link the driver statically, or "m" to build a
dynamically linked module called "g_printer".
For more information, see Documentation/usb/gadget_printer.txt
which includes sample code for accessing the device file.
# put drivers that need isochronous transfer support (for audio # put drivers that need isochronous transfer support (for audio
# or video class gadget drivers), or specific hardware, here. # or video class gadget drivers), or specific hardware, here.
...@@ -546,4 +571,4 @@ config USB_MIDI_GADGET ...@@ -546,4 +571,4 @@ config USB_MIDI_GADGET
endchoice endchoice
endmenu endif # USB_GADGET
...@@ -28,6 +28,8 @@ g_midi-objs := gmidi.o usbstring.o config.o epautoconf.o ...@@ -28,6 +28,8 @@ g_midi-objs := gmidi.o usbstring.o config.o epautoconf.o
gadgetfs-objs := inode.o gadgetfs-objs := inode.o
g_file_storage-objs := file_storage.o usbstring.o config.o \ g_file_storage-objs := file_storage.o usbstring.o config.o \
epautoconf.o epautoconf.o
g_printer-objs := printer.o usbstring.o config.o \
epautoconf.o
ifeq ($(CONFIG_USB_ETH_RNDIS),y) ifeq ($(CONFIG_USB_ETH_RNDIS),y)
g_ether-objs += rndis.o g_ether-objs += rndis.o
...@@ -38,5 +40,6 @@ obj-$(CONFIG_USB_ETH) += g_ether.o ...@@ -38,5 +40,6 @@ obj-$(CONFIG_USB_ETH) += g_ether.o
obj-$(CONFIG_USB_GADGETFS) += gadgetfs.o obj-$(CONFIG_USB_GADGETFS) += gadgetfs.o
obj-$(CONFIG_USB_FILE_STORAGE) += g_file_storage.o obj-$(CONFIG_USB_FILE_STORAGE) += g_file_storage.o
obj-$(CONFIG_USB_G_SERIAL) += g_serial.o obj-$(CONFIG_USB_G_SERIAL) += g_serial.o
obj-$(CONFIG_USB_G_PRINTER) += g_printer.o
obj-$(CONFIG_USB_MIDI_GADGET) += g_midi.o obj-$(CONFIG_USB_MIDI_GADGET) += g_midi.o
...@@ -1244,7 +1244,7 @@ udc_queue(struct usb_ep *usbep, struct usb_request *usbreq, gfp_t gfp) ...@@ -1244,7 +1244,7 @@ udc_queue(struct usb_ep *usbep, struct usb_request *usbreq, gfp_t gfp)
/* stop OUT naking */ /* stop OUT naking */
if (!ep->in) { if (!ep->in) {
if (!use_dma && udc_rxfifo_pending) { if (!use_dma && udc_rxfifo_pending) {
DBG(dev, "udc_queue(): pending bytes in" DBG(dev, "udc_queue(): pending bytes in "
"rxfifo after nyet\n"); "rxfifo after nyet\n");
/* /*
* read pending bytes afer nyet: * read pending bytes afer nyet:
...@@ -2038,6 +2038,7 @@ int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) ...@@ -2038,6 +2038,7 @@ int usb_gadget_unregister_driver(struct usb_gadget_driver *driver)
spin_unlock_irqrestore(&dev->lock, flags); spin_unlock_irqrestore(&dev->lock, flags);
driver->unbind(&dev->gadget); driver->unbind(&dev->gadget);
dev->gadget.dev.driver = NULL;
dev->driver = NULL; dev->driver = NULL;
/* set SD */ /* set SD */
......
This diff is collapsed.
...@@ -53,7 +53,7 @@ ...@@ -53,7 +53,7 @@
#define AT91_UDP_RXRSM (1 << 9) /* USB Resume Interrupt Status */ #define AT91_UDP_RXRSM (1 << 9) /* USB Resume Interrupt Status */
#define AT91_UDP_EXTRSM (1 << 10) /* External Resume Interrupt Status [AT91RM9200 only] */ #define AT91_UDP_EXTRSM (1 << 10) /* External Resume Interrupt Status [AT91RM9200 only] */
#define AT91_UDP_SOFINT (1 << 11) /* Start of Frame Interrupt Status */ #define AT91_UDP_SOFINT (1 << 11) /* Start of Frame Interrupt Status */
#define AT91_UDP_ENDBUSRES (1 << 12) /* End of Bus Reset Interrpt Status */ #define AT91_UDP_ENDBUSRES (1 << 12) /* End of Bus Reset Interrupt Status */
#define AT91_UDP_WAKEUP (1 << 13) /* USB Wakeup Interrupt Status [AT91RM9200 only] */ #define AT91_UDP_WAKEUP (1 << 13) /* USB Wakeup Interrupt Status [AT91RM9200 only] */
#define AT91_UDP_ICR 0x20 /* Interrupt Clear Register */ #define AT91_UDP_ICR 0x20 /* Interrupt Clear Register */
...@@ -158,13 +158,7 @@ struct at91_request { ...@@ -158,13 +158,7 @@ struct at91_request {
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
#ifdef DEBUG #ifdef VERBOSE_DEBUG
#define DBG(stuff...) printk(KERN_DEBUG "udc: " stuff)
#else
#define DBG(stuff...) do{}while(0)
#endif
#ifdef VERBOSE
# define VDBG DBG # define VDBG DBG
#else #else
# define VDBG(stuff...) do{}while(0) # define VDBG(stuff...) do{}while(0)
...@@ -176,9 +170,10 @@ struct at91_request { ...@@ -176,9 +170,10 @@ struct at91_request {
# define PACKET(stuff...) do{}while(0) # define PACKET(stuff...) do{}while(0)
#endif #endif
#define ERR(stuff...) printk(KERN_ERR "udc: " stuff) #define ERR(stuff...) pr_err("udc: " stuff)
#define WARN(stuff...) printk(KERN_WARNING "udc: " stuff) #define WARN(stuff...) pr_warning("udc: " stuff)
#define INFO(stuff...) printk(KERN_INFO "udc: " stuff) #define INFO(stuff...) pr_info("udc: " stuff)
#define DBG(stuff...) pr_debug("udc: " stuff)
#endif #endif
...@@ -1384,8 +1384,7 @@ static int handle_ep0_setup(struct usba_udc *udc, struct usba_ep *ep, ...@@ -1384,8 +1384,7 @@ static int handle_ep0_setup(struct usba_udc *udc, struct usba_ep *ep,
return retval; return retval;
stall: stall:
printk(KERN_ERR pr_err("udc: %s: Invalid setup request: %02x.%02x v%04x i%04x l%d, "
"udc: %s: Invalid setup request: %02x.%02x v%04x i%04x l%d, "
"halting endpoint...\n", "halting endpoint...\n",
ep->ep.name, crq->bRequestType, crq->bRequest, ep->ep.name, crq->bRequestType, crq->bRequest,
le16_to_cpu(crq->wValue), le16_to_cpu(crq->wIndex), le16_to_cpu(crq->wValue), le16_to_cpu(crq->wIndex),
...@@ -1456,8 +1455,7 @@ static void usba_control_irq(struct usba_udc *udc, struct usba_ep *ep) ...@@ -1456,8 +1455,7 @@ static void usba_control_irq(struct usba_udc *udc, struct usba_ep *ep)
set_protocol_stall(udc, ep); set_protocol_stall(udc, ep);
break; break;
default: default:
printk(KERN_ERR pr_err("udc: %s: TXCOMP: Invalid endpoint state %d, "
"udc: %s: TXCOMP: Invalid endpoint state %d, "
"halting endpoint...\n", "halting endpoint...\n",
ep->ep.name, ep->state); ep->ep.name, ep->state);
set_protocol_stall(udc, ep); set_protocol_stall(udc, ep);
...@@ -1486,8 +1484,7 @@ static void usba_control_irq(struct usba_udc *udc, struct usba_ep *ep) ...@@ -1486,8 +1484,7 @@ static void usba_control_irq(struct usba_udc *udc, struct usba_ep *ep)
default: default:
usba_ep_writel(ep, CLR_STA, USBA_RX_BK_RDY); usba_ep_writel(ep, CLR_STA, USBA_RX_BK_RDY);
usba_ep_writel(ep, CTL_DIS, USBA_RX_BK_RDY); usba_ep_writel(ep, CTL_DIS, USBA_RX_BK_RDY);
printk(KERN_ERR pr_err("udc: %s: RXRDY: Invalid endpoint state %d, "
"udc: %s: RXRDY: Invalid endpoint state %d, "
"halting endpoint...\n", "halting endpoint...\n",
ep->ep.name, ep->state); ep->ep.name, ep->state);
set_protocol_stall(udc, ep); set_protocol_stall(udc, ep);
...@@ -1532,7 +1529,7 @@ static void usba_control_irq(struct usba_udc *udc, struct usba_ep *ep) ...@@ -1532,7 +1529,7 @@ static void usba_control_irq(struct usba_udc *udc, struct usba_ep *ep)
pkt_len = USBA_BFEXT(BYTE_COUNT, usba_ep_readl(ep, STA)); pkt_len = USBA_BFEXT(BYTE_COUNT, usba_ep_readl(ep, STA));
DBG(DBG_HW, "Packet length: %u\n", pkt_len); DBG(DBG_HW, "Packet length: %u\n", pkt_len);
if (pkt_len != sizeof(crq)) { if (pkt_len != sizeof(crq)) {
printk(KERN_WARNING "udc: Invalid packet length %u " pr_warning("udc: Invalid packet length %u "
"(expected %lu)\n", pkt_len, sizeof(crq)); "(expected %lu)\n", pkt_len, sizeof(crq));
set_protocol_stall(udc, ep); set_protocol_stall(udc, ep);
return; return;
......
...@@ -216,7 +216,6 @@ ...@@ -216,7 +216,6 @@
#define FIFO_IOMEM_ID 0 #define FIFO_IOMEM_ID 0
#define CTRL_IOMEM_ID 1 #define CTRL_IOMEM_ID 1
#ifdef DEBUG
#define DBG_ERR 0x0001 /* report all error returns */ #define DBG_ERR 0x0001 /* report all error returns */
#define DBG_HW 0x0002 /* debug hardware initialization */ #define DBG_HW 0x0002 /* debug hardware initialization */
#define DBG_GADGET 0x0004 /* calls to/from gadget driver */ #define DBG_GADGET 0x0004 /* calls to/from gadget driver */
...@@ -230,14 +229,12 @@ ...@@ -230,14 +229,12 @@
#define DBG_NONE 0x0000 #define DBG_NONE 0x0000
#define DEBUG_LEVEL (DBG_ERR) #define DEBUG_LEVEL (DBG_ERR)
#define DBG(level, fmt, ...) \ #define DBG(level, fmt, ...) \
do { \ do { \
if ((level) & DEBUG_LEVEL) \ if ((level) & DEBUG_LEVEL) \
printk(KERN_DEBUG "udc: " fmt, ## __VA_ARGS__); \ pr_debug("udc: " fmt, ## __VA_ARGS__); \
} while (0) } while (0)
#else
#define DBG(level, fmt...)
#endif
enum usba_ctrl_state { enum usba_ctrl_state {
WAIT_FOR_SETUP, WAIT_FOR_SETUP,
......
This diff is collapsed.
...@@ -1067,19 +1067,19 @@ set_ether_config (struct eth_dev *dev, gfp_t gfp_flags) ...@@ -1067,19 +1067,19 @@ set_ether_config (struct eth_dev *dev, gfp_t gfp_flags)
/* on error, disable any endpoints */ /* on error, disable any endpoints */
if (result < 0) { if (result < 0) {
if (!subset_active(dev)) if (!subset_active(dev) && dev->status_ep)
(void) usb_ep_disable (dev->status_ep); (void) usb_ep_disable (dev->status_ep);
dev->status = NULL; dev->status = NULL;
(void) usb_ep_disable (dev->in_ep); (void) usb_ep_disable (dev->in_ep);
(void) usb_ep_disable (dev->out_ep); (void) usb_ep_disable (dev->out_ep);
dev->in = NULL; dev->in = NULL;
dev->out = NULL; dev->out = NULL;
} else }
/* activate non-CDC configs right away /* activate non-CDC configs right away
* this isn't strictly according to the RNDIS spec * this isn't strictly according to the RNDIS spec
*/ */
if (!cdc_active (dev)) { else if (!cdc_active (dev)) {
netif_carrier_on (dev->net); netif_carrier_on (dev->net);
if (netif_running (dev->net)) { if (netif_running (dev->net)) {
spin_unlock (&dev->lock); spin_unlock (&dev->lock);
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -1422,6 +1422,7 @@ int usb_gadget_unregister_driver(struct usb_gadget_driver *driver) ...@@ -1422,6 +1422,7 @@ int usb_gadget_unregister_driver(struct usb_gadget_driver *driver)
spin_unlock_irqrestore(&dev->lock, flags); spin_unlock_irqrestore(&dev->lock, flags);
driver->unbind(&dev->gadget); driver->unbind(&dev->gadget);
dev->gadget.dev.driver = NULL;
DBG(dev, "unregistered driver '%s'\n", driver->driver.name); DBG(dev, "unregistered driver '%s'\n", driver->driver.name);
return 0; return 0;
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment