| /* |
| * The setup file for USB related hardware on PMC-Sierra MSP processors. |
| * |
| * Copyright 2006 PMC-Sierra, Inc. |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License as published by the |
| * Free Software Foundation; either version 2 of the License, or (at your |
| * option) any later version. |
| * |
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED |
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN |
| * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, |
| * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT |
| * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF |
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON |
| * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| * |
| * You should have received a copy of the GNU General Public License along |
| * with this program; if not, write to the Free Software Foundation, Inc., |
| * 675 Mass Ave, Cambridge, MA 02139, USA. |
| */ |
| #if defined(CONFIG_USB_EHCI_HCD) || defined(CONFIG_USB_GADGET) |
| |
| #include <linux/init.h> |
| #include <linux/ioport.h> |
| #include <linux/platform_device.h> |
| |
| #include <asm/mipsregs.h> |
| |
| #include <msp_regs.h> |
| #include <msp_int.h> |
| #include <msp_prom.h> |
| #include <msp_usb.h> |
| |
| |
| #if defined(CONFIG_USB_EHCI_HCD) |
| static struct resource msp_usbhost0_resources[] = { |
| [0] = { /* EHCI-HS operational and capabilities registers */ |
| .start = MSP_USB0_HS_START, |
| .end = MSP_USB0_HS_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [1] = { |
| .start = MSP_INT_USB, |
| .end = MSP_INT_USB, |
| .flags = IORESOURCE_IRQ, |
| }, |
| [2] = { /* MSBus-to-AMBA bridge register space */ |
| .start = MSP_USB0_MAB_START, |
| .end = MSP_USB0_MAB_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [3] = { /* Identification and general hardware parameters */ |
| .start = MSP_USB0_ID_START, |
| .end = MSP_USB0_ID_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| }; |
| |
| static u64 msp_usbhost0_dma_mask = 0xffffffffUL; |
| |
| static struct mspusb_device msp_usbhost0_device = { |
| .dev = { |
| .name = "pmcmsp-ehci", |
| .id = 0, |
| .dev = { |
| .dma_mask = &msp_usbhost0_dma_mask, |
| .coherent_dma_mask = 0xffffffffUL, |
| }, |
| .num_resources = ARRAY_SIZE(msp_usbhost0_resources), |
| .resource = msp_usbhost0_resources, |
| }, |
| }; |
| |
| /* MSP7140/MSP82XX has two USB2 hosts. */ |
| #ifdef CONFIG_MSP_HAS_DUAL_USB |
| static u64 msp_usbhost1_dma_mask = 0xffffffffUL; |
| |
| static struct resource msp_usbhost1_resources[] = { |
| [0] = { /* EHCI-HS operational and capabilities registers */ |
| .start = MSP_USB1_HS_START, |
| .end = MSP_USB1_HS_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [1] = { |
| .start = MSP_INT_USB, |
| .end = MSP_INT_USB, |
| .flags = IORESOURCE_IRQ, |
| }, |
| [2] = { /* MSBus-to-AMBA bridge register space */ |
| .start = MSP_USB1_MAB_START, |
| .end = MSP_USB1_MAB_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [3] = { /* Identification and general hardware parameters */ |
| .start = MSP_USB1_ID_START, |
| .end = MSP_USB1_ID_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| }; |
| |
| static struct mspusb_device msp_usbhost1_device = { |
| .dev = { |
| .name = "pmcmsp-ehci", |
| .id = 1, |
| .dev = { |
| .dma_mask = &msp_usbhost1_dma_mask, |
| .coherent_dma_mask = 0xffffffffUL, |
| }, |
| .num_resources = ARRAY_SIZE(msp_usbhost1_resources), |
| .resource = msp_usbhost1_resources, |
| }, |
| }; |
| #endif /* CONFIG_MSP_HAS_DUAL_USB */ |
| #endif /* CONFIG_USB_EHCI_HCD */ |
| |
| #if defined(CONFIG_USB_GADGET) |
| static struct resource msp_usbdev0_resources[] = { |
| [0] = { /* EHCI-HS operational and capabilities registers */ |
| .start = MSP_USB0_HS_START, |
| .end = MSP_USB0_HS_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [1] = { |
| .start = MSP_INT_USB, |
| .end = MSP_INT_USB, |
| .flags = IORESOURCE_IRQ, |
| }, |
| [2] = { /* MSBus-to-AMBA bridge register space */ |
| .start = MSP_USB0_MAB_START, |
| .end = MSP_USB0_MAB_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [3] = { /* Identification and general hardware parameters */ |
| .start = MSP_USB0_ID_START, |
| .end = MSP_USB0_ID_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| }; |
| |
| static u64 msp_usbdev_dma_mask = 0xffffffffUL; |
| |
| /* This may need to be converted to a mspusb_device, too. */ |
| static struct mspusb_device msp_usbdev0_device = { |
| .dev = { |
| .name = "msp71xx_udc", |
| .id = 0, |
| .dev = { |
| .dma_mask = &msp_usbdev_dma_mask, |
| .coherent_dma_mask = 0xffffffffUL, |
| }, |
| .num_resources = ARRAY_SIZE(msp_usbdev0_resources), |
| .resource = msp_usbdev0_resources, |
| }, |
| }; |
| |
| #ifdef CONFIG_MSP_HAS_DUAL_USB |
| static struct resource msp_usbdev1_resources[] = { |
| [0] = { /* EHCI-HS operational and capabilities registers */ |
| .start = MSP_USB1_HS_START, |
| .end = MSP_USB1_HS_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [1] = { |
| .start = MSP_INT_USB, |
| .end = MSP_INT_USB, |
| .flags = IORESOURCE_IRQ, |
| }, |
| [2] = { /* MSBus-to-AMBA bridge register space */ |
| .start = MSP_USB1_MAB_START, |
| .end = MSP_USB1_MAB_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| [3] = { /* Identification and general hardware parameters */ |
| .start = MSP_USB1_ID_START, |
| .end = MSP_USB1_ID_END, |
| .flags = IORESOURCE_MEM, |
| }, |
| }; |
| |
| /* This may need to be converted to a mspusb_device, too. */ |
| static struct mspusb_device msp_usbdev1_device = { |
| .dev = { |
| .name = "msp71xx_udc", |
| .id = 0, |
| .dev = { |
| .dma_mask = &msp_usbdev_dma_mask, |
| .coherent_dma_mask = 0xffffffffUL, |
| }, |
| .num_resources = ARRAY_SIZE(msp_usbdev1_resources), |
| .resource = msp_usbdev1_resources, |
| }, |
| }; |
| |
| #endif /* CONFIG_MSP_HAS_DUAL_USB */ |
| #endif /* CONFIG_USB_GADGET */ |
| |
| static int __init msp_usb_setup(void) |
| { |
| char *strp; |
| char envstr[32]; |
| struct platform_device *msp_devs[NUM_USB_DEVS]; |
| unsigned int val; |
| |
| /* construct environment name usbmode */ |
| /* set usbmode <host/device> as pmon environment var */ |
| /* |
| * Could this perhaps be integrated into the "features" env var? |
| * Use the features key "U", and follow with "H" for host-mode, |
| * "D" for device-mode. If it works for Ethernet, why not USB... |
| * -- hammtrev, 2007/03/22 |
| */ |
| snprintf((char *)&envstr[0], sizeof(envstr), "usbmode"); |
| |
| /* set default host mode */ |
| val = 1; |
| |
| /* get environment string */ |
| strp = prom_getenv((char *)&envstr[0]); |
| if (strp) { |
| /* compare string */ |
| if (!strcmp(strp, "device")) |
| val = 0; |
| } |
| |
| if (val) { |
| #if defined(CONFIG_USB_EHCI_HCD) |
| msp_devs[0] = &msp_usbhost0_device.dev; |
| ppfinit("platform add USB HOST done %s.\n", msp_devs[0]->name); |
| #ifdef CONFIG_MSP_HAS_DUAL_USB |
| msp_devs[1] = &msp_usbhost1_device.dev; |
| ppfinit("platform add USB HOST done %s.\n", msp_devs[1]->name); |
| #endif |
| #else |
| ppfinit("%s: echi_hcd not supported\n", __FILE__); |
| #endif /* CONFIG_USB_EHCI_HCD */ |
| } else { |
| #if defined(CONFIG_USB_GADGET) |
| /* get device mode structure */ |
| msp_devs[0] = &msp_usbdev0_device.dev; |
| ppfinit("platform add USB DEVICE done %s.\n" |
| , msp_devs[0]->name); |
| #ifdef CONFIG_MSP_HAS_DUAL_USB |
| msp_devs[1] = &msp_usbdev1_device.dev; |
| ppfinit("platform add USB DEVICE done %s.\n" |
| , msp_devs[1]->name); |
| #endif |
| #else |
| ppfinit("%s: usb_gadget not supported\n", __FILE__); |
| #endif /* CONFIG_USB_GADGET */ |
| } |
| /* add device */ |
| platform_add_devices(msp_devs, ARRAY_SIZE(msp_devs)); |
| |
| return 0; |
| } |
| |
| subsys_initcall(msp_usb_setup); |
| #endif /* CONFIG_USB_EHCI_HCD || CONFIG_USB_GADGET */ |