Hi Andrew,

On Fri,  4 Oct 2019 23:09:34 +0200, Andrew Lunn <and...@lunn.ch> wrote:
> Some of the marvell switches have bits controlling the hash algorithm
> the ATU uses for MAC addresses. In some industrial settings, where all
> the devices are from the same manufacture, and hence use the same OUI,
> the default hashing algorithm is not optimal. Allow the other
> algorithms to be selected via devlink.
> 
> Signed-off-by: Andrew Lunn <and...@lunn.ch>
> ---
>  .../networking/devlink-params-mv88e6xxx.txt   |   6 +
>  MAINTAINERS                                   |   1 +
>  drivers/net/dsa/mv88e6xxx/chip.c              | 134 +++++++++++++++++-
>  drivers/net/dsa/mv88e6xxx/chip.h              |   4 +
>  drivers/net/dsa/mv88e6xxx/global1.h           |   3 +
>  drivers/net/dsa/mv88e6xxx/global1_atu.c       |  32 +++++
>  6 files changed, 179 insertions(+), 1 deletion(-)
>  create mode 100644 Documentation/networking/devlink-params-mv88e6xxx.txt
> 
> diff --git a/Documentation/networking/devlink-params-mv88e6xxx.txt 
> b/Documentation/networking/devlink-params-mv88e6xxx.txt
> new file mode 100644
> index 000000000000..cc5c1ac87c36
> --- /dev/null
> +++ b/Documentation/networking/devlink-params-mv88e6xxx.txt
> @@ -0,0 +1,6 @@
> +ATU_hash             [DEVICE, DRIVER-SPECIFIC]
> +                     Select one of four possible hashing algorithms for
> +                     MAC addresses in the Address Translation Unity.
> +                     A value of 3 seems to work better than the default of
> +                     1 when many MAC addresses have the same OUI.
> +                     Configuration mode: runtime
> diff --git a/MAINTAINERS b/MAINTAINERS
> index 496e8f156925..2246dc121c30 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -9736,6 +9736,7 @@ S:      Maintained
>  F:   drivers/net/dsa/mv88e6xxx/
>  F:   include/linux/platform_data/mv88e6xxx.h
>  F:   Documentation/devicetree/bindings/net/dsa/marvell.txt
> +F:   Documentation/networking/devlink-params-mv88e6xxx.txt
>  
>  MARVELL ARMADA DRM SUPPORT
>  M:   Russell King <li...@armlinux.org.uk>
> diff --git a/drivers/net/dsa/mv88e6xxx/chip.c 
> b/drivers/net/dsa/mv88e6xxx/chip.c
> index 6787d560e9e3..c9755a4285a9 100644
> --- a/drivers/net/dsa/mv88e6xxx/chip.c
> +++ b/drivers/net/dsa/mv88e6xxx/chip.c
> @@ -1370,6 +1370,22 @@ static int mv88e6xxx_atu_new(struct mv88e6xxx_chip 
> *chip, u16 *fid)
>       return mv88e6xxx_g1_atu_flush(chip, *fid, true);
>  }
>  
> +static int mv88e6xxx_atu_get_hash(struct mv88e6xxx_chip *chip, u8 *hash)
> +{
> +     if (chip->info->ops->atu_get_hash)
> +             return chip->info->ops->atu_get_hash(chip, hash);
> +
> +     return -EOPNOTSUPP;
> +}
> +
> +static int mv88e6xxx_atu_set_hash(struct mv88e6xxx_chip *chip, u8 hash)
> +{
> +     if (chip->info->ops->atu_set_hash)
> +             return chip->info->ops->atu_set_hash(chip, hash);
> +
> +     return -EOPNOTSUPP;
> +}
> +
>  static int mv88e6xxx_port_check_hw_vlan(struct dsa_switch *ds, int port,
>                                       u16 vid_begin, u16 vid_end)
>  {
> @@ -2641,6 +2657,81 @@ static int mv88e6390_setup_errata(struct 
> mv88e6xxx_chip *chip)
>       return mv88e6xxx_software_reset(chip);
>  }
>  
> +enum mv88e6xxx_devlink_param_id {
> +     MV88E6XXX_DEVLINK_PARAM_ID_BASE = DEVLINK_PARAM_GENERIC_ID_MAX,
> +     MV88E6XXX_DEVLINK_PARAM_ID_ATU_HASH,
> +};
> +
> +static int mv88e6xxx_devlink_param_get(struct dsa_switch *ds, u32 id,
> +                                    struct devlink_param_gset_ctx *ctx)
> +{
> +     struct mv88e6xxx_chip *chip = ds->priv;
> +     int err;
> +     u8 hash;
> +
> +     mv88e6xxx_reg_lock(chip);
> +
> +     switch (id) {
> +     case MV88E6XXX_DEVLINK_PARAM_ID_ATU_HASH:
> +             err = mv88e6xxx_atu_get_hash(chip, &hash);
> +             if (err < 0)
> +                     break;

No need to check for error signedness, just do if (err).

> +
> +             ctx->val.vu8 = hash;

Is ctx->val.vu8 an u8 as well? If so you can just write:

                err = mv88e6xxx_atu_get_hash(chip, &ctx->val.vu8);
> +             break;
> +     default:
> +             err = -EOPNOTSUPP;

Missing a break statement here.

> +     }
> +
> +     mv88e6xxx_reg_unlock(chip);
> +
> +     return err;
> +}
> +
> +static int mv88e6xxx_devlink_param_set(struct dsa_switch *ds, u32 id,
> +                                    struct devlink_param_gset_ctx *ctx)
> +{
> +     struct mv88e6xxx_chip *chip = ds->priv;
> +     int err;
> +
> +     mv88e6xxx_reg_lock(chip);
> +
> +     switch (id) {
> +     case MV88E6XXX_DEVLINK_PARAM_ID_ATU_HASH:
> +             err = mv88e6xxx_atu_set_hash(chip, ctx->val.vu8);
> +             break;
> +     default:
> +             err = -EOPNOTSUPP;

Missing a break statement here too.

> +     }
> +
> +     mv88e6xxx_reg_unlock(chip);
> +
> +     return err;
> +}
> +
> +static const struct devlink_param mv88e6xxx_devlink_params[] = {
> +     DSA_DEVLINK_PARAM_DRIVER(MV88E6XXX_DEVLINK_PARAM_ID_ATU_HASH,
> +                              "ATU_hash", DEVLINK_PARAM_TYPE_U8,
> +                              BIT(DEVLINK_PARAM_CMODE_RUNTIME)),
> +};
> +
> +static int mv88e6xxx_setup_devlink_params(struct dsa_switch *ds)
> +{
> +     return dsa_devlink_params_register(ds, mv88e6xxx_devlink_params,
> +                                        
> ARRAY_SIZE(mv88e6xxx_devlink_params));
> +}
> +
> +static void mv88e6xxx_teardown_devlink_params(struct dsa_switch *ds)
> +{
> +     dsa_devlink_params_unregister(ds, mv88e6xxx_devlink_params,
> +                                   ARRAY_SIZE(mv88e6xxx_devlink_params));
> +}
> +
> +static void mv88e6xxx_teardown(struct dsa_switch *ds)
> +{
> +     mv88e6xxx_teardown_devlink_params(ds);
> +}
> +
>  static int mv88e6xxx_setup(struct dsa_switch *ds)
>  {
>       struct mv88e6xxx_chip *chip = ds->priv;
> @@ -2757,7 +2848,11 @@ static int mv88e6xxx_setup(struct dsa_switch *ds)
>  unlock:
>       mv88e6xxx_reg_unlock(chip);
>  
> -     return err;
> +     /* Has to be called without holding the register lock, since
> +      * it takes the devlink lock, and we later take the locks in
> +      * the reverse order when getting/setting parameters.
> +      */
> +     return mv88e6xxx_setup_devlink_params(ds);
>  }
>  
>  static int mv88e6xxx_mdio_read(struct mii_bus *bus, int phy, int reg)
> @@ -3117,6 +3212,8 @@ static const struct mv88e6xxx_ops mv88e6123_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .phylink_validate = mv88e6185_phylink_validate,
> @@ -3246,6 +3343,8 @@ static const struct mv88e6xxx_ops mv88e6161_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .avb_ops = &mv88e6165_avb_ops,
> @@ -3280,6 +3379,8 @@ static const struct mv88e6xxx_ops mv88e6165_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .avb_ops = &mv88e6165_avb_ops,
> @@ -3322,6 +3423,8 @@ static const struct mv88e6xxx_ops mv88e6171_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .phylink_validate = mv88e6185_phylink_validate,
> @@ -3366,6 +3469,8 @@ static const struct mv88e6xxx_ops mv88e6172_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6352_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .serdes_get_lane = mv88e6352_serdes_get_lane,
> @@ -3409,6 +3514,8 @@ static const struct mv88e6xxx_ops mv88e6175_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .phylink_validate = mv88e6185_phylink_validate,
> @@ -3453,6 +3560,8 @@ static const struct mv88e6xxx_ops mv88e6176_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6352_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .serdes_get_lane = mv88e6352_serdes_get_lane,
> @@ -3538,6 +3647,8 @@ static const struct mv88e6xxx_ops mv88e6190_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6390_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6390_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
>       .serdes_power = mv88e6390_serdes_power,
> @@ -3587,6 +3698,8 @@ static const struct mv88e6xxx_ops mv88e6190x_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6390_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6390_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
>       .serdes_power = mv88e6390_serdes_power,
> @@ -3635,6 +3748,8 @@ static const struct mv88e6xxx_ops mv88e6191_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6390_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6390_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
>       .serdes_power = mv88e6390_serdes_power,
> @@ -3686,6 +3801,8 @@ static const struct mv88e6xxx_ops mv88e6240_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6352_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .serdes_get_lane = mv88e6352_serdes_get_lane,
> @@ -3777,6 +3894,8 @@ static const struct mv88e6xxx_ops mv88e6290_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6390_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6390_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
>       .serdes_power = mv88e6390_serdes_power,
> @@ -3963,6 +4082,8 @@ static const struct mv88e6xxx_ops mv88e6350_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .phylink_validate = mv88e6185_phylink_validate,
> @@ -4003,6 +4124,8 @@ static const struct mv88e6xxx_ops mv88e6351_ops = {
>       .mgmt_rsvd2cpu = mv88e6352_g2_mgmt_rsvd2cpu,
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .avb_ops = &mv88e6352_avb_ops,
> @@ -4049,6 +4172,8 @@ static const struct mv88e6xxx_ops mv88e6352_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6352_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6352_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
>       .serdes_get_lane = mv88e6352_serdes_get_lane,
> @@ -4105,6 +4230,8 @@ static const struct mv88e6xxx_ops mv88e6390_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6390_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6390_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
>       .serdes_power = mv88e6390_serdes_power,
> @@ -4158,6 +4285,8 @@ static const struct mv88e6xxx_ops mv88e6390x_ops = {
>       .pot_clear = mv88e6xxx_g2_pot_clear,
>       .reset = mv88e6352_g1_reset,
>       .rmu_disable = mv88e6390_g1_rmu_disable,
> +     .atu_get_hash = mv88e6165_g1_atu_get_hash,
> +     .atu_set_hash = mv88e6165_g1_atu_set_hash,
>       .vtu_getnext = mv88e6390_g1_vtu_getnext,
>       .vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
>       .serdes_power = mv88e6390_serdes_power,
> @@ -4933,6 +5062,7 @@ static int mv88e6xxx_port_egress_floods(struct 
> dsa_switch *ds, int port,
>  static const struct dsa_switch_ops mv88e6xxx_switch_ops = {
>       .get_tag_protocol       = mv88e6xxx_get_tag_protocol,
>       .setup                  = mv88e6xxx_setup,
> +     .teardown               = mv88e6xxx_teardown,
>       .phylink_validate       = mv88e6xxx_validate,
>       .phylink_mac_link_state = mv88e6xxx_link_state,
>       .phylink_mac_config     = mv88e6xxx_mac_config,
> @@ -4975,6 +5105,8 @@ static const struct dsa_switch_ops mv88e6xxx_switch_ops 
> = {
>       .port_txtstamp          = mv88e6xxx_port_txtstamp,
>       .port_rxtstamp          = mv88e6xxx_port_rxtstamp,
>       .get_ts_info            = mv88e6xxx_get_ts_info,
> +     .devlink_param_get      = mv88e6xxx_devlink_param_get,
> +     .devlink_param_set      = mv88e6xxx_devlink_param_set,
>  };
>  
>  static int mv88e6xxx_register_switch(struct mv88e6xxx_chip *chip)
> diff --git a/drivers/net/dsa/mv88e6xxx/chip.h 
> b/drivers/net/dsa/mv88e6xxx/chip.h
> index e9b1a1ac9a8e..52f7726cc099 100644
> --- a/drivers/net/dsa/mv88e6xxx/chip.h
> +++ b/drivers/net/dsa/mv88e6xxx/chip.h
> @@ -497,6 +497,10 @@ struct mv88e6xxx_ops {
>       int (*serdes_get_stats)(struct mv88e6xxx_chip *chip,  int port,
>                               uint64_t *data);
>  
> +     /* Address Translation Unit operations */
> +     int (*atu_get_hash)(struct mv88e6xxx_chip *chip, u8 *hash);
> +     int (*atu_set_hash)(struct mv88e6xxx_chip *chip, u8 hash);
> +
>       /* VLAN Translation Unit operations */
>       int (*vtu_getnext)(struct mv88e6xxx_chip *chip,
>                          struct mv88e6xxx_vtu_entry *entry);
> diff --git a/drivers/net/dsa/mv88e6xxx/global1.h 
> b/drivers/net/dsa/mv88e6xxx/global1.h
> index 0870fcc8bfc8..9f2af711293b 100644
> --- a/drivers/net/dsa/mv88e6xxx/global1.h
> +++ b/drivers/net/dsa/mv88e6xxx/global1.h
> @@ -109,6 +109,7 @@
>  /* Offset 0x0A: ATU Control Register */
>  #define MV88E6XXX_G1_ATU_CTL         0x0a
>  #define MV88E6XXX_G1_ATU_CTL_LEARN2ALL       0x0008
> +#define MV88E6161_G1_ATU_CTL_HASH_MASK       0x3
>  
>  /* Offset 0x0B: ATU Operation Register */
>  #define MV88E6XXX_G1_ATU_OP                          0x0b
> @@ -318,6 +319,8 @@ int mv88e6xxx_g1_atu_remove(struct mv88e6xxx_chip *chip, 
> u16 fid, int port,
>                           bool all);
>  int mv88e6xxx_g1_atu_prob_irq_setup(struct mv88e6xxx_chip *chip);
>  void mv88e6xxx_g1_atu_prob_irq_free(struct mv88e6xxx_chip *chip);
> +int mv88e6165_g1_atu_get_hash(struct mv88e6xxx_chip *chip, u8 *hash);
> +int mv88e6165_g1_atu_set_hash(struct mv88e6xxx_chip *chip, u8 hash);
>  
>  int mv88e6185_g1_vtu_getnext(struct mv88e6xxx_chip *chip,
>                            struct mv88e6xxx_vtu_entry *entry);
> diff --git a/drivers/net/dsa/mv88e6xxx/global1_atu.c 
> b/drivers/net/dsa/mv88e6xxx/global1_atu.c
> index 792a96ef418f..d8a03bbba83c 100644
> --- a/drivers/net/dsa/mv88e6xxx/global1_atu.c
> +++ b/drivers/net/dsa/mv88e6xxx/global1_atu.c
> @@ -73,6 +73,38 @@ int mv88e6xxx_g1_atu_set_age_time(struct mv88e6xxx_chip 
> *chip,
>       return 0;
>  }
>  
> +int mv88e6165_g1_atu_get_hash(struct mv88e6xxx_chip *chip, u8 *hash)
> +{
> +     int err;
> +     u16 val;
> +
> +     err = mv88e6xxx_g1_read(chip, MV88E6XXX_G1_ATU_CTL, &val);
> +     if (err)
> +             return err;
> +
> +     *hash = val & MV88E6161_G1_ATU_CTL_HASH_MASK;
> +
> +     return 0;
> +}
> +
> +int mv88e6165_g1_atu_set_hash(struct mv88e6xxx_chip *chip, u8 hash)
> +{
> +     int err;
> +     u16 val;
> +
> +     if (hash & ~MV88E6161_G1_ATU_CTL_HASH_MASK)
> +             return -EINVAL;
> +
> +     err = mv88e6xxx_g1_read(chip, MV88E6XXX_G1_ATU_CTL, &val);
> +     if (err)
> +             return err;
> +
> +     val &= ~MV88E6161_G1_ATU_CTL_HASH_MASK;
> +     val |= hash;
> +
> +     return mv88e6xxx_g1_write(chip, MV88E6XXX_G1_ATU_CTL, val);
> +}
> +
>  /* Offset 0x0B: ATU Operation Register */
>  
>  static int mv88e6xxx_g1_atu_op_wait(struct mv88e6xxx_chip *chip)


Thanks,

        Vivien

Reply via email to