aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorYoshihiro Shimoda <yoshihiro.shimoda.uh@renesas.com>2016-04-29 04:52:25 -0400
committerKishon Vijay Abraham I <kishon@ti.com>2016-04-30 10:42:26 -0400
commit2b38543c8db1c7dff48aa767fcbfba13f50514ca (patch)
treed3fd232d1f267aec9e38c88848d9748a864d4545
parent6dcfd7c300bf3588d4f2500e14d90ffea5595e84 (diff)
phy: rcar-gen3-usb2: add extcon support
This patch adds extcon support for otg related channel. Signed-off-by: Yoshihiro Shimoda <yoshihiro.shimoda.uh@renesas.com> Reviewed-by: Chanwoo Choi <cw00.choi@samsung.com> Signed-off-by: Kishon Vijay Abraham I <kishon@ti.com>
-rw-r--r--drivers/phy/Kconfig1
-rw-r--r--drivers/phy/phy-rcar-gen3-usb2.c26
2 files changed, 27 insertions, 0 deletions
diff --git a/drivers/phy/Kconfig b/drivers/phy/Kconfig
index a72d73cdaef3..447cb7c1a18d 100644
--- a/drivers/phy/Kconfig
+++ b/drivers/phy/Kconfig
@@ -121,6 +121,7 @@ config PHY_RCAR_GEN2
121config PHY_RCAR_GEN3_USB2 121config PHY_RCAR_GEN3_USB2
122 tristate "Renesas R-Car generation 3 USB 2.0 PHY driver" 122 tristate "Renesas R-Car generation 3 USB 2.0 PHY driver"
123 depends on ARCH_RENESAS 123 depends on ARCH_RENESAS
124 depends on EXTCON
124 select GENERIC_PHY 125 select GENERIC_PHY
125 help 126 help
126 Support for USB 2.0 PHY found on Renesas R-Car generation 3 SoCs. 127 Support for USB 2.0 PHY found on Renesas R-Car generation 3 SoCs.
diff --git a/drivers/phy/phy-rcar-gen3-usb2.c b/drivers/phy/phy-rcar-gen3-usb2.c
index 7b14244be285..76bb88f0700a 100644
--- a/drivers/phy/phy-rcar-gen3-usb2.c
+++ b/drivers/phy/phy-rcar-gen3-usb2.c
@@ -12,6 +12,7 @@
12 * published by the Free Software Foundation. 12 * published by the Free Software Foundation.
13 */ 13 */
14 14
15#include <linux/extcon.h>
15#include <linux/interrupt.h> 16#include <linux/interrupt.h>
16#include <linux/io.h> 17#include <linux/io.h>
17#include <linux/module.h> 18#include <linux/module.h>
@@ -77,6 +78,7 @@
77 78
78struct rcar_gen3_chan { 79struct rcar_gen3_chan {
79 void __iomem *base; 80 void __iomem *base;
81 struct extcon_dev *extcon;
80 struct phy *phy; 82 struct phy *phy;
81 struct regulator *vbus; 83 struct regulator *vbus;
82 bool has_otg; 84 bool has_otg;
@@ -127,6 +129,9 @@ static void rcar_gen3_init_for_host(struct rcar_gen3_chan *ch)
127 rcar_gen3_set_linectrl(ch, 1, 1); 129 rcar_gen3_set_linectrl(ch, 1, 1);
128 rcar_gen3_set_host_mode(ch, 1); 130 rcar_gen3_set_host_mode(ch, 1);
129 rcar_gen3_enable_vbus_ctrl(ch, 1); 131 rcar_gen3_enable_vbus_ctrl(ch, 1);
132
133 extcon_set_cable_state_(ch->extcon, EXTCON_USB_HOST, true);
134 extcon_set_cable_state_(ch->extcon, EXTCON_USB, false);
130} 135}
131 136
132static void rcar_gen3_init_for_peri(struct rcar_gen3_chan *ch) 137static void rcar_gen3_init_for_peri(struct rcar_gen3_chan *ch)
@@ -134,6 +139,9 @@ static void rcar_gen3_init_for_peri(struct rcar_gen3_chan *ch)
134 rcar_gen3_set_linectrl(ch, 0, 1); 139 rcar_gen3_set_linectrl(ch, 0, 1);
135 rcar_gen3_set_host_mode(ch, 0); 140 rcar_gen3_set_host_mode(ch, 0);
136 rcar_gen3_enable_vbus_ctrl(ch, 0); 141 rcar_gen3_enable_vbus_ctrl(ch, 0);
142
143 extcon_set_cable_state_(ch->extcon, EXTCON_USB_HOST, false);
144 extcon_set_cable_state_(ch->extcon, EXTCON_USB, true);
137} 145}
138 146
139static bool rcar_gen3_check_vbus(struct rcar_gen3_chan *ch) 147static bool rcar_gen3_check_vbus(struct rcar_gen3_chan *ch)
@@ -272,6 +280,12 @@ static const struct of_device_id rcar_gen3_phy_usb2_match_table[] = {
272}; 280};
273MODULE_DEVICE_TABLE(of, rcar_gen3_phy_usb2_match_table); 281MODULE_DEVICE_TABLE(of, rcar_gen3_phy_usb2_match_table);
274 282
283static const unsigned int rcar_gen3_phy_cable[] = {
284 EXTCON_USB,
285 EXTCON_USB_HOST,
286 EXTCON_NONE,
287};
288
275static int rcar_gen3_phy_usb2_probe(struct platform_device *pdev) 289static int rcar_gen3_phy_usb2_probe(struct platform_device *pdev)
276{ 290{
277 struct device *dev = &pdev->dev; 291 struct device *dev = &pdev->dev;
@@ -297,11 +311,23 @@ static int rcar_gen3_phy_usb2_probe(struct platform_device *pdev)
297 /* call request_irq for OTG */ 311 /* call request_irq for OTG */
298 irq = platform_get_irq(pdev, 0); 312 irq = platform_get_irq(pdev, 0);
299 if (irq >= 0) { 313 if (irq >= 0) {
314 int ret;
315
300 irq = devm_request_irq(dev, irq, rcar_gen3_phy_usb2_irq, 316 irq = devm_request_irq(dev, irq, rcar_gen3_phy_usb2_irq,
301 IRQF_SHARED, dev_name(dev), channel); 317 IRQF_SHARED, dev_name(dev), channel);
302 if (irq < 0) 318 if (irq < 0)
303 dev_err(dev, "No irq handler (%d)\n", irq); 319 dev_err(dev, "No irq handler (%d)\n", irq);
304 channel->has_otg = true; 320 channel->has_otg = true;
321 channel->extcon = devm_extcon_dev_allocate(dev,
322 rcar_gen3_phy_cable);
323 if (IS_ERR(channel->extcon))
324 return PTR_ERR(channel->extcon);
325
326 ret = devm_extcon_dev_register(dev, channel->extcon);
327 if (ret < 0) {
328 dev_err(dev, "Failed to register extcon\n");
329 return ret;
330 }
305 } 331 }
306 332
307 /* devm_phy_create() will call pm_runtime_enable(dev); */ 333 /* devm_phy_create() will call pm_runtime_enable(dev); */