.. | .. |
---|
| 1 | +// SPDX-License-Identifier: GPL-2.0-or-later |
---|
1 | 2 | /* |
---|
2 | 3 | * Marvell 88E6xxx Switch Global 2 Registers support |
---|
3 | 4 | * |
---|
.. | .. |
---|
8 | 9 | * |
---|
9 | 10 | * Copyright (c) 2017 National Instruments |
---|
10 | 11 | * Brandon Streiff <brandon.streiff@ni.com> |
---|
11 | | - * |
---|
12 | | - * This program is free software; you can redistribute it and/or modify |
---|
13 | | - * it under the terms of the GNU General Public License as published by |
---|
14 | | - * the Free Software Foundation; either version 2 of the License, or |
---|
15 | | - * (at your option) any later version. |
---|
16 | 12 | */ |
---|
| 13 | + |
---|
| 14 | +#include <linux/bitfield.h> |
---|
17 | 15 | |
---|
18 | 16 | #include "global2.h" |
---|
19 | 17 | |
---|
.. | .. |
---|
31 | 29 | /* mv88e6xxx_g2_avb_read -- Read one or multiple 16-bit words. |
---|
32 | 30 | * The hardware supports snapshotting up to four contiguous registers. |
---|
33 | 31 | */ |
---|
| 32 | +static int mv88e6xxx_g2_avb_wait(struct mv88e6xxx_chip *chip) |
---|
| 33 | +{ |
---|
| 34 | + int bit = __bf_shf(MV88E6352_G2_AVB_CMD_BUSY); |
---|
| 35 | + |
---|
| 36 | + return mv88e6xxx_g2_wait_bit(chip, MV88E6352_G2_AVB_CMD, bit, 0); |
---|
| 37 | +} |
---|
| 38 | + |
---|
34 | 39 | static int mv88e6xxx_g2_avb_read(struct mv88e6xxx_chip *chip, u16 readop, |
---|
35 | 40 | u16 *data, int len) |
---|
36 | 41 | { |
---|
37 | 42 | int err; |
---|
38 | 43 | int i; |
---|
39 | 44 | |
---|
| 45 | + err = mv88e6xxx_g2_avb_wait(chip); |
---|
| 46 | + if (err) |
---|
| 47 | + return err; |
---|
| 48 | + |
---|
40 | 49 | /* Hardware can only snapshot four words. */ |
---|
41 | 50 | if (len > 4) |
---|
42 | 51 | return -E2BIG; |
---|
43 | 52 | |
---|
44 | | - err = mv88e6xxx_g2_update(chip, MV88E6352_G2_AVB_CMD, readop); |
---|
| 53 | + err = mv88e6xxx_g2_write(chip, MV88E6352_G2_AVB_CMD, |
---|
| 54 | + MV88E6352_G2_AVB_CMD_BUSY | readop); |
---|
| 55 | + if (err) |
---|
| 56 | + return err; |
---|
| 57 | + |
---|
| 58 | + err = mv88e6xxx_g2_avb_wait(chip); |
---|
45 | 59 | if (err) |
---|
46 | 60 | return err; |
---|
47 | 61 | |
---|
.. | .. |
---|
61 | 75 | { |
---|
62 | 76 | int err; |
---|
63 | 77 | |
---|
| 78 | + err = mv88e6xxx_g2_avb_wait(chip); |
---|
| 79 | + if (err) |
---|
| 80 | + return err; |
---|
| 81 | + |
---|
64 | 82 | err = mv88e6xxx_g2_write(chip, MV88E6352_G2_AVB_DATA, data); |
---|
65 | 83 | if (err) |
---|
66 | 84 | return err; |
---|
67 | 85 | |
---|
68 | | - return mv88e6xxx_g2_update(chip, MV88E6352_G2_AVB_CMD, writeop); |
---|
| 86 | + err = mv88e6xxx_g2_write(chip, MV88E6352_G2_AVB_CMD, |
---|
| 87 | + MV88E6352_G2_AVB_CMD_BUSY | writeop); |
---|
| 88 | + |
---|
| 89 | + return mv88e6xxx_g2_avb_wait(chip); |
---|
69 | 90 | } |
---|
70 | 91 | |
---|
71 | 92 | static int mv88e6352_g2_avb_port_ptp_read(struct mv88e6xxx_chip *chip, |
---|