lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Date:	Fri, 08 Aug 2014 18:25:06 +0200
From:	Heiko Stübner <heiko@...ech.de>
To:	Romain Perier <romain.perier@...il.com>
Cc:	davem@...emloft.net, max.schwarz@...ine.de, b.galvani@...il.com,
	eric.dumazet@...il.com, netdev@...r.kernel.org
Subject: Re: [PATCH 2/2] ethernet: arc: Add support for Rockchip SoC glue layer device tree bindings

Hi Romain,

this new emac-variant is missing a dt-bindings document.
And additional to Varka's comments, more inline:

Am Freitag, 8. August 2014, 12:27:55 schrieb Romain Perier:
> This patch defines a platform glue layer for Rockchip SoCs which support
> arc-emac driver. It ensures that regulator for the rmii is on before trying
> to connect to the ethernet controller. It applies right speed and mode
> changes to the grf when ethernet settings changes.
> 
> Signed-off-by: Romain Perier <romain.perier@...il.com>
> ---
>  arch/arm/boot/dts/rk3188-radxarock.dts   |   2 +
>  arch/arm/boot/dts/rk3xxx.dtsi            |   3 +
>  drivers/net/ethernet/arc/Kconfig         |   9 ++
>  drivers/net/ethernet/arc/Makefile        |   1 +
>  drivers/net/ethernet/arc/emac_rockchip.c | 191
> +++++++++++++++++++++++++++++++ 5 files changed, 206 insertions(+)
>  create mode 100644 drivers/net/ethernet/arc/emac_rockchip.c
> 
> diff --git a/arch/arm/boot/dts/rk3188-radxarock.dts
> b/arch/arm/boot/dts/rk3188-radxarock.dts index d149155..773a433 100644
> --- a/arch/arm/boot/dts/rk3188-radxarock.dts
> +++ b/arch/arm/boot/dts/rk3188-radxarock.dts
> @@ -110,12 +110,14 @@
> 
>  &emac {
>  	status = "okay";
> +	compatible = "rockchip,rk3188-emac";
> 
>  	pinctrl-names = "default";
>  	pinctrl-0 = <&emac_xfer>, <&emac_mdio>, <&phy_int>;
> 
>  	mac-address = [ c6 ef 91 8e 60 4b ];
>  	phy = <&phy0>;
> +	phy-supply = <&vcc_rmii>;
> 
>  	phy0: ethernet-phy@0 {
>  		reg = <0>;
> diff --git a/arch/arm/boot/dts/rk3xxx.dtsi b/arch/arm/boot/dts/rk3xxx.dtsi
> index 80ed3dc..66c8b85 100644
> --- a/arch/arm/boot/dts/rk3xxx.dtsi
> +++ b/arch/arm/boot/dts/rk3xxx.dtsi
> @@ -101,8 +101,11 @@
>  		#address-cells = <1>;
>  		#size-cells = <0>;
> 
> +		rockchip,grf = <&grf>;
> +
>  		clocks = <&cru HCLK_EMAC>;
>  		max-speed = <100>;
> +		phy-mode = "rmii";
> 
>  		status = "disabled";
>  	};

please don't include dts changes in driver patches ... they should be 
individual patches, as they're normally going through a separate tree.

Also in this case, you based your dts changes on my development-branch and 
there is currently no emac support at all in the mainline trees, so you should 
add the complete nodes.


> diff --git a/drivers/net/ethernet/arc/Kconfig
> b/drivers/net/ethernet/arc/Kconfig index d73d971..196e9e7 100644
> --- a/drivers/net/ethernet/arc/Kconfig
> +++ b/drivers/net/ethernet/arc/Kconfig
> @@ -28,4 +28,13 @@ config ARC_EMAC
>  	  non-standard on-chip ethernet device ARC EMAC 10/100 is used.
>  	  Say Y here if you have such a board.  If unsure, say N.
> 
> +config EMAC_ROCKCHIP
> +       tristate "Rockchip EMAC support"
> +       depends on MFD_SYSCON && ARCH_ROCKCHIP
> +       ---help---
> +         Support for Rockchip RK3066/RK3188 EMAC ethernet controllers.
> +         This selects Rockchip SoC glue layer support for the
> +         emac device driver. This driver is used for RK3066/RK3188
> +         EMAC ethernet controller.
> +
>  endif # NET_VENDOR_ARC
> diff --git a/drivers/net/ethernet/arc/Makefile
> b/drivers/net/ethernet/arc/Makefile index b6f15b4..cf10cf3 100644
> --- a/drivers/net/ethernet/arc/Makefile
> +++ b/drivers/net/ethernet/arc/Makefile
> @@ -4,3 +4,4 @@
> 
>  obj-$(CONFIG_NET_VENDOR_ARC) += emac_main.o emac_mdio.o
>  obj-$(CONFIG_ARC_EMAC) += emac_arc.o
> +obj-$(CONFIG_EMAC_ROCKCHIP) += emac_rockchip.o
> diff --git a/drivers/net/ethernet/arc/emac_rockchip.c
> b/drivers/net/ethernet/arc/emac_rockchip.c new file mode 100644
> index 0000000..2ea724f
> --- /dev/null
> +++ b/drivers/net/ethernet/arc/emac_rockchip.c
> @@ -0,0 +1,191 @@
> +/**
> + * emac-rockchip.c - Rockchip EMAC specific glue layer
> + *
> + * Copyright (C) 2014 Romain Perier
> + *
> + * Romain Perier  <romain.perier@...il.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 "emac.h"
> +#include <linux/of_net.h>
> +#include <linux/platform_device.h>
> +#include <linux/regulator/consumer.h>
> +#include <linux/mfd/syscon.h>
> +#include <linux/regmap.h>
> +#include <linux/module.h>
> +#include <linux/clk.h>
> +
> +#define DRV_NAME        "rockchip_emac"
> +#define DRV_VERSION     "1.0"
> +
> +#define GRF_MODE_MII   BIT(0)
> +#define GRF_MODE_RMII  0x0
> +#define GRF_SPEED_10M  0x0
> +#define GRF_SPEED_100M BIT(1)
> +
> +struct rockchip_priv_data {
> +	struct regmap *grf;
> +	unsigned int grf_offset;
> +	int interface;
> +	struct regulator *regulator;
> +};
> +
> +static const unsigned int rockchip_emac_soc_grf_offset[] = {
> +	0x154, /* rk3066 */
> +	0x0a4, /* rk3188 */
> +};
> +
> +static const struct of_device_id rockchip_emac_dt_ids[] = {
> +	{ .compatible = "rockchip,rk3066-emac", .data = (void
> *)rockchip_emac_soc_grf_offset }, +	{ .compatible = "rockchip,rk3188-emac",
> .data = (void *)rockchip_emac_soc_grf_offset + 1 }, +	{ /* Sentinel */ }
> +};
> +
> +MODULE_DEVICE_TABLE(of, rockchip_emac_dt_ids);
> +
> +static int rockchip_grf_set_phy_speed(const struct rockchip_priv_data
> *emac, unsigned int speed) +{
> +	/* write-enable bits */
> +	u32 data = BIT(17) | BIT(16);
> +
> +	switch(speed) {
> +	case 10:
> +		data |= GRF_SPEED_10M << 1;
> +		break;
> +	case 100:
> +		data |= GRF_SPEED_100M << 1;

I don't understand the "<< 1" here, the speed-bit in the register is already 
bit(1) as defined in the GRF_SPEED_100M at the top, so this code would move the 
bit to bit(2)


> +		break;
> +	default:
> +		return -ENOTSUPP;
> +	}
> +
> +	switch(emac->interface) {
> +	case PHY_INTERFACE_MODE_RMII:
> +		data |= GRF_MODE_RMII;
> +		break;
> +	case PHY_INTERFACE_MODE_MII:
> +		data |= GRF_MODE_MII;
> +		break;
> +	default:
> +		return -ENOTSUPP;
> +	}

setting the interface mode could be done only once in _probe, especially when 
you take into account the necessary clock settings described below.


> +
> +	return regmap_write(emac->grf, emac->grf_offset, data);
> +}
> +
> +static void rockchip_set_mac_speed(void *priv, unsigned int speed)
> +{
> +	struct rockchip_priv_data *emac = priv;
> +	int ret = 0;
> +
> +	ret = rockchip_grf_set_phy_speed(emac, speed);
> +	if (ret) {
> +		if (ret == -ENOTSUPP)
> +			pr_err("phy interface (%d) or speed (%u) not supported\n",
> emac->interface, speed); +		return ret;
> +	}
> +}
> +
> +static int rockchip_emac_probe(struct platform_device *pdev)
> +{
> +	struct rockchip_priv_data *emac = NULL;
> +	struct emac_platform_data *emac_plat_data = NULL;
> +	struct device *dev = &pdev->dev;
> +	const struct of_device_id *match = NULL;
> +	int ret = 0;
> +	u32 data = 0;
> +
> +	emac = dev_get_platdata(dev);

Rockchip support is devicetree-only, so you could simply error out in a 
theoretical non-dt case with:

if (!pdev->of_node)
	return -ENODEV;

Otherwise if you really want to get a previous defined platform-data struct, 
you should not overwrite it unconditionally below.


> +
> +	if (!emac) {
> +		emac = devm_kzalloc(dev, sizeof(*emac), GFP_KERNEL);
> +		if (!emac)
> +			return -ENOMEM;
> +	}
> +
> +	emac_plat_data = devm_kzalloc(dev, sizeof(*emac_plat_data), GFP_KERNEL);
> +	if (!emac_plat_data)
> +		return -ENOMEM;
> +	emac_plat_data->name = DRV_NAME;
> +	emac_plat_data->version = DRV_VERSION;
> +	emac_plat_data->set_mac_speed = rockchip_set_mac_speed;
> +	emac_plat_data->priv = emac;
> +
> +	emac->interface = of_get_phy_mode(dev->of_node);
> +	emac_plat_data->interface = emac->interface;
> +
> +	emac->grf = syscon_regmap_lookup_by_phandle(dev->of_node, "rockchip,grf");
> +	if (IS_ERR(emac->grf)) {
> +		dev_err(dev, "unable to find syscon grf (%ld)\n", PTR_ERR(emac->grf));
> +		return PTR_ERR_OR_ZERO(emac->grf);
> +	}
> +
> +	match = of_match_node(rockchip_emac_dt_ids, dev->of_node);
> +	emac->grf_offset = *(unsigned int *)match;
> +
> +	emac_plat_data->clk = of_clk_get(dev->of_node, 0);
> +	if (IS_ERR(emac_plat_data->clk)) {
> +		dev_err(dev, "failed to retrieve clock from device tree\n");
> +		return PTR_ERR_OR_ZERO(emac_plat_data->clk);
> +	}

The clock handling needs a bit more work.

For one, the rockchip emac needs two clocks (the hclk and the macref clock). 
Additionally the macref clk needs to be set to different values depending on 
the interface type.

In RMII mode it should be set to 50MHz and I guess in MII mode to 25MHz [0].

And secondly, you could use devm_clk_get because mixing devm_ and non-devm_ 
function doesn't work that well (and you're missing the clk_put here)

