Commit 5328f35b authored by Greg Kroah-Hartman's avatar Greg Kroah-Hartman

Merge tag 'usb-for-v3.13' of git://git.kernel.org/pub/scm/linux/kernel/git/balbi/usb into usb-next

Felipe writes:

usb: patches for v3.13

Final conversions to configfs for mass storage, acm_ms, and
multi gadgets.

MUSB should now work out of the box on AM335x-based boards
(beagle bone white and black) with DMA thanks to Sebastian's
work.

We can now enable VERBOSE_DEBUG on builds of drivers/usb/gadget/
by selecting CONFIG_USB_GADGET_VERBOSE.

s3c-hsotg got quite a few non-critical fixes but also learned
a few new tricks (isochronous transfers, multi count support).

The Marvel USB3 Controller driver got a memory leak fix.

devm_usb_get_phy() learned not to return NULL, ever.

Other than these patches, we have the usual set of cleanups
ranging from removal of unnecessary *_set_drvdata() to using
SIMPLE_DEV_PM_OPS.
Signed-of-by: default avatarFelipe Balbi <balbi@ti.com>
parents e3967e7b 80d7d8a7
What: /config/usb-gadget/gadget/functions/mass_storage.name
Date: Oct 2013
KenelVersion: 3.13
Description:
The attributes:
stall - Set to permit function to halt bulk endpoints.
Disabled on some USB devices known not to work
correctly. You should set it to true.
num_buffers - Number of pipeline buffers. Valid numbers
are 2..4. Available only if
CONFIG_USB_GADGET_DEBUG_FILES is set.
What: /config/usb-gadget/gadget/functions/mass_storage.name/lun.name
Date: Oct 2013
KenelVersion: 3.13
Description:
The attributes:
file - The path to the backing file for the LUN.
Required if LUN is not marked as removable.
ro - Flag specifying access to the LUN shall be
read-only. This is implied if CD-ROM emulation
is enabled as well as when it was impossible
to open "filename" in R/W mode.
removable - Flag specifying that LUN shall be indicated as
being removable.
cdrom - Flag specifying that LUN shall be reported as
being a CD-ROM.
nofua - Flag specifying that FUA flag
in SCSI WRITE(10,12)
...@@ -15,7 +15,7 @@ Optional properties: ...@@ -15,7 +15,7 @@ Optional properties:
- vcc-supply: phandle to the regulator that provides RESET to the PHY. - vcc-supply: phandle to the regulator that provides RESET to the PHY.
- reset-supply: phandle to the regulator that provides power to the PHY. - reset-gpios: Should specify the GPIO for reset.
Example: Example:
...@@ -25,10 +25,9 @@ Example: ...@@ -25,10 +25,9 @@ Example:
clocks = <&osc 0>; clocks = <&osc 0>;
clock-names = "main_clk"; clock-names = "main_clk";
vcc-supply = <&hsusb1_vcc_regulator>; vcc-supply = <&hsusb1_vcc_regulator>;
reset-supply = <&hsusb1_reset_regulator>; reset-gpios = <&gpio1 7 GPIO_ACTIVE_LOW>;
}; };
hsusb1_phy is a NOP USB PHY device that gets its clock from an oscillator hsusb1_phy is a NOP USB PHY device that gets its clock from an oscillator
and expects that clock to be configured to 19.2MHz by the NOP PHY driver. and expects that clock to be configured to 19.2MHz by the NOP PHY driver.
hsusb1_vcc_regulator provides power to the PHY and hsusb1_reset_regulator hsusb1_vcc_regulator provides power to the PHY and GPIO 7 controls RESET.
controls RESET.
...@@ -289,18 +289,12 @@ static struct regulator_consumer_supply beagle_vsim_supply[] = { ...@@ -289,18 +289,12 @@ static struct regulator_consumer_supply beagle_vsim_supply[] = {
static struct gpio_led gpio_leds[]; static struct gpio_led gpio_leds[];
/* PHY's VCC regulator might be added later, so flag that we need it */
static struct usb_phy_gen_xceiv_platform_data hsusb2_phy_data = {
.needs_vcc = true,
};
static struct usbhs_phy_data phy_data[] = { static struct usbhs_phy_data phy_data[] = {
{ {
.port = 2, .port = 2,
.reset_gpio = 147, .reset_gpio = 147,
.vcc_gpio = -1, /* updated in beagle_twl_gpio_setup */ .vcc_gpio = -1, /* updated in beagle_twl_gpio_setup */
.vcc_polarity = 1, /* updated in beagle_twl_gpio_setup */ .vcc_polarity = 1, /* updated in beagle_twl_gpio_setup */
.platform_data = &hsusb2_phy_data,
}, },
}; };
......
...@@ -435,6 +435,7 @@ int usbhs_init_phys(struct usbhs_phy_data *phy, int num_phys) ...@@ -435,6 +435,7 @@ int usbhs_init_phys(struct usbhs_phy_data *phy, int num_phys)
struct platform_device *pdev; struct platform_device *pdev;
char *phy_id; char *phy_id;
struct platform_device_info pdevinfo; struct platform_device_info pdevinfo;
struct usb_phy_gen_xceiv_platform_data nop_pdata;
for (i = 0; i < num_phys; i++) { for (i = 0; i < num_phys; i++) {
...@@ -455,11 +456,18 @@ int usbhs_init_phys(struct usbhs_phy_data *phy, int num_phys) ...@@ -455,11 +456,18 @@ int usbhs_init_phys(struct usbhs_phy_data *phy, int num_phys)
return -ENOMEM; return -ENOMEM;
} }
/* set platform data */
memset(&nop_pdata, 0, sizeof(nop_pdata));
if (gpio_is_valid(phy->vcc_gpio))
nop_pdata.needs_vcc = true;
nop_pdata.gpio_reset = phy->reset_gpio;
nop_pdata.type = USB_PHY_TYPE_USB2;
/* create a NOP PHY device */ /* create a NOP PHY device */
memset(&pdevinfo, 0, sizeof(pdevinfo)); memset(&pdevinfo, 0, sizeof(pdevinfo));
pdevinfo.name = nop_name; pdevinfo.name = nop_name;
pdevinfo.id = phy->port; pdevinfo.id = phy->port;
pdevinfo.data = phy->platform_data; pdevinfo.data = &nop_pdata;
pdevinfo.size_data = pdevinfo.size_data =
sizeof(struct usb_phy_gen_xceiv_platform_data); sizeof(struct usb_phy_gen_xceiv_platform_data);
scnprintf(phy_id, MAX_STR, "usb_phy_gen_xceiv.%d", scnprintf(phy_id, MAX_STR, "usb_phy_gen_xceiv.%d",
...@@ -474,14 +482,6 @@ int usbhs_init_phys(struct usbhs_phy_data *phy, int num_phys) ...@@ -474,14 +482,6 @@ int usbhs_init_phys(struct usbhs_phy_data *phy, int num_phys)
usb_bind_phy("ehci-omap.0", phy->port - 1, phy_id); usb_bind_phy("ehci-omap.0", phy->port - 1, phy_id);
/* Do we need RESET regulator ? */
if (gpio_is_valid(phy->reset_gpio)) {
scnprintf(rail_name, MAX_STR,
"hsusb%d_reset", phy->port);
usbhs_add_regulator(rail_name, phy_id, "reset",
phy->reset_gpio, 1);
}
/* Do we need VCC regulator ? */ /* Do we need VCC regulator ? */
if (gpio_is_valid(phy->vcc_gpio)) { if (gpio_is_valid(phy->vcc_gpio)) {
scnprintf(rail_name, MAX_STR, "hsusb%d_vcc", phy->port); scnprintf(rail_name, MAX_STR, "hsusb%d_vcc", phy->port);
......
...@@ -58,7 +58,6 @@ struct usbhs_phy_data { ...@@ -58,7 +58,6 @@ struct usbhs_phy_data {
int reset_gpio; int reset_gpio;
int vcc_gpio; int vcc_gpio;
bool vcc_polarity; /* 1 active high, 0 active low */ bool vcc_polarity; /* 1 active high, 0 active low */
void *platform_data;
}; };
extern void usb_musb_init(struct omap_musb_board_data *board_data); extern void usb_musb_init(struct omap_musb_board_data *board_data);
......
...@@ -724,6 +724,8 @@ static int twl4030_usb_probe(struct platform_device *pdev) ...@@ -724,6 +724,8 @@ static int twl4030_usb_probe(struct platform_device *pdev)
if (device_create_file(&pdev->dev, &dev_attr_vbus)) if (device_create_file(&pdev->dev, &dev_attr_vbus))
dev_warn(&pdev->dev, "could not create sysfs file\n"); dev_warn(&pdev->dev, "could not create sysfs file\n");
ATOMIC_INIT_NOTIFIER_HEAD(&twl->phy.notifier);
/* Our job is to use irqs and status from the power module /* Our job is to use irqs and status from the power module
* to keep the transceiver disabled when nothing's connected. * to keep the transceiver disabled when nothing's connected.
* *
......
...@@ -584,7 +584,7 @@ static int dwc3_remove(struct platform_device *pdev) ...@@ -584,7 +584,7 @@ static int dwc3_remove(struct platform_device *pdev)
usb_phy_set_suspend(dwc->usb2_phy, 1); usb_phy_set_suspend(dwc->usb2_phy, 1);
usb_phy_set_suspend(dwc->usb3_phy, 1); usb_phy_set_suspend(dwc->usb3_phy, 1);
pm_runtime_put(&pdev->dev); pm_runtime_put_sync(&pdev->dev);
pm_runtime_disable(&pdev->dev); pm_runtime_disable(&pdev->dev);
dwc3_debugfs_exit(dwc); dwc3_debugfs_exit(dwc);
...@@ -691,7 +691,6 @@ static int dwc3_resume(struct device *dev) ...@@ -691,7 +691,6 @@ static int dwc3_resume(struct device *dev)
usb_phy_init(dwc->usb3_phy); usb_phy_init(dwc->usb3_phy);
usb_phy_init(dwc->usb2_phy); usb_phy_init(dwc->usb2_phy);
msleep(100);
spin_lock_irqsave(&dwc->lock, flags); spin_lock_irqsave(&dwc->lock, flags);
......
...@@ -58,6 +58,20 @@ config USB_GADGET_DEBUG ...@@ -58,6 +58,20 @@ config USB_GADGET_DEBUG
trying to track down. Never enable these messages for a trying to track down. Never enable these messages for a
production build. production build.
config USB_GADGET_VERBOSE
bool "Verbose debugging Messages (DEVELOPMENT)"
depends on USB_GADGET_DEBUG
help
Many controller and gadget drivers will print verbose debugging
messages if you use this option to ask for those messages.
Avoid enabling these messages, even if you're actively
debugging such a driver. Many drivers will emit so many
messages that the driver timings are affected, which will
either create new failure modes or remove the one you're
trying to track down. Never enable these messages for a
production build.
config USB_GADGET_DEBUG_FILES config USB_GADGET_DEBUG_FILES
boolean "Debugging information files (DEVELOPMENT)" boolean "Debugging information files (DEVELOPMENT)"
depends on PROC_FS depends on PROC_FS
...@@ -525,6 +539,9 @@ config USB_F_SUBSET ...@@ -525,6 +539,9 @@ config USB_F_SUBSET
config USB_F_RNDIS config USB_F_RNDIS
tristate tristate
config USB_F_MASS_STORAGE
tristate
choice choice
tristate "USB Gadget Drivers" tristate "USB Gadget Drivers"
default USB_ETH default USB_ETH
...@@ -662,6 +679,16 @@ config USB_CONFIGFS_PHONET ...@@ -662,6 +679,16 @@ config USB_CONFIGFS_PHONET
help help
The Phonet protocol implementation for USB device. The Phonet protocol implementation for USB device.
config USB_CONFIGFS_MASS_STORAGE
boolean "Mass storage"
depends on USB_CONFIGFS
select USB_F_MASS_STORAGE
help
The Mass Storage Gadget acts as a USB Mass Storage disk drive.
As its storage repository it can use a regular file or a block
device (in much the same way as the "loop" device driver),
specified as a module parameter or sysfs option.
config USB_ZERO config USB_ZERO
tristate "Gadget Zero (DEVELOPMENT)" tristate "Gadget Zero (DEVELOPMENT)"
select USB_LIBCOMPOSITE select USB_LIBCOMPOSITE
...@@ -878,6 +905,7 @@ config USB_MASS_STORAGE ...@@ -878,6 +905,7 @@ config USB_MASS_STORAGE
tristate "Mass Storage Gadget" tristate "Mass Storage Gadget"
depends on BLOCK depends on BLOCK
select USB_LIBCOMPOSITE select USB_LIBCOMPOSITE
select USB_F_MASS_STORAGE
help help
The Mass Storage Gadget acts as a USB Mass Storage disk drive. The Mass Storage Gadget acts as a USB Mass Storage disk drive.
As its storage repository it can use a regular file or a block As its storage repository it can use a regular file or a block
...@@ -1001,6 +1029,7 @@ config USB_G_ACM_MS ...@@ -1001,6 +1029,7 @@ config USB_G_ACM_MS
select USB_LIBCOMPOSITE select USB_LIBCOMPOSITE
select USB_U_SERIAL select USB_U_SERIAL
select USB_F_ACM select USB_F_ACM
select USB_F_MASS_STORAGE
help help
This driver provides two functions in one configuration: This driver provides two functions in one configuration:
a mass storage, and a CDC ACM (serial port) link. a mass storage, and a CDC ACM (serial port) link.
...@@ -1015,8 +1044,8 @@ config USB_G_MULTI ...@@ -1015,8 +1044,8 @@ config USB_G_MULTI
select USB_LIBCOMPOSITE select USB_LIBCOMPOSITE
select USB_U_SERIAL select USB_U_SERIAL
select USB_U_ETHER select USB_U_ETHER
select USB_U_RNDIS
select USB_F_ACM select USB_F_ACM
select USB_F_MASS_STORAGE
help help
The Multifunction Composite Gadget provides Ethernet (RNDIS The Multifunction Composite Gadget provides Ethernet (RNDIS
and/or CDC Ethernet), mass storage and ACM serial link and/or CDC Ethernet), mass storage and ACM serial link
...@@ -1035,6 +1064,8 @@ config USB_G_MULTI ...@@ -1035,6 +1064,8 @@ config USB_G_MULTI
config USB_G_MULTI_RNDIS config USB_G_MULTI_RNDIS
bool "RNDIS + CDC Serial + Storage configuration" bool "RNDIS + CDC Serial + Storage configuration"
depends on USB_G_MULTI depends on USB_G_MULTI
select USB_U_RNDIS
select USB_F_RNDIS
default y default y
help help
This option enables a configuration with RNDIS, CDC Serial and This option enables a configuration with RNDIS, CDC Serial and
...@@ -1048,6 +1079,7 @@ config USB_G_MULTI_CDC ...@@ -1048,6 +1079,7 @@ config USB_G_MULTI_CDC
bool "CDC Ethernet + CDC Serial + Storage configuration" bool "CDC Ethernet + CDC Serial + Storage configuration"
depends on USB_G_MULTI depends on USB_G_MULTI
default n default n
select USB_F_ECM
help help
This option enables a configuration with CDC Ethernet (ECM), CDC This option enables a configuration with CDC Ethernet (ECM), CDC
Serial and Mass Storage functions available in the Multifunction Serial and Mass Storage functions available in the Multifunction
......
# #
# USB peripheral controller drivers # USB peripheral controller drivers
# #
ccflags-$(CONFIG_USB_GADGET_DEBUG) := -DDEBUG ccflags-$(CONFIG_USB_GADGET_DEBUG) := -DDEBUG
ccflags-$(CONFIG_USB_GADGET_VERBOSE) += -DVERBOSE_DEBUG
obj-$(CONFIG_USB_GADGET) += udc-core.o obj-$(CONFIG_USB_GADGET) += udc-core.o
obj-$(CONFIG_USB_LIBCOMPOSITE) += libcomposite.o obj-$(CONFIG_USB_LIBCOMPOSITE) += libcomposite.o
...@@ -60,6 +61,8 @@ usb_f_ecm_subset-y := f_subset.o ...@@ -60,6 +61,8 @@ usb_f_ecm_subset-y := f_subset.o
obj-$(CONFIG_USB_F_SUBSET) += usb_f_ecm_subset.o obj-$(CONFIG_USB_F_SUBSET) += usb_f_ecm_subset.o
usb_f_rndis-y := f_rndis.o usb_f_rndis-y := f_rndis.o
obj-$(CONFIG_USB_F_RNDIS) += usb_f_rndis.o obj-$(CONFIG_USB_F_RNDIS) += usb_f_rndis.o
usb_f_mass_storage-y := f_mass_storage.o storage_common.o
obj-$(CONFIG_USB_F_MASS_STORAGE)+= usb_f_mass_storage.o
# #
# USB gadget drivers # USB gadget drivers
......
...@@ -31,16 +31,7 @@ ...@@ -31,16 +31,7 @@
#define ACM_MS_VENDOR_NUM 0x1d6b /* Linux Foundation */ #define ACM_MS_VENDOR_NUM 0x1d6b /* Linux Foundation */
#define ACM_MS_PRODUCT_NUM 0x0106 /* Composite Gadget: ACM + MS*/ #define ACM_MS_PRODUCT_NUM 0x0106 /* Composite Gadget: ACM + MS*/
/*-------------------------------------------------------------------------*/ #include "f_mass_storage.h"
/*
* Kbuild is not very cooperative with respect to linking separately
* compiled library objects into one module. So for now we won't use
* separate compilation ... ensuring init/exit sections work to shrink
* the runtime footprint, and giving us at least some parts of what
* a "gcc --combine ... part1.c part2.c part3.c ... " build would.
*/
#include "f_mass_storage.c"
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
USB_GADGET_COMPOSITE_OPTIONS(); USB_GADGET_COMPOSITE_OPTIONS();
...@@ -104,18 +95,35 @@ static struct usb_gadget_strings *dev_strings[] = { ...@@ -104,18 +95,35 @@ static struct usb_gadget_strings *dev_strings[] = {
/****************************** Configurations ******************************/ /****************************** Configurations ******************************/
static struct fsg_module_parameters fsg_mod_data = { .stall = 1 }; static struct fsg_module_parameters fsg_mod_data = { .stall = 1 };
FSG_MODULE_PARAMETERS(/* no prefix */, fsg_mod_data); #ifdef CONFIG_USB_GADGET_DEBUG_FILES
static struct fsg_common fsg_common; static unsigned int fsg_num_buffers = CONFIG_USB_GADGET_STORAGE_NUM_BUFFERS;
#else
/*
* Number of buffers we will use.
* 2 is usually enough for good buffering pipeline
*/
#define fsg_num_buffers CONFIG_USB_GADGET_STORAGE_NUM_BUFFERS
#endif /* CONFIG_USB_DEBUG */
FSG_MODULE_PARAMETERS(/* no prefix */, fsg_mod_data);
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
static struct usb_function *f_acm; static struct usb_function *f_acm;
static struct usb_function_instance *f_acm_inst; static struct usb_function_instance *f_acm_inst;
static struct usb_function_instance *fi_msg;
static struct usb_function *f_msg;
/* /*
* We _always_ have both ACM and mass storage functions. * We _always_ have both ACM and mass storage functions.
*/ */
static int __init acm_ms_do_config(struct usb_configuration *c) static int __init acm_ms_do_config(struct usb_configuration *c)
{ {
struct fsg_opts *opts;
int status; int status;
if (gadget_is_otg(c->cdev->gadget)) { if (gadget_is_otg(c->cdev->gadget)) {
...@@ -123,31 +131,37 @@ static int __init acm_ms_do_config(struct usb_configuration *c) ...@@ -123,31 +131,37 @@ static int __init acm_ms_do_config(struct usb_configuration *c)
c->bmAttributes |= USB_CONFIG_ATT_WAKEUP; c->bmAttributes |= USB_CONFIG_ATT_WAKEUP;
} }
f_acm_inst = usb_get_function_instance("acm"); opts = fsg_opts_from_func_inst(fi_msg);
if (IS_ERR(f_acm_inst))
return PTR_ERR(f_acm_inst);
f_acm = usb_get_function(f_acm_inst); f_acm = usb_get_function(f_acm_inst);
if (IS_ERR(f_acm)) { if (IS_ERR(f_acm))
status = PTR_ERR(f_acm); return PTR_ERR(f_acm);
goto err_func;
f_msg = usb_get_function(fi_msg);
if (IS_ERR(f_msg)) {
status = PTR_ERR(f_msg);
goto put_acm;
} }
status = usb_add_function(c, f_acm); status = usb_add_function(c, f_acm);
if (status < 0) if (status < 0)
goto err_conf; goto put_msg;
status = fsg_bind_config(c->cdev, c, &fsg_common); status = fsg_common_run_thread(opts->common);
if (status < 0) if (status)
goto err_fsg; goto remove_acm;
status = usb_add_function(c, f_msg);
if (status)
goto remove_acm;
return 0; return 0;
err_fsg: remove_acm:
usb_remove_function(c, f_acm); usb_remove_function(c, f_acm);
err_conf: put_msg:
usb_put_function(f_msg);
put_acm:
usb_put_function(f_acm); usb_put_function(f_acm);
err_func:
usb_put_function_instance(f_acm_inst);
return status; return status;
} }
...@@ -163,45 +177,82 @@ static struct usb_configuration acm_ms_config_driver = { ...@@ -163,45 +177,82 @@ static struct usb_configuration acm_ms_config_driver = {
static int __init acm_ms_bind(struct usb_composite_dev *cdev) static int __init acm_ms_bind(struct usb_composite_dev *cdev)
{ {
struct usb_gadget *gadget = cdev->gadget; struct usb_gadget *gadget = cdev->gadget;
struct fsg_opts *opts;
struct fsg_config config;
int status; int status;
void *retp;
/* set up mass storage function */ f_acm_inst = usb_get_function_instance("acm");
retp = fsg_common_from_params(&fsg_common, cdev, &fsg_mod_data); if (IS_ERR(f_acm_inst))
if (IS_ERR(retp)) { return PTR_ERR(f_acm_inst);
status = PTR_ERR(retp);
return PTR_ERR(retp); fi_msg = usb_get_function_instance("mass_storage");
if (IS_ERR(fi_msg)) {
status = PTR_ERR(fi_msg);
goto fail_get_msg;
} }
/* set up mass storage function */
fsg_config_from_params(&config, &fsg_mod_data, fsg_num_buffers);
opts = fsg_opts_from_func_inst(fi_msg);
opts->no_configfs = true;
status = fsg_common_set_num_buffers(opts->common, fsg_num_buffers);
if (status)
goto fail;
status = fsg_common_set_nluns(opts->common, config.nluns);
if (status)
goto fail_set_nluns;
status = fsg_common_set_cdev(opts->common, cdev, config.can_stall);
if (status)
goto fail_set_cdev;
fsg_common_set_sysfs(opts->common, true);
status = fsg_common_create_luns(opts->common, &config);
if (status)
goto fail_set_cdev;
fsg_common_set_inquiry_string(opts->common, config.vendor_name,
config.product_name);
/* /*
* Allocate string descriptor numbers ... note that string * Allocate string descriptor numbers ... note that string
* contents can be overridden by the composite_dev glue. * contents can be overridden by the composite_dev glue.
*/ */
status = usb_string_ids_tab(cdev, strings_dev); status = usb_string_ids_tab(cdev, strings_dev);
if (status < 0) if (status < 0)
goto fail1; goto fail_string_ids;
device_desc.iManufacturer = strings_dev[USB_GADGET_MANUFACTURER_IDX].id; device_desc.iManufacturer = strings_dev[USB_GADGET_MANUFACTURER_IDX].id;
device_desc.iProduct = strings_dev[USB_GADGET_PRODUCT_IDX].id; device_desc.iProduct = strings_dev[USB_GADGET_PRODUCT_IDX].id;
/* register our configuration */ /* register our configuration */
status = usb_add_config(cdev, &acm_ms_config_driver, acm_ms_do_config); status = usb_add_config(cdev, &acm_ms_config_driver, acm_ms_do_config);
if (status < 0) if (status < 0)
goto fail1; goto fail_string_ids;
usb_composite_overwrite_options(cdev, &coverwrite); usb_composite_overwrite_options(cdev, &coverwrite);
dev_info(&gadget->dev, "%s, version: " DRIVER_VERSION "\n", dev_info(&gadget->dev, "%s, version: " DRIVER_VERSION "\n",
DRIVER_DESC); DRIVER_DESC);
fsg_common_put(&fsg_common);
return 0; return 0;
/* error recovery */ /* error recovery */
fail1: fail_string_ids:
fsg_common_put(&fsg_common); fsg_common_remove_luns(opts->common);
fail_set_cdev:
fsg_common_free_luns(opts->common);
fail_set_nluns:
fsg_common_free_buffers(opts->common);
fail:
usb_put_function_instance(fi_msg);
fail_get_msg:
usb_put_function_instance(f_acm_inst);
return status; return status;
} }
static int __exit acm_ms_unbind(struct usb_composite_dev *cdev) static int __exit acm_ms_unbind(struct usb_composite_dev *cdev)
{ {
usb_put_function(f_msg);
usb_put_function_instance(fi_msg);
usb_put_function(f_acm); usb_put_function(f_acm);
usb_put_function_instance(f_acm_inst); usb_put_function_instance(f_acm_inst);
return 0; return 0;
......
...@@ -557,7 +557,7 @@ static struct config_group *function_make( ...@@ -557,7 +557,7 @@ static struct config_group *function_make(
fi = usb_get_function_instance(func_name); fi = usb_get_function_instance(func_name);
if (IS_ERR(fi)) if (IS_ERR(fi))
return ERR_PTR(PTR_ERR(fi)); return ERR_CAST(fi);
ret = config_item_set_name(&fi->group.cg_item, name); ret = config_item_set_name(&fi->group.cg_item, name);
if (ret) { if (ret) {
...@@ -991,6 +991,14 @@ static struct configfs_subsystem gadget_subsys = { ...@@ -991,6 +991,14 @@ static struct configfs_subsystem gadget_subsys = {
.su_mutex = __MUTEX_INITIALIZER(gadget_subsys.su_mutex), .su_mutex = __MUTEX_INITIALIZER(gadget_subsys.su_mutex),
}; };
void unregister_gadget_item(struct config_item *item)
{
struct gadget_info *gi = to_gadget_info(item);
unregister_gadget(gi);
}
EXPORT_SYMBOL(unregister_gadget_item);
static int __init gadget_cfs_init(void) static int __init gadget_cfs_init(void)
{ {
int ret; int ret;
......
#ifndef USB__GADGET__CONFIGFS__H
#define USB__GADGET__CONFIGFS__H
void unregister_gadget_item(struct config_item *item);
#endif /* USB__GADGET__CONFIGFS__H */
This diff is collapsed.
#ifndef USB_F_MASS_STORAGE_H
#define USB_F_MASS_STORAGE_H
#include <linux/usb/composite.h>
#include "storage_common.h"
struct fsg_module_parameters {
char *file[FSG_MAX_LUNS];
bool ro[FSG_MAX_LUNS];
bool removable[FSG_MAX_LUNS];
bool cdrom[FSG_MAX_LUNS];
bool nofua[FSG_MAX_LUNS];
unsigned int file_count, ro_count, removable_count, cdrom_count;
unsigned int nofua_count;
unsigned int luns; /* nluns */
bool stall; /* can_stall */
};
#define _FSG_MODULE_PARAM_ARRAY(prefix, params, name, type, desc) \
module_param_array_named(prefix ## name, params.name, type, \
&prefix ## params.name ## _count, \
S_IRUGO); \
MODULE_PARM_DESC(prefix ## name, desc)
#define _FSG_MODULE_PARAM(prefix, params, name, type, desc) \
module_param_named(prefix ## name, params.name, type, \
S_IRUGO); \
MODULE_PARM_DESC(prefix ## name, desc)
#define __FSG_MODULE_PARAMETERS(prefix, params) \
_FSG_MODULE_PARAM_ARRAY(prefix, params, file, charp, \
"names of backing files or devices"); \
_FSG_MODULE_PARAM_ARRAY(prefix, params, ro, bool, \
"true to force read-only"); \
_FSG_MODULE_PARAM_ARRAY(prefix, params, removable, bool, \
"true to simulate removable media"); \
_FSG_MODULE_PARAM_ARRAY(prefix, params, cdrom, bool, \
"true to simulate CD-ROM instead of disk"); \
_FSG_MODULE_PARAM_ARRAY(prefix, params, nofua, bool, \
"true to ignore SCSI WRITE(10,12) FUA bit"); \
_FSG_MODULE_PARAM(prefix, params, luns, uint, \
"number of LUNs"); \
_FSG_MODULE_PARAM(prefix, params, stall, bool, \
"false to prevent bulk stalls")
#ifdef CONFIG_USB_GADGET_DEBUG_FILES
#define FSG_MODULE_PARAMETERS(prefix, params) \
__FSG_MODULE_PARAMETERS(prefix, params); \
module_param_named(num_buffers, fsg_num_buffers, uint, S_IRUGO);\
MODULE_PARM_DESC(num_buffers, "Number of pipeline buffers")
#else
#define FSG_MODULE_PARAMETERS(prefix, params) \
__FSG_MODULE_PARAMETERS(prefix, params)
#endif
struct fsg_common;
/* FSF callback functions */
struct fsg_operations {
/*
* Callback function to call when thread exits. If no
* callback is set or it returns value lower then zero MSF
* will force eject all LUNs it operates on (including those
* marked as non-removable or with prevent_medium_removal flag
* set).
*/
int (*thread_exits)(struct fsg_common *common);
};
struct fsg_lun_opts {
struct config_group group;
struct fsg_lun *lun;
int lun_id;
};
struct fsg_opts {
struct fsg_common *common;
struct usb_function_instance func_inst;
struct fsg_lun_opts lun0;
struct config_group *default_groups[2];
bool no_configfs; /* for legacy gadgets */
/*
* Read/write access to configfs attributes is handled by configfs.
*
* This is to protect the data from concurrent access by read/write
* and create symlink/remove symlink.
*/
struct mutex lock;
int refcnt;
};
struct fsg_lun_config {
const char *filename;
char ro;
char removable;
char cdrom;
char nofua;
};
struct fsg_config {
unsigned nluns;
struct fsg_lun_config luns[FSG_MAX_LUNS];
/* Callback functions. */
const struct fsg_operations *ops;
/* Gadget's private data. */
void *private_data;
const char *vendor_name; /* 8 characters or less */
const char *product_name; /* 16 characters or less */
char can_stall;
unsigned int fsg_num_buffers;
};
static inline struct fsg_opts *
fsg_opts_from_func_inst(const struct usb_function_instance *fi)
{
return container_of(fi, struct fsg_opts, func_inst);
}
void fsg_common_get(struct fsg_common *common);
void fsg_common_put(struct fsg_common *common);
void fsg_common_set_sysfs(struct fsg_common *common, bool sysfs);
int fsg_common_set_num_buffers(struct fsg_common *common, unsigned int n);
void fsg_common_free_buffers(struct fsg_common *common);
int fsg_common_set_cdev(struct fsg_common *common,
struct usb_composite_dev *cdev, bool can_stall);
void fsg_common_remove_lun(struct fsg_lun *lun, bool sysfs);
void fsg_common_remove_luns(struct fsg_common *common);
void fsg_common_free_luns(struct fsg_common *common);
int fsg_common_set_nluns(struct fsg_common *common, int nluns);
void fsg_common_set_ops(struct fsg_common *common,
const struct fsg_operations *ops);
int fsg_common_create_lun(struct fsg_common *common, struct fsg_lun_config *cfg,
unsigned int id, const char *name,
const char **name_pfx);
int fsg_common_create_luns(struct fsg_common *common, struct fsg_config *cfg);
void fsg_common_set_inquiry_string(struct fsg_common *common, const char *vn,
const char *pn);
int fsg_common_run_thread(struct fsg_common *common);
void fsg_config_from_params(struct fsg_config *cfg,
const struct fsg_module_parameters *params,
unsigned int fsg_num_buffers);
#endif /* USB_F_MASS_STORAGE_H */
...@@ -37,16 +37,16 @@ ...@@ -37,16 +37,16 @@
#define DRIVER_DESC "Mass Storage Gadget" #define DRIVER_DESC "Mass Storage Gadget"
#define DRIVER_VERSION "2009/09/11" #define DRIVER_VERSION "2009/09/11"
/*-------------------------------------------------------------------------*/
/* /*
* kbuild is not very cooperative with respect to linking separately * Thanks to NetChip Technologies for donating this product ID.
* compiled library objects into one module. So for now we won't use *
* separate compilation ... ensuring init/exit sections work to shrink * DO NOT REUSE THESE IDs with any other driver!! Ever!!
* the runtime footprint, and giving us at least some parts of what * Instead: allocate your own, using normal USB-IF procedures.
* a "gcc --combine ... part1.c part2.c part3.c ... " build would.
*/ */
#include "f_mass_storage.c" #define FSG_VENDOR_ID 0x0525 /* NetChip */
#define FSG_PRODUCT_ID 0xa4a5 /* Linux-USB File-backed Storage Gadget */
#include "f_mass_storage.h"
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
USB_GADGET_COMPOSITE_OPTIONS(); USB_GADGET_COMPOSITE_OPTIONS();
...@@ -97,11 +97,28 @@ static struct usb_gadget_strings *dev_strings[] = { ...@@ -97,11 +97,28 @@ static struct usb_gadget_strings *dev_strings[] = {
NULL, NULL,
}; };
static struct usb_function_instance *fi_msg;
static struct usb_function *f_msg;
/****************************** Configurations ******************************/ /****************************** Configurations ******************************/
static struct fsg_module_parameters mod_data = { static struct fsg_module_parameters mod_data = {
.stall = 1 .stall = 1
}; };
#ifdef CONFIG_USB_GADGET_DEBUG_FILES
static unsigned int fsg_num_buffers = CONFIG_USB_GADGET_STORAGE_NUM_BUFFERS;
#else
/*
* Number of buffers we will use.
* 2 is usually enough for good buffering pipeline
*/
#define fsg_num_buffers CONFIG_USB_GADGET_STORAGE_NUM_BUFFERS
#endif /* CONFIG_USB_GADGET_DEBUG_FILES */
FSG_MODULE_PARAMETERS(/* no prefix */, mod_data); FSG_MODULE_PARAMETERS(/* no prefix */, mod_data);
static unsigned long msg_registered; static unsigned long msg_registered;
...@@ -115,13 +132,7 @@ static int msg_thread_exits(struct fsg_common *common) ...@@ -115,13 +132,7 @@ static int msg_thread_exits(struct fsg_common *common)
static int __init msg_do_config(struct usb_configuration *c) static int __init msg_do_config(struct usb_configuration *c)
{ {
static const struct fsg_operations ops = { struct fsg_opts *opts;
.thread_exits = msg_thread_exits,
};
static struct fsg_common common;
struct fsg_common *retp;
struct fsg_config config;
int ret; int ret;
if (gadget_is_otg(c->cdev->gadget)) { if (gadget_is_otg(c->cdev->gadget)) {
...@@ -129,15 +140,24 @@ static int __init msg_do_config(struct usb_configuration *c) ...@@ -129,15 +140,24 @@ static int __init msg_do_config(struct usb_configuration *c)
c->bmAttributes |= USB_CONFIG_ATT_WAKEUP; c->bmAttributes |= USB_CONFIG_ATT_WAKEUP;
} }
fsg_config_from_params(&config, &mod_data); opts = fsg_opts_from_func_inst(fi_msg);
config.ops = &ops;
f_msg = usb_get_function(fi_msg);
if (IS_ERR(f_msg))
return PTR_ERR(f_msg);
ret = fsg_common_run_thread(opts->common);
if (ret)
goto put_func;
ret = usb_add_function(c, f_msg);
if (ret)
goto put_func;
retp = fsg_common_init(&common, c->cdev, &config); return 0;
if (IS_ERR(retp))
return PTR_ERR(retp);
ret = fsg_bind_config(c->cdev, c, &common); put_func:
fsg_common_put(&common); usb_put_function(f_msg);
return ret; return ret;
} }
...@@ -152,23 +172,79 @@ static struct usb_configuration msg_config_driver = { ...@@ -152,23 +172,79 @@ static struct usb_configuration msg_config_driver = {
static int __init msg_bind(struct usb_composite_dev *cdev) static int __init msg_bind(struct usb_composite_dev *cdev)
{ {
static const struct fsg_operations ops = {
.thread_exits = msg_thread_exits,
};
struct fsg_opts *opts;
struct fsg_config config;
int status; int status;
fi_msg = usb_get_function_instance("mass_storage");
if (IS_ERR(fi_msg))
return PTR_ERR(fi_msg);
fsg_config_from_params(&config, &mod_data, fsg_num_buffers);
opts = fsg_opts_from_func_inst(fi_msg);
opts->no_configfs = true;
status = fsg_common_set_num_buffers(opts->common, fsg_num_buffers);
if (status)
goto fail;
status = fsg_common_set_nluns(opts->common, config.nluns);
if (status)
goto fail_set_nluns;
fsg_common_set_ops(opts->common, &ops);
status = fsg_common_set_cdev(opts->common, cdev, config.can_stall);
if (status)
goto fail_set_cdev;
fsg_common_set_sysfs(opts->common, true);
status = fsg_common_create_luns(opts->common, &config);
if (status)
goto fail_set_cdev;
fsg_common_set_inquiry_string(opts->common, config.vendor_name,
config.product_name);
status = usb_string_ids_tab(cdev, strings_dev); status = usb_string_ids_tab(cdev, strings_dev);
if (status < 0) if (status < 0)
return status; goto fail_string_ids;
msg_device_desc.iProduct = strings_dev[USB_GADGET_PRODUCT_IDX].id; msg_device_desc.iProduct = strings_dev[USB_GADGET_PRODUCT_IDX].id;
status = usb_add_config(cdev, &msg_config_driver, msg_do_config); status = usb_add_config(cdev, &msg_config_driver, msg_do_config);
if (status < 0) if (status < 0)
return status; goto fail_string_ids;
usb_composite_overwrite_options(cdev, &coverwrite); usb_composite_overwrite_options(cdev, &coverwrite);
dev_info(&cdev->gadget->dev, dev_info(&cdev->gadget->dev,
DRIVER_DESC ", version: " DRIVER_VERSION "\n"); DRIVER_DESC ", version: " DRIVER_VERSION "\n");
set_bit(0, &msg_registered); set_bit(0, &msg_registered);
return 0; return 0;
fail_string_ids:
fsg_common_remove_luns(opts->common);
fail_set_cdev:
fsg_common_free_luns(opts->common);
fail_set_nluns:
fsg_common_free_buffers(opts->common);
fail:
usb_put_function_instance(fi_msg);
return status;
} }
static int msg_unbind(struct usb_composite_dev *cdev)
{
if (!IS_ERR(f_msg))
usb_put_function(f_msg);
if (!IS_ERR(fi_msg))
usb_put_function_instance(fi_msg);
return 0;
}
/****************************** Some noise ******************************/ /****************************** Some noise ******************************/
...@@ -179,6 +255,7 @@ static __refdata struct usb_composite_driver msg_driver = { ...@@ -179,6 +255,7 @@ static __refdata struct usb_composite_driver msg_driver = {
.needs_serial = 1, .needs_serial = 1,
.strings = dev_strings, .strings = dev_strings,
.bind = msg_bind, .bind = msg_bind,
.unbind = msg_unbind,
}; };
MODULE_DESCRIPTION(DRIVER_DESC); MODULE_DESCRIPTION(DRIVER_DESC);
......
This diff is collapsed.
...@@ -310,6 +310,7 @@ static struct mv_u3d_trb *mv_u3d_build_trb_one(struct mv_u3d_req *req, ...@@ -310,6 +310,7 @@ static struct mv_u3d_trb *mv_u3d_build_trb_one(struct mv_u3d_req *req,
*/ */
trb_hw = dma_pool_alloc(u3d->trb_pool, GFP_ATOMIC, dma); trb_hw = dma_pool_alloc(u3d->trb_pool, GFP_ATOMIC, dma);
if (!trb_hw) { if (!trb_hw) {
kfree(trb);
dev_err(u3d->dev, dev_err(u3d->dev,
"%s, dma_pool_alloc fail\n", __func__); "%s, dma_pool_alloc fail\n", __func__);
return NULL; return NULL;
...@@ -454,6 +455,7 @@ static int mv_u3d_req_to_trb(struct mv_u3d_req *req) ...@@ -454,6 +455,7 @@ static int mv_u3d_req_to_trb(struct mv_u3d_req *req)
trb_hw = kcalloc(trb_num, sizeof(*trb_hw), GFP_ATOMIC); trb_hw = kcalloc(trb_num, sizeof(*trb_hw), GFP_ATOMIC);
if (!trb_hw) { if (!trb_hw) {
kfree(trb);
dev_err(u3d->dev, dev_err(u3d->dev,
"%s, trb_hw alloc fail\n", __func__); "%s, trb_hw alloc fail\n", __func__);
return -ENOMEM; return -ENOMEM;
......
This diff is collapsed.
This diff is collapsed.
#ifndef USB_STORAGE_COMMON_H
#define USB_STORAGE_COMMON_H
#include <linux/device.h>
#include <linux/usb/storage.h>
#include <scsi/scsi.h>
#include <asm/unaligned.h>
#ifndef DEBUG
#undef VERBOSE_DEBUG
#undef DUMP_MSGS
#endif /* !DEBUG */
#ifdef VERBOSE_DEBUG
#define VLDBG LDBG
#else
#define VLDBG(lun, fmt, args...) do { } while (0)
#endif /* VERBOSE_DEBUG */
#define _LMSG(func, lun, fmt, args...) \
do { \
if ((lun)->name_pfx && *(lun)->name_pfx) \
func("%s/%s: " fmt, *(lun)->name_pfx, \
(lun)->name, ## args); \
else \
func("%s: " fmt, (lun)->name, ## args); \
} while (0)
#define LDBG(lun, fmt, args...) _LMSG(pr_debug, lun, fmt, ## args)
#define LERROR(lun, fmt, args...) _LMSG(pr_err, lun, fmt, ## args)
#define LWARN(lun, fmt, args...) _LMSG(pr_warn, lun, fmt, ## args)
#define LINFO(lun, fmt, args...) _LMSG(pr_info, lun, fmt, ## args)
#ifdef DUMP_MSGS
# define dump_msg(fsg, /* const char * */ label, \
/* const u8 * */ buf, /* unsigned */ length) \
do { \
if (length < 512) { \
DBG(fsg, "%s, length %u:\n", label, length); \
print_hex_dump(KERN_DEBUG, "", DUMP_PREFIX_OFFSET, \
16, 1, buf, length, 0); \
} \
} while (0)
# define dump_cdb(fsg) do { } while (0)
#else
# define dump_msg(fsg, /* const char * */ label, \
/* const u8 * */ buf, /* unsigned */ length) do { } while (0)
# ifdef VERBOSE_DEBUG
# define dump_cdb(fsg) \
print_hex_dump(KERN_DEBUG, "SCSI CDB: ", DUMP_PREFIX_NONE, \
16, 1, (fsg)->cmnd, (fsg)->cmnd_size, 0) \
# else
# define dump_cdb(fsg) do { } while (0)
# endif /* VERBOSE_DEBUG */
#endif /* DUMP_MSGS */
/* Length of a SCSI Command Data Block */
#define MAX_COMMAND_SIZE 16
/* SCSI Sense Key/Additional Sense Code/ASC Qualifier values */
#define SS_NO_SENSE 0
#define SS_COMMUNICATION_FAILURE 0x040800
#define SS_INVALID_COMMAND 0x052000
#define SS_INVALID_FIELD_IN_CDB 0x052400
#define SS_LOGICAL_BLOCK_ADDRESS_OUT_OF_RANGE 0x052100
#define SS_LOGICAL_UNIT_NOT_SUPPORTED 0x052500
#define SS_MEDIUM_NOT_PRESENT 0x023a00
#define SS_MEDIUM_REMOVAL_PREVENTED 0x055302
#define SS_NOT_READY_TO_READY_TRANSITION 0x062800
#define SS_RESET_OCCURRED 0x062900
#define SS_SAVING_PARAMETERS_NOT_SUPPORTED 0x053900
#define SS_UNRECOVERED_READ_ERROR 0x031100
#define SS_WRITE_ERROR 0x030c02
#define SS_WRITE_PROTECTED 0x072700
#define SK(x) ((u8) ((x) >> 16)) /* Sense Key byte, etc. */
#define ASC(x) ((u8) ((x) >> 8))
#define ASCQ(x) ((u8) (x))
struct fsg_lun {
struct file *filp;
loff_t file_length;
loff_t num_sectors;
unsigned int initially_ro:1;
unsigned int ro:1;
unsigned int removable:1;
unsigned int cdrom:1;
unsigned int prevent_medium_removal:1;
unsigned int registered:1;
unsigned int info_valid:1;
unsigned int nofua:1;
u32 sense_data;
u32 sense_data_info;
u32 unit_attention_data;
unsigned int blkbits; /* Bits of logical block size
of bound block device */
unsigned int blksize; /* logical block size of bound block device */
struct device dev;
const char *name; /* "lun.name" */
const char **name_pfx; /* "function.name" */
};
static inline bool fsg_lun_is_open(struct fsg_lun *curlun)
{
return curlun->filp != NULL;
}
/* Big enough to hold our biggest descriptor */
#define EP0_BUFSIZE 256
#define DELAYED_STATUS (EP0_BUFSIZE + 999) /* An impossibly large value */
/* Default size of buffer length. */
#define FSG_BUFLEN ((u32)16384)
/* Maximal number of LUNs supported in mass storage function */
#define FSG_MAX_LUNS 8
enum fsg_buffer_state {
BUF_STATE_EMPTY = 0,
BUF_STATE_FULL,
BUF_STATE_BUSY
};
struct fsg_buffhd {
void *buf;
enum fsg_buffer_state state;
struct fsg_buffhd *next;
/*
* The NetChip 2280 is faster, and handles some protocol faults
* better, if we don't submit any short bulk-out read requests.
* So we will record the intended request length here.
*/
unsigned int bulk_out_intended_length;
struct usb_request *inreq;
int inreq_busy;
struct usb_request *outreq;
int outreq_busy;
};
enum fsg_state {
/* This one isn't used anywhere */
FSG_STATE_COMMAND_PHASE = -10,
FSG_STATE_DATA_PHASE,
FSG_STATE_STATUS_PHASE,
FSG_STATE_IDLE = 0,
FSG_STATE_ABORT_BULK_OUT,
FSG_STATE_RESET,
FSG_STATE_INTERFACE_CHANGE,
FSG_STATE_CONFIG_CHANGE,
FSG_STATE_DISCONNECT,
FSG_STATE_EXIT,
FSG_STATE_TERMINATED
};
enum data_direction {
DATA_DIR_UNKNOWN = 0,
DATA_DIR_FROM_HOST,
DATA_DIR_TO_HOST,
DATA_DIR_NONE
};
static inline u32 get_unaligned_be24(u8 *buf)
{
return 0xffffff & (u32) get_unaligned_be32(buf - 1);
}
static inline struct fsg_lun *fsg_lun_from_dev(struct device *dev)
{
return container_of(dev, struct fsg_lun, dev);
}
enum {
FSG_STRING_INTERFACE
};
extern struct usb_interface_descriptor fsg_intf_desc;
extern struct usb_endpoint_descriptor fsg_fs_bulk_in_desc;
extern struct usb_endpoint_descriptor fsg_fs_bulk_out_desc;
extern struct usb_descriptor_header *fsg_fs_function[];
extern struct usb_endpoint_descriptor fsg_hs_bulk_in_desc;
extern struct usb_endpoint_descriptor fsg_hs_bulk_out_desc;
extern struct usb_descriptor_header *fsg_hs_function[];
extern struct usb_endpoint_descriptor fsg_ss_bulk_in_desc;
extern struct usb_ss_ep_comp_descriptor fsg_ss_bulk_in_comp_desc;
extern struct usb_endpoint_descriptor fsg_ss_bulk_out_desc;
extern struct usb_ss_ep_comp_descriptor fsg_ss_bulk_out_comp_desc;
extern struct usb_descriptor_header *fsg_ss_function[];
void fsg_lun_close(struct fsg_lun *curlun);
int fsg_lun_open(struct fsg_lun *curlun, const char *filename);
int fsg_lun_fsync_sub(struct fsg_lun *curlun);
void store_cdrom_address(u8 *dest, int msf, u32 addr);
ssize_t fsg_show_ro(struct fsg_lun *curlun, char *buf);
ssize_t fsg_show_nofua(struct fsg_lun *curlun, char *buf);
ssize_t fsg_show_file(struct fsg_lun *curlun, struct rw_semaphore *filesem,
char *buf);
ssize_t fsg_show_cdrom(struct fsg_lun *curlun, char *buf);
ssize_t fsg_show_removable(struct fsg_lun *curlun, char *buf);
ssize_t fsg_store_ro(struct fsg_lun *curlun, struct rw_semaphore *filesem,
const char *buf, size_t count);
ssize_t fsg_store_nofua(struct fsg_lun *curlun, const char *buf, size_t count);
ssize_t fsg_store_file(struct fsg_lun *curlun, struct rw_semaphore *filesem,
const char *buf, size_t count);
ssize_t fsg_store_cdrom(struct fsg_lun *curlun, struct rw_semaphore *filesem,
const char *buf, size_t count);
ssize_t fsg_store_removable(struct fsg_lun *curlun, const char *buf,
size_t count);
#endif /* USB_STORAGE_COMMON_H */
...@@ -356,7 +356,8 @@ static int udc_bind_to_driver(struct usb_udc *udc, struct usb_gadget_driver *dri ...@@ -356,7 +356,8 @@ static int udc_bind_to_driver(struct usb_udc *udc, struct usb_gadget_driver *dri
kobject_uevent(&udc->dev.kobj, KOBJ_CHANGE); kobject_uevent(&udc->dev.kobj, KOBJ_CHANGE);
return 0; return 0;
err1: err1:
dev_err(&udc->dev, "failed to start %s: %d\n", if (ret != -EISNAM)
dev_err(&udc->dev, "failed to start %s: %d\n",
udc->driver->function, ret); udc->driver->function, ret);
udc->driver = NULL; udc->driver = NULL;
udc->dev.driver = NULL; udc->dev.driver = NULL;
......
...@@ -95,6 +95,18 @@ unsigned autoresume = DEFAULT_AUTORESUME; ...@@ -95,6 +95,18 @@ unsigned autoresume = DEFAULT_AUTORESUME;
module_param(autoresume, uint, S_IRUGO); module_param(autoresume, uint, S_IRUGO);
MODULE_PARM_DESC(autoresume, "zero, or seconds before remote wakeup"); MODULE_PARM_DESC(autoresume, "zero, or seconds before remote wakeup");
/* Maximum Autoresume time */
unsigned max_autoresume;
module_param(max_autoresume, uint, S_IRUGO);
MODULE_PARM_DESC(max_autoresume, "maximum seconds before remote wakeup");
/* Interval between two remote wakeups */
unsigned autoresume_interval_ms;
module_param(autoresume_interval_ms, uint, S_IRUGO);
MODULE_PARM_DESC(autoresume_interval_ms,
"milliseconds to increase successive wakeup delays");
static unsigned autoresume_step_ms;
/*-------------------------------------------------------------------------*/ /*-------------------------------------------------------------------------*/
static struct usb_device_descriptor device_desc = { static struct usb_device_descriptor device_desc = {
...@@ -183,8 +195,16 @@ static void zero_suspend(struct usb_composite_dev *cdev) ...@@ -183,8 +195,16 @@ static void zero_suspend(struct usb_composite_dev *cdev)
return; return;
if (autoresume) { if (autoresume) {
mod_timer(&autoresume_timer, jiffies + (HZ * autoresume)); if (max_autoresume &&
DBG(cdev, "suspend, wakeup in %d seconds\n", autoresume); (autoresume_step_ms > max_autoresume * 1000))
autoresume_step_ms = autoresume * 1000;
mod_timer(&autoresume_timer, jiffies +
msecs_to_jiffies(autoresume_step_ms));
DBG(cdev, "suspend, wakeup in %d milliseconds\n",
autoresume_step_ms);
autoresume_step_ms += autoresume_interval_ms;
} else } else
DBG(cdev, "%s\n", __func__); DBG(cdev, "%s\n", __func__);
} }
...@@ -316,6 +336,7 @@ static int __init zero_bind(struct usb_composite_dev *cdev) ...@@ -316,6 +336,7 @@ static int __init zero_bind(struct usb_composite_dev *cdev)
if (autoresume) { if (autoresume) {
sourcesink_driver.bmAttributes |= USB_CONFIG_ATT_WAKEUP; sourcesink_driver.bmAttributes |= USB_CONFIG_ATT_WAKEUP;
loopback_driver.bmAttributes |= USB_CONFIG_ATT_WAKEUP; loopback_driver.bmAttributes |= USB_CONFIG_ATT_WAKEUP;
autoresume_step_ms = autoresume * 1000;
} }
/* support OTG systems */ /* support OTG systems */
......
...@@ -91,7 +91,7 @@ config USB_MUSB_BLACKFIN ...@@ -91,7 +91,7 @@ config USB_MUSB_BLACKFIN
depends on (BF54x && !BF544) || (BF52x && ! BF522 && !BF523) depends on (BF54x && !BF544) || (BF52x && ! BF522 && !BF523)
config USB_MUSB_UX500 config USB_MUSB_UX500
tristate "U8500 and U5500" tristate "Ux500 platforms"
endchoice endchoice
...@@ -113,7 +113,7 @@ choice ...@@ -113,7 +113,7 @@ choice
allow using DMA on multiplatform kernels. allow using DMA on multiplatform kernels.
config USB_UX500_DMA config USB_UX500_DMA
bool 'ST Ericsson U8500 and U5500' bool 'ST Ericsson Ux500'
depends on USB_MUSB_UX500 depends on USB_MUSB_UX500
help help
Enable DMA transfers on UX500 platforms. Enable DMA transfers on UX500 platforms.
......
...@@ -89,7 +89,6 @@ struct am35x_glue { ...@@ -89,7 +89,6 @@ struct am35x_glue {
struct clk *phy_clk; struct clk *phy_clk;
struct clk *clk; struct clk *clk;
}; };
#define glue_to_musb(g) platform_get_drvdata(g->musb)
/* /*
* am35x_musb_enable - enable interrupts * am35x_musb_enable - enable interrupts
...@@ -452,14 +451,18 @@ static const struct musb_platform_ops am35x_ops = { ...@@ -452,14 +451,18 @@ static const struct musb_platform_ops am35x_ops = {
.set_vbus = am35x_musb_set_vbus, .set_vbus = am35x_musb_set_vbus,
}; };
static u64 am35x_dmamask = DMA_BIT_MASK(32); static const struct platform_device_info am35x_dev_info = {
.name = "musb-hdrc",
.id = PLATFORM_DEVID_AUTO,
.dma_mask = DMA_BIT_MASK(32),
};
static int am35x_probe(struct platform_device *pdev) static int am35x_probe(struct platform_device *pdev)
{ {
struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev); struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct platform_device *musb; struct platform_device *musb;
struct am35x_glue *glue; struct am35x_glue *glue;
struct platform_device_info pinfo;
struct clk *phy_clk; struct clk *phy_clk;
struct clk *clk; struct clk *clk;
...@@ -471,12 +474,6 @@ static int am35x_probe(struct platform_device *pdev) ...@@ -471,12 +474,6 @@ static int am35x_probe(struct platform_device *pdev)
goto err0; goto err0;
} }
musb = platform_device_alloc("musb-hdrc", PLATFORM_DEVID_AUTO);
if (!musb) {
dev_err(&pdev->dev, "failed to allocate musb device\n");
goto err1;
}
phy_clk = clk_get(&pdev->dev, "fck"); phy_clk = clk_get(&pdev->dev, "fck");
if (IS_ERR(phy_clk)) { if (IS_ERR(phy_clk)) {
dev_err(&pdev->dev, "failed to get PHY clock\n"); dev_err(&pdev->dev, "failed to get PHY clock\n");
...@@ -503,12 +500,7 @@ static int am35x_probe(struct platform_device *pdev) ...@@ -503,12 +500,7 @@ static int am35x_probe(struct platform_device *pdev)
goto err6; goto err6;
} }
musb->dev.parent = &pdev->dev;
musb->dev.dma_mask = &am35x_dmamask;
musb->dev.coherent_dma_mask = am35x_dmamask;
glue->dev = &pdev->dev; glue->dev = &pdev->dev;
glue->musb = musb;
glue->phy_clk = phy_clk; glue->phy_clk = phy_clk;
glue->clk = clk; glue->clk = clk;
...@@ -516,22 +508,17 @@ static int am35x_probe(struct platform_device *pdev) ...@@ -516,22 +508,17 @@ static int am35x_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, glue); platform_set_drvdata(pdev, glue);
ret = platform_device_add_resources(musb, pdev->resource, pinfo = am35x_dev_info;
pdev->num_resources); pinfo.parent = &pdev->dev;
if (ret) { pinfo.res = pdev->resource;
dev_err(&pdev->dev, "failed to add resources\n"); pinfo.num_res = pdev->num_resources;
goto err7; pinfo.data = pdata;
} pinfo.size_data = sizeof(*pdata);
ret = platform_device_add_data(musb, pdata, sizeof(*pdata)); glue->musb = musb = platform_device_register_full(&pinfo);
if (ret) { if (IS_ERR(musb)) {
dev_err(&pdev->dev, "failed to add platform_data\n"); ret = PTR_ERR(musb);
goto err7; dev_err(&pdev->dev, "failed to register musb device: %d\n", ret);
}
ret = platform_device_add(musb);
if (ret) {
dev_err(&pdev->dev, "failed to register musb device\n");
goto err7; goto err7;
} }
...@@ -550,9 +537,6 @@ static int am35x_probe(struct platform_device *pdev) ...@@ -550,9 +537,6 @@ static int am35x_probe(struct platform_device *pdev)
clk_put(phy_clk); clk_put(phy_clk);
err3: err3:
platform_device_put(musb);
err1:
kfree(glue); kfree(glue);
err0: err0:
...@@ -615,23 +599,16 @@ static int am35x_resume(struct device *dev) ...@@ -615,23 +599,16 @@ static int am35x_resume(struct device *dev)
return 0; return 0;
} }
static struct dev_pm_ops am35x_pm_ops = {
.suspend = am35x_suspend,
.resume = am35x_resume,
};
#define DEV_PM_OPS &am35x_pm_ops
#else
#define DEV_PM_OPS NULL
#endif #endif
static SIMPLE_DEV_PM_OPS(am35x_pm_ops, am35x_suspend, am35x_resume);
static struct platform_driver am35x_driver = { static struct platform_driver am35x_driver = {
.probe = am35x_probe, .probe = am35x_probe,
.remove = am35x_remove, .remove = am35x_remove,
.driver = { .driver = {
.name = "musb-am35x", .name = "musb-am35x",
.pm = DEV_PM_OPS, .pm = &am35x_pm_ops,
}, },
}; };
......
...@@ -561,23 +561,16 @@ static int bfin_resume(struct device *dev) ...@@ -561,23 +561,16 @@ static int bfin_resume(struct device *dev)
return 0; return 0;
} }
static struct dev_pm_ops bfin_pm_ops = {
.suspend = bfin_suspend,
.resume = bfin_resume,
};
#define DEV_PM_OPS &bfin_pm_ops
#else
#define DEV_PM_OPS NULL
#endif #endif
static SIMPLE_DEV_PM_OPS(bfin_pm_ops, bfin_suspend, bfin_resume);
static struct platform_driver bfin_driver = { static struct platform_driver bfin_driver = {
.probe = bfin_probe, .probe = bfin_probe,
.remove = __exit_p(bfin_remove), .remove = __exit_p(bfin_remove),
.driver = { .driver = {
.name = "musb-blackfin", .name = "musb-blackfin",
.pm = DEV_PM_OPS, .pm = &bfin_pm_ops,
}, },
}; };
......
...@@ -472,7 +472,11 @@ static const struct musb_platform_ops da8xx_ops = { ...@@ -472,7 +472,11 @@ static const struct musb_platform_ops da8xx_ops = {
.set_vbus = da8xx_musb_set_vbus, .set_vbus = da8xx_musb_set_vbus,
}; };
static u64 da8xx_dmamask = DMA_BIT_MASK(32); static const struct platform_device_info da8xx_dev_info = {
.name = "musb-hdrc",
.id = PLATFORM_DEVID_AUTO,
.dma_mask = DMA_BIT_MASK(32),
};
static int da8xx_probe(struct platform_device *pdev) static int da8xx_probe(struct platform_device *pdev)
{ {
...@@ -480,7 +484,7 @@ static int da8xx_probe(struct platform_device *pdev) ...@@ -480,7 +484,7 @@ static int da8xx_probe(struct platform_device *pdev)
struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev); struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct platform_device *musb; struct platform_device *musb;
struct da8xx_glue *glue; struct da8xx_glue *glue;
struct platform_device_info pinfo;
struct clk *clk; struct clk *clk;
int ret = -ENOMEM; int ret = -ENOMEM;
...@@ -491,12 +495,6 @@ static int da8xx_probe(struct platform_device *pdev) ...@@ -491,12 +495,6 @@ static int da8xx_probe(struct platform_device *pdev)
goto err0; goto err0;
} }
musb = platform_device_alloc("musb-hdrc", PLATFORM_DEVID_AUTO);
if (!musb) {
dev_err(&pdev->dev, "failed to allocate musb device\n");
goto err1;
}
clk = clk_get(&pdev->dev, "usb20"); clk = clk_get(&pdev->dev, "usb20");
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
dev_err(&pdev->dev, "failed to get clock\n"); dev_err(&pdev->dev, "failed to get clock\n");
...@@ -510,12 +508,7 @@ static int da8xx_probe(struct platform_device *pdev) ...@@ -510,12 +508,7 @@ static int da8xx_probe(struct platform_device *pdev)
goto err4; goto err4;
} }
musb->dev.parent = &pdev->dev;
musb->dev.dma_mask = &da8xx_dmamask;
musb->dev.coherent_dma_mask = da8xx_dmamask;
glue->dev = &pdev->dev; glue->dev = &pdev->dev;
glue->musb = musb;
glue->clk = clk; glue->clk = clk;
pdata->platform_ops = &da8xx_ops; pdata->platform_ops = &da8xx_ops;
...@@ -535,22 +528,17 @@ static int da8xx_probe(struct platform_device *pdev) ...@@ -535,22 +528,17 @@ static int da8xx_probe(struct platform_device *pdev)
musb_resources[1].end = pdev->resource[1].end; musb_resources[1].end = pdev->resource[1].end;
musb_resources[1].flags = pdev->resource[1].flags; musb_resources[1].flags = pdev->resource[1].flags;
ret = platform_device_add_resources(musb, musb_resources, pinfo = da8xx_dev_info;
ARRAY_SIZE(musb_resources)); pinfo.parent = &pdev->dev;
if (ret) { pinfo.res = musb_resources;
dev_err(&pdev->dev, "failed to add resources\n"); pinfo.num_res = ARRAY_SIZE(musb_resources);
goto err5; pinfo.data = pdata;
} pinfo.size_data = sizeof(*pdata);
ret = platform_device_add_data(musb, pdata, sizeof(*pdata)); glue->musb = musb = platform_device_register_full(&pinfo);
if (ret) { if (IS_ERR(musb)) {
dev_err(&pdev->dev, "failed to add platform_data\n"); ret = PTR_ERR(musb);
goto err5; dev_err(&pdev->dev, "failed to register musb device: %d\n", ret);
}
ret = platform_device_add(musb);
if (ret) {
dev_err(&pdev->dev, "failed to register musb device\n");
goto err5; goto err5;
} }
...@@ -563,9 +551,6 @@ static int da8xx_probe(struct platform_device *pdev) ...@@ -563,9 +551,6 @@ static int da8xx_probe(struct platform_device *pdev)
clk_put(clk); clk_put(clk);
err3: err3:
platform_device_put(musb);
err1:
kfree(glue); kfree(glue);
err0: err0:
......
...@@ -505,14 +505,19 @@ static const struct musb_platform_ops davinci_ops = { ...@@ -505,14 +505,19 @@ static const struct musb_platform_ops davinci_ops = {
.set_vbus = davinci_musb_set_vbus, .set_vbus = davinci_musb_set_vbus,
}; };
static u64 davinci_dmamask = DMA_BIT_MASK(32); static const struct platform_device_info davinci_dev_info = {
.name = "musb-hdrc",
.id = PLATFORM_DEVID_AUTO,
.dma_mask = DMA_BIT_MASK(32),
};
static int davinci_probe(struct platform_device *pdev) static int davinci_probe(struct platform_device *pdev)
{ {
struct resource musb_resources[2]; struct resource musb_resources[3];
struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev); struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct platform_device *musb; struct platform_device *musb;
struct davinci_glue *glue; struct davinci_glue *glue;
struct platform_device_info pinfo;
struct clk *clk; struct clk *clk;
int ret = -ENOMEM; int ret = -ENOMEM;
...@@ -523,12 +528,6 @@ static int davinci_probe(struct platform_device *pdev) ...@@ -523,12 +528,6 @@ static int davinci_probe(struct platform_device *pdev)
goto err0; goto err0;
} }
musb = platform_device_alloc("musb-hdrc", PLATFORM_DEVID_AUTO);
if (!musb) {
dev_err(&pdev->dev, "failed to allocate musb device\n");
goto err1;
}
clk = clk_get(&pdev->dev, "usb"); clk = clk_get(&pdev->dev, "usb");
if (IS_ERR(clk)) { if (IS_ERR(clk)) {
dev_err(&pdev->dev, "failed to get clock\n"); dev_err(&pdev->dev, "failed to get clock\n");
...@@ -542,12 +541,7 @@ static int davinci_probe(struct platform_device *pdev) ...@@ -542,12 +541,7 @@ static int davinci_probe(struct platform_device *pdev)
goto err4; goto err4;
} }
musb->dev.parent = &pdev->dev;
musb->dev.dma_mask = &davinci_dmamask;
musb->dev.coherent_dma_mask = davinci_dmamask;
glue->dev = &pdev->dev; glue->dev = &pdev->dev;
glue->musb = musb;
glue->clk = clk; glue->clk = clk;
pdata->platform_ops = &davinci_ops; pdata->platform_ops = &davinci_ops;
...@@ -567,22 +561,26 @@ static int davinci_probe(struct platform_device *pdev) ...@@ -567,22 +561,26 @@ static int davinci_probe(struct platform_device *pdev)
musb_resources[1].end = pdev->resource[1].end; musb_resources[1].end = pdev->resource[1].end;
musb_resources[1].flags = pdev->resource[1].flags; musb_resources[1].flags = pdev->resource[1].flags;
ret = platform_device_add_resources(musb, musb_resources, /*
ARRAY_SIZE(musb_resources)); * For DM6467 3 resources are passed. A placeholder for the 3rd
if (ret) { * resource is always there, so it's safe to always copy it...
dev_err(&pdev->dev, "failed to add resources\n"); */
goto err5; musb_resources[2].name = pdev->resource[2].name;
} musb_resources[2].start = pdev->resource[2].start;
musb_resources[2].end = pdev->resource[2].end;
ret = platform_device_add_data(musb, pdata, sizeof(*pdata)); musb_resources[2].flags = pdev->resource[2].flags;
if (ret) {
dev_err(&pdev->dev, "failed to add platform_data\n"); pinfo = davinci_dev_info;
goto err5; pinfo.parent = &pdev->dev;
} pinfo.res = musb_resources;
pinfo.num_res = ARRAY_SIZE(musb_resources);
ret = platform_device_add(musb); pinfo.data = pdata;
if (ret) { pinfo.size_data = sizeof(*pdata);
dev_err(&pdev->dev, "failed to register musb device\n");
glue->musb = musb = platform_device_register_full(&pinfo);
if (IS_ERR(musb)) {
ret = PTR_ERR(musb);
dev_err(&pdev->dev, "failed to register musb device: %d\n", ret);
goto err5; goto err5;
} }
...@@ -595,9 +593,6 @@ static int davinci_probe(struct platform_device *pdev) ...@@ -595,9 +593,6 @@ static int davinci_probe(struct platform_device *pdev)
clk_put(clk); clk_put(clk);
err3: err3:
platform_device_put(musb);
err1:
kfree(glue); kfree(glue);
err0: err0:
......
...@@ -46,7 +46,7 @@ static struct platform_driver am335x_child_driver = { ...@@ -46,7 +46,7 @@ static struct platform_driver am335x_child_driver = {
.remove = am335x_child_remove, .remove = am335x_child_remove,
.driver = { .driver = {
.name = "am335x-usb-childs", .name = "am335x-usb-childs",
.of_match_table = of_match_ptr(am335x_child_of_match), .of_match_table = am335x_child_of_match,
}, },
}; };
......
...@@ -1809,8 +1809,7 @@ static void musb_free(struct musb *musb) ...@@ -1809,8 +1809,7 @@ static void musb_free(struct musb *musb)
disable_irq_wake(musb->nIrq); disable_irq_wake(musb->nIrq);
free_irq(musb->nIrq, musb); free_irq(musb->nIrq, musb);
} }
if (musb->dma_controller) cancel_work_sync(&musb->irq_work);
dma_controller_destroy(musb->dma_controller);
musb_host_free(musb); musb_host_free(musb);
} }
...@@ -1885,8 +1884,13 @@ musb_init_controller(struct device *dev, int nIrq, void __iomem *ctrl) ...@@ -1885,8 +1884,13 @@ musb_init_controller(struct device *dev, int nIrq, void __iomem *ctrl)
pm_runtime_get_sync(musb->controller); pm_runtime_get_sync(musb->controller);
if (use_dma && dev->dma_mask) if (use_dma && dev->dma_mask) {
musb->dma_controller = dma_controller_create(musb, musb->mregs); musb->dma_controller = dma_controller_create(musb, musb->mregs);
if (IS_ERR(musb->dma_controller)) {
status = PTR_ERR(musb->dma_controller);
goto fail2_5;
}
}
/* be sure interrupts are disabled before connecting ISR */ /* be sure interrupts are disabled before connecting ISR */
musb_platform_disable(musb); musb_platform_disable(musb);
...@@ -1946,6 +1950,8 @@ musb_init_controller(struct device *dev, int nIrq, void __iomem *ctrl) ...@@ -1946,6 +1950,8 @@ musb_init_controller(struct device *dev, int nIrq, void __iomem *ctrl)
if (status < 0) if (status < 0)
goto fail3; goto fail3;
status = musb_gadget_setup(musb); status = musb_gadget_setup(musb);
if (status)
musb_host_cleanup(musb);
break; break;
default: default:
dev_err(dev, "unsupported port mode %d\n", musb->port_mode); dev_err(dev, "unsupported port mode %d\n", musb->port_mode);
...@@ -1972,10 +1978,12 @@ musb_init_controller(struct device *dev, int nIrq, void __iomem *ctrl) ...@@ -1972,10 +1978,12 @@ musb_init_controller(struct device *dev, int nIrq, void __iomem *ctrl)
fail4: fail4:
musb_gadget_cleanup(musb); musb_gadget_cleanup(musb);
musb_host_cleanup(musb);
fail3: fail3:
if (musb->dma_controller) if (musb->dma_controller)
dma_controller_destroy(musb->dma_controller); dma_controller_destroy(musb->dma_controller);
fail2_5:
pm_runtime_put_sync(musb->controller); pm_runtime_put_sync(musb->controller);
fail2: fail2:
...@@ -2032,6 +2040,9 @@ static int musb_remove(struct platform_device *pdev) ...@@ -2032,6 +2040,9 @@ static int musb_remove(struct platform_device *pdev)
musb_exit_debugfs(musb); musb_exit_debugfs(musb);
musb_shutdown(pdev); musb_shutdown(pdev);
if (musb->dma_controller)
dma_controller_destroy(musb->dma_controller);
musb_free(musb); musb_free(musb);
device_init_wakeup(dev, 0); device_init_wakeup(dev, 0);
return 0; return 0;
......
...@@ -484,6 +484,7 @@ static int cppi41_dma_controller_start(struct cppi41_dma_controller *controller) ...@@ -484,6 +484,7 @@ static int cppi41_dma_controller_start(struct cppi41_dma_controller *controller)
if (ret) if (ret)
goto err; goto err;
ret = -EINVAL;
if (port > MUSB_DMA_NUM_CHANNELS || !port) if (port > MUSB_DMA_NUM_CHANNELS || !port)
goto err; goto err;
if (is_tx) if (is_tx)
...@@ -503,6 +504,7 @@ static int cppi41_dma_controller_start(struct cppi41_dma_controller *controller) ...@@ -503,6 +504,7 @@ static int cppi41_dma_controller_start(struct cppi41_dma_controller *controller)
dc = dma_request_slave_channel(dev, str); dc = dma_request_slave_channel(dev, str);
if (!dc) { if (!dc) {
dev_err(dev, "Falied to request %s.\n", str); dev_err(dev, "Falied to request %s.\n", str);
ret = -EPROBE_DEFER;
goto err; goto err;
} }
cppi41_channel->dc = dc; cppi41_channel->dc = dc;
...@@ -510,7 +512,7 @@ static int cppi41_dma_controller_start(struct cppi41_dma_controller *controller) ...@@ -510,7 +512,7 @@ static int cppi41_dma_controller_start(struct cppi41_dma_controller *controller)
return 0; return 0;
err: err:
cppi41_release_all_dma_chans(controller); cppi41_release_all_dma_chans(controller);
return -EINVAL; return ret;
} }
void dma_controller_destroy(struct dma_controller *c) void dma_controller_destroy(struct dma_controller *c)
...@@ -526,7 +528,7 @@ struct dma_controller *dma_controller_create(struct musb *musb, ...@@ -526,7 +528,7 @@ struct dma_controller *dma_controller_create(struct musb *musb,
void __iomem *base) void __iomem *base)
{ {
struct cppi41_dma_controller *controller; struct cppi41_dma_controller *controller;
int ret; int ret = 0;
if (!musb->controller->of_node) { if (!musb->controller->of_node) {
dev_err(musb->controller, "Need DT for the DMA engine.\n"); dev_err(musb->controller, "Need DT for the DMA engine.\n");
...@@ -553,5 +555,7 @@ struct dma_controller *dma_controller_create(struct musb *musb, ...@@ -553,5 +555,7 @@ struct dma_controller *dma_controller_create(struct musb *musb,
plat_get_fail: plat_get_fail:
kfree(controller); kfree(controller);
kzalloc_fail: kzalloc_fail:
if (ret == -EPROBE_DEFER)
return ERR_PTR(ret);
return NULL; return NULL;
} }
...@@ -121,6 +121,43 @@ struct dsps_glue { ...@@ -121,6 +121,43 @@ struct dsps_glue {
unsigned long last_timer; /* last timer data for each instance */ unsigned long last_timer; /* last timer data for each instance */
}; };
static void dsps_musb_try_idle(struct musb *musb, unsigned long timeout)
{
struct device *dev = musb->controller;
struct dsps_glue *glue = dev_get_drvdata(dev->parent);
if (timeout == 0)
timeout = jiffies + msecs_to_jiffies(3);
/* Never idle if active, or when VBUS timeout is not set as host */
if (musb->is_active || (musb->a_wait_bcon == 0 &&
musb->xceiv->state == OTG_STATE_A_WAIT_BCON)) {
dev_dbg(musb->controller, "%s active, deleting timer\n",
usb_otg_state_string(musb->xceiv->state));
del_timer(&glue->timer);
glue->last_timer = jiffies;
return;
}
if (musb->port_mode != MUSB_PORT_MODE_DUAL_ROLE)
return;
if (!musb->g.dev.driver)
return;
if (time_after(glue->last_timer, timeout) &&
timer_pending(&glue->timer)) {
dev_dbg(musb->controller,
"Longer idle timer already pending, ignoring...\n");
return;
}
glue->last_timer = timeout;
dev_dbg(musb->controller, "%s inactive, starting idle timer for %u ms\n",
usb_otg_state_string(musb->xceiv->state),
jiffies_to_msecs(timeout - jiffies));
mod_timer(&glue->timer, timeout);
}
/** /**
* dsps_musb_enable - enable interrupts * dsps_musb_enable - enable interrupts
*/ */
...@@ -143,6 +180,7 @@ static void dsps_musb_enable(struct musb *musb) ...@@ -143,6 +180,7 @@ static void dsps_musb_enable(struct musb *musb)
/* Force the DRVVBUS IRQ so we can start polling for ID change. */ /* Force the DRVVBUS IRQ so we can start polling for ID change. */
dsps_writel(reg_base, wrp->coreintr_set, dsps_writel(reg_base, wrp->coreintr_set,
(1 << wrp->drvvbus) << wrp->usb_shift); (1 << wrp->drvvbus) << wrp->usb_shift);
dsps_musb_try_idle(musb, 0);
} }
/** /**
...@@ -171,6 +209,7 @@ static void otg_timer(unsigned long _musb) ...@@ -171,6 +209,7 @@ static void otg_timer(unsigned long _musb)
const struct dsps_musb_wrapper *wrp = glue->wrp; const struct dsps_musb_wrapper *wrp = glue->wrp;
u8 devctl; u8 devctl;
unsigned long flags; unsigned long flags;
int skip_session = 0;
/* /*
* We poll because DSPS IP's won't expose several OTG-critical * We poll because DSPS IP's won't expose several OTG-critical
...@@ -183,10 +222,12 @@ static void otg_timer(unsigned long _musb) ...@@ -183,10 +222,12 @@ static void otg_timer(unsigned long _musb)
spin_lock_irqsave(&musb->lock, flags); spin_lock_irqsave(&musb->lock, flags);
switch (musb->xceiv->state) { switch (musb->xceiv->state) {
case OTG_STATE_A_WAIT_BCON: case OTG_STATE_A_WAIT_BCON:
devctl &= ~MUSB_DEVCTL_SESSION; dsps_writeb(musb->mregs, MUSB_DEVCTL, 0);
dsps_writeb(musb->mregs, MUSB_DEVCTL, devctl); skip_session = 1;
/* fall */
devctl = dsps_readb(musb->mregs, MUSB_DEVCTL); case OTG_STATE_A_IDLE:
case OTG_STATE_B_IDLE:
if (devctl & MUSB_DEVCTL_BDEVICE) { if (devctl & MUSB_DEVCTL_BDEVICE) {
musb->xceiv->state = OTG_STATE_B_IDLE; musb->xceiv->state = OTG_STATE_B_IDLE;
MUSB_DEV_MODE(musb); MUSB_DEV_MODE(musb);
...@@ -194,60 +235,21 @@ static void otg_timer(unsigned long _musb) ...@@ -194,60 +235,21 @@ static void otg_timer(unsigned long _musb)
musb->xceiv->state = OTG_STATE_A_IDLE; musb->xceiv->state = OTG_STATE_A_IDLE;
MUSB_HST_MODE(musb); MUSB_HST_MODE(musb);
} }
if (!(devctl & MUSB_DEVCTL_SESSION) && !skip_session)
dsps_writeb(mregs, MUSB_DEVCTL, MUSB_DEVCTL_SESSION);
mod_timer(&glue->timer, jiffies + wrp->poll_seconds * HZ);
break; break;
case OTG_STATE_A_WAIT_VFALL: case OTG_STATE_A_WAIT_VFALL:
musb->xceiv->state = OTG_STATE_A_WAIT_VRISE; musb->xceiv->state = OTG_STATE_A_WAIT_VRISE;
dsps_writel(musb->ctrl_base, wrp->coreintr_set, dsps_writel(musb->ctrl_base, wrp->coreintr_set,
MUSB_INTR_VBUSERROR << wrp->usb_shift); MUSB_INTR_VBUSERROR << wrp->usb_shift);
break; break;
case OTG_STATE_B_IDLE:
devctl = dsps_readb(mregs, MUSB_DEVCTL);
if (devctl & MUSB_DEVCTL_BDEVICE)
mod_timer(&glue->timer,
jiffies + wrp->poll_seconds * HZ);
else
musb->xceiv->state = OTG_STATE_A_IDLE;
break;
default: default:
break; break;
} }
spin_unlock_irqrestore(&musb->lock, flags); spin_unlock_irqrestore(&musb->lock, flags);
} }
static void dsps_musb_try_idle(struct musb *musb, unsigned long timeout)
{
struct device *dev = musb->controller;
struct dsps_glue *glue = dev_get_drvdata(dev->parent);
if (timeout == 0)
timeout = jiffies + msecs_to_jiffies(3);
/* Never idle if active, or when VBUS timeout is not set as host */
if (musb->is_active || (musb->a_wait_bcon == 0 &&
musb->xceiv->state == OTG_STATE_A_WAIT_BCON)) {
dev_dbg(musb->controller, "%s active, deleting timer\n",
usb_otg_state_string(musb->xceiv->state));
del_timer(&glue->timer);
glue->last_timer = jiffies;
return;
}
if (musb->port_mode == MUSB_PORT_MODE_HOST)
return;
if (time_after(glue->last_timer, timeout) &&
timer_pending(&glue->timer)) {
dev_dbg(musb->controller,
"Longer idle timer already pending, ignoring...\n");
return;
}
glue->last_timer = timeout;
dev_dbg(musb->controller, "%s inactive, starting idle timer for %u ms\n",
usb_otg_state_string(musb->xceiv->state),
jiffies_to_msecs(timeout - jiffies));
mod_timer(&glue->timer, timeout);
}
static irqreturn_t dsps_interrupt(int irq, void *hci) static irqreturn_t dsps_interrupt(int irq, void *hci)
{ {
struct musb *musb = hci; struct musb *musb = hci;
...@@ -631,7 +633,7 @@ static struct platform_driver dsps_usbss_driver = { ...@@ -631,7 +633,7 @@ static struct platform_driver dsps_usbss_driver = {
.remove = dsps_remove, .remove = dsps_remove,
.driver = { .driver = {
.name = "musb-dsps", .name = "musb-dsps",
.of_match_table = of_match_ptr(musb_dsps_of_match), .of_match_table = musb_dsps_of_match,
}, },
}; };
......
...@@ -220,6 +220,23 @@ int musb_hub_status_data(struct usb_hcd *hcd, char *buf) ...@@ -220,6 +220,23 @@ int musb_hub_status_data(struct usb_hcd *hcd, char *buf)
return retval; return retval;
} }
static int musb_has_gadget(struct musb *musb)
{
/*
* In host-only mode we start a connection right away. In OTG mode
* we have to wait until we loaded a gadget. We don't really need a
* gadget if we operate as a host but we should not start a session
* as a device without a gadget or else we explode.
*/
#ifdef CONFIG_USB_MUSB_HOST
return 1;
#else
if (musb->port_mode == MUSB_PORT_MODE_HOST)
return 1;
return musb->g.dev.driver != NULL;
#endif
}
int musb_hub_control( int musb_hub_control(
struct usb_hcd *hcd, struct usb_hcd *hcd,
u16 typeReq, u16 typeReq,
...@@ -362,7 +379,7 @@ int musb_hub_control( ...@@ -362,7 +379,7 @@ int musb_hub_control(
* initialization logic, e.g. for OTG, or change any * initialization logic, e.g. for OTG, or change any
* logic relating to VBUS power-up. * logic relating to VBUS power-up.
*/ */
if (!hcd->self.is_b_host) if (!hcd->self.is_b_host && musb_has_gadget(musb))
musb_start(musb); musb_start(musb);
break; break;
case USB_PORT_FEAT_RESET: case USB_PORT_FEAT_RESET:
......
...@@ -306,6 +306,9 @@ static void omap_musb_set_mailbox(struct omap2430_glue *glue) ...@@ -306,6 +306,9 @@ static void omap_musb_set_mailbox(struct omap2430_glue *glue)
default: default:
dev_dbg(dev, "ID float\n"); dev_dbg(dev, "ID float\n");
} }
atomic_notifier_call_chain(&musb->xceiv->notifier,
musb->xceiv->last_event, NULL);
} }
......
...@@ -1152,7 +1152,11 @@ static const struct musb_platform_ops tusb_ops = { ...@@ -1152,7 +1152,11 @@ static const struct musb_platform_ops tusb_ops = {
.set_vbus = tusb_musb_set_vbus, .set_vbus = tusb_musb_set_vbus,
}; };
static u64 tusb_dmamask = DMA_BIT_MASK(32); static const struct platform_device_info tusb_dev_info = {
.name = "musb-hdrc",
.id = PLATFORM_DEVID_AUTO,
.dma_mask = DMA_BIT_MASK(32),
};
static int tusb_probe(struct platform_device *pdev) static int tusb_probe(struct platform_device *pdev)
{ {
...@@ -1160,7 +1164,7 @@ static int tusb_probe(struct platform_device *pdev) ...@@ -1160,7 +1164,7 @@ static int tusb_probe(struct platform_device *pdev)
struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev); struct musb_hdrc_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct platform_device *musb; struct platform_device *musb;
struct tusb6010_glue *glue; struct tusb6010_glue *glue;
struct platform_device_info pinfo;
int ret = -ENOMEM; int ret = -ENOMEM;
glue = kzalloc(sizeof(*glue), GFP_KERNEL); glue = kzalloc(sizeof(*glue), GFP_KERNEL);
...@@ -1169,18 +1173,7 @@ static int tusb_probe(struct platform_device *pdev) ...@@ -1169,18 +1173,7 @@ static int tusb_probe(struct platform_device *pdev)
goto err0; goto err0;
} }
musb = platform_device_alloc("musb-hdrc", PLATFORM_DEVID_AUTO);
if (!musb) {
dev_err(&pdev->dev, "failed to allocate musb device\n");
goto err1;
}
musb->dev.parent = &pdev->dev;
musb->dev.dma_mask = &tusb_dmamask;
musb->dev.coherent_dma_mask = tusb_dmamask;
glue->dev = &pdev->dev; glue->dev = &pdev->dev;
glue->musb = musb;
pdata->platform_ops = &tusb_ops; pdata->platform_ops = &tusb_ops;
...@@ -1204,31 +1197,23 @@ static int tusb_probe(struct platform_device *pdev) ...@@ -1204,31 +1197,23 @@ static int tusb_probe(struct platform_device *pdev)
musb_resources[2].end = pdev->resource[2].end; musb_resources[2].end = pdev->resource[2].end;
musb_resources[2].flags = pdev->resource[2].flags; musb_resources[2].flags = pdev->resource[2].flags;
ret = platform_device_add_resources(musb, musb_resources, pinfo = tusb_dev_info;
ARRAY_SIZE(musb_resources)); pinfo.parent = &pdev->dev;
if (ret) { pinfo.res = musb_resources;
dev_err(&pdev->dev, "failed to add resources\n"); pinfo.num_res = ARRAY_SIZE(musb_resources);
goto err3; pinfo.data = pdata;
} pinfo.size_data = sizeof(*pdata);
ret = platform_device_add_data(musb, pdata, sizeof(*pdata)); glue->musb = musb = platform_device_register_full(&pinfo);
if (ret) { if (IS_ERR(musb)) {
dev_err(&pdev->dev, "failed to add platform_data\n"); ret = PTR_ERR(musb);
goto err3; dev_err(&pdev->dev, "failed to register musb device: %d\n", ret);
}
ret = platform_device_add(musb);
if (ret) {
dev_err(&pdev->dev, "failed to register musb device\n");
goto err3; goto err3;
} }
return 0; return 0;
err3: err3:
platform_device_put(musb);
err1:
kfree(glue); kfree(glue);
err0: err0:
......
...@@ -376,17 +376,10 @@ static int ux500_resume(struct device *dev) ...@@ -376,17 +376,10 @@ static int ux500_resume(struct device *dev)
return 0; return 0;
} }
static const struct dev_pm_ops ux500_pm_ops = {
.suspend = ux500_suspend,
.resume = ux500_resume,
};
#define DEV_PM_OPS (&ux500_pm_ops)
#else
#define DEV_PM_OPS NULL
#endif #endif
static SIMPLE_DEV_PM_OPS(ux500_pm_ops, ux500_suspend, ux500_resume);
static const struct of_device_id ux500_match[] = { static const struct of_device_id ux500_match[] = {
{ .compatible = "stericsson,db8500-musb", }, { .compatible = "stericsson,db8500-musb", },
{} {}
...@@ -397,7 +390,7 @@ static struct platform_driver ux500_driver = { ...@@ -397,7 +390,7 @@ static struct platform_driver ux500_driver = {
.remove = ux500_remove, .remove = ux500_remove,
.driver = { .driver = {
.name = "musb-ux500", .name = "musb-ux500",
.pm = DEV_PM_OPS, .pm = &ux500_pm_ops,
.of_match_table = ux500_match, .of_match_table = ux500_match,
}, },
}; };
......
...@@ -194,6 +194,19 @@ config USB_RCAR_PHY ...@@ -194,6 +194,19 @@ config USB_RCAR_PHY
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 phy-rcar-usb. module will be called phy-rcar-usb.
config USB_RCAR_GEN2_PHY
tristate "Renesas R-Car Gen2 USB PHY support"
depends on ARCH_R8A7790 || ARCH_R8A7791 || COMPILE_TEST
select USB_PHY
help
Say Y here to add support for the Renesas R-Car Gen2 USB PHY driver.
It is typically used to control internal USB PHY for USBHS,
and to configure shared USB channels 0 and 2.
This driver supports R8A7790 and R8A7791.
To compile this driver as a module, choose M here: the
module will be called phy-rcar-gen2-usb.
config USB_ULPI config USB_ULPI
bool "Generic ULPI Transceiver Driver" bool "Generic ULPI Transceiver Driver"
depends on ARM depends on ARM
......
...@@ -27,5 +27,6 @@ obj-$(CONFIG_USB_MSM_OTG) += phy-msm-usb.o ...@@ -27,5 +27,6 @@ obj-$(CONFIG_USB_MSM_OTG) += phy-msm-usb.o
obj-$(CONFIG_USB_MV_OTG) += phy-mv-usb.o obj-$(CONFIG_USB_MV_OTG) += phy-mv-usb.o
obj-$(CONFIG_USB_MXS_PHY) += phy-mxs-usb.o obj-$(CONFIG_USB_MXS_PHY) += phy-mxs-usb.o
obj-$(CONFIG_USB_RCAR_PHY) += phy-rcar-usb.o obj-$(CONFIG_USB_RCAR_PHY) += phy-rcar-usb.o
obj-$(CONFIG_USB_RCAR_GEN2_PHY) += phy-rcar-gen2-usb.o
obj-$(CONFIG_USB_ULPI) += phy-ulpi.o obj-$(CONFIG_USB_ULPI) += phy-ulpi.o
obj-$(CONFIG_USB_ULPI_VIEWPORT) += phy-ulpi-viewport.o obj-$(CONFIG_USB_ULPI_VIEWPORT) += phy-ulpi-viewport.o
...@@ -26,6 +26,41 @@ struct am335x_control_usb { ...@@ -26,6 +26,41 @@ struct am335x_control_usb {
#define USBPHY_OTGVDET_EN (1 << 19) #define USBPHY_OTGVDET_EN (1 << 19)
#define USBPHY_OTGSESSEND_EN (1 << 20) #define USBPHY_OTGSESSEND_EN (1 << 20)
#define AM335X_PHY0_WK_EN (1 << 0)
#define AM335X_PHY1_WK_EN (1 << 8)
static void am335x_phy_wkup(struct phy_control *phy_ctrl, u32 id, bool on)
{
struct am335x_control_usb *usb_ctrl;
u32 val;
u32 reg;
usb_ctrl = container_of(phy_ctrl, struct am335x_control_usb, phy_ctrl);
switch (id) {
case 0:
reg = AM335X_PHY0_WK_EN;
break;
case 1:
reg = AM335X_PHY1_WK_EN;
break;
default:
WARN_ON(1);
return;
}
spin_lock(&usb_ctrl->lock);
val = readl(usb_ctrl->wkup);
if (on)
val |= reg;
else
val &= ~reg;
writel(val, usb_ctrl->wkup);
spin_unlock(&usb_ctrl->lock);
}
static void am335x_phy_power(struct phy_control *phy_ctrl, u32 id, bool on) static void am335x_phy_power(struct phy_control *phy_ctrl, u32 id, bool on)
{ {
struct am335x_control_usb *usb_ctrl; struct am335x_control_usb *usb_ctrl;
...@@ -59,6 +94,7 @@ static void am335x_phy_power(struct phy_control *phy_ctrl, u32 id, bool on) ...@@ -59,6 +94,7 @@ static void am335x_phy_power(struct phy_control *phy_ctrl, u32 id, bool on)
static const struct phy_control ctrl_am335x = { static const struct phy_control ctrl_am335x = {
.phy_power = am335x_phy_power, .phy_power = am335x_phy_power,
.phy_wkup = am335x_phy_wkup,
}; };
static const struct of_device_id omap_control_usb_id_table[] = { static const struct of_device_id omap_control_usb_id_table[] = {
...@@ -117,6 +153,12 @@ static int am335x_control_usb_probe(struct platform_device *pdev) ...@@ -117,6 +153,12 @@ static int am335x_control_usb_probe(struct platform_device *pdev)
ctrl_usb->phy_reg = devm_ioremap_resource(&pdev->dev, res); ctrl_usb->phy_reg = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(ctrl_usb->phy_reg)) if (IS_ERR(ctrl_usb->phy_reg))
return PTR_ERR(ctrl_usb->phy_reg); return PTR_ERR(ctrl_usb->phy_reg);
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "wakeup");
ctrl_usb->wkup = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(ctrl_usb->wkup))
return PTR_ERR(ctrl_usb->wkup);
spin_lock_init(&ctrl_usb->lock); spin_lock_init(&ctrl_usb->lock);
ctrl_usb->phy_ctrl = *phy_ctrl; ctrl_usb->phy_ctrl = *phy_ctrl;
...@@ -129,7 +171,7 @@ static struct platform_driver am335x_control_driver = { ...@@ -129,7 +171,7 @@ static struct platform_driver am335x_control_driver = {
.driver = { .driver = {
.name = "am335x-control-usb", .name = "am335x-control-usb",
.owner = THIS_MODULE, .owner = THIS_MODULE,
.of_match_table = of_match_ptr(omap_control_usb_id_table), .of_match_table = omap_control_usb_id_table,
}, },
}; };
......
...@@ -53,21 +53,20 @@ static int am335x_phy_probe(struct platform_device *pdev) ...@@ -53,21 +53,20 @@ static int am335x_phy_probe(struct platform_device *pdev)
} }
ret = usb_phy_gen_create_phy(dev, &am_phy->usb_phy_gen, ret = usb_phy_gen_create_phy(dev, &am_phy->usb_phy_gen,
USB_PHY_TYPE_USB2, 0, false, false); USB_PHY_TYPE_USB2, 0, false);
if (ret) if (ret)
return ret; return ret;
ret = usb_add_phy_dev(&am_phy->usb_phy_gen.phy); ret = usb_add_phy_dev(&am_phy->usb_phy_gen.phy);
if (ret) if (ret)
goto err_add; return ret;
am_phy->usb_phy_gen.phy.init = am335x_init; am_phy->usb_phy_gen.phy.init = am335x_init;
am_phy->usb_phy_gen.phy.shutdown = am335x_shutdown; am_phy->usb_phy_gen.phy.shutdown = am335x_shutdown;
platform_set_drvdata(pdev, am_phy); platform_set_drvdata(pdev, am_phy);
return 0; return 0;
err_add:
usb_phy_gen_cleanup_phy(&am_phy->usb_phy_gen);
return ret; return ret;
} }
...@@ -79,6 +78,40 @@ static int am335x_phy_remove(struct platform_device *pdev) ...@@ -79,6 +78,40 @@ static int am335x_phy_remove(struct platform_device *pdev)
return 0; return 0;
} }
#ifdef CONFIG_PM_RUNTIME
static int am335x_phy_runtime_suspend(struct device *dev)
{
struct platform_device *pdev = to_platform_device(dev);
struct am335x_phy *am_phy = platform_get_drvdata(pdev);
if (device_may_wakeup(dev))
phy_ctrl_wkup(am_phy->phy_ctrl, am_phy->id, true);
phy_ctrl_power(am_phy->phy_ctrl, am_phy->id, false);
return 0;
}
static int am335x_phy_runtime_resume(struct device *dev)
{
struct platform_device *pdev = to_platform_device(dev);
struct am335x_phy *am_phy = platform_get_drvdata(pdev);
phy_ctrl_power(am_phy->phy_ctrl, am_phy->id, true);
if (device_may_wakeup(dev))
phy_ctrl_wkup(am_phy->phy_ctrl, am_phy->id, false);
return 0;
}
static const struct dev_pm_ops am335x_pm_ops = {
SET_RUNTIME_PM_OPS(am335x_phy_runtime_suspend,
am335x_phy_runtime_resume, NULL)
};
#define DEV_PM_OPS (&am335x_pm_ops)
#else
#define DEV_PM_OPS NULL
#endif
static const struct of_device_id am335x_phy_ids[] = { static const struct of_device_id am335x_phy_ids[] = {
{ .compatible = "ti,am335x-usb-phy" }, { .compatible = "ti,am335x-usb-phy" },
{ } { }
...@@ -91,7 +124,8 @@ static struct platform_driver am335x_phy_driver = { ...@@ -91,7 +124,8 @@ static struct platform_driver am335x_phy_driver = {
.driver = { .driver = {
.name = "am335x-phy-driver", .name = "am335x-phy-driver",
.owner = THIS_MODULE, .owner = THIS_MODULE,
.of_match_table = of_match_ptr(am335x_phy_ids), .pm = DEV_PM_OPS,
.of_match_table = am335x_phy_ids,
}, },
}; };
......
This diff is collapsed.
...@@ -401,6 +401,6 @@ struct fsl_otg_config { ...@@ -401,6 +401,6 @@ struct fsl_otg_config {
#define GET_A_BUS_REQ _IOR(OTG_IOCTL_MAGIC, 8, int) #define GET_A_BUS_REQ _IOR(OTG_IOCTL_MAGIC, 8, int)
#define GET_B_BUS_REQ _IOR(OTG_IOCTL_MAGIC, 9, int) #define GET_B_BUS_REQ _IOR(OTG_IOCTL_MAGIC, 9, int)
void fsl_otg_add_timer(void *timer); void fsl_otg_add_timer(struct otg_fsm *fsm, void *timer);
void fsl_otg_del_timer(void *timer); void fsl_otg_del_timer(struct otg_fsm *fsm, void *timer);
void fsl_otg_pulse_vbus(void); void fsl_otg_pulse_vbus(void);
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -6,15 +6,14 @@ struct usb_phy_gen_xceiv { ...@@ -6,15 +6,14 @@ struct usb_phy_gen_xceiv {
struct device *dev; struct device *dev;
struct clk *clk; struct clk *clk;
struct regulator *vcc; struct regulator *vcc;
struct regulator *reset; int gpio_reset;
bool reset_active_low;
}; };
int usb_gen_phy_init(struct usb_phy *phy); int usb_gen_phy_init(struct usb_phy *phy);
void usb_gen_phy_shutdown(struct usb_phy *phy); void usb_gen_phy_shutdown(struct usb_phy *phy);
int usb_phy_gen_create_phy(struct device *dev, struct usb_phy_gen_xceiv *nop, int usb_phy_gen_create_phy(struct device *dev, struct usb_phy_gen_xceiv *nop,
enum usb_phy_type type, u32 clk_rate, bool needs_vcc, enum usb_phy_type type, u32 clk_rate, bool needs_vcc);
bool needs_reset);
void usb_phy_gen_cleanup_phy(struct usb_phy_gen_xceiv *nop);
#endif #endif
This diff is collapsed.
...@@ -1090,7 +1090,7 @@ static struct platform_driver tegra_usb_phy_driver = { ...@@ -1090,7 +1090,7 @@ static struct platform_driver tegra_usb_phy_driver = {
.driver = { .driver = {
.name = "tegra-phy", .name = "tegra-phy",
.owner = THIS_MODULE, .owner = THIS_MODULE,
.of_match_table = of_match_ptr(tegra_usb_phy_id_table), .of_match_table = tegra_usb_phy_id_table,
}, },
}; };
module_platform_driver(tegra_usb_phy_driver); module_platform_driver(tegra_usb_phy_driver);
......
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
#include <linux/err.h> #include <linux/err.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/of.h>
/* usb register definitions */ /* usb register definitions */
#define USB_VENDOR_ID_LSB 0x00 #define USB_VENDOR_ID_LSB 0x00
......
...@@ -98,7 +98,7 @@ struct usb_phy *devm_usb_get_phy(struct device *dev, enum usb_phy_type type) ...@@ -98,7 +98,7 @@ struct usb_phy *devm_usb_get_phy(struct device *dev, enum usb_phy_type type)
ptr = devres_alloc(devm_usb_phy_release, sizeof(*ptr), GFP_KERNEL); ptr = devres_alloc(devm_usb_phy_release, sizeof(*ptr), GFP_KERNEL);
if (!ptr) if (!ptr)
return NULL; return ERR_PTR(-ENOMEM);
phy = usb_get_phy(type); phy = usb_get_phy(type);
if (!IS_ERR(phy)) { if (!IS_ERR(phy)) {
......
/*
* Copyright (C) 2013 Renesas Solutions Corp.
* Copyright (C) 2013 Cogent Embedded, Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#ifndef __USB_RCAR_GEN2_PHY_H
#define __USB_RCAR_GEN2_PHY_H
#include <linux/types.h>
struct rcar_gen2_phy_platform_data {
/* USB channel 0 configuration */
bool chan0_pci:1; /* true: PCI USB host 0, false: USBHS */
/* USB channel 2 configuration */
bool chan2_pci:1; /* true: PCI USB host 2, false: USBSS */
};
#endif
...@@ -9,7 +9,8 @@ struct usb_phy_gen_xceiv_platform_data { ...@@ -9,7 +9,8 @@ struct usb_phy_gen_xceiv_platform_data {
/* if set fails with -EPROBE_DEFER if can't get regulator */ /* if set fails with -EPROBE_DEFER if can't get regulator */
unsigned int needs_vcc:1; unsigned int needs_vcc:1;
unsigned int needs_reset:1; unsigned int needs_reset:1; /* deprecated */
int gpio_reset;
}; };
#if defined(CONFIG_NOP_USB_XCEIV) || (defined(CONFIG_NOP_USB_XCEIV_MODULE) && defined(MODULE)) #if defined(CONFIG_NOP_USB_XCEIV) || (defined(CONFIG_NOP_USB_XCEIV_MODULE) && defined(MODULE))
......
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