forked from torvalds/linux
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
phy: ralink-usb: add driver for Mediatek/Ralink
Add a driver to setup the USB phy on Mediatek/Ralink SoCs. The driver sets up power and host mode, but also needs to configure PHY registers for the MT7628 and MT7688. Signed-off-by: John Crispin <[email protected]> Signed-off-by: Harvey Hunt <[email protected]> Signed-off-by: Kishon Vijay Abraham I <[email protected]>
- Loading branch information
Showing
5 changed files
with
263 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -18,6 +18,7 @@ obj-y += broadcom/ \ | |
marvell/ \ | ||
motorola/ \ | ||
qualcomm/ \ | ||
ralink/ \ | ||
samsung/ \ | ||
st/ \ | ||
ti/ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
# | ||
# PHY drivers for Ralink platforms. | ||
# | ||
config PHY_RALINK_USB | ||
tristate "Ralink USB PHY driver" | ||
depends on RALINK || COMPILE_TEST | ||
select GENERIC_PHY | ||
select MFD_SYSCON | ||
help | ||
This option enables support for the Ralink USB PHY found inside | ||
RT3352, MT7620, MT7628 and MT7688. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
obj-$(CONFIG_PHY_RALINK_USB) += phy-ralink-usb.o |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,249 @@ | ||
/* | ||
* Copyright (C) 2017 John Crispin <[email protected]> | ||
* | ||
* Based on code from | ||
* Allwinner Technology Co., Ltd. <www.allwinnertech.com> | ||
* | ||
* This program is free software; you can redistribute it and/or modify | ||
* it under the terms of the GNU General Public License as published by | ||
* the Free Software Foundation; either version 2 of the License, or | ||
* (at your option) any later version. | ||
* | ||
* 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/delay.h> | ||
#include <linux/err.h> | ||
#include <linux/io.h> | ||
#include <linux/kernel.h> | ||
#include <linux/mfd/syscon.h> | ||
#include <linux/module.h> | ||
#include <linux/mutex.h> | ||
#include <linux/of_platform.h> | ||
#include <linux/phy/phy.h> | ||
#include <linux/platform_device.h> | ||
#include <linux/regmap.h> | ||
#include <linux/reset.h> | ||
|
||
#define RT_SYSC_REG_SYSCFG1 0x014 | ||
#define RT_SYSC_REG_CLKCFG1 0x030 | ||
#define RT_SYSC_REG_USB_PHY_CFG 0x05c | ||
|
||
#define OFS_U2_PHY_AC0 0x800 | ||
#define OFS_U2_PHY_AC1 0x804 | ||
#define OFS_U2_PHY_AC2 0x808 | ||
#define OFS_U2_PHY_ACR0 0x810 | ||
#define OFS_U2_PHY_ACR1 0x814 | ||
#define OFS_U2_PHY_ACR2 0x818 | ||
#define OFS_U2_PHY_ACR3 0x81C | ||
#define OFS_U2_PHY_ACR4 0x820 | ||
#define OFS_U2_PHY_AMON0 0x824 | ||
#define OFS_U2_PHY_DCR0 0x860 | ||
#define OFS_U2_PHY_DCR1 0x864 | ||
#define OFS_U2_PHY_DTM0 0x868 | ||
#define OFS_U2_PHY_DTM1 0x86C | ||
|
||
#define RT_RSTCTRL_UDEV BIT(25) | ||
#define RT_RSTCTRL_UHST BIT(22) | ||
#define RT_SYSCFG1_USB0_HOST_MODE BIT(10) | ||
|
||
#define MT7620_CLKCFG1_UPHY0_CLK_EN BIT(25) | ||
#define MT7620_CLKCFG1_UPHY1_CLK_EN BIT(22) | ||
#define RT_CLKCFG1_UPHY1_CLK_EN BIT(20) | ||
#define RT_CLKCFG1_UPHY0_CLK_EN BIT(18) | ||
|
||
#define USB_PHY_UTMI_8B60M BIT(1) | ||
#define UDEV_WAKEUP BIT(0) | ||
|
||
struct ralink_usb_phy { | ||
struct reset_control *rstdev; | ||
struct reset_control *rsthost; | ||
u32 clk; | ||
struct phy *phy; | ||
void __iomem *base; | ||
struct regmap *sysctl; | ||
}; | ||
|
||
static void u2_phy_w32(struct ralink_usb_phy *phy, u32 val, u32 reg) | ||
{ | ||
writel(val, phy->base + reg); | ||
} | ||
|
||
static u32 u2_phy_r32(struct ralink_usb_phy *phy, u32 reg) | ||
{ | ||
return readl(phy->base + reg); | ||
} | ||
|
||
static void ralink_usb_phy_init(struct ralink_usb_phy *phy) | ||
{ | ||
u2_phy_r32(phy, OFS_U2_PHY_AC2); | ||
u2_phy_r32(phy, OFS_U2_PHY_ACR0); | ||
u2_phy_r32(phy, OFS_U2_PHY_DCR0); | ||
|
||
u2_phy_w32(phy, 0x00ffff02, OFS_U2_PHY_DCR0); | ||
u2_phy_r32(phy, OFS_U2_PHY_DCR0); | ||
u2_phy_w32(phy, 0x00555502, OFS_U2_PHY_DCR0); | ||
u2_phy_r32(phy, OFS_U2_PHY_DCR0); | ||
u2_phy_w32(phy, 0x00aaaa02, OFS_U2_PHY_DCR0); | ||
u2_phy_r32(phy, OFS_U2_PHY_DCR0); | ||
u2_phy_w32(phy, 0x00000402, OFS_U2_PHY_DCR0); | ||
u2_phy_r32(phy, OFS_U2_PHY_DCR0); | ||
u2_phy_w32(phy, 0x0048086a, OFS_U2_PHY_AC0); | ||
u2_phy_w32(phy, 0x4400001c, OFS_U2_PHY_AC1); | ||
u2_phy_w32(phy, 0xc0200000, OFS_U2_PHY_ACR3); | ||
u2_phy_w32(phy, 0x02000000, OFS_U2_PHY_DTM0); | ||
} | ||
|
||
static int ralink_usb_phy_power_on(struct phy *_phy) | ||
{ | ||
struct ralink_usb_phy *phy = phy_get_drvdata(_phy); | ||
u32 t; | ||
|
||
/* enable the phy */ | ||
regmap_update_bits(phy->sysctl, RT_SYSC_REG_CLKCFG1, | ||
phy->clk, phy->clk); | ||
|
||
/* setup host mode */ | ||
regmap_update_bits(phy->sysctl, RT_SYSC_REG_SYSCFG1, | ||
RT_SYSCFG1_USB0_HOST_MODE, | ||
RT_SYSCFG1_USB0_HOST_MODE); | ||
|
||
/* deassert the reset lines */ | ||
reset_control_deassert(phy->rsthost); | ||
reset_control_deassert(phy->rstdev); | ||
|
||
/* | ||
* The SDK kernel had a delay of 100ms. however on device | ||
* testing showed that 10ms is enough | ||
*/ | ||
mdelay(10); | ||
|
||
if (phy->base) | ||
ralink_usb_phy_init(phy); | ||
|
||
/* print some status info */ | ||
regmap_read(phy->sysctl, RT_SYSC_REG_USB_PHY_CFG, &t); | ||
dev_info(&phy->phy->dev, "remote usb device wakeup %s\n", | ||
(t & UDEV_WAKEUP) ? ("enabled") : ("disabled")); | ||
if (t & USB_PHY_UTMI_8B60M) | ||
dev_info(&phy->phy->dev, "UTMI 8bit 60MHz\n"); | ||
else | ||
dev_info(&phy->phy->dev, "UTMI 16bit 30MHz\n"); | ||
|
||
return 0; | ||
} | ||
|
||
static int ralink_usb_phy_power_off(struct phy *_phy) | ||
{ | ||
struct ralink_usb_phy *phy = phy_get_drvdata(_phy); | ||
|
||
/* disable the phy */ | ||
regmap_update_bits(phy->sysctl, RT_SYSC_REG_CLKCFG1, | ||
phy->clk, 0); | ||
|
||
/* assert the reset lines */ | ||
reset_control_assert(phy->rstdev); | ||
reset_control_assert(phy->rsthost); | ||
|
||
return 0; | ||
} | ||
|
||
static struct phy_ops ralink_usb_phy_ops = { | ||
.power_on = ralink_usb_phy_power_on, | ||
.power_off = ralink_usb_phy_power_off, | ||
.owner = THIS_MODULE, | ||
}; | ||
|
||
static const struct of_device_id ralink_usb_phy_of_match[] = { | ||
{ | ||
.compatible = "ralink,rt3352-usbphy", | ||
.data = (void *) (RT_CLKCFG1_UPHY1_CLK_EN | | ||
RT_CLKCFG1_UPHY0_CLK_EN) | ||
}, | ||
{ | ||
.compatible = "mediatek,mt7620-usbphy", | ||
.data = (void *) (MT7620_CLKCFG1_UPHY1_CLK_EN | | ||
MT7620_CLKCFG1_UPHY0_CLK_EN) | ||
}, | ||
{ | ||
.compatible = "mediatek,mt7628-usbphy", | ||
.data = (void *) (MT7620_CLKCFG1_UPHY1_CLK_EN | | ||
MT7620_CLKCFG1_UPHY0_CLK_EN) }, | ||
{ }, | ||
}; | ||
MODULE_DEVICE_TABLE(of, ralink_usb_phy_of_match); | ||
|
||
static int ralink_usb_phy_probe(struct platform_device *pdev) | ||
{ | ||
struct device *dev = &pdev->dev; | ||
struct resource *res; | ||
struct phy_provider *phy_provider; | ||
const struct of_device_id *match; | ||
struct ralink_usb_phy *phy; | ||
|
||
match = of_match_device(ralink_usb_phy_of_match, &pdev->dev); | ||
if (!match) | ||
return -ENODEV; | ||
|
||
phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL); | ||
if (!phy) | ||
return -ENOMEM; | ||
|
||
phy->clk = (u32) match->data; | ||
phy->base = NULL; | ||
|
||
phy->sysctl = syscon_regmap_lookup_by_phandle(dev->of_node, "ralink,sysctl"); | ||
if (IS_ERR(phy->sysctl)) { | ||
dev_err(dev, "failed to get sysctl registers\n"); | ||
return PTR_ERR(phy->sysctl); | ||
} | ||
|
||
/* The MT7628 and MT7688 require extra setup of PHY registers. */ | ||
if (of_device_is_compatible(dev->of_node, "mediatek,mt7628-usbphy")) { | ||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | ||
phy->base = devm_ioremap_resource(&pdev->dev, res); | ||
if (IS_ERR(phy->base)) { | ||
dev_err(dev, "failed to remap register memory\n"); | ||
return PTR_ERR(phy->base); | ||
} | ||
} | ||
|
||
phy->rsthost = devm_reset_control_get(&pdev->dev, "host"); | ||
if (IS_ERR(phy->rsthost)) { | ||
dev_err(dev, "host reset is missing\n"); | ||
return PTR_ERR(phy->rsthost); | ||
} | ||
|
||
phy->rstdev = devm_reset_control_get(&pdev->dev, "device"); | ||
if (IS_ERR(phy->rstdev)) { | ||
dev_err(dev, "device reset is missing\n"); | ||
return PTR_ERR(phy->rstdev); | ||
} | ||
|
||
phy->phy = devm_phy_create(dev, NULL, &ralink_usb_phy_ops); | ||
if (IS_ERR(phy->phy)) { | ||
dev_err(dev, "failed to create PHY\n"); | ||
return PTR_ERR(phy->phy); | ||
} | ||
phy_set_drvdata(phy->phy, phy); | ||
|
||
phy_provider = devm_of_phy_provider_register(dev, of_phy_simple_xlate); | ||
|
||
return PTR_ERR_OR_ZERO(phy_provider); | ||
} | ||
|
||
static struct platform_driver ralink_usb_phy_driver = { | ||
.probe = ralink_usb_phy_probe, | ||
.driver = { | ||
.of_match_table = ralink_usb_phy_of_match, | ||
.name = "ralink-usb-phy", | ||
} | ||
}; | ||
module_platform_driver(ralink_usb_phy_driver); | ||
|
||
MODULE_DESCRIPTION("Ralink USB phy driver"); | ||
MODULE_AUTHOR("John Crispin <[email protected]>"); | ||
MODULE_LICENSE("GPL v2"); |