From: Heiko Stuebner <heiko@sntech.de>
To: Kishon Vijay Abraham I <kishon@ti.com>
Cc: linux-arm-kernel@lists.infradead.org,
linux-kernel@vger.kernel.org, linux-rockchip@lists.infradead.org,
arnd@arndb.de, romain.perier@gmail.com
Subject: Re: [PATCH v5.1] phy: rockchip-usb: add handler for usb-uart functionality
Date: Mon, 22 Feb 2016 12:53:21 +0100 [thread overview]
Message-ID: <2151229.sTi0mmqIUM@phil> (raw)
In-Reply-To: <56CAF4DE.2060708@ti.com>
Am Montag, 22. Februar 2016, 17:15:34 schrieb Kishon Vijay Abraham I:
> > changes in v5.1:
> > - fix corruptions that happened when sending v5
>
> I still see the corruption.
>
> This is how I see the patch after downloading (from both mutt and
> thunderbird)
interesting ... I see this now too, but "git am" was happy with the encoding
it seems, I did a
git am foo.mbox
git format-patch -1
diff -u new.patch orig.patch
and the result did match (minus the comment section).
Anyway, I'll go with git send-email now ;-)
Sorry about that
Heiko
> @@ -3486,6 +3486,12 @@ bytes respectively. Such letter
> suffixes can also be= entirely omitted.
> =20
> ro [KNL] Mount root device read-only on boot
> =20
> + rockchip.usb_uart
> + Enable the uart passthrough on the designated usb
> port + on Rockchip SoCs. When active, the signals
> of the + debug-uart get routed to the D+ and D-
> pins of the usb + port and the regular usb
> controller gets disabled. +
> root=3D [KNL] Root filesystem
> See name_to_dev_t comment in init/do_mounts.c.
> =20
> diff --git a/drivers/phy/phy-rockchip-usb.c
> b/drivers/phy/phy-rockchip-usb.= c
> index 33a80eb..f62d899 100644
> --- a/drivers/phy/phy-rockchip-usb.c
> +++ b/drivers/phy/phy-rockchip-usb.c
> @@ -30,21 +30,23 @@
> #include <linux/regmap.h>
> #include <linux/mfd/syscon.h>
> =20
> -/*
> - * The higher 16-bit of this register is used for write protection
> - * only if BIT(13 + 16) set to 1 the BIT(13) can be written.
> - */
> -#define SIDDQ_WRITE_ENA BIT(29)
> -#define SIDDQ_ON BIT(13)
> -#define SIDDQ_OFF (0 << 13)
> +static int enable_usb_uart;
> +
> +#define HIWORD_UPDATE(val, mask) \
> + ((val) | (mask) << 16)
> +
> +#define UOC_CON0_SIDDQ BIT(13)
> =20
> struct rockchip_usb_phys {
> int reg;
> const char *pll_name;
> };
> =20
> <snip>
> .
> .
>
> I didn't see this problem with the other patches I applied today.
>
> Thanks
> Kishon
>
> > changes in v5:
> > - only compile debug uart functionality if the phy is compiled in
> >
> > fixes initcall conflict and debug functionality also is only really
> > usable if it's available early
> >
> > changes in v4:
> > - drop the rest of the phy-series, as the patches have gotten applied
> >
> > So far, this hasn't gotten eyeballs yet, so citing discussion-parts from
> > the v3 coverletter:
> >
> > The patch, while not associated with the new pll handling, also builds
> > on the groundwork introduced there and adds support for the function
> > repurposing one of the phys as passthrough for uart-data. This enables
> > attaching a ttl converter to the D+ and D- pins of an usb cable to
> > receive uart data this way, when it is not really possible to attach
> > a regular serial console to a board.
> >
> > One point of critique in my first iteration [0] of this was, that
> > due to when the reconfiguration happens we may miss parts of the logs
> > when earlycon is enabled. So far early_initcall gets used as the
> > unflattened devicetree is necessary to set this up. Doing this for
> > example in the early_param directly would require parsing the flattened
> > devicetree to get needed nodes and properties.
> >
> > I still maintain that if you're working on anything before smp-bringup
> > you should use a real dev-board instead or try to solder uart cables
> > on hopefully available test-points .
> >
> > Documentation/kernel-parameters.txt | 6 +
> > drivers/phy/phy-rockchip-usb.c | 233
> > ++++++++++++++++++++++++++++++------ 2 files changed, 203
> > insertions(+), 36 deletions(-)
> >
> > diff --git a/Documentation/kernel-parameters.txt
> > b/Documentation/kernel-parameters.txt index 87d40a7..d91417b 100644
> > --- a/Documentation/kernel-parameters.txt
> > +++ b/Documentation/kernel-parameters.txt
> > @@ -3486,6 +3486,12 @@ bytes respectively. Such letter suffixes can also
> > be entirely omitted.>
> > ro [KNL] Mount root device read-only on boot
> >
> > + rockchip.usb_uart
> > + Enable the uart passthrough on the designated usb port
> > + on Rockchip SoCs. When active, the signals of the
> > + debug-uart get routed to the D+ and D- pins of the usb
> > + port and the regular usb controller gets disabled.
> > +
> >
> > root= [KNL] Root filesystem
> >
> > See name_to_dev_t comment in init/do_mounts.c.
> >
> > diff --git a/drivers/phy/phy-rockchip-usb.c
> > b/drivers/phy/phy-rockchip-usb.c index 33a80eb..f62d899 100644
> > --- a/drivers/phy/phy-rockchip-usb.c
> > +++ b/drivers/phy/phy-rockchip-usb.c
> > @@ -30,21 +30,23 @@
> >
> > #include <linux/regmap.h>
> > #include <linux/mfd/syscon.h>
> >
> > -/*
> > - * The higher 16-bit of this register is used for write protection
> > - * only if BIT(13 + 16) set to 1 the BIT(13) can be written.
> > - */
> > -#define SIDDQ_WRITE_ENA BIT(29)
> > -#define SIDDQ_ON BIT(13)
> > -#define SIDDQ_OFF (0 << 13)
> > +static int enable_usb_uart;
> > +
> > +#define HIWORD_UPDATE(val, mask) \
> > + ((val) | (mask) << 16)
> > +
> > +#define UOC_CON0_SIDDQ BIT(13)
> >
> > struct rockchip_usb_phys {
> >
> > int reg;
> > const char *pll_name;
> >
> > };
> >
> > +struct rockchip_usb_phy_base;
> >
> > struct rockchip_usb_phy_pdata {
> >
> > struct rockchip_usb_phys *phys;
> >
> > + int (*init_usb_uart)(struct regmap *grf);
> > + int usb_uart_phy;
> >
> > };
> >
> > struct rockchip_usb_phy_base {
> >
> > @@ -61,13 +63,15 @@ struct rockchip_usb_phy {
> >
> > struct clk *clk480m;
> > struct clk_hw clk480m_hw;
> > struct phy *phy;
> >
> > + bool uart_enabled;
> >
> > };
> >
> > static int rockchip_usb_phy_power(struct rockchip_usb_phy *phy,
> >
> > bool siddq)
> >
> > {
> >
> > - return regmap_write(phy->base->reg_base, phy->reg_offset,
> > - SIDDQ_WRITE_ENA | (siddq ? SIDDQ_ON : SIDDQ_OFF));
> > + u32 val = HIWORD_UPDATE(siddq ? UOC_CON0_SIDDQ : 0,
UOC_CON0_SIDDQ);
> > +
> > + return regmap_write(phy->base->reg_base, phy->reg_offset, val);
> >
> > }
> >
> > static unsigned long rockchip_usb_phy480m_recalc_rate(struct clk_hw
> > *hw,
> >
> > @@ -108,7 +112,7 @@ static int rockchip_usb_phy480m_is_enabled(struct
> > clk_hw *hw)>
> > if (ret < 0)
> >
> > return ret;
> >
> > - return (val & SIDDQ_ON) ? 0 : 1;
> > + return (val & UOC_CON0_SIDDQ) ? 0 : 1;
> >
> > }
> >
> > static const struct clk_ops rockchip_usb_phy480m_ops = {
> >
> > @@ -122,6 +126,9 @@ static int rockchip_usb_phy_power_off(struct phy
> > *_phy)>
> > {
> >
> > struct rockchip_usb_phy *phy = phy_get_drvdata(_phy);
> >
> > + if (phy->uart_enabled)
> > + return -EBUSY;
> > +
> >
> > clk_disable_unprepare(phy->clk480m);
> >
> > return 0;
> >
> > @@ -131,6 +138,9 @@ static int rockchip_usb_phy_power_on(struct phy
> > *_phy)>
> > {
> >
> > struct rockchip_usb_phy *phy = phy_get_drvdata(_phy);
> >
> > + if (phy->uart_enabled)
> > + return -EBUSY;
> > +
> >
> > return clk_prepare_enable(phy->clk480m);
> >
> > }
> >
> > @@ -144,8 +154,10 @@ static void rockchip_usb_phy_action(void *data)
> >
> > {
> >
> > struct rockchip_usb_phy *rk_phy = data;
> >
> > - of_clk_del_provider(rk_phy->np);
> > - clk_unregister(rk_phy->clk480m);
> > + if (!rk_phy->uart_enabled) {
> > + of_clk_del_provider(rk_phy->np);
> > + clk_unregister(rk_phy->clk480m);
> > + }
> >
> > if (rk_phy->clk)
> >
> > clk_put(rk_phy->clk);
> >
> > @@ -194,30 +206,35 @@ static int rockchip_usb_phy_init(struct
> > rockchip_usb_phy_base *base,>
> > return -EINVAL;
> >
> > }
> >
> > - if (rk_phy->clk) {
> > - clk_name = __clk_get_name(rk_phy->clk);
> > - init.flags = 0;
> > - init.parent_names = &clk_name;
> > - init.num_parents = 1;
> > + if (enable_usb_uart && base->pdata->usb_uart_phy == i) {
> > + dev_dbg(base->dev, "phy%d used as uart output\n", i);
> > + rk_phy->uart_enabled = true;
> >
> > } else {
> >
> > - init.flags = CLK_IS_ROOT;
> > - init.parent_names = NULL;
> > - init.num_parents = 0;
> > - }
> > + if (rk_phy->clk) {
> > + clk_name = __clk_get_name(rk_phy->clk);
> > + init.flags = 0;
> > + init.parent_names = &clk_name;
> > + init.num_parents = 1;
> > + } else {
> > + init.flags = CLK_IS_ROOT;
> > + init.parent_names = NULL;
> > + init.num_parents = 0;
> > + }
> >
> > - init.ops = &rockchip_usb_phy480m_ops;
> > - rk_phy->clk480m_hw.init = &init;
> > + init.ops = &rockchip_usb_phy480m_ops;
> > + rk_phy->clk480m_hw.init = &init;
> >
> > - rk_phy->clk480m = clk_register(base->dev, &rk_phy->clk480m_hw);
> > - if (IS_ERR(rk_phy->clk480m)) {
> > - err = PTR_ERR(rk_phy->clk480m);
> > - goto err_clk;
> > - }
> > + rk_phy->clk480m = clk_register(base->dev, &rk_phy->clk480m_hw);
> > + if (IS_ERR(rk_phy->clk480m)) {
> > + err = PTR_ERR(rk_phy->clk480m);
> > + goto err_clk;
> > + }
> >
> > - err = of_clk_add_provider(child, of_clk_src_simple_get,
> > - rk_phy->clk480m);
> > - if (err < 0)
> > - goto err_clk_prov;
> > + err = of_clk_add_provider(child, of_clk_src_simple_get,
> > + rk_phy->clk480m);
> > + if (err < 0)
> > + goto err_clk_prov;
> > + }
> >
> > err = devm_add_action(base->dev, rockchip_usb_phy_action, rk_phy);
> > if (err)
> >
> > @@ -230,13 +247,21 @@ static int rockchip_usb_phy_init(struct
> > rockchip_usb_phy_base *base,>
> > }
> > phy_set_drvdata(rk_phy->phy, rk_phy);
> >
> > - /* only power up usb phy when it use, so disable it when init*/
> > - return rockchip_usb_phy_power(rk_phy, 1);
> > + /*
> > + * When acting as uart-pipe, just keep clock on otherwise
> > + * only power up usb phy when it use, so disable it when init
> > + */
> > + if (rk_phy->uart_enabled)
> > + return clk_prepare_enable(rk_phy->clk);
> > + else
> > + return rockchip_usb_phy_power(rk_phy, 1);
> >
> > err_devm_action:
> > - of_clk_del_provider(child);
> > + if (!rk_phy->uart_enabled)
> > + of_clk_del_provider(child);
> >
> > err_clk_prov:
> > - clk_unregister(rk_phy->clk480m);
> > + if (!rk_phy->uart_enabled)
> > + clk_unregister(rk_phy->clk480m);
> >
> > err_clk:
> > if (rk_phy->clk)
> >
> > clk_put(rk_phy->clk);
> >
> > @@ -259,6 +284,86 @@ static const struct rockchip_usb_phy_pdata
> > rk3188_pdata = {>
> > },
> >
> > };
> >
> > +#define RK3288_UOC0_CON0 0x320
> > +#define RK3288_UOC0_CON0_COMMON_ON_N BIT(0)
> > +#define RK3288_UOC0_CON0_DISABLE BIT(4)
> > +
> > +#define RK3288_UOC0_CON2 0x328
> > +#define RK3288_UOC0_CON2_SOFT_CON_SEL BIT(2)
> > +
> > +#define RK3288_UOC0_CON3 0x32c
> > +#define RK3288_UOC0_CON3_UTMI_SUSPENDN BIT(0)
> > +#define RK3288_UOC0_CON3_UTMI_OPMODE_NODRIVING (1 << 1)
> > +#define RK3288_UOC0_CON3_UTMI_OPMODE_MASK (3 << 1)
> > +#define RK3288_UOC0_CON3_UTMI_XCVRSEELCT_FSTRANSC (1 << 3)
> > +#define RK3288_UOC0_CON3_UTMI_XCVRSEELCT_MASK (3 << 3)
> > +#define RK3288_UOC0_CON3_UTMI_TERMSEL_FULLSPEED BIT(5)
> > +#define RK3288_UOC0_CON3_BYPASSDMEN BIT(6)
> > +#define RK3288_UOC0_CON3_BYPASSSEL BIT(7)
> > +
> > +/*
> > + * Enable the bypass of uart2 data through the otg usb phy.
> > + * Original description in the TRM.
> > + * 1. Disable the OTG block by setting OTGDISABLE0 to 1’b1.
> > + * 2. Disable the pull-up resistance on the D+ line by setting
> > + * OPMODE0[1:0] to 2’b01.
> > + * 3. To ensure that the XO, Bias, and PLL blocks are powered down in
> > Suspend + * mode, set COMMONONN to 1’b1.
> > + * 4. Place the USB PHY in Suspend mode by setting SUSPENDM0 to 1’b0.
> > + * 5. Set BYPASSSEL0 to 1’b1.
> > + * 6. To transmit data, controls BYPASSDMEN0, and BYPASSDMDATA0.
> > + * To receive data, monitor FSVPLUS0.
> > + *
> > + * The actual code in the vendor kernel does some things differently.
> > + */
> > +static int __init rk3288_init_usb_uart(struct regmap *grf)
> > +{
> > + u32 val;
> > + int ret;
> > +
> > + /*
> > + * COMMON_ON and DISABLE settings are described in the TRM,
> > + * but were not present in the original code.
> > + * Also disable the analog phy components to save power.
> > + */
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON0_COMMON_ON_N
> > + | RK3288_UOC0_CON0_DISABLE
> > + | UOC_CON0_SIDDQ,
> > + RK3288_UOC0_CON0_COMMON_ON_N
> > + | RK3288_UOC0_CON0_DISABLE
> > + | UOC_CON0_SIDDQ);
> > + ret = regmap_write(grf, RK3288_UOC0_CON0, val);
> > + if (ret)
> > + return ret;
> > +
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON2_SOFT_CON_SEL,
> > + RK3288_UOC0_CON2_SOFT_CON_SEL);
> > + ret = regmap_write(grf, RK3288_UOC0_CON2, val);
> > + if (ret)
> > + return ret;
> > +
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON3_UTMI_OPMODE_NODRIVING
> > + | RK3288_UOC0_CON3_UTMI_XCVRSEELCT_FSTRANSC
> > + | RK3288_UOC0_CON3_UTMI_TERMSEL_FULLSPEED,
> > + RK3288_UOC0_CON3_UTMI_SUSPENDN
> > + | RK3288_UOC0_CON3_UTMI_OPMODE_MASK
> > + | RK3288_UOC0_CON3_UTMI_XCVRSEELCT_MASK
> > + | RK3288_UOC0_CON3_UTMI_TERMSEL_FULLSPEED);
> > + ret = regmap_write(grf, RK3288_UOC0_CON3, val);
> > + if (ret)
> > + return ret;
> > +
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON3_BYPASSSEL
> > + | RK3288_UOC0_CON3_BYPASSDMEN,
> > + RK3288_UOC0_CON3_BYPASSSEL
> > + | RK3288_UOC0_CON3_BYPASSDMEN);
> > + ret = regmap_write(grf, RK3288_UOC0_CON3, val);
> > + if (ret)
> > + return ret;
> > +
> > + return 0;
> > +}
> > +
> >
> > static const struct rockchip_usb_phy_pdata rk3288_pdata = {
> >
> > .phys = (struct rockchip_usb_phys[]){
> >
> > { .reg = 0x320, .pll_name = "sclk_otgphy0_480m" },
> >
> > @@ -266,6 +371,8 @@ static const struct rockchip_usb_phy_pdata
> > rk3288_pdata = {>
> > { .reg = 0x348, .pll_name = "sclk_otgphy2_480m" },
> > { /* sentinel */ }
> >
> > },
> >
> > + .init_usb_uart = rk3288_init_usb_uart,
> > + .usb_uart_phy = 0,
> >
> > };
> >
> > static int rockchip_usb_phy_probe(struct platform_device *pdev)
> >
> > @@ -328,6 +435,60 @@ static struct platform_driver rockchip_usb_driver =
> > {>
> > module_platform_driver(rockchip_usb_driver);
> >
> > +#ifndef MODULE
> > +static int __init rockchip_init_usb_uart(void)
> > +{
> > + const struct of_device_id *match;
> > + const struct rockchip_usb_phy_pdata *data;
> > + struct device_node *np;
> > + struct regmap *grf;
> > + int ret;
> > +
> > + if (!enable_usb_uart)
> > + return 0;
> > +
> > + np = of_find_matching_node_and_match(NULL, rockchip_usb_phy_dt_ids,
> > + &match);
> > + if (!np) {
> > + pr_err("%s: failed to find usbphy node\n", __func__);
> > + return -ENOTSUPP;
> > + }
> > +
> > + pr_debug("%s: using settings for %s\n", __func__, match-
>compatible);
> > + data = match->data;
> > +
> > + if (!data->init_usb_uart) {
> > + pr_err("%s: usb-uart not available on %s\n",
> > + __func__, match->compatible);
> > + return -ENOTSUPP;
> > + }
> > +
> > + grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
> > + if (IS_ERR(grf)) {
> > + pr_err("%s: Missing rockchip,grf property, %lu\n",
> > + __func__, PTR_ERR(grf));
> > + return PTR_ERR(grf);
> > + }
> > +
> > + ret = data->init_usb_uart(grf);
> > + if (ret) {
> > + pr_err("%s: could not init usb_uart, %d\n", __func__, ret);
> > + enable_usb_uart = 0;
> > + return ret;
> > + }
> > +
> > + return 0;
> > +}
> > +early_initcall(rockchip_init_usb_uart);
> > +
> > +static int __init rockchip_usb_uart(char *buf)
> > +{
> > + enable_usb_uart = true;
> > + return 0;
> > +}
> > +early_param("rockchip.usb_uart", rockchip_usb_uart);
> > +#endif
> > +
> >
> > MODULE_AUTHOR("Yunzhi Li <lyz@rock-chips.com>");
> > MODULE_DESCRIPTION("Rockchip USB 2.0 PHY driver");
> > MODULE_LICENSE("GPL v2");
WARNING: multiple messages have this Message-ID (diff)
From: heiko@sntech.de (Heiko Stuebner)
To: linux-arm-kernel@lists.infradead.org
Subject: [PATCH v5.1] phy: rockchip-usb: add handler for usb-uart functionality
Date: Mon, 22 Feb 2016 12:53:21 +0100 [thread overview]
Message-ID: <2151229.sTi0mmqIUM@phil> (raw)
In-Reply-To: <56CAF4DE.2060708@ti.com>
Am Montag, 22. Februar 2016, 17:15:34 schrieb Kishon Vijay Abraham I:
> > changes in v5.1:
> > - fix corruptions that happened when sending v5
>
> I still see the corruption.
>
> This is how I see the patch after downloading (from both mutt and
> thunderbird)
interesting ... I see this now too, but "git am" was happy with the encoding
it seems, I did a
git am foo.mbox
git format-patch -1
diff -u new.patch orig.patch
and the result did match (minus the comment section).
Anyway, I'll go with git send-email now ;-)
Sorry about that
Heiko
> @@ -3486,6 +3486,12 @@ bytes respectively. Such letter
> suffixes can also be= entirely omitted.
> =20
> ro [KNL] Mount root device read-only on boot
> =20
> + rockchip.usb_uart
> + Enable the uart passthrough on the designated usb
> port + on Rockchip SoCs. When active, the signals
> of the + debug-uart get routed to the D+ and D-
> pins of the usb + port and the regular usb
> controller gets disabled. +
> root=3D [KNL] Root filesystem
> See name_to_dev_t comment in init/do_mounts.c.
> =20
> diff --git a/drivers/phy/phy-rockchip-usb.c
> b/drivers/phy/phy-rockchip-usb.= c
> index 33a80eb..f62d899 100644
> --- a/drivers/phy/phy-rockchip-usb.c
> +++ b/drivers/phy/phy-rockchip-usb.c
> @@ -30,21 +30,23 @@
> #include <linux/regmap.h>
> #include <linux/mfd/syscon.h>
> =20
> -/*
> - * The higher 16-bit of this register is used for write protection
> - * only if BIT(13 + 16) set to 1 the BIT(13) can be written.
> - */
> -#define SIDDQ_WRITE_ENA BIT(29)
> -#define SIDDQ_ON BIT(13)
> -#define SIDDQ_OFF (0 << 13)
> +static int enable_usb_uart;
> +
> +#define HIWORD_UPDATE(val, mask) \
> + ((val) | (mask) << 16)
> +
> +#define UOC_CON0_SIDDQ BIT(13)
> =20
> struct rockchip_usb_phys {
> int reg;
> const char *pll_name;
> };
> =20
> <snip>
> .
> .
>
> I didn't see this problem with the other patches I applied today.
>
> Thanks
> Kishon
>
> > changes in v5:
> > - only compile debug uart functionality if the phy is compiled in
> >
> > fixes initcall conflict and debug functionality also is only really
> > usable if it's available early
> >
> > changes in v4:
> > - drop the rest of the phy-series, as the patches have gotten applied
> >
> > So far, this hasn't gotten eyeballs yet, so citing discussion-parts from
> > the v3 coverletter:
> >
> > The patch, while not associated with the new pll handling, also builds
> > on the groundwork introduced there and adds support for the function
> > repurposing one of the phys as passthrough for uart-data. This enables
> > attaching a ttl converter to the D+ and D- pins of an usb cable to
> > receive uart data this way, when it is not really possible to attach
> > a regular serial console to a board.
> >
> > One point of critique in my first iteration [0] of this was, that
> > due to when the reconfiguration happens we may miss parts of the logs
> > when earlycon is enabled. So far early_initcall gets used as the
> > unflattened devicetree is necessary to set this up. Doing this for
> > example in the early_param directly would require parsing the flattened
> > devicetree to get needed nodes and properties.
> >
> > I still maintain that if you're working on anything before smp-bringup
> > you should use a real dev-board instead or try to solder uart cables
> > on hopefully available test-points .
> >
> > Documentation/kernel-parameters.txt | 6 +
> > drivers/phy/phy-rockchip-usb.c | 233
> > ++++++++++++++++++++++++++++++------ 2 files changed, 203
> > insertions(+), 36 deletions(-)
> >
> > diff --git a/Documentation/kernel-parameters.txt
> > b/Documentation/kernel-parameters.txt index 87d40a7..d91417b 100644
> > --- a/Documentation/kernel-parameters.txt
> > +++ b/Documentation/kernel-parameters.txt
> > @@ -3486,6 +3486,12 @@ bytes respectively. Such letter suffixes can also
> > be entirely omitted.>
> > ro [KNL] Mount root device read-only on boot
> >
> > + rockchip.usb_uart
> > + Enable the uart passthrough on the designated usb port
> > + on Rockchip SoCs. When active, the signals of the
> > + debug-uart get routed to the D+ and D- pins of the usb
> > + port and the regular usb controller gets disabled.
> > +
> >
> > root= [KNL] Root filesystem
> >
> > See name_to_dev_t comment in init/do_mounts.c.
> >
> > diff --git a/drivers/phy/phy-rockchip-usb.c
> > b/drivers/phy/phy-rockchip-usb.c index 33a80eb..f62d899 100644
> > --- a/drivers/phy/phy-rockchip-usb.c
> > +++ b/drivers/phy/phy-rockchip-usb.c
> > @@ -30,21 +30,23 @@
> >
> > #include <linux/regmap.h>
> > #include <linux/mfd/syscon.h>
> >
> > -/*
> > - * The higher 16-bit of this register is used for write protection
> > - * only if BIT(13 + 16) set to 1 the BIT(13) can be written.
> > - */
> > -#define SIDDQ_WRITE_ENA BIT(29)
> > -#define SIDDQ_ON BIT(13)
> > -#define SIDDQ_OFF (0 << 13)
> > +static int enable_usb_uart;
> > +
> > +#define HIWORD_UPDATE(val, mask) \
> > + ((val) | (mask) << 16)
> > +
> > +#define UOC_CON0_SIDDQ BIT(13)
> >
> > struct rockchip_usb_phys {
> >
> > int reg;
> > const char *pll_name;
> >
> > };
> >
> > +struct rockchip_usb_phy_base;
> >
> > struct rockchip_usb_phy_pdata {
> >
> > struct rockchip_usb_phys *phys;
> >
> > + int (*init_usb_uart)(struct regmap *grf);
> > + int usb_uart_phy;
> >
> > };
> >
> > struct rockchip_usb_phy_base {
> >
> > @@ -61,13 +63,15 @@ struct rockchip_usb_phy {
> >
> > struct clk *clk480m;
> > struct clk_hw clk480m_hw;
> > struct phy *phy;
> >
> > + bool uart_enabled;
> >
> > };
> >
> > static int rockchip_usb_phy_power(struct rockchip_usb_phy *phy,
> >
> > bool siddq)
> >
> > {
> >
> > - return regmap_write(phy->base->reg_base, phy->reg_offset,
> > - SIDDQ_WRITE_ENA | (siddq ? SIDDQ_ON : SIDDQ_OFF));
> > + u32 val = HIWORD_UPDATE(siddq ? UOC_CON0_SIDDQ : 0,
UOC_CON0_SIDDQ);
> > +
> > + return regmap_write(phy->base->reg_base, phy->reg_offset, val);
> >
> > }
> >
> > static unsigned long rockchip_usb_phy480m_recalc_rate(struct clk_hw
> > *hw,
> >
> > @@ -108,7 +112,7 @@ static int rockchip_usb_phy480m_is_enabled(struct
> > clk_hw *hw)>
> > if (ret < 0)
> >
> > return ret;
> >
> > - return (val & SIDDQ_ON) ? 0 : 1;
> > + return (val & UOC_CON0_SIDDQ) ? 0 : 1;
> >
> > }
> >
> > static const struct clk_ops rockchip_usb_phy480m_ops = {
> >
> > @@ -122,6 +126,9 @@ static int rockchip_usb_phy_power_off(struct phy
> > *_phy)>
> > {
> >
> > struct rockchip_usb_phy *phy = phy_get_drvdata(_phy);
> >
> > + if (phy->uart_enabled)
> > + return -EBUSY;
> > +
> >
> > clk_disable_unprepare(phy->clk480m);
> >
> > return 0;
> >
> > @@ -131,6 +138,9 @@ static int rockchip_usb_phy_power_on(struct phy
> > *_phy)>
> > {
> >
> > struct rockchip_usb_phy *phy = phy_get_drvdata(_phy);
> >
> > + if (phy->uart_enabled)
> > + return -EBUSY;
> > +
> >
> > return clk_prepare_enable(phy->clk480m);
> >
> > }
> >
> > @@ -144,8 +154,10 @@ static void rockchip_usb_phy_action(void *data)
> >
> > {
> >
> > struct rockchip_usb_phy *rk_phy = data;
> >
> > - of_clk_del_provider(rk_phy->np);
> > - clk_unregister(rk_phy->clk480m);
> > + if (!rk_phy->uart_enabled) {
> > + of_clk_del_provider(rk_phy->np);
> > + clk_unregister(rk_phy->clk480m);
> > + }
> >
> > if (rk_phy->clk)
> >
> > clk_put(rk_phy->clk);
> >
> > @@ -194,30 +206,35 @@ static int rockchip_usb_phy_init(struct
> > rockchip_usb_phy_base *base,>
> > return -EINVAL;
> >
> > }
> >
> > - if (rk_phy->clk) {
> > - clk_name = __clk_get_name(rk_phy->clk);
> > - init.flags = 0;
> > - init.parent_names = &clk_name;
> > - init.num_parents = 1;
> > + if (enable_usb_uart && base->pdata->usb_uart_phy == i) {
> > + dev_dbg(base->dev, "phy%d used as uart output\n", i);
> > + rk_phy->uart_enabled = true;
> >
> > } else {
> >
> > - init.flags = CLK_IS_ROOT;
> > - init.parent_names = NULL;
> > - init.num_parents = 0;
> > - }
> > + if (rk_phy->clk) {
> > + clk_name = __clk_get_name(rk_phy->clk);
> > + init.flags = 0;
> > + init.parent_names = &clk_name;
> > + init.num_parents = 1;
> > + } else {
> > + init.flags = CLK_IS_ROOT;
> > + init.parent_names = NULL;
> > + init.num_parents = 0;
> > + }
> >
> > - init.ops = &rockchip_usb_phy480m_ops;
> > - rk_phy->clk480m_hw.init = &init;
> > + init.ops = &rockchip_usb_phy480m_ops;
> > + rk_phy->clk480m_hw.init = &init;
> >
> > - rk_phy->clk480m = clk_register(base->dev, &rk_phy->clk480m_hw);
> > - if (IS_ERR(rk_phy->clk480m)) {
> > - err = PTR_ERR(rk_phy->clk480m);
> > - goto err_clk;
> > - }
> > + rk_phy->clk480m = clk_register(base->dev, &rk_phy->clk480m_hw);
> > + if (IS_ERR(rk_phy->clk480m)) {
> > + err = PTR_ERR(rk_phy->clk480m);
> > + goto err_clk;
> > + }
> >
> > - err = of_clk_add_provider(child, of_clk_src_simple_get,
> > - rk_phy->clk480m);
> > - if (err < 0)
> > - goto err_clk_prov;
> > + err = of_clk_add_provider(child, of_clk_src_simple_get,
> > + rk_phy->clk480m);
> > + if (err < 0)
> > + goto err_clk_prov;
> > + }
> >
> > err = devm_add_action(base->dev, rockchip_usb_phy_action, rk_phy);
> > if (err)
> >
> > @@ -230,13 +247,21 @@ static int rockchip_usb_phy_init(struct
> > rockchip_usb_phy_base *base,>
> > }
> > phy_set_drvdata(rk_phy->phy, rk_phy);
> >
> > - /* only power up usb phy when it use, so disable it when init*/
> > - return rockchip_usb_phy_power(rk_phy, 1);
> > + /*
> > + * When acting as uart-pipe, just keep clock on otherwise
> > + * only power up usb phy when it use, so disable it when init
> > + */
> > + if (rk_phy->uart_enabled)
> > + return clk_prepare_enable(rk_phy->clk);
> > + else
> > + return rockchip_usb_phy_power(rk_phy, 1);
> >
> > err_devm_action:
> > - of_clk_del_provider(child);
> > + if (!rk_phy->uart_enabled)
> > + of_clk_del_provider(child);
> >
> > err_clk_prov:
> > - clk_unregister(rk_phy->clk480m);
> > + if (!rk_phy->uart_enabled)
> > + clk_unregister(rk_phy->clk480m);
> >
> > err_clk:
> > if (rk_phy->clk)
> >
> > clk_put(rk_phy->clk);
> >
> > @@ -259,6 +284,86 @@ static const struct rockchip_usb_phy_pdata
> > rk3188_pdata = {>
> > },
> >
> > };
> >
> > +#define RK3288_UOC0_CON0 0x320
> > +#define RK3288_UOC0_CON0_COMMON_ON_N BIT(0)
> > +#define RK3288_UOC0_CON0_DISABLE BIT(4)
> > +
> > +#define RK3288_UOC0_CON2 0x328
> > +#define RK3288_UOC0_CON2_SOFT_CON_SEL BIT(2)
> > +
> > +#define RK3288_UOC0_CON3 0x32c
> > +#define RK3288_UOC0_CON3_UTMI_SUSPENDN BIT(0)
> > +#define RK3288_UOC0_CON3_UTMI_OPMODE_NODRIVING (1 << 1)
> > +#define RK3288_UOC0_CON3_UTMI_OPMODE_MASK (3 << 1)
> > +#define RK3288_UOC0_CON3_UTMI_XCVRSEELCT_FSTRANSC (1 << 3)
> > +#define RK3288_UOC0_CON3_UTMI_XCVRSEELCT_MASK (3 << 3)
> > +#define RK3288_UOC0_CON3_UTMI_TERMSEL_FULLSPEED BIT(5)
> > +#define RK3288_UOC0_CON3_BYPASSDMEN BIT(6)
> > +#define RK3288_UOC0_CON3_BYPASSSEL BIT(7)
> > +
> > +/*
> > + * Enable the bypass of uart2 data through the otg usb phy.
> > + * Original description in the TRM.
> > + * 1. Disable the OTG block by setting OTGDISABLE0 to 1?b1.
> > + * 2. Disable the pull-up resistance on the D+ line by setting
> > + * OPMODE0[1:0] to 2?b01.
> > + * 3. To ensure that the XO, Bias, and PLL blocks are powered down in
> > Suspend + * mode, set COMMONONN to 1?b1.
> > + * 4. Place the USB PHY in Suspend mode by setting SUSPENDM0 to 1?b0.
> > + * 5. Set BYPASSSEL0 to 1?b1.
> > + * 6. To transmit data, controls BYPASSDMEN0, and BYPASSDMDATA0.
> > + * To receive data, monitor FSVPLUS0.
> > + *
> > + * The actual code in the vendor kernel does some things differently.
> > + */
> > +static int __init rk3288_init_usb_uart(struct regmap *grf)
> > +{
> > + u32 val;
> > + int ret;
> > +
> > + /*
> > + * COMMON_ON and DISABLE settings are described in the TRM,
> > + * but were not present in the original code.
> > + * Also disable the analog phy components to save power.
> > + */
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON0_COMMON_ON_N
> > + | RK3288_UOC0_CON0_DISABLE
> > + | UOC_CON0_SIDDQ,
> > + RK3288_UOC0_CON0_COMMON_ON_N
> > + | RK3288_UOC0_CON0_DISABLE
> > + | UOC_CON0_SIDDQ);
> > + ret = regmap_write(grf, RK3288_UOC0_CON0, val);
> > + if (ret)
> > + return ret;
> > +
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON2_SOFT_CON_SEL,
> > + RK3288_UOC0_CON2_SOFT_CON_SEL);
> > + ret = regmap_write(grf, RK3288_UOC0_CON2, val);
> > + if (ret)
> > + return ret;
> > +
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON3_UTMI_OPMODE_NODRIVING
> > + | RK3288_UOC0_CON3_UTMI_XCVRSEELCT_FSTRANSC
> > + | RK3288_UOC0_CON3_UTMI_TERMSEL_FULLSPEED,
> > + RK3288_UOC0_CON3_UTMI_SUSPENDN
> > + | RK3288_UOC0_CON3_UTMI_OPMODE_MASK
> > + | RK3288_UOC0_CON3_UTMI_XCVRSEELCT_MASK
> > + | RK3288_UOC0_CON3_UTMI_TERMSEL_FULLSPEED);
> > + ret = regmap_write(grf, RK3288_UOC0_CON3, val);
> > + if (ret)
> > + return ret;
> > +
> > + val = HIWORD_UPDATE(RK3288_UOC0_CON3_BYPASSSEL
> > + | RK3288_UOC0_CON3_BYPASSDMEN,
> > + RK3288_UOC0_CON3_BYPASSSEL
> > + | RK3288_UOC0_CON3_BYPASSDMEN);
> > + ret = regmap_write(grf, RK3288_UOC0_CON3, val);
> > + if (ret)
> > + return ret;
> > +
> > + return 0;
> > +}
> > +
> >
> > static const struct rockchip_usb_phy_pdata rk3288_pdata = {
> >
> > .phys = (struct rockchip_usb_phys[]){
> >
> > { .reg = 0x320, .pll_name = "sclk_otgphy0_480m" },
> >
> > @@ -266,6 +371,8 @@ static const struct rockchip_usb_phy_pdata
> > rk3288_pdata = {>
> > { .reg = 0x348, .pll_name = "sclk_otgphy2_480m" },
> > { /* sentinel */ }
> >
> > },
> >
> > + .init_usb_uart = rk3288_init_usb_uart,
> > + .usb_uart_phy = 0,
> >
> > };
> >
> > static int rockchip_usb_phy_probe(struct platform_device *pdev)
> >
> > @@ -328,6 +435,60 @@ static struct platform_driver rockchip_usb_driver =
> > {>
> > module_platform_driver(rockchip_usb_driver);
> >
> > +#ifndef MODULE
> > +static int __init rockchip_init_usb_uart(void)
> > +{
> > + const struct of_device_id *match;
> > + const struct rockchip_usb_phy_pdata *data;
> > + struct device_node *np;
> > + struct regmap *grf;
> > + int ret;
> > +
> > + if (!enable_usb_uart)
> > + return 0;
> > +
> > + np = of_find_matching_node_and_match(NULL, rockchip_usb_phy_dt_ids,
> > + &match);
> > + if (!np) {
> > + pr_err("%s: failed to find usbphy node\n", __func__);
> > + return -ENOTSUPP;
> > + }
> > +
> > + pr_debug("%s: using settings for %s\n", __func__, match-
>compatible);
> > + data = match->data;
> > +
> > + if (!data->init_usb_uart) {
> > + pr_err("%s: usb-uart not available on %s\n",
> > + __func__, match->compatible);
> > + return -ENOTSUPP;
> > + }
> > +
> > + grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
> > + if (IS_ERR(grf)) {
> > + pr_err("%s: Missing rockchip,grf property, %lu\n",
> > + __func__, PTR_ERR(grf));
> > + return PTR_ERR(grf);
> > + }
> > +
> > + ret = data->init_usb_uart(grf);
> > + if (ret) {
> > + pr_err("%s: could not init usb_uart, %d\n", __func__, ret);
> > + enable_usb_uart = 0;
> > + return ret;
> > + }
> > +
> > + return 0;
> > +}
> > +early_initcall(rockchip_init_usb_uart);
> > +
> > +static int __init rockchip_usb_uart(char *buf)
> > +{
> > + enable_usb_uart = true;
> > + return 0;
> > +}
> > +early_param("rockchip.usb_uart", rockchip_usb_uart);
> > +#endif
> > +
> >
> > MODULE_AUTHOR("Yunzhi Li <lyz@rock-chips.com>");
> > MODULE_DESCRIPTION("Rockchip USB 2.0 PHY driver");
> > MODULE_LICENSE("GPL v2");
next prev parent reply other threads:[~2016-02-22 11:53 UTC|newest]
Thread overview: 7+ messages / expand[flat|nested] mbox.gz Atom feed top
2016-02-22 11:41 [PATCH v5.1] phy: rockchip-usb: add handler for usb-uart functionality Heiko Stuebner
2016-02-22 11:41 ` Heiko Stuebner
2016-02-22 11:45 ` Kishon Vijay Abraham I
2016-02-22 11:45 ` Kishon Vijay Abraham I
2016-02-22 11:45 ` Kishon Vijay Abraham I
2016-02-22 11:53 ` Heiko Stuebner [this message]
2016-02-22 11:53 ` Heiko Stuebner
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=2151229.sTi0mmqIUM@phil \
--to=heiko@sntech.de \
--cc=arnd@arndb.de \
--cc=kishon@ti.com \
--cc=linux-arm-kernel@lists.infradead.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-rockchip@lists.infradead.org \
--cc=romain.perier@gmail.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.