1 // SPDX-License-Identifier: GPL-2.0
2 #include <linux/module.h>
3 #include <linux/platform_device.h>
4 #include <linux/dma-mapping.h>
5 #include <linux/usb/otg.h>
6 #include <linux/usb/usb_phy_generic.h>
7 #include <linux/slab.h>
10 #include <linux/of_address.h>
11 #include <linux/usb/of.h>
13 #include "phy-am335x-control.h"
14 #include "phy-generic.h"
17 struct usb_phy_generic usb_phy_gen
;
18 struct phy_control
*phy_ctrl
;
20 enum usb_dr_mode dr_mode
;
23 static int am335x_init(struct usb_phy
*phy
)
25 struct am335x_phy
*am_phy
= dev_get_drvdata(phy
->dev
);
27 phy_ctrl_power(am_phy
->phy_ctrl
, am_phy
->id
, am_phy
->dr_mode
, true);
31 static void am335x_shutdown(struct usb_phy
*phy
)
33 struct am335x_phy
*am_phy
= dev_get_drvdata(phy
->dev
);
35 phy_ctrl_power(am_phy
->phy_ctrl
, am_phy
->id
, am_phy
->dr_mode
, false);
38 static int am335x_phy_probe(struct platform_device
*pdev
)
40 struct am335x_phy
*am_phy
;
41 struct device
*dev
= &pdev
->dev
;
44 am_phy
= devm_kzalloc(dev
, sizeof(*am_phy
), GFP_KERNEL
);
48 am_phy
->phy_ctrl
= am335x_get_phy_control(dev
);
49 if (!am_phy
->phy_ctrl
)
52 am_phy
->id
= of_alias_get_id(pdev
->dev
.of_node
, "phy");
54 dev_err(&pdev
->dev
, "Missing PHY id: %d\n", am_phy
->id
);
58 am_phy
->dr_mode
= of_usb_get_dr_mode_by_phy(pdev
->dev
.of_node
, -1);
60 ret
= usb_phy_gen_create_phy(dev
, &am_phy
->usb_phy_gen
, NULL
);
64 ret
= usb_add_phy_dev(&am_phy
->usb_phy_gen
.phy
);
67 am_phy
->usb_phy_gen
.phy
.init
= am335x_init
;
68 am_phy
->usb_phy_gen
.phy
.shutdown
= am335x_shutdown
;
70 platform_set_drvdata(pdev
, am_phy
);
71 device_init_wakeup(dev
, true);
74 * If we leave PHY wakeup enabled then AM33XX wakes up
75 * immediately from DS0. To avoid this we mark dev->power.can_wakeup
76 * to false. The same is checked in suspend routine to decide
77 * on whether to enable PHY wakeup or not.
78 * PHY wakeup works fine in standby mode, there by allowing us to
79 * handle remote wakeup, wakeup on disconnect and connect.
82 device_set_wakeup_enable(dev
, false);
83 phy_ctrl_power(am_phy
->phy_ctrl
, am_phy
->id
, am_phy
->dr_mode
, false);
88 static int am335x_phy_remove(struct platform_device
*pdev
)
90 struct am335x_phy
*am_phy
= platform_get_drvdata(pdev
);
92 usb_remove_phy(&am_phy
->usb_phy_gen
.phy
);
96 #ifdef CONFIG_PM_SLEEP
97 static int am335x_phy_suspend(struct device
*dev
)
99 struct am335x_phy
*am_phy
= dev_get_drvdata(dev
);
102 * Enable phy wakeup only if dev->power.can_wakeup is true.
103 * Make sure to enable wakeup to support remote wakeup in
104 * standby mode ( same is not supported in OFF(DS0) mode).
106 * echo enabled > /sys/bus/platform/devices/<usb-phy-id>/power/wakeup
109 if (device_may_wakeup(dev
))
110 phy_ctrl_wkup(am_phy
->phy_ctrl
, am_phy
->id
, true);
112 phy_ctrl_power(am_phy
->phy_ctrl
, am_phy
->id
, am_phy
->dr_mode
, false);
117 static int am335x_phy_resume(struct device
*dev
)
119 struct am335x_phy
*am_phy
= dev_get_drvdata(dev
);
121 phy_ctrl_power(am_phy
->phy_ctrl
, am_phy
->id
, am_phy
->dr_mode
, true);
123 if (device_may_wakeup(dev
))
124 phy_ctrl_wkup(am_phy
->phy_ctrl
, am_phy
->id
, false);
130 static SIMPLE_DEV_PM_OPS(am335x_pm_ops
, am335x_phy_suspend
, am335x_phy_resume
);
132 static const struct of_device_id am335x_phy_ids
[] = {
133 { .compatible
= "ti,am335x-usb-phy" },
136 MODULE_DEVICE_TABLE(of
, am335x_phy_ids
);
138 static struct platform_driver am335x_phy_driver
= {
139 .probe
= am335x_phy_probe
,
140 .remove
= am335x_phy_remove
,
142 .name
= "am335x-phy-driver",
143 .pm
= &am335x_pm_ops
,
144 .of_match_table
= am335x_phy_ids
,
148 module_platform_driver(am335x_phy_driver
);
149 MODULE_LICENSE("GPL v2");