So combined:

emac_plat_data->clk = devm_clk_get(dev, "ahb_hclk");
if (IS_ERR(...))
	...

emac->refclk = devm_clk_get(dev, "macref");
...

	switch(emac->interface) {
	case PHY_INTERFACE_MODE_RMII:
		data = GRF_MODE_RMII;
		rate = 50000000;
		break;
	case PHY_INTERFACE_MODE_MII:
		data = GRF_MODE_MII;
		rate = 25000000;
		break;
	default:
		return -ENOTSUPP;
	}

	ret = regmap_write(emac->grf, emac->grf_offset, data);
	if (ret)
		foo

	ret = clk_set_rate(emac->refclk, rate);
	if (ret)
		foo


Heiko


[0] 
http://en.wikipedia.org/wiki/Media_Independent_Interface#Reduced_Media_Independent_Interface

> +
> +        /* Optional regulator for PHY */
> +	emac->regulator = devm_regulator_get_optional(dev, "phy");
> +	if (IS_ERR(emac->regulator)) {
> +		if (PTR_ERR(emac->regulator) == -EPROBE_DEFER)
> +			return ERR_PTR(-EPROBE_DEFER);
> +		dev_info(dev, "no regulator found\n");
> +		emac->regulator = NULL;
> +	}
> +
> +	if (emac->regulator) {
> +		ret = regulator_enable(emac->regulator);
> +		if (ret)
> +			return ret;
> +	}
> +
> +	ret = rockchip_grf_set_phy_speed(emac, 100);
> +	if (ret) {
> +		if (ret == -ENOTSUPP)
> +			dev_err(dev, "phy interface not supported (%d)\n", emac-
>interface);
> +		return ret;
> +	}
> +
> +	return emac_drv_probe(dev, emac_plat_data);
> +}
> +
> +static int rockchip_emac_remove(struct platform_device *pdev)
> +{
> +	struct rockchip_priv_data *emac = dev_get_platdata(&pdev->dev);
> +	struct net_device *ndev = dev_get_drvdata(&pdev->dev);
> +
> +	if (emac->regulator)
> +		regulator_disable(emac->regulator);
> +	return emac_drv_remove(ndev);
> +}
> +
> +static struct platform_driver rockchip_emac_driver = {
> +	.probe = rockchip_emac_probe,
> +	.remove = rockchip_emac_remove,
> +	.driver = {
> +		.name = DRV_NAME,
> +		.owner = THIS_MODULE,
> +		.of_match_table  = rockchip_emac_dt_ids,
> +	},
> +};
> +
> +module_platform_driver(rockchip_emac_driver);
> +
> +MODULE_AUTHOR("Romain Perier <romain.perier@...il.com>");
> +MODULE_DESCRIPTION("Rockchip EMAC driver");
> +MODULE_LICENSE("GPL");

--
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