2008-11-25 22:39:18 +01:00
|
|
|
#include <linux/pm.h>
|
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/* Functions local to drivers/usb/core/ */
|
|
|
|
|
2008-01-31 00:21:33 +01:00
|
|
|
extern int usb_create_sysfs_dev_files(struct usb_device *dev);
|
|
|
|
extern void usb_remove_sysfs_dev_files(struct usb_device *dev);
|
2011-04-14 17:47:09 +02:00
|
|
|
extern void usb_create_sysfs_intf_files(struct usb_interface *intf);
|
2008-01-31 00:21:33 +01:00
|
|
|
extern void usb_remove_sysfs_intf_files(struct usb_interface *intf);
|
2008-12-05 20:10:34 +01:00
|
|
|
extern int usb_create_ep_devs(struct device *parent,
|
2008-01-31 00:21:33 +01:00
|
|
|
struct usb_host_endpoint *endpoint,
|
2006-06-14 21:14:34 +02:00
|
|
|
struct usb_device *udev);
|
2008-12-05 20:10:34 +01:00
|
|
|
extern void usb_remove_ep_devs(struct usb_host_endpoint *endpoint);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2007-07-30 23:05:22 +02:00
|
|
|
extern void usb_enable_endpoint(struct usb_device *dev,
|
2008-12-31 17:31:33 +01:00
|
|
|
struct usb_host_endpoint *ep, bool reset_toggle);
|
|
|
|
extern void usb_enable_interface(struct usb_device *dev,
|
|
|
|
struct usb_interface *intf, bool reset_toggles);
|
2009-01-15 23:03:33 +01:00
|
|
|
extern void usb_disable_endpoint(struct usb_device *dev, unsigned int epaddr,
|
|
|
|
bool reset_hardware);
|
2008-01-31 00:21:33 +01:00
|
|
|
extern void usb_disable_interface(struct usb_device *dev,
|
2009-01-15 23:03:33 +01:00
|
|
|
struct usb_interface *intf, bool reset_hardware);
|
2005-04-17 00:20:36 +02:00
|
|
|
extern void usb_release_interface_cache(struct kref *ref);
|
2008-01-31 00:21:33 +01:00
|
|
|
extern void usb_disable_device(struct usb_device *dev, int skip_ep0);
|
|
|
|
extern int usb_deauthorize_device(struct usb_device *);
|
|
|
|
extern int usb_authorize_device(struct usb_device *);
|
2007-01-26 14:26:21 +01:00
|
|
|
extern void usb_detect_quirks(struct usb_device *udev);
|
2009-10-27 20:20:13 +01:00
|
|
|
extern int usb_remove_device(struct usb_device *udev);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
extern int usb_get_device_descriptor(struct usb_device *dev,
|
|
|
|
unsigned int size);
|
2011-09-23 23:19:47 +02:00
|
|
|
extern int usb_get_bos_descriptor(struct usb_device *dev);
|
|
|
|
extern void usb_release_bos_descriptor(struct usb_device *dev);
|
2005-10-24 22:24:14 +02:00
|
|
|
extern char *usb_cache_string(struct usb_device *udev, int index);
|
2005-04-17 00:20:36 +02:00
|
|
|
extern int usb_set_configuration(struct usb_device *dev, int configuration);
|
2007-08-03 06:44:27 +02:00
|
|
|
extern int usb_choose_configuration(struct usb_device *udev);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
|
|
|
extern void usb_kick_khubd(struct usb_device *dev);
|
2007-01-26 14:26:21 +01:00
|
|
|
extern int usb_match_device(struct usb_device *dev,
|
|
|
|
const struct usb_device_id *id);
|
2008-06-23 22:00:40 +02:00
|
|
|
extern void usb_forced_unbind_intf(struct usb_interface *intf);
|
|
|
|
extern void usb_rebind_intf(struct usb_interface *intf);
|
2005-04-17 00:20:36 +02:00
|
|
|
|
2009-06-29 16:56:54 +02:00
|
|
|
extern int usb_hub_claim_port(struct usb_device *hdev, unsigned port,
|
|
|
|
void *owner);
|
|
|
|
extern int usb_hub_release_port(struct usb_device *hdev, unsigned port,
|
|
|
|
void *owner);
|
|
|
|
extern void usb_hub_release_all_ports(struct usb_device *hdev, void *owner);
|
|
|
|
extern bool usb_device_is_owned(struct usb_device *udev);
|
|
|
|
|
2005-04-19 02:39:24 +02:00
|
|
|
extern int usb_hub_init(void);
|
|
|
|
extern void usb_hub_cleanup(void);
|
|
|
|
extern int usb_major_init(void);
|
|
|
|
extern void usb_major_cleanup(void);
|
|
|
|
|
2006-07-02 04:14:24 +02:00
|
|
|
#ifdef CONFIG_PM
|
|
|
|
|
2008-08-12 20:34:10 +02:00
|
|
|
extern int usb_suspend(struct device *dev, pm_message_t msg);
|
2008-11-25 22:39:18 +01:00
|
|
|
extern int usb_resume(struct device *dev, pm_message_t msg);
|
2008-08-12 20:34:10 +02:00
|
|
|
|
2008-11-25 22:39:18 +01:00
|
|
|
extern int usb_port_suspend(struct usb_device *dev, pm_message_t msg);
|
|
|
|
extern int usb_port_resume(struct usb_device *dev, pm_message_t msg);
|
2006-09-26 20:50:20 +02:00
|
|
|
|
2006-07-02 04:14:24 +02:00
|
|
|
#else
|
|
|
|
|
2008-11-25 22:39:18 +01:00
|
|
|
static inline int usb_port_suspend(struct usb_device *udev, pm_message_t msg)
|
2007-05-30 22:51:28 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-11-25 22:39:18 +01:00
|
|
|
static inline int usb_port_resume(struct usb_device *udev, pm_message_t msg)
|
2007-05-30 22:51:28 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-07-02 04:14:24 +02:00
|
|
|
#endif
|
|
|
|
|
2006-08-30 21:47:02 +02:00
|
|
|
#ifdef CONFIG_USB_SUSPEND
|
|
|
|
|
2006-11-20 17:38:46 +01:00
|
|
|
extern void usb_autosuspend_device(struct usb_device *udev);
|
|
|
|
extern int usb_autoresume_device(struct usb_device *udev);
|
2010-01-08 18:57:28 +01:00
|
|
|
extern int usb_remote_wakeup(struct usb_device *dev);
|
2011-03-18 19:55:36 +01:00
|
|
|
extern int usb_runtime_suspend(struct device *dev);
|
|
|
|
extern int usb_runtime_resume(struct device *dev);
|
|
|
|
extern int usb_runtime_idle(struct device *dev);
|
2006-08-30 21:47:02 +02:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
2007-02-20 21:03:32 +01:00
|
|
|
#define usb_autosuspend_device(udev) do {} while (0)
|
2006-11-20 17:38:46 +01:00
|
|
|
static inline int usb_autoresume_device(struct usb_device *udev)
|
2006-09-26 20:50:20 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2006-08-30 21:47:02 +02:00
|
|
|
|
2010-01-08 18:57:28 +01:00
|
|
|
static inline int usb_remote_wakeup(struct usb_device *udev)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-08-30 21:47:02 +02:00
|
|
|
#endif
|
|
|
|
|
2006-07-02 04:08:06 +02:00
|
|
|
extern struct bus_type usb_bus_type;
|
2007-03-13 15:59:31 +01:00
|
|
|
extern struct device_type usb_device_type;
|
|
|
|
extern struct device_type usb_if_device_type;
|
2009-05-04 19:48:32 +02:00
|
|
|
extern struct device_type usb_ep_device_type;
|
2006-07-02 04:08:49 +02:00
|
|
|
extern struct usb_device_driver usb_generic_driver;
|
|
|
|
|
2006-08-27 04:48:11 +02:00
|
|
|
static inline int is_usb_device(const struct device *dev)
|
2006-07-02 04:08:49 +02:00
|
|
|
{
|
2007-03-13 15:59:31 +01:00
|
|
|
return dev->type == &usb_device_type;
|
2006-07-02 04:08:49 +02:00
|
|
|
}
|
|
|
|
|
2009-05-04 19:48:32 +02:00
|
|
|
static inline int is_usb_interface(const struct device *dev)
|
|
|
|
{
|
|
|
|
return dev->type == &usb_if_device_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int is_usb_endpoint(const struct device *dev)
|
|
|
|
{
|
|
|
|
return dev->type == &usb_ep_device_type;
|
|
|
|
}
|
|
|
|
|
2006-07-02 04:08:49 +02:00
|
|
|
/* Do the same for device drivers and interface drivers. */
|
|
|
|
|
|
|
|
static inline int is_usb_device_driver(struct device_driver *drv)
|
|
|
|
{
|
|
|
|
return container_of(drv, struct usbdrv_wrap, driver)->
|
|
|
|
for_devices;
|
|
|
|
}
|
2005-09-23 07:45:26 +02:00
|
|
|
|
2010-12-19 21:54:48 +01:00
|
|
|
/* translate USB error codes to codes user space understands */
|
|
|
|
static inline int usb_translate_errors(int error_code)
|
|
|
|
{
|
|
|
|
switch (error_code) {
|
|
|
|
case 0:
|
|
|
|
case -ENOMEM:
|
|
|
|
case -ENODEV:
|
|
|
|
return error_code;
|
|
|
|
default:
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-09-14 04:56:33 +02:00
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/* for labeling diagnostics */
|
|
|
|
extern const char *usbcore_name;
|
|
|
|
|
2008-04-30 21:37:19 +02:00
|
|
|
/* sysfs stuff */
|
2009-06-24 19:06:31 +02:00
|
|
|
extern const struct attribute_group *usb_device_groups[];
|
|
|
|
extern const struct attribute_group *usb_interface_groups[];
|
2008-04-30 21:37:19 +02:00
|
|
|
|
2005-04-17 00:20:36 +02:00
|
|
|
/* usbfs stuff */
|
usbfs: private mutex for open, release, and remove
The usbfs code doesn't provide sufficient mutual exclusion among open,
release, and remove. Release vs. remove is okay because they both
acquire the device lock, but open is not exclusive with either one. All
three routines modify the udev->filelist linked list, so they must not
run concurrently.
Apparently someone gave this a minimum amount of thought in the past by
explicitly acquiring the BKL at the start of the usbdev_open routine.
Oddly enough, there's a comment pointing out that locking is unnecessary
because chrdev_open already has acquired the BKL.
But this ignores the point that the files in /proc/bus/usb/* are not
char device files; they are regular files and so they don't get any
special locking. Furthermore it's necessary to acquire the same lock in
the release and remove routines, which the code does not do.
Yet another problem arises because the same file_operations structure is
accessible through both the /proc/bus/usb/* and /dev/usb/usbdev* file
nodes. Even when one of them has been removed, it's still possible for
userspace to open the other. So simple locking around the individual
remove routines is insufficient; we need to lock the entire
usb_notify_remove_device notifier chain.
Rather than rely on the BKL, this patch (as723) introduces a new private
mutex for the purpose. Holding the BKL while invoking a notifier chain
doesn't seem like a good idea.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2006-07-02 04:05:01 +02:00
|
|
|
extern struct mutex usbfs_mutex;
|
2005-04-17 00:20:36 +02:00
|
|
|
extern struct usb_driver usbfs_driver;
|
2006-08-06 01:37:11 +02:00
|
|
|
extern const struct file_operations usbfs_devices_fops;
|
2007-03-13 15:59:31 +01:00
|
|
|
extern const struct file_operations usbdev_file_operations;
|
2005-04-17 00:20:36 +02:00
|
|
|
extern void usbfs_conn_disc_event(void);
|
|
|
|
|
2007-03-13 15:59:31 +01:00
|
|
|
extern int usb_devio_init(void);
|
|
|
|
extern void usb_devio_cleanup(void);
|
2005-04-19 02:39:24 +02:00
|
|
|
|
2005-06-21 06:15:16 +02:00
|
|
|
/* internal notify stuff */
|
|
|
|
extern void usb_notify_add_device(struct usb_device *udev);
|
|
|
|
extern void usb_notify_remove_device(struct usb_device *udev);
|
|
|
|
extern void usb_notify_add_bus(struct usb_bus *ubus);
|
|
|
|
extern void usb_notify_remove_bus(struct usb_bus *ubus);
|
|
|
|
|