diff options
author | Jingchang Lu <b35083@freescale.com> | 2012-06-28 10:15:11 +0800 |
---|---|---|
committer | Justin Waters <justin.waters@timesys.com> | 2012-09-12 16:49:54 -0400 |
commit | a3a01b2d057f18efaf6143dda3bda98c1d22056a (patch) | |
tree | a27cfe215a6cc99f6e1f20b4e93d6fdc071653d8 /arch/arm/mach-mvf | |
parent | 10d6fbb19dfff064f9d78f8ac50662b999b4e597 (diff) |
ENGR00181401,ENGR00181396-1: Add USB OTG controller support for MVF platform
OTG1 acts as gadget and OTG2 acts as host on TWR-MVF600 board.
Signed-off-by: Jingchang Lu <b35083@freescale.com>
Diffstat (limited to 'arch/arm/mach-mvf')
-rw-r--r-- | arch/arm/mach-mvf/usb.h | 61 | ||||
-rw-r--r-- | arch/arm/mach-mvf/usb_dr.c | 230 | ||||
-rw-r--r-- | arch/arm/mach-mvf/usb_dr2.c | 285 |
3 files changed, 576 insertions, 0 deletions
diff --git a/arch/arm/mach-mvf/usb.h b/arch/arm/mach-mvf/usb.h new file mode 100644 index 000000000000..14d9bff94bf0 --- /dev/null +++ b/arch/arm/mach-mvf/usb.h @@ -0,0 +1,61 @@ +/* + * Copyright 2012 Freescale Semiconductor, 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 program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + + * 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., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#include <mach/common.h> + +extern int usbotg_init(struct platform_device *pdev); +extern void usbotg_uninit(struct fsl_usb2_platform_data *pdata); +extern int usbotg2_init(struct platform_device *pdev); +extern void usbotg2_uninit(struct fsl_usb2_platform_data *pdata); +extern struct platform_device *host_pdev_register(struct resource *res, + int n_res, struct fsl_usb2_platform_data *config); + +extern int fsl_usb_host_init(struct platform_device *pdev); +extern void fsl_usb_host_uninit(struct fsl_usb2_platform_data *pdata); +extern int gpio_usbotg_utmi_active(void); +extern void gpio_usbotg_utmi_inactive(void); + +extern void __init mvf_usb_dr_init(void); +extern void __init mvf_usb_dr2_init(void); +extern void __init mvf_usb_h1_init(void); + +typedef void (*driver_vbus_func)(bool); +extern void mvf_set_host1_vbus_func(driver_vbus_func); +extern void mvf_set_otghost_vbus_func(driver_vbus_func); +extern struct platform_device anatop_thermal_device; +extern struct platform_device mvf_usbdr_otg_device; +extern struct platform_device mvf_usbdr_udc_device; +extern struct platform_device mvf_usbdr_host_device; +extern struct platform_device mvf_usbdr_wakeup_device; +extern struct platform_device mvf_usbh1_device; +extern struct platform_device mvf_usbh1_wakeup_device; + +/* + * Used to set pdata->operating_mode before registering the platform_device. + * If OTG is configured, the controller operates in OTG mode, + * otherwise it's either host or device. + */ +#ifdef CONFIG_USB_OTG +#define DR_UDC_MODE FSL_USB2_DR_OTG +#define DR_HOST_MODE FSL_USB2_DR_OTG +#else +#define DR_UDC_MODE FSL_USB2_DR_DEVICE +#define DR_HOST_MODE FSL_USB2_DR_HOST +#endif + +extern void __iomem *imx_otg_base; diff --git a/arch/arm/mach-mvf/usb_dr.c b/arch/arm/mach-mvf/usb_dr.c new file mode 100644 index 000000000000..2467acc1bf28 --- /dev/null +++ b/arch/arm/mach-mvf/usb_dr.c @@ -0,0 +1,230 @@ +/* + * Copyright 2012 Freescale Semiconductor, 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 program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + + * 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., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#include <linux/kernel.h> +#include <linux/types.h> +#include <linux/clk.h> +#include <linux/platform_device.h> +#include <linux/fsl_devices.h> +#include <linux/delay.h> +#include <linux/io.h> +#include <mach/arc_otg.h> +#include <mach/hardware.h> +#include <mach/clock.h> +#include "devices-mvf.h" +#include "crm_regs.h" +#include "regs-anadig.h" +#include "usb.h" + +static int usbotg_init_ext(struct platform_device *pdev); +static void usbotg_uninit_ext(struct platform_device *pdev); +static void usbotg_clock_gate(bool on); +static void _dr_discharge_line(bool enable); + +/* The usb_phy0_clk do not have enable/disable function at clock.c + * and PLL output for usb0's phy should be always enabled. + * usb_phy0_clk only stands for usb uses pll3 as its parent. + */ +static struct clk *usb_phy0_clk; +static u8 otg_used; + +/* Beginning of Common operation for DR port */ + +/* + * platform data structs + * - Which one to use is determined by CONFIG options in usb.h + * - operating_mode plugged at run time + */ +static struct fsl_usb2_platform_data dr_utmi_config = { + .name = "DR", + .init = usbotg_init_ext, + .exit = usbotg_uninit_ext, + .phy_mode = FSL_USB2_PHY_UTMI_WIDE, + .power_budget = 500, /* 500 mA max power */ + .usb_clock_for_pm = usbotg_clock_gate, + .transceiver = "utmi", + .phy_regs = MVF_USBPHY0_BASE_ADDR, + .dr_discharge_line = _dr_discharge_line, +}; + +static void usbotg_internal_phy_clock_gate(bool on) +{ + u32 reg; + + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY0_BASE_ADDR); + reg = __raw_readl(phy_reg + HW_USBPHY_CTRL); + + if (on) + reg &= ~BM_USBPHY_CTRL_CLKGATE; + else + reg |= BM_USBPHY_CTRL_CLKGATE; + + __raw_writel(reg, phy_reg + HW_USBPHY_CTRL); +} + +static int usb_phy_enable(struct fsl_usb2_platform_data *pdata) +{ + u32 tmp; + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY0_BASE_ADDR); + void __iomem *phy_ctrl; + + /* Stop then Reset */ + UOG_USBCMD &= ~UCMD_RUN_STOP; + while (UOG_USBCMD & UCMD_RUN_STOP) + ; + + UOG_USBCMD |= UCMD_RESET; + while ((UOG_USBCMD) & (UCMD_RESET)) + ; + /* Reset USBPHY module */ + phy_ctrl = phy_reg + HW_USBPHY_CTRL; + tmp = __raw_readl(phy_ctrl); + tmp |= BM_USBPHY_CTRL_SFTRST; + __raw_writel(tmp, phy_ctrl); + udelay(10); + + /* Remove CLKGATE and SFTRST */ + tmp = __raw_readl(phy_ctrl); + tmp &= ~(BM_USBPHY_CTRL_CLKGATE | BM_USBPHY_CTRL_SFTRST); + __raw_writel(tmp, phy_ctrl); + udelay(10); + + /* Power up the PHY */ + __raw_writel(0, phy_reg + HW_USBPHY_PWD); + + return 0; +} +/* Notes: configure USB clock*/ +static int usbotg_init_ext(struct platform_device *pdev) +{ + struct clk *usb_clk; + u32 ret; + + usb_clk = clk_get(NULL, "mvf-usb.0"); + clk_enable(usb_clk); + usb_phy0_clk = usb_clk; + + ret = usbotg_init(pdev); + if (ret) { + printk(KERN_ERR "otg init fails......\n"); + return ret; + } + if (!otg_used) { + usbotg_internal_phy_clock_gate(true); + usb_phy_enable(pdev->dev.platform_data); + /* + * after the phy reset,can not read the value for id/vbus at + * the register of otgsc ,cannot read at once ,need delay 3 ms + */ + mdelay(3); + } + otg_used++; + + return ret; +} + +static void usbotg_uninit_ext(struct platform_device *pdev) +{ + struct fsl_usb2_platform_data *pdata = pdev->dev.platform_data; + + clk_disable(usb_phy0_clk); + clk_put(usb_phy0_clk); + + usbotg_uninit(pdata); + otg_used--; +} + +static void usbotg_clock_gate(bool on) +{ + pr_debug("%s: on is %d\n", __func__, on); + if (on) + clk_enable(usb_phy0_clk); + else + clk_disable(usb_phy0_clk); +} +/* +void mvf_set_otghost_vbus_func(driver_vbus_func driver_vbus) +{ + dr_utmi_config.platform_driver_vbus = driver_vbus; +} +*/ +static void _dr_discharge_line(bool enable) +{ + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY0_BASE_ADDR); + if (enable) { + __raw_writel(BF_USBPHY_DEBUG_ENHSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_SET); + __raw_writel(BF_USBPHY_DEBUG_HSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_SET); + } else { + __raw_writel(BF_USBPHY_DEBUG_ENHSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_CLR); + __raw_writel(BF_USBPHY_DEBUG_HSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_CLR); + } + +} + +/* Below two macros are used at otg mode to indicate usb mode*/ +#define ENABLED_BY_HOST (0x1 << 0) +#define ENABLED_BY_DEVICE (0x1 << 1) + +#ifdef CONFIG_USB_EHCI_ARC_OTG +#endif /* CONFIG_USB_EHCI_ARC_OTG */ + + +#ifdef CONFIG_USB_GADGET_ARC +#endif /* CONFIG_USB_GADGET_ARC */ + +void __init mvf_usb_dr_init(void) +{ + struct platform_device *pdev; + u32 reg; +#ifdef CONFIG_USB_GADGET_ARC + dr_utmi_config.operating_mode = DR_UDC_MODE; + dr_utmi_config.platform_suspend = NULL; + dr_utmi_config.platform_resume = NULL; + + pdev = mvf_add_fsl_usb2_udc(&dr_utmi_config); + + __raw_writel(((0x01 << 30) | 0x2), ANADIG_USB1_MISC); + + __raw_writel(0x00000894, USBPHY1_CTRL); + udelay(10); + + __raw_writel(0x0, USBPHY1_PWD); + __raw_writel(0x1, USBPHY1_IP); + udelay(20); + __raw_writel(0x7, USBPHY1_IP); + + reg = __raw_readl(USBPHY1_DEBUG); + reg &= ~0x40000000; /* clear gate clock */ + __raw_writel(reg, USBPHY1_DEBUG); + + __raw_writel(0x10000007, USBPHY1_TX); + reg = __raw_readl(USBPHY1_CTRL); + reg |= ((0xD1 << 11) | 0x6); + __raw_writel(reg, USBPHY1_CTRL); + + reg = __raw_readl(ANADIG_USB1_VBUS_DETECT); + reg |= 0xE8; + __raw_writel(reg, ANADIG_USB1_VBUS_DETECT); + __raw_writel(0x001c0000, ANADIG_USB1_CHRG_DETECT); +#endif + +} diff --git a/arch/arm/mach-mvf/usb_dr2.c b/arch/arm/mach-mvf/usb_dr2.c new file mode 100644 index 000000000000..d26cfde125e0 --- /dev/null +++ b/arch/arm/mach-mvf/usb_dr2.c @@ -0,0 +1,285 @@ +/* + * Copyright 2012 Freescale Semiconductor, 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 program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + + * 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., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#include <linux/kernel.h> +#include <linux/types.h> +#include <linux/clk.h> +#include <linux/platform_device.h> +#include <linux/fsl_devices.h> +#include <linux/delay.h> +#include <linux/io.h> +#include <mach/arc_otg.h> +#include <mach/hardware.h> +#include <mach/clock.h> +#include "devices-mvf.h" +#include "crm_regs.h" +#include "regs-anadig.h" +#include "usb.h" + +static int usbotg2_init_ext(struct platform_device *pdev); +static void usbotg2_uninit_ext(struct platform_device *pdev); +static void usbotg_clock_gate(bool on); +static void _dr_discharge_line(bool enable); + +static struct clk *usb_phy1_clk; +static u8 otg2_used; + +/* Beginning of Common operation for DR port */ + +/* + * platform data structs + * - Which one to use is determined by CONFIG options in usb.h + * - operating_mode plugged at run time + */ +static struct fsl_usb2_platform_data dr_utmi_config = { + .name = "DR", + .init = usbotg2_init_ext, + .exit = usbotg2_uninit_ext, + .phy_mode = FSL_USB2_PHY_UTMI_WIDE, + .power_budget = 500, /* 500 mA max power */ + .usb_clock_for_pm = usbotg_clock_gate, + .transceiver = "utmi", + .phy_regs = MVF_USBPHY1_BASE_ADDR, + .dr_discharge_line = _dr_discharge_line, +}; + +static void usbotg_internal_phy_clock_gate(bool on) +{ + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY1_BASE_ADDR); + + if (on) + __raw_writel(BM_USBPHY_CTRL_CLKGATE, + phy_reg + HW_USBPHY_CTRL_CLR); + else + __raw_writel(BM_USBPHY_CTRL_CLKGATE, + phy_reg + HW_USBPHY_CTRL_SET); + +} + +static int usb_phy_enable(struct fsl_usb2_platform_data *pdata) +{ + u32 tmp; + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY1_BASE_ADDR); + void __iomem *phy_ctrl; + + /* Stop then Reset */ + UOG2_USBCMD &= ~UCMD_RUN_STOP; + while (UOG2_USBCMD & UCMD_RUN_STOP) + ; + + UOG2_USBCMD |= UCMD_RESET; + while ((UOG2_USBCMD) & (UCMD_RESET)) + ; + /* Reset USBPHY module */ + phy_ctrl = phy_reg + HW_USBPHY_CTRL; + tmp = __raw_readl(phy_ctrl); + tmp |= BM_USBPHY_CTRL_SFTRST; + __raw_writel(tmp, phy_ctrl); + udelay(10); + + /* Remove CLKGATE and SFTRST */ + tmp = __raw_readl(phy_ctrl); + tmp &= ~(BM_USBPHY_CTRL_CLKGATE | BM_USBPHY_CTRL_SFTRST); + __raw_writel(tmp, phy_ctrl); + udelay(10); + + /* Power up the PHY */ + __raw_writel(0, phy_reg + HW_USBPHY_PWD); + if ((pdata->operating_mode == FSL_USB2_DR_HOST) || + (pdata->operating_mode == FSL_USB2_DR_OTG)) { + /* enable FS/LS device */ + __raw_writel( + BM_USBPHY_CTRL_ENUTMILEVEL2 | BM_USBPHY_CTRL_ENUTMILEVEL3, + phy_reg + HW_USBPHY_CTRL_SET); + } + + return 0; +} +/* Notes: configure USB clock*/ +static int usbotg2_init_ext(struct platform_device *pdev) +{ + struct clk *usb_clk; + u32 ret; + + usb_clk = clk_get(NULL, "mvf-usb.1"); + clk_enable(usb_clk); + usb_phy1_clk = usb_clk; + + ret = usbotg2_init(pdev); + if (ret) { + printk(KERN_ERR "otg init fails......\n"); + return ret; + } + if (!otg2_used) { + usbotg_internal_phy_clock_gate(true); + usb_phy_enable(pdev->dev.platform_data); + /* + * after the phy reset,can't read the value for id/vbus at + * the register of otgsc at once ,need delay 3 ms + */ + mdelay(3); + } + otg2_used++; + + return ret; +} + +static void usbotg2_uninit_ext(struct platform_device *pdev) +{ + struct fsl_usb2_platform_data *pdata = pdev->dev.platform_data; + + clk_disable(usb_phy1_clk); + clk_put(usb_phy1_clk); + + usbotg2_uninit(pdata); + otg2_used--; +} + +static void usbotg_clock_gate(bool on) +{ + + if (on) + clk_enable(usb_phy1_clk); + else + clk_disable(usb_phy1_clk); + +} + +static void _dr_discharge_line(bool enable) +{ + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY1_BASE_ADDR); + + if (enable) { + __raw_writel(BF_USBPHY_DEBUG_ENHSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_SET); + __raw_writel(BF_USBPHY_DEBUG_HSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_SET); + } else { + __raw_writel(BF_USBPHY_DEBUG_ENHSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_CLR); + __raw_writel(BF_USBPHY_DEBUG_HSTPULLDOWN(0x3), + phy_reg + HW_USBPHY_DEBUG_CLR); + } + +} + +/* Below two macros are used at otg mode to indicate usb mode*/ +#define ENABLED_BY_HOST (0x1 << 0) +#define ENABLED_BY_DEVICE (0x1 << 1) +/* End of Common operation for DR port */ + +#ifdef CONFIG_USB_EHCI_ARC_OTG +/* Beginning of host related operation for DR port */ +static void _host_platform_suspend(struct fsl_usb2_platform_data *pdata) +{ + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY1_BASE_ADDR); + u32 tmp; + + tmp = (BM_USBPHY_PWD_TXPWDFS + | BM_USBPHY_PWD_TXPWDIBIAS + | BM_USBPHY_PWD_TXPWDV2I + | BM_USBPHY_PWD_RXPWDENV + | BM_USBPHY_PWD_RXPWD1PT1 + | BM_USBPHY_PWD_RXPWDDIFF + | BM_USBPHY_PWD_RXPWDRX); + __raw_writel(tmp, phy_reg + HW_USBPHY_PWD_SET); +} + +static void _host_platform_resume(struct fsl_usb2_platform_data *pdata) +{ + void __iomem *phy_reg = MVF_IO_ADDRESS(MVF_USBPHY1_BASE_ADDR); + u32 tmp; + + tmp = (BM_USBPHY_PWD_TXPWDFS + | BM_USBPHY_PWD_TXPWDIBIAS + | BM_USBPHY_PWD_TXPWDV2I + | BM_USBPHY_PWD_RXPWDENV + | BM_USBPHY_PWD_RXPWD1PT1 + | BM_USBPHY_PWD_RXPWDDIFF + | BM_USBPHY_PWD_RXPWDRX); + __raw_writel(tmp, phy_reg + HW_USBPHY_PWD_CLR); +} + +/* End of host related operation for DR port */ +#endif /* CONFIG_USB_EHCI_ARC_OTG */ + + +void __init mvf_usb_dr2_init(void) +{ + struct platform_device *pdev; + u32 reg; + +#ifdef CONFIG_USB_EHCI_ARC_OTG + dr_utmi_config.operating_mode = DR_HOST_MODE; + dr_utmi_config.platform_suspend = _host_platform_suspend; + dr_utmi_config.platform_resume = _host_platform_resume; + + pdev = mvf_add_fsl_ehci_otg(&dr_utmi_config); + + __raw_writel(0x0220C802, USBPHY2_CTRL); + udelay(20); + + __raw_writel(0x0, USBPHY2_PWD); + + reg = __raw_readl(USBPHY2_DEBUG); + reg &= ~0x40000000; /* clear clock gate */ + __raw_writel(reg, USBPHY2_DEBUG); + + reg = __raw_readl(ANADIG_USB2_MISC); + reg |= (0x01 << 30); /* Enable CLK_TO_UTMI */ + __raw_writel(reg, ANADIG_USB2_MISC); + + __raw_writel(0x10000007, USBPHY2_TX); + /*__raw_writel(0x100F0F07, USBPHY2_TX);*/ + + /* Enable disconnect detect */ + reg = __raw_readl(USBPHY2_CTRL); + reg &= ~0x040; /* clear OTG ID change IRQ */ + reg |= (0x1 << 14); /* Enable UTMI+ level2 */ + reg |= (0x1 << 9); + reg |= (0x1 << 15); /* Enable UTMI+ level3 */ + reg &= ~((0xD1 << 11) | 0x6); + __raw_writel(reg, USBPHY2_CTRL); + __raw_writel(0x1 << 3, USBPHY2_CTRL + 0x8); + + /* Disable VBUS and CHARGER detect */ + reg = __raw_readl(ANADIG_USB2_VBUS_DETECT); + reg |= 0xE8; + __raw_writel(reg, ANADIG_USB2_VBUS_DETECT); + __raw_writel(0x001c0000, ANADIG_USB2_CHRG_DETECT); + +#endif +} + +/* USB HIGH_SPEED disconnect detect on/off */ +void fsl_platform_set_usb_phy_dis(struct fsl_usb2_platform_data *pdata, + bool enable) +{ + u32 reg; + reg = __raw_readl(USBPHY2_CTRL); + + if (enable) + reg |= ((0xD1 << 11) | 0x6); + else + reg &= ~((0xD1 << 11) | 0x6); + + __raw_writel(reg, USBPHY2_CTRL); + + __raw_writel(0x1 << 3, USBPHY2_CTRL + 0x8); + +} |