2017-11-03 11:28:30 +01:00
// SPDX-License-Identifier: GPL-2.0
2013-07-30 21:43:45 +02:00
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/dma-mapping.h>
# include <linux/usb/otg.h>
2014-04-16 15:28:32 -05:00
# include <linux/usb/usb_phy_generic.h>
2013-07-30 21:43:45 +02:00
# include <linux/slab.h>
# include <linux/clk.h>
# include <linux/of.h>
# include <linux/of_address.h>
2015-12-08 10:31:50 -06:00
# include <linux/usb/of.h>
2013-07-30 21:43:45 +02:00
2015-11-20 16:13:07 -06:00
# include "phy-am335x-control.h"
2013-07-30 21:43:45 +02:00
# include "phy-generic.h"
struct am335x_phy {
2014-04-16 15:20:44 -05:00
struct usb_phy_generic usb_phy_gen ;
2013-07-30 21:43:45 +02:00
struct phy_control * phy_ctrl ;
int id ;
2015-12-08 10:31:50 -06:00
enum usb_dr_mode dr_mode ;
2013-07-30 21:43:45 +02:00
} ;
static int am335x_init ( struct usb_phy * phy )
{
struct am335x_phy * am_phy = dev_get_drvdata ( phy - > dev ) ;
2015-12-08 10:31:50 -06:00
phy_ctrl_power ( am_phy - > phy_ctrl , am_phy - > id , am_phy - > dr_mode , true ) ;
2013-07-30 21:43:45 +02:00
return 0 ;
}
static void am335x_shutdown ( struct usb_phy * phy )
{
struct am335x_phy * am_phy = dev_get_drvdata ( phy - > dev ) ;
2015-12-08 10:31:50 -06:00
phy_ctrl_power ( am_phy - > phy_ctrl , am_phy - > id , am_phy - > dr_mode , false ) ;
2013-07-30 21:43:45 +02:00
}
static int am335x_phy_probe ( struct platform_device * pdev )
{
struct am335x_phy * am_phy ;
struct device * dev = & pdev - > dev ;
int ret ;
am_phy = devm_kzalloc ( dev , sizeof ( * am_phy ) , GFP_KERNEL ) ;
if ( ! am_phy )
return - ENOMEM ;
am_phy - > phy_ctrl = am335x_get_phy_control ( dev ) ;
if ( ! am_phy - > phy_ctrl )
return - EPROBE_DEFER ;
2015-12-08 10:31:50 -06:00
2013-07-30 21:43:45 +02:00
am_phy - > id = of_alias_get_id ( pdev - > dev . of_node , " phy " ) ;
if ( am_phy - > id < 0 ) {
dev_err ( & pdev - > dev , " Missing PHY id: %d \n " , am_phy - > id ) ;
return am_phy - > id ;
}
2016-06-10 11:46:25 +02:00
am_phy - > dr_mode = of_usb_get_dr_mode_by_phy ( pdev - > dev . of_node , - 1 ) ;
2015-12-08 10:31:50 -06:00
2013-10-24 09:45:29 -05:00
ret = usb_phy_gen_create_phy ( dev , & am_phy - > usb_phy_gen , NULL ) ;
2013-07-30 21:43:45 +02:00
if ( ret )
return ret ;
am_phy - > usb_phy_gen . phy . init = am335x_init ;
am_phy - > usb_phy_gen . phy . shutdown = am335x_shutdown ;
platform_set_drvdata ( pdev , am_phy ) ;
2013-12-18 18:52:09 +05:30
device_init_wakeup ( dev , true ) ;
/*
* If we leave PHY wakeup enabled then AM33XX wakes up
* immediately from DS0 . To avoid this we mark dev - > power . can_wakeup
* to false . The same is checked in suspend routine to decide
* on whether to enable PHY wakeup or not .
* PHY wakeup works fine in standby mode , there by allowing us to
* handle remote wakeup , wakeup on disconnect and connect .
*/
device_set_wakeup_enable ( dev , false ) ;
2015-12-08 10:31:50 -06:00
phy_ctrl_power ( am_phy - > phy_ctrl , am_phy - > id , am_phy - > dr_mode , false ) ;
2013-08-11 15:26:04 +01:00
2019-01-16 11:54:07 -06:00
return usb_add_phy_dev ( & am_phy - > usb_phy_gen . phy ) ;
2013-07-30 21:43:45 +02:00
}
static int am335x_phy_remove ( struct platform_device * pdev )
{
struct am335x_phy * am_phy = platform_get_drvdata ( pdev ) ;
usb_remove_phy ( & am_phy - > usb_phy_gen . phy ) ;
return 0 ;
}
2013-12-18 18:52:09 +05:30
# ifdef CONFIG_PM_SLEEP
static int am335x_phy_suspend ( struct device * dev )
2013-08-19 12:39:44 +02:00
{
2018-04-19 16:06:26 +02:00
struct am335x_phy * am_phy = dev_get_drvdata ( dev ) ;
2013-08-19 12:39:44 +02:00
2013-12-18 18:52:09 +05:30
/*
* Enable phy wakeup only if dev - > power . can_wakeup is true .
* Make sure to enable wakeup to support remote wakeup in
* standby mode ( same is not supported in OFF ( DS0 ) mode ) .
* Enable it by doing
* echo enabled > / sys / bus / platform / devices / < usb - phy - id > / power / wakeup
*/
2013-08-19 12:39:44 +02:00
if ( device_may_wakeup ( dev ) )
phy_ctrl_wkup ( am_phy - > phy_ctrl , am_phy - > id , true ) ;
2013-12-18 18:52:09 +05:30
2015-12-08 10:31:50 -06:00
phy_ctrl_power ( am_phy - > phy_ctrl , am_phy - > id , am_phy - > dr_mode , false ) ;
2013-12-18 18:52:09 +05:30
2013-08-19 12:39:44 +02:00
return 0 ;
}
2013-12-18 18:52:09 +05:30
static int am335x_phy_resume ( struct device * dev )
2013-08-19 12:39:44 +02:00
{
2018-04-19 16:06:26 +02:00
struct am335x_phy * am_phy = dev_get_drvdata ( dev ) ;
2013-08-19 12:39:44 +02:00
2015-12-08 10:31:50 -06:00
phy_ctrl_power ( am_phy - > phy_ctrl , am_phy - > id , am_phy - > dr_mode , true ) ;
2013-12-18 18:52:09 +05:30
2013-08-19 12:39:44 +02:00
if ( device_may_wakeup ( dev ) )
phy_ctrl_wkup ( am_phy - > phy_ctrl , am_phy - > id , false ) ;
2013-12-18 18:52:09 +05:30
2013-08-19 12:39:44 +02:00
return 0 ;
}
2013-12-18 18:52:09 +05:30
# endif
2014-07-08 20:51:34 +09:00
static SIMPLE_DEV_PM_OPS ( am335x_pm_ops , am335x_phy_suspend , am335x_phy_resume ) ;
2013-07-30 21:43:45 +02:00
static const struct of_device_id am335x_phy_ids [ ] = {
{ . compatible = " ti,am335x-usb-phy " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , am335x_phy_ids ) ;
static struct platform_driver am335x_phy_driver = {
. probe = am335x_phy_probe ,
. remove = am335x_phy_remove ,
. driver = {
. name = " am335x-phy-driver " ,
2014-07-08 20:51:34 +09:00
. pm = & am335x_pm_ops ,
2013-09-30 09:44:46 +05:30
. of_match_table = am335x_phy_ids ,
2013-07-30 21:43:45 +02:00
} ,
} ;
module_platform_driver ( am335x_phy_driver ) ;
MODULE_LICENSE ( " GPL v2 " ) ;