2012-07-07 18:56:48 +04:00
/*
* Copyright 2012 Freescale Semiconductor , Inc .
* Copyright ( C ) 2012 Marek Vasut < marex @ denx . de >
* on behalf of DENX Software Engineering GmbH
*
* The code contained herein is licensed under the GNU General Public
* License . You may obtain a copy of the GNU General Public License
* Version 2 or later at the following locations :
*
* http : //www.opensource.org/licenses/gpl-license.html
* http : //www.gnu.org/copyleft/gpl.html
*/
# include <linux/module.h>
# include <linux/of_platform.h>
# include <linux/of_gpio.h>
# include <linux/platform_device.h>
# include <linux/pm_runtime.h>
# include <linux/dma-mapping.h>
# include <linux/usb/chipidea.h>
# include <linux/clk.h>
# include "ci.h"
2013-06-24 15:46:36 +04:00
# include "ci_hdrc_imx.h"
2012-07-07 18:56:48 +04:00
2014-01-10 09:51:28 +04:00
# define CI_HDRC_IMX_IMX28_WRITE_FIX BIT(0)
struct ci_hdrc_imx_platform_flag {
unsigned int flags ;
} ;
static const struct ci_hdrc_imx_platform_flag imx27_usb_data = {
} ;
static const struct ci_hdrc_imx_platform_flag imx28_usb_data = {
. flags = CI_HDRC_IMX_IMX28_WRITE_FIX ,
} ;
static const struct of_device_id ci_hdrc_imx_dt_ids [ ] = {
{ . compatible = " fsl,imx28-usb " , . data = & imx28_usb_data } ,
{ . compatible = " fsl,imx27-usb " , . data = & imx27_usb_data } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , ci_hdrc_imx_dt_ids ) ;
2013-06-24 15:46:36 +04:00
struct ci_hdrc_imx_data {
2012-07-07 18:56:48 +04:00
struct usb_phy * phy ;
struct platform_device * ci_pdev ;
struct clk * clk ;
2013-08-14 13:44:16 +04:00
struct imx_usbmisc_data * usbmisc_data ;
2012-07-07 18:56:48 +04:00
} ;
2012-09-12 15:58:05 +04:00
/* Common functions shared by usbmisc drivers */
2013-08-14 13:44:16 +04:00
static struct imx_usbmisc_data * usbmisc_get_init_data ( struct device * dev )
2012-09-12 15:58:05 +04:00
{
2014-09-22 04:14:15 +04:00
struct platform_device * misc_pdev ;
2012-09-12 15:58:05 +04:00
struct device_node * np = dev - > of_node ;
struct of_phandle_args args ;
2013-08-14 13:44:16 +04:00
struct imx_usbmisc_data * data ;
2012-09-12 15:58:05 +04:00
int ret ;
2013-08-14 13:44:16 +04:00
/*
* In case the fsl , usbmisc property is not present this device doesn ' t
* need usbmisc . Return NULL ( which is no error here )
*/
if ( ! of_get_property ( np , " fsl,usbmisc " , NULL ) )
return NULL ;
data = devm_kzalloc ( dev , sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data )
return ERR_PTR ( - ENOMEM ) ;
2012-09-12 15:58:05 +04:00
ret = of_parse_phandle_with_args ( np , " fsl,usbmisc " , " #index-cells " ,
0 , & args ) ;
if ( ret ) {
dev_err ( dev , " Failed to parse property fsl,usbmisc, errno %d \n " ,
ret ) ;
2013-08-14 13:44:16 +04:00
return ERR_PTR ( ret ) ;
2012-09-12 15:58:05 +04:00
}
2013-08-14 13:44:16 +04:00
data - > index = args . args [ 0 ] ;
2014-09-22 04:14:15 +04:00
misc_pdev = of_find_device_by_node ( args . np ) ;
2012-09-12 15:58:05 +04:00
of_node_put ( args . np ) ;
2014-09-22 04:14:15 +04:00
if ( ! misc_pdev )
return ERR_PTR ( - EPROBE_DEFER ) ;
data - > dev = & misc_pdev - > dev ;
2012-09-12 15:58:05 +04:00
if ( of_find_property ( np , " disable-over-current " , NULL ) )
2013-08-14 13:44:16 +04:00
data - > disable_oc = 1 ;
2012-09-12 15:58:05 +04:00
2013-03-30 14:54:01 +04:00
if ( of_find_property ( np , " external-vbus-divider " , NULL ) )
2013-08-14 13:44:16 +04:00
data - > evdo = 1 ;
2013-03-30 14:54:01 +04:00
2013-08-14 13:44:16 +04:00
return data ;
2012-09-12 15:58:05 +04:00
}
/* End of common functions shared by usbmisc drivers*/
2013-06-24 15:46:36 +04:00
static int ci_hdrc_imx_probe ( struct platform_device * pdev )
2012-07-07 18:56:48 +04:00
{
2013-06-24 15:46:36 +04:00
struct ci_hdrc_imx_data * data ;
struct ci_hdrc_platform_data pdata = {
2014-03-11 09:47:38 +04:00
. name = dev_name ( & pdev - > dev ) ,
2013-06-13 18:59:58 +04:00
. capoffset = DEF_CAPOFFSET ,
2013-06-24 15:46:36 +04:00
. flags = CI_HDRC_REQUIRE_TRANSCEIVER |
CI_HDRC_DISABLE_STREAMING ,
2013-06-13 18:59:58 +04:00
} ;
2012-07-07 18:56:48 +04:00
int ret ;
2014-01-10 09:51:28 +04:00
const struct of_device_id * of_id =
of_match_device ( ci_hdrc_imx_dt_ids , & pdev - > dev ) ;
const struct ci_hdrc_imx_platform_flag * imx_platform_flag = of_id - > data ;
2012-07-07 18:56:48 +04:00
data = devm_kzalloc ( & pdev - > dev , sizeof ( * data ) , GFP_KERNEL ) ;
if ( ! data ) {
2013-06-24 15:46:36 +04:00
dev_err ( & pdev - > dev , " Failed to allocate ci_hdrc-imx data! \n " ) ;
2012-07-07 18:56:48 +04:00
return - ENOMEM ;
}
2013-08-14 13:44:16 +04:00
data - > usbmisc_data = usbmisc_get_init_data ( & pdev - > dev ) ;
if ( IS_ERR ( data - > usbmisc_data ) )
return PTR_ERR ( data - > usbmisc_data ) ;
2012-07-07 18:56:48 +04:00
data - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( data - > clk ) ) {
dev_err ( & pdev - > dev ,
" Failed to get clock, err=%ld \n " , PTR_ERR ( data - > clk ) ) ;
return PTR_ERR ( data - > clk ) ;
}
ret = clk_prepare_enable ( data - > clk ) ;
if ( ret ) {
dev_err ( & pdev - > dev ,
" Failed to prepare or enable clock, err=%d \n " , ret ) ;
return ret ;
}
2013-06-25 13:58:05 +04:00
data - > phy = devm_usb_get_phy_by_phandle ( & pdev - > dev , " fsl,usbphy " , 0 ) ;
2013-09-24 08:47:54 +04:00
if ( IS_ERR ( data - > phy ) ) {
ret = PTR_ERR ( data - > phy ) ;
2014-07-22 06:09:43 +04:00
/* Return -EINVAL if no usbphy is available */
if ( ret = = - ENODEV )
ret = - EINVAL ;
2013-06-13 19:00:00 +04:00
goto err_clk ;
2012-07-07 18:56:48 +04:00
}
2013-06-13 18:59:58 +04:00
pdata . phy = data - > phy ;
2012-07-07 18:56:48 +04:00
2014-01-10 09:51:28 +04:00
if ( imx_platform_flag - > flags & CI_HDRC_IMX_IMX28_WRITE_FIX )
pdata . flags | = CI_HDRC_IMX28_WRITE_FIX ;
2013-06-27 15:36:37 +04:00
ret = dma_coerce_mask_and_coherent ( & pdev - > dev , DMA_BIT_MASK ( 32 ) ) ;
2013-06-10 19:28:49 +04:00
if ( ret )
goto err_clk ;
2012-09-12 15:58:05 +04:00
2013-08-14 13:44:16 +04:00
if ( data - > usbmisc_data ) {
ret = imx_usbmisc_init ( data - > usbmisc_data ) ;
2012-09-12 15:58:05 +04:00
if ( ret ) {
2013-08-14 13:44:16 +04:00
dev_err ( & pdev - > dev , " usbmisc init failed, ret=%d \n " ,
ret ) ;
2013-09-24 08:47:54 +04:00
goto err_clk ;
2012-09-12 15:58:05 +04:00
}
}
2013-06-24 15:46:36 +04:00
data - > ci_pdev = ci_hdrc_add_device ( & pdev - > dev ,
2012-07-07 18:56:48 +04:00
pdev - > resource , pdev - > num_resources ,
2013-06-13 18:59:58 +04:00
& pdata ) ;
2013-06-13 18:59:48 +04:00
if ( IS_ERR ( data - > ci_pdev ) ) {
ret = PTR_ERR ( data - > ci_pdev ) ;
2012-07-07 18:56:48 +04:00
dev_err ( & pdev - > dev ,
" Can't register ci_hdrc platform device, err=%d \n " ,
ret ) ;
2013-09-24 08:47:54 +04:00
goto err_clk ;
2012-07-07 18:56:48 +04:00
}
2013-08-14 13:44:16 +04:00
if ( data - > usbmisc_data ) {
ret = imx_usbmisc_init_post ( data - > usbmisc_data ) ;
2013-03-30 14:54:01 +04:00
if ( ret ) {
2013-08-14 13:44:16 +04:00
dev_err ( & pdev - > dev , " usbmisc post failed, ret=%d \n " ,
ret ) ;
2013-06-13 18:59:48 +04:00
goto disable_device ;
2013-03-30 14:54:01 +04:00
}
}
2012-07-07 18:56:48 +04:00
platform_set_drvdata ( pdev , data ) ;
pm_runtime_no_callbacks ( & pdev - > dev ) ;
pm_runtime_enable ( & pdev - > dev ) ;
return 0 ;
2013-06-13 18:59:48 +04:00
disable_device :
2013-06-24 15:46:36 +04:00
ci_hdrc_remove_device ( data - > ci_pdev ) ;
2013-06-13 19:00:00 +04:00
err_clk :
2012-07-07 18:56:48 +04:00
clk_disable_unprepare ( data - > clk ) ;
return ret ;
}
2013-06-24 15:46:36 +04:00
static int ci_hdrc_imx_remove ( struct platform_device * pdev )
2012-07-07 18:56:48 +04:00
{
2013-06-24 15:46:36 +04:00
struct ci_hdrc_imx_data * data = platform_get_drvdata ( pdev ) ;
2012-07-07 18:56:48 +04:00
pm_runtime_disable ( & pdev - > dev ) ;
2013-06-24 15:46:36 +04:00
ci_hdrc_remove_device ( data - > ci_pdev ) ;
2012-07-07 18:56:48 +04:00
clk_disable_unprepare ( data - > clk ) ;
return 0 ;
}
2013-06-24 15:46:36 +04:00
static struct platform_driver ci_hdrc_imx_driver = {
. probe = ci_hdrc_imx_probe ,
. remove = ci_hdrc_imx_remove ,
2012-07-07 18:56:48 +04:00
. driver = {
. name = " imx_usb " ,
. owner = THIS_MODULE ,
2013-06-24 15:46:36 +04:00
. of_match_table = ci_hdrc_imx_dt_ids ,
2012-07-07 18:56:48 +04:00
} ,
} ;
2013-06-24 15:46:36 +04:00
module_platform_driver ( ci_hdrc_imx_driver ) ;
2012-07-07 18:56:48 +04:00
MODULE_ALIAS ( " platform:imx-usb " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
2013-06-24 15:46:36 +04:00
MODULE_DESCRIPTION ( " CI HDRC i.MX USB binding " ) ;
2012-07-07 18:56:48 +04:00
MODULE_AUTHOR ( " Marek Vasut <marex@denx.de> " ) ;
MODULE_AUTHOR ( " Richard Zhao <richard.zhao@freescale.com> " ) ;