2013-12-19 18:23:09 +04:00
/*
* phy - bcm - kona - usb2 . c - Broadcom Kona USB2 Phy Driver
*
* Copyright ( C ) 2013 Linaro Limited
* Matt Porter < mporter @ linaro . org >
*
* This software is licensed under the terms of the GNU General Public
* License version 2 , as published by the Free Software Foundation , and
* may be copied , distributed , and modified under those terms .
*
* 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 .
*/
# include <linux/clk.h>
# include <linux/delay.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/phy/phy.h>
# include <linux/platform_device.h>
# define OTGCTL (0)
# define OTGCTL_OTGSTAT2 BIT(31)
# define OTGCTL_OTGSTAT1 BIT(30)
# define OTGCTL_PRST_N_SW BIT(11)
# define OTGCTL_HRESET_N BIT(10)
# define OTGCTL_UTMI_LINE_STATE1 BIT(9)
# define OTGCTL_UTMI_LINE_STATE0 BIT(8)
# define P1CTL (8)
# define P1CTL_SOFT_RESET BIT(1)
# define P1CTL_NON_DRIVING BIT(0)
struct bcm_kona_usb {
void __iomem * regs ;
} ;
static void bcm_kona_usb_phy_power ( struct bcm_kona_usb * phy , int on )
{
u32 val ;
val = readl ( phy - > regs + OTGCTL ) ;
if ( on ) {
/* Configure and power PHY */
val & = ~ ( OTGCTL_OTGSTAT2 | OTGCTL_OTGSTAT1 |
OTGCTL_UTMI_LINE_STATE1 | OTGCTL_UTMI_LINE_STATE0 ) ;
val | = OTGCTL_PRST_N_SW | OTGCTL_HRESET_N ;
} else {
val & = ~ ( OTGCTL_PRST_N_SW | OTGCTL_HRESET_N ) ;
}
writel ( val , phy - > regs + OTGCTL ) ;
}
static int bcm_kona_usb_phy_init ( struct phy * gphy )
{
struct bcm_kona_usb * phy = phy_get_drvdata ( gphy ) ;
u32 val ;
/* Soft reset PHY */
val = readl ( phy - > regs + P1CTL ) ;
val & = ~ P1CTL_NON_DRIVING ;
val | = P1CTL_SOFT_RESET ;
writel ( val , phy - > regs + P1CTL ) ;
writel ( val & ~ P1CTL_SOFT_RESET , phy - > regs + P1CTL ) ;
/* Reset needs to be asserted for 2ms */
mdelay ( 2 ) ;
writel ( val | P1CTL_SOFT_RESET , phy - > regs + P1CTL ) ;
return 0 ;
}
static int bcm_kona_usb_phy_power_on ( struct phy * gphy )
{
struct bcm_kona_usb * phy = phy_get_drvdata ( gphy ) ;
bcm_kona_usb_phy_power ( phy , 1 ) ;
return 0 ;
}
static int bcm_kona_usb_phy_power_off ( struct phy * gphy )
{
struct bcm_kona_usb * phy = phy_get_drvdata ( gphy ) ;
bcm_kona_usb_phy_power ( phy , 0 ) ;
return 0 ;
}
static struct phy_ops ops = {
. init = bcm_kona_usb_phy_init ,
. power_on = bcm_kona_usb_phy_power_on ,
. power_off = bcm_kona_usb_phy_power_off ,
. owner = THIS_MODULE ,
} ;
static int bcm_kona_usb2_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct bcm_kona_usb * phy ;
struct resource * res ;
struct phy * gphy ;
struct phy_provider * phy_provider ;
phy = devm_kzalloc ( dev , sizeof ( * phy ) , GFP_KERNEL ) ;
if ( ! phy )
return - ENOMEM ;
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
phy - > regs = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( phy - > regs ) )
return PTR_ERR ( phy - > regs ) ;
platform_set_drvdata ( pdev , phy ) ;
2014-07-14 14:25:02 +04:00
gphy = devm_phy_create ( dev , NULL , & ops , NULL ) ;
2013-12-19 18:23:09 +04:00
if ( IS_ERR ( gphy ) )
return PTR_ERR ( gphy ) ;
/* The Kona PHY supports an 8-bit wide UTMI interface */
phy_set_bus_width ( gphy , 8 ) ;
phy_set_drvdata ( gphy , phy ) ;
phy_provider = devm_of_phy_provider_register ( dev ,
of_phy_simple_xlate ) ;
2014-02-03 12:25:05 +04:00
return PTR_ERR_OR_ZERO ( phy_provider ) ;
2013-12-19 18:23:09 +04:00
}
static const struct of_device_id bcm_kona_usb2_dt_ids [ ] = {
{ . compatible = " brcm,kona-usb2-phy " } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , bcm_kona_usb2_dt_ids ) ;
static struct platform_driver bcm_kona_usb2_driver = {
. probe = bcm_kona_usb2_probe ,
. driver = {
. name = " bcm-kona-usb2 " ,
. of_match_table = bcm_kona_usb2_dt_ids ,
} ,
} ;
module_platform_driver ( bcm_kona_usb2_driver ) ;
MODULE_ALIAS ( " platform:bcm-kona-usb2 " ) ;
MODULE_AUTHOR ( " Matt Porter <mporter@linaro.org> " ) ;
MODULE_DESCRIPTION ( " BCM Kona USB 2.0 PHY driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;