On Thu, Dec 17, 2020 at 08:51:28AM +0100, Steen Hegelund wrote:

> +static struct sparx5_io_resource sparx5_iomap[] =  {

This could be made const i think,.

> +     { TARGET_DEV2G5,         0,         0 }, /* 0x610004000: dev2g5_0 */
> +     { TARGET_DEV5G,          0x4000,    0 }, /* 0x610008000: dev5g_0 */
> +     { TARGET_PCS5G_BR,       0x8000,    0 }, /* 0x61000c000: pcs5g_br_0 */
> +     { TARGET_DEV2G5 + 1,     0xc000,    0 }, /* 0x610010000: dev2g5_1 */

> +static int sparx5_create_targets(struct sparx5 *sparx5)
> +{
> +     int idx, jdx;
> +     struct resource *iores[IO_RANGES];
> +     void __iomem *iomem[IO_RANGES];
> +     void __iomem *begin[IO_RANGES];
> +     int range_id[IO_RANGES];

Reverse Christmas tree. idx, jdx need to come last.

> +
> +     /* Check if done previously (deferred by serdes load) */
> +     if (sparx5->regs[sparx5_iomap[0].id])
> +             return 0;

Could you explain this a bit more. Do you mean -EPROBE_DEFER?

> +static int sparx5_probe_port(struct sparx5 *sparx5,
> +                          struct device_node *portnp,
> +                          struct phy *serdes,
> +                          u32 portno,
> +                          struct sparx5_port_config *conf)
> +{
> +     struct sparx5_port *spx5_port;
> +     struct net_device *ndev;
> +     int err;
> +
> +     err = sparx5_create_targets(sparx5);
> +     if (err)
> +             return err;

This sees odd here. Don't sparx5_create_targets() create all the
targets, where as this creates one specific port? Seems like
sparx5_create_targets() should be in the devices as a whole probe, not
the port probe.

> +     spx5_port = netdev_priv(ndev);
> +     spx5_port->of_node = portnp;
> +     spx5_port->serdes = serdes;
> +     spx5_port->pvid = NULL_VID;
> +     spx5_port->signd_internal = true;
> +     spx5_port->signd_active_high = true;
> +     spx5_port->signd_enable = true;
> +     spx5_port->flow_control = false;
> +     spx5_port->max_vlan_tags = SPX5_PORT_MAX_TAGS_NONE;
> +     spx5_port->vlan_type = SPX5_VLAN_PORT_TYPE_UNAWARE;
> +     spx5_port->custom_etype = 0x8880; /* Vitesse */
> +     conf->portmode = conf->phy_mode;
> +     spx5_port->conf.speed = SPEED_UNKNOWN;
> +     spx5_port->conf.power_down = true;
> +     sparx5->ports[portno] = spx5_port;
> +     return 0;

I'm also not sure this has the correct name. This does not look like a
typical probe function.


> +}
> +
> +static int sparx5_init_switchcore(struct sparx5 *sparx5)
> +{
> +     u32 value, pending, jdx, idx;
> +     struct {
> +             bool gazwrap;
> +             void __iomem *init_reg;
> +             u32  init_val;
> +     } ram, ram_init_list[] = {
> +             {false, spx5_reg_get(sparx5, ANA_AC_STAT_RESET),
> +              ANA_AC_STAT_RESET_RESET},
> +             {false, spx5_reg_get(sparx5, ASM_STAT_CFG),
> +              ASM_STAT_CFG_STAT_CNT_CLR_SHOT},
> +             {true,  spx5_reg_get(sparx5, QSYS_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, REW_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, VOP_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, ANA_AC_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, ASM_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, EACL_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, VCAP_SUPER_RAM_INIT), 0},
> +             {true,  spx5_reg_get(sparx5, DSM_RAM_INIT), 0}
> +     };

Looks like this could be const as well. And this does not really fit
reverse christmas tree.

> +
> +     spx5_rmw(EACL_POL_EACL_CFG_EACL_FORCE_INIT_SET(1),
> +              EACL_POL_EACL_CFG_EACL_FORCE_INIT,
> +              sparx5,
> +              EACL_POL_EACL_CFG);
> +
> +     spx5_rmw(EACL_POL_EACL_CFG_EACL_FORCE_INIT_SET(0),
> +              EACL_POL_EACL_CFG_EACL_FORCE_INIT,
> +              sparx5,
> +              EACL_POL_EACL_CFG);
> +
> +     /* Initialize memories, if not done already */
> +     value = spx5_rd(sparx5, HSCH_RESET_CFG);
> +
> +     if (!(value & HSCH_RESET_CFG_CORE_ENA)) {
> +             for (idx = 0; idx < 10; idx++) {
> +                     pending = ARRAY_SIZE(ram_init_list);
> +                     for (jdx = 0; jdx < ARRAY_SIZE(ram_init_list); jdx++) {
> +                             ram = ram_init_list[jdx];
> +                             if (ram.gazwrap)
> +                                     ram.init_val = QSYS_RAM_INIT_RAM_INIT;
> +
> +                             if (idx == 0) {
> +                                     writel(ram.init_val, ram.init_reg);
> +                             } else {
> +                                     value = readl(ram.init_reg);
> +                                     if ((value & ram.init_val) !=
> +                                         ram.init_val) {
> +                                             pending--;
> +                                     }
> +                             }
> +                     }
> +                     if (!pending)
> +                             break;
> +                     usleep_range(USEC_PER_MSEC, 2 * USEC_PER_MSEC);
> +             }

You are getting pretty deeply nested here. Might be better to pull
this out into a helpers.

> +
> +             if (pending > 0) {
> +                     /* Still initializing, should be complete in
> +                      * less than 1ms
> +                      */
> +                     dev_err(sparx5->dev, "Memory initialization error\n");
> +                     return -EINVAL;
> +             }
> +     }
> +
> +     /* Reset counters */
> +     spx5_wr(ANA_AC_STAT_RESET_RESET_SET(1), sparx5, ANA_AC_STAT_RESET);
> +     spx5_wr(ASM_STAT_CFG_STAT_CNT_CLR_SHOT_SET(1), sparx5, ASM_STAT_CFG);
> +
> +     /* Enable switch-core and queue system */
> +     spx5_wr(HSCH_RESET_CFG_CORE_ENA_SET(1), sparx5, HSCH_RESET_CFG);
> +
> +     return 0;
> +}
> +
> +static int sparx5_init_coreclock(struct sparx5 *sparx5)
> +{
> +     u32 clk_div, clk_period, pol_upd_int, idx;
> +     enum sparx5_core_clockfreq freq = sparx5->coreclock;

More reverse christmas tree. Please review the whole driver.

> +
> +     /* Verify if core clock frequency is supported on target.
> +      * If 'VTSS_CORE_CLOCK_DEFAULT' then the highest supported
> +      * freq. is used
> +      */
> +     switch (sparx5->target_ct) {
> +     case SPX5_TARGET_CT_7546:
> +             if (sparx5->coreclock == SPX5_CORE_CLOCK_DEFAULT)
> +                     freq = SPX5_CORE_CLOCK_250MHZ;
> +             else if (sparx5->coreclock != SPX5_CORE_CLOCK_250MHZ)
> +                     freq = 0; /* Not supported */
> +             break;
> +     case SPX5_TARGET_CT_7549:
> +     case SPX5_TARGET_CT_7552:
> +     case SPX5_TARGET_CT_7556:
> +             if (sparx5->coreclock == SPX5_CORE_CLOCK_DEFAULT)
> +                     freq = SPX5_CORE_CLOCK_500MHZ;
> +             else if (sparx5->coreclock != SPX5_CORE_CLOCK_500MHZ)
> +                     freq = 0; /* Not supported */
> +             break;
> +     case SPX5_TARGET_CT_7558:
> +     case SPX5_TARGET_CT_7558TSN:
> +             if (sparx5->coreclock == SPX5_CORE_CLOCK_DEFAULT)
> +                     freq = SPX5_CORE_CLOCK_625MHZ;
> +             else if (sparx5->coreclock != SPX5_CORE_CLOCK_625MHZ)
> +                     freq = 0; /* Not supported */
> +             break;
> +     case SPX5_TARGET_CT_7546TSN:
> +             if (sparx5->coreclock == SPX5_CORE_CLOCK_DEFAULT)
> +                     freq = SPX5_CORE_CLOCK_625MHZ;
> +             break;
> +     case SPX5_TARGET_CT_7549TSN:
> +     case SPX5_TARGET_CT_7552TSN:
> +     case SPX5_TARGET_CT_7556TSN:
> +             if (sparx5->coreclock == SPX5_CORE_CLOCK_DEFAULT)
> +                     freq = SPX5_CORE_CLOCK_625MHZ;
> +             else if (sparx5->coreclock == SPX5_CORE_CLOCK_250MHZ)
> +                     freq = 0; /* Not supported */
> +             break;
> +     default:
> +             dev_err(sparx5->dev, "Target (%#04x) not supported\n", 
> sparx5->target_ct);

netdev is staying with 80 character lines. Please fold this, here and
every where else, where possible. The exception is, you should not
split a string.

> +             return -ENODEV;
> +     }
> +
> +     switch (freq) {
> +     case SPX5_CORE_CLOCK_250MHZ:
> +             clk_div = 10;
> +             pol_upd_int = 312;
> +             break;
> +     case SPX5_CORE_CLOCK_500MHZ:
> +             clk_div = 5;
> +             pol_upd_int = 624;
> +             break;
> +     case SPX5_CORE_CLOCK_625MHZ:
> +             clk_div = 4;
> +             pol_upd_int = 780;
> +             break;
> +     default:
> +             dev_err(sparx5->dev, "%s: Frequency (%d) not supported on 
> target (%#04x)\n",
> +                     __func__,
> +                     sparx5->coreclock, sparx5->target_ct);
> +             return 0;

-EINVAL? Or is it not fatal to use an unsupported frequency?

> +static int sparx5_init(struct sparx5 *sparx5)
> +{
> +     u32 idx;
> +
> +     if (sparx5_create_targets(sparx5))
> +             return -ENODEV;

Hum, sparx5_create_targets() again?

> +
> +     /* Read chip ID to check CPU interface */
> +     sparx5->chip_id = spx5_rd(sparx5, GCB_CHIP_ID);
> +
> +     sparx5->target_ct = (enum spx5_target_chiptype)
> +             GCB_CHIP_ID_PART_ID_GET(sparx5->chip_id);
> +
> +     /* Initialize Switchcore and internal RAMs */
> +     if (sparx5_init_switchcore(sparx5)) {
> +             dev_err(sparx5->dev, "Switchcore initialization error\n");
> +             return -EINVAL;
> +     }
> +
> +     /* Initialize the LC-PLL (core clock) and set affected registers */
> +     if (sparx5_init_coreclock(sparx5)) {
> +             dev_err(sparx5->dev, "LC-PLL initialization error\n");
> +             return -EINVAL;
> +     }
> +
> +     /* Setup own UPSIDs */
> +     for (idx = 0; idx < 3; idx++) {
> +             spx5_wr(idx, sparx5, ANA_AC_OWN_UPSID(idx));
> +             spx5_wr(idx, sparx5, ANA_CL_OWN_UPSID(idx));
> +             spx5_wr(idx, sparx5, ANA_L2_OWN_UPSID(idx));
> +             spx5_wr(idx, sparx5, REW_OWN_UPSID(idx));
> +     }
> +
> +     /* Enable switch ports */
> +     for (idx = SPX5_PORTS; idx < SPX5_PORTS_ALL; idx++) {
> +             spx5_rmw(QFWD_SWITCH_PORT_MODE_PORT_ENA_SET(1),
> +                      QFWD_SWITCH_PORT_MODE_PORT_ENA,
> +                      sparx5,
> +                      QFWD_SWITCH_PORT_MODE(idx));
> +     }

What happens when you enable the ports? Why is this here, and not in
the port specific open call?

> +/* Some boards needs to map the SGPIO for signal detect explicitly to the
> + * port module
> + */
> +static void sparx5_board_init(struct sparx5 *sparx5)
> +{
> +     int idx;
> +
> +     if (!sparx5->sd_sgpio_remapping)
> +             return;
> +
> +     /* Enable SGPIO Signal Detect remapping */
> +     spx5_rmw(GCB_HW_SGPIO_SD_CFG_SD_MAP_SEL,
> +              GCB_HW_SGPIO_SD_CFG_SD_MAP_SEL,
> +              sparx5,
> +              GCB_HW_SGPIO_SD_CFG);
> +
> +     /* Refer to LOS SGPIO */
> +     for (idx = 0; idx < SPX5_PORTS; idx++) {
> +             if (sparx5->ports[idx]) {
> +                     if (sparx5->ports[idx]->conf.sd_sgpio != ~0) {
> +                             spx5_wr(sparx5->ports[idx]->conf.sd_sgpio,
> +                                     sparx5,
> +                                     GCB_HW_SGPIO_TO_SD_MAP_CFG(idx));
> +                     }
> +             }
> +     }
> +}

I've not looked at how you do SFP integration yet. Is this the LOS
from the SFP socket? Is there a Linux GPIO controller exported by this
driver, so the SFP driver can use the GPIOs?

> +
> +static int mchp_sparx5_probe(struct platform_device *pdev)
> +{
> +     struct device_node *np = pdev->dev.of_node;
> +     struct sparx5 *sparx5;
> +     struct device_node *ports, *portnp;
> +     const u8 *mac_addr;
> +     int err = 0;
> +
> +     if (!np && !pdev->dev.platform_data)
> +             return -ENODEV;
> +
> +     sparx5 = devm_kzalloc(&pdev->dev, sizeof(*sparx5), GFP_KERNEL);
> +     if (!sparx5)
> +             return -ENOMEM;
> +
> +     platform_set_drvdata(pdev, sparx5);
> +     sparx5->pdev = pdev;
> +     sparx5->dev = &pdev->dev;
> +
> +     /* Default values, some from DT */
> +     sparx5->coreclock = SPX5_CORE_CLOCK_DEFAULT;
> +
> +     mac_addr = of_get_mac_address(np);
> +     if (IS_ERR_OR_NULL(mac_addr)) {
> +             dev_info(sparx5->dev, "MAC addr was not set, use random MAC\n");
> +             eth_random_addr(sparx5->base_mac);
> +             sparx5->base_mac[5] = 0;
> +     } else {
> +             ether_addr_copy(sparx5->base_mac, mac_addr);
> +     }

The binding document does not say anything about a MAC address at the
top level. What is this used for?

+
> +     if (sparx5_init(sparx5)) {
> +             dev_err(sparx5->dev, "Init failed\n");
> +             return -ENODEV;
> +     }
> +     ports = of_get_child_by_name(np, "ethernet-ports");
> +     if (!ports) {
> +             dev_err(sparx5->dev, "no ethernet-ports child node found\n");
> +             return -ENODEV;
> +     }
> +     sparx5->port_count = of_get_child_count(ports);
> +
> +     for_each_available_child_of_node(ports, portnp) {
> +             struct sparx5_port_config config = {};
> +             u32 portno;
> +             struct phy *serdes;
> +
> +             err = of_property_read_u32(portnp, "reg", &portno);
> +             if (err) {
> +                     dev_err(sparx5->dev, "port reg property error\n");
> +                     continue;
> +             }
> +             err = of_property_read_u32(portnp, "max-speed",
> +                                        &config.max_speed);
> +             if (err) {
> +                     dev_err(sparx5->dev, "port max-speed property error\n");
> +                     continue;
> +             }
> +             config.speed = SPEED_UNKNOWN;
> +             err = of_property_read_u32(portnp, "sd_sgpio", 
> &config.sd_sgpio);

Not in the binding documentation. I think i need to withdraw my Reviewed-by :-(

> +             if (err)
> +                     config.sd_sgpio = ~0;
> +             else
> +                     sparx5->sd_sgpio_remapping = true;
> +             serdes = devm_of_phy_get(sparx5->dev, portnp, NULL);
> +             if (IS_ERR(serdes)) {
> +                     err = PTR_ERR(serdes);
> +                     if (err != -EPROBE_DEFER)
> +                             dev_err(sparx5->dev,
> +                                     "missing SerDes phys for port%d\n",
> +                                     portno);
> +                     return err;
> +             }
> +
> +             err = of_get_phy_mode(portnp, &config.phy_mode);
> +             if (err)
> +                     config.power_down = true;

You should indicate in the binding it is optional. And what happens
when it is missing.

> +             config.media_type = ETH_MEDIA_DAC;
> +             config.serdes_reset = true;
> +             config.portmode = config.phy_mode;
> +             err = sparx5_probe_port(sparx5, portnp, serdes, portno, 
> &config);
> +             if (err) {
> +                     dev_err(sparx5->dev, "port probe error\n");
> +                     goto cleanup_ports;
> +             }
> +     }
> +     sparx5_board_init(sparx5);
> +
> +cleanup_ports:
> +     return err;

Seems missed named, no cleanup.

> +static int __init sparx5_switch_reset(void)
> +{
> +     const char *syscon_cpu = "microchip,sparx5-cpu-syscon",
> +             *syscon_gcb = "microchip,sparx5-gcb-syscon";
> +     struct regmap *cpu_ctrl, *gcb_ctrl;
> +     u32 val;
> +
> +     cpu_ctrl = syscon_regmap_lookup_by_compatible(syscon_cpu);
> +     if (IS_ERR(cpu_ctrl)) {
> +             pr_err("No '%s' syscon map\n", syscon_cpu);
> +             return PTR_ERR(cpu_ctrl);
> +     }
> +
> +     gcb_ctrl = syscon_regmap_lookup_by_compatible(syscon_gcb);
> +     if (IS_ERR(gcb_ctrl)) {
> +             pr_err("No '%s' syscon map\n", syscon_gcb);
> +             return PTR_ERR(gcb_ctrl);
> +     }
> +
> +     /* Make sure the core is PROTECTED from reset */
> +     regmap_update_bits(cpu_ctrl, RESET_PROT_STAT,
> +                        SYS_RST_PROT_VCORE, SYS_RST_PROT_VCORE);
> +
> +     regmap_write(gcb_ctrl, spx5_offset(GCB_SOFT_RST),
> +                  GCB_SOFT_RST_SOFT_SWC_RST_SET(1));
> +
> +     return readx_poll_timeout(sparx5_read_gcb_soft_rst, gcb_ctrl, val,
> +                               GCB_SOFT_RST_SOFT_SWC_RST_GET(val) == 0,
> +                               1, 100);
> +}
> +postcore_initcall(sparx5_switch_reset);

That is pretty unusual. Why cannot this be done at probe time?

> +/* Clock period in picoseconds */
> +static inline u32 sparx5_clk_period(enum sparx5_core_clockfreq cclock)
> +{
> +     switch (cclock) {
> +     case SPX5_CORE_CLOCK_250MHZ:
> +             return 4000;
> +     case SPX5_CORE_CLOCK_500MHZ:
> +             return 2000;
> +     case SPX5_CORE_CLOCK_625MHZ:
> +     default:
> +             return 1600;
> +     }
> +}

Is this something which is used in the hot path?

> --- /dev/null
> +++ b/drivers/net/ethernet/microchip/sparx5/sparx5_main_regs.h
> @@ -0,0 +1,3922 @@
> +/* SPDX-License-Identifier: GPL-2.0+
> + * Microchip Sparx5 Switch driver
> + *
> + * Copyright (c) 2020 Microchip Technology Inc.
> + */
> +
> +/* This file is autogenerated by cml-utils 2020-11-19 10:41:34 +0100.
> + * Commit ID: f34790e69dc252103e2cc3e85b1a5e4d9e3aa190
> + */

How reproducible this is generation process? If you have to run it
again, will it keep the same order of lines?

       Andrew 

Reply via email to