Skip to content

Commit 0d632c3

Browse files
bstreiffdavem330
authored andcommitted
net: dsa: mv88e6xxx: add accessors for PTP/TAI registers
This patch implements support for accessing the Precision Time Protocol and Time Application Interface registers via the AVB register interface in the Global 2 register. The register interface differs slightly between different models; older models use a 3-bit operations field, while newer models use a 2-bit field. The operations values and the special "global port" values are different between the two. This is a similar split to the differences in the "Ingress Rate" register between models, so, like in that case, we call the two variants "6352" and "6390" and create an ops structure to abstract between the two. checkpatch fixups by Andrew Lunn Signed-off-by: Brandon Streiff <[email protected]> Signed-off-by: Andrew Lunn <[email protected]> Signed-off-by: David S. Miller <[email protected]>
1 parent b000be9 commit 0d632c3

File tree

5 files changed

+251
-1
lines changed

5 files changed

+251
-1
lines changed

drivers/net/dsa/mv88e6xxx/Makefile

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ mv88e6xxx-objs += global1.o
55
mv88e6xxx-objs += global1_atu.o
66
mv88e6xxx-objs += global1_vtu.o
77
mv88e6xxx-$(CONFIG_NET_DSA_MV88E6XXX_GLOBAL2) += global2.o
8+
mv88e6xxx-$(CONFIG_NET_DSA_MV88E6XXX_GLOBAL2) += global2_avb.o
89
mv88e6xxx-objs += phy.o
910
mv88e6xxx-objs += port.o
1011
mv88e6xxx-objs += serdes.o

drivers/net/dsa/mv88e6xxx/chip.c

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2843,6 +2843,7 @@ static const struct mv88e6xxx_ops mv88e6240_ops = {
28432843
.vtu_getnext = mv88e6352_g1_vtu_getnext,
28442844
.vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
28452845
.serdes_power = mv88e6352_serdes_power,
2846+
.avb_ops = &mv88e6352_avb_ops,
28462847
};
28472848

28482849
static const struct mv88e6xxx_ops mv88e6290_ops = {
@@ -2879,6 +2880,7 @@ static const struct mv88e6xxx_ops mv88e6290_ops = {
28792880
.vtu_getnext = mv88e6390_g1_vtu_getnext,
28802881
.vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
28812882
.serdes_power = mv88e6390_serdes_power,
2883+
.avb_ops = &mv88e6390_avb_ops,
28822884
};
28832885

28842886
static const struct mv88e6xxx_ops mv88e6320_ops = {
@@ -2913,6 +2915,7 @@ static const struct mv88e6xxx_ops mv88e6320_ops = {
29132915
.reset = mv88e6352_g1_reset,
29142916
.vtu_getnext = mv88e6185_g1_vtu_getnext,
29152917
.vtu_loadpurge = mv88e6185_g1_vtu_loadpurge,
2918+
.avb_ops = &mv88e6352_avb_ops,
29162919
};
29172920

29182921
static const struct mv88e6xxx_ops mv88e6321_ops = {
@@ -2945,6 +2948,7 @@ static const struct mv88e6xxx_ops mv88e6321_ops = {
29452948
.reset = mv88e6352_g1_reset,
29462949
.vtu_getnext = mv88e6185_g1_vtu_getnext,
29472950
.vtu_loadpurge = mv88e6185_g1_vtu_loadpurge,
2951+
.avb_ops = &mv88e6352_avb_ops,
29482952
};
29492953

29502954
static const struct mv88e6xxx_ops mv88e6341_ops = {
@@ -2981,6 +2985,7 @@ static const struct mv88e6xxx_ops mv88e6341_ops = {
29812985
.reset = mv88e6352_g1_reset,
29822986
.vtu_getnext = mv88e6352_g1_vtu_getnext,
29832987
.vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
2988+
.avb_ops = &mv88e6390_avb_ops,
29842989
};
29852990

29862991
static const struct mv88e6xxx_ops mv88e6350_ops = {
@@ -3049,6 +3054,7 @@ static const struct mv88e6xxx_ops mv88e6351_ops = {
30493054
.reset = mv88e6352_g1_reset,
30503055
.vtu_getnext = mv88e6352_g1_vtu_getnext,
30513056
.vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
3057+
.avb_ops = &mv88e6352_avb_ops,
30523058
};
30533059

30543060
static const struct mv88e6xxx_ops mv88e6352_ops = {
@@ -3086,6 +3092,7 @@ static const struct mv88e6xxx_ops mv88e6352_ops = {
30863092
.vtu_getnext = mv88e6352_g1_vtu_getnext,
30873093
.vtu_loadpurge = mv88e6352_g1_vtu_loadpurge,
30883094
.serdes_power = mv88e6352_serdes_power,
3095+
.avb_ops = &mv88e6352_avb_ops,
30893096
};
30903097

30913098
static const struct mv88e6xxx_ops mv88e6390_ops = {
@@ -3124,6 +3131,7 @@ static const struct mv88e6xxx_ops mv88e6390_ops = {
31243131
.vtu_getnext = mv88e6390_g1_vtu_getnext,
31253132
.vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
31263133
.serdes_power = mv88e6390_serdes_power,
3134+
.avb_ops = &mv88e6390_avb_ops,
31273135
};
31283136

31293137
static const struct mv88e6xxx_ops mv88e6390x_ops = {
@@ -3162,6 +3170,7 @@ static const struct mv88e6xxx_ops mv88e6390x_ops = {
31623170
.vtu_getnext = mv88e6390_g1_vtu_getnext,
31633171
.vtu_loadpurge = mv88e6390_g1_vtu_loadpurge,
31643172
.serdes_power = mv88e6390_serdes_power,
3173+
.avb_ops = &mv88e6390_avb_ops,
31653174
};
31663175

31673176
static const struct mv88e6xxx_info mv88e6xxx_table[] = {

drivers/net/dsa/mv88e6xxx/chip.h

Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,7 @@ struct mv88e6xxx_vtu_entry {
146146

147147
struct mv88e6xxx_bus_ops;
148148
struct mv88e6xxx_irq_ops;
149+
struct mv88e6xxx_avb_ops;
149150

150151
struct mv88e6xxx_irq {
151152
u16 masked;
@@ -344,6 +345,9 @@ struct mv88e6xxx_ops {
344345
struct mv88e6xxx_vtu_entry *entry);
345346
int (*vtu_loadpurge)(struct mv88e6xxx_chip *chip,
346347
struct mv88e6xxx_vtu_entry *entry);
348+
349+
/* Interface to the AVB/PTP registers */
350+
const struct mv88e6xxx_avb_ops *avb_ops;
347351
};
348352

349353
struct mv88e6xxx_irq_ops {
@@ -355,6 +359,24 @@ struct mv88e6xxx_irq_ops {
355359
void (*irq_free)(struct mv88e6xxx_chip *chip);
356360
};
357361

362+
struct mv88e6xxx_avb_ops {
363+
/* Access port-scoped Precision Time Protocol registers */
364+
int (*port_ptp_read)(struct mv88e6xxx_chip *chip, int port, int addr,
365+
u16 *data, int len);
366+
int (*port_ptp_write)(struct mv88e6xxx_chip *chip, int port, int addr,
367+
u16 data);
368+
369+
/* Access global Precision Time Protocol registers */
370+
int (*ptp_read)(struct mv88e6xxx_chip *chip, int addr, u16 *data,
371+
int len);
372+
int (*ptp_write)(struct mv88e6xxx_chip *chip, int addr, u16 data);
373+
374+
/* Access global Time Application Interface registers */
375+
int (*tai_read)(struct mv88e6xxx_chip *chip, int addr, u16 *data,
376+
int len);
377+
int (*tai_write)(struct mv88e6xxx_chip *chip, int addr, u16 data);
378+
};
379+
358380
#define STATS_TYPE_PORT BIT(0)
359381
#define STATS_TYPE_BANK0 BIT(1)
360382
#define STATS_TYPE_BANK1 BIT(2)

drivers/net/dsa/mv88e6xxx/global2.h

Lines changed: 26 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -149,7 +149,26 @@
149149
#define MV88E6390_G2_EEPROM_ADDR_MASK 0xffff
150150

151151
/* Offset 0x16: AVB Command Register */
152-
#define MV88E6352_G2_AVB_CMD 0x16
152+
#define MV88E6352_G2_AVB_CMD 0x16
153+
#define MV88E6352_G2_AVB_CMD_BUSY 0x8000
154+
#define MV88E6352_G2_AVB_CMD_OP_READ 0x4000
155+
#define MV88E6352_G2_AVB_CMD_OP_READ_INCR 0x6000
156+
#define MV88E6352_G2_AVB_CMD_OP_WRITE 0x3000
157+
#define MV88E6390_G2_AVB_CMD_OP_READ 0x0000
158+
#define MV88E6390_G2_AVB_CMD_OP_READ_INCR 0x4000
159+
#define MV88E6390_G2_AVB_CMD_OP_WRITE 0x6000
160+
#define MV88E6352_G2_AVB_CMD_PORT_MASK 0x0f00
161+
#define MV88E6352_G2_AVB_CMD_PORT_TAIGLOBAL 0xe
162+
#define MV88E6352_G2_AVB_CMD_PORT_PTPGLOBAL 0xf
163+
#define MV88E6390_G2_AVB_CMD_PORT_MASK 0x1f00
164+
#define MV88E6390_G2_AVB_CMD_PORT_TAIGLOBAL 0x1e
165+
#define MV88E6390_G2_AVB_CMD_PORT_PTPGLOBAL 0x1f
166+
#define MV88E6352_G2_AVB_CMD_BLOCK_PTP 0
167+
#define MV88E6352_G2_AVB_CMD_BLOCK_AVB 1
168+
#define MV88E6352_G2_AVB_CMD_BLOCK_QAV 2
169+
#define MV88E6352_G2_AVB_CMD_BLOCK_QVB 3
170+
#define MV88E6352_G2_AVB_CMD_BLOCK_MASK 0x00e0
171+
#define MV88E6352_G2_AVB_CMD_ADDR_MASK 0x001f
153172

154173
/* Offset 0x17: AVB Data Register */
155174
#define MV88E6352_G2_AVB_DATA 0x17
@@ -272,6 +291,9 @@ int mv88e6xxx_g2_pot_clear(struct mv88e6xxx_chip *chip);
272291
extern const struct mv88e6xxx_irq_ops mv88e6097_watchdog_ops;
273292
extern const struct mv88e6xxx_irq_ops mv88e6390_watchdog_ops;
274293

294+
extern const struct mv88e6xxx_avb_ops mv88e6352_avb_ops;
295+
extern const struct mv88e6xxx_avb_ops mv88e6390_avb_ops;
296+
275297
#else /* !CONFIG_NET_DSA_MV88E6XXX_GLOBAL2 */
276298

277299
static inline int mv88e6xxx_g2_require(struct mv88e6xxx_chip *chip)
@@ -407,6 +429,9 @@ static inline int mv88e6xxx_g2_pot_clear(struct mv88e6xxx_chip *chip)
407429
static const struct mv88e6xxx_irq_ops mv88e6097_watchdog_ops = {};
408430
static const struct mv88e6xxx_irq_ops mv88e6390_watchdog_ops = {};
409431

432+
static const struct mv88e6xxx_avb_ops mv88e6352_avb_ops = {};
433+
static const struct mv88e6xxx_avb_ops mv88e6390_avb_ops = {};
434+
410435
#endif /* CONFIG_NET_DSA_MV88E6XXX_GLOBAL2 */
411436

412437
#endif /* _MV88E6XXX_GLOBAL2_H */
Lines changed: 193 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,193 @@
1+
/*
2+
* Marvell 88E6xxx Switch Global 2 Registers support
3+
*
4+
* Copyright (c) 2008 Marvell Semiconductor
5+
*
6+
* Copyright (c) 2016-2017 Savoir-faire Linux Inc.
7+
* Vivien Didelot <[email protected]>
8+
*
9+
* Copyright (c) 2017 National Instruments
10+
* Brandon Streiff <[email protected]>
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+
*/
17+
18+
#include "global2.h"
19+
20+
/* Offset 0x16: AVB Command Register
21+
* Offset 0x17: AVB Data Register
22+
*
23+
* There are two different versions of this register interface:
24+
* "6352": 3-bit "op" field, 4-bit "port" field.
25+
* "6390": 2-bit "op" field, 5-bit "port" field.
26+
*
27+
* The "op" codes are different between the two, as well as the special
28+
* port fields for global PTP and TAI configuration.
29+
*/
30+
31+
/* mv88e6xxx_g2_avb_read -- Read one or multiple 16-bit words.
32+
* The hardware supports snapshotting up to four contiguous registers.
33+
*/
34+
static int mv88e6xxx_g2_avb_read(struct mv88e6xxx_chip *chip, u16 readop,
35+
u16 *data, int len)
36+
{
37+
int err;
38+
int i;
39+
40+
/* Hardware can only snapshot four words. */
41+
if (len > 4)
42+
return -E2BIG;
43+
44+
err = mv88e6xxx_g2_update(chip, MV88E6352_G2_AVB_CMD, readop);
45+
if (err)
46+
return err;
47+
48+
for (i = 0; i < len; ++i) {
49+
err = mv88e6xxx_g2_read(chip, MV88E6352_G2_AVB_DATA,
50+
&data[i]);
51+
if (err)
52+
return err;
53+
}
54+
55+
return 0;
56+
}
57+
58+
/* mv88e6xxx_g2_avb_write -- Write one 16-bit word. */
59+
static int mv88e6xxx_g2_avb_write(struct mv88e6xxx_chip *chip, u16 writeop,
60+
u16 data)
61+
{
62+
int err;
63+
64+
err = mv88e6xxx_g2_write(chip, MV88E6352_G2_AVB_DATA, data);
65+
if (err)
66+
return err;
67+
68+
return mv88e6xxx_g2_update(chip, MV88E6352_G2_AVB_CMD, writeop);
69+
}
70+
71+
static int mv88e6352_g2_avb_port_ptp_read(struct mv88e6xxx_chip *chip,
72+
int port, int addr, u16 *data,
73+
int len)
74+
{
75+
u16 readop = (len == 1 ? MV88E6352_G2_AVB_CMD_OP_READ :
76+
MV88E6352_G2_AVB_CMD_OP_READ_INCR) |
77+
(port << 8) | (MV88E6352_G2_AVB_CMD_BLOCK_PTP << 5) |
78+
addr;
79+
80+
return mv88e6xxx_g2_avb_read(chip, readop, data, len);
81+
}
82+
83+
static int mv88e6352_g2_avb_port_ptp_write(struct mv88e6xxx_chip *chip,
84+
int port, int addr, u16 data)
85+
{
86+
u16 writeop = MV88E6352_G2_AVB_CMD_OP_WRITE | (port << 8) |
87+
(MV88E6352_G2_AVB_CMD_BLOCK_PTP << 5) | addr;
88+
89+
return mv88e6xxx_g2_avb_write(chip, writeop, data);
90+
}
91+
92+
static int mv88e6352_g2_avb_ptp_read(struct mv88e6xxx_chip *chip, int addr,
93+
u16 *data, int len)
94+
{
95+
return mv88e6352_g2_avb_port_ptp_read(chip,
96+
MV88E6352_G2_AVB_CMD_PORT_PTPGLOBAL,
97+
addr, data, len);
98+
}
99+
100+
static int mv88e6352_g2_avb_ptp_write(struct mv88e6xxx_chip *chip, int addr,
101+
u16 data)
102+
{
103+
return mv88e6352_g2_avb_port_ptp_write(chip,
104+
MV88E6352_G2_AVB_CMD_PORT_PTPGLOBAL,
105+
addr, data);
106+
}
107+
108+
static int mv88e6352_g2_avb_tai_read(struct mv88e6xxx_chip *chip, int addr,
109+
u16 *data, int len)
110+
{
111+
return mv88e6352_g2_avb_port_ptp_read(chip,
112+
MV88E6352_G2_AVB_CMD_PORT_TAIGLOBAL,
113+
addr, data, len);
114+
}
115+
116+
static int mv88e6352_g2_avb_tai_write(struct mv88e6xxx_chip *chip, int addr,
117+
u16 data)
118+
{
119+
return mv88e6352_g2_avb_port_ptp_write(chip,
120+
MV88E6352_G2_AVB_CMD_PORT_TAIGLOBAL,
121+
addr, data);
122+
}
123+
124+
const struct mv88e6xxx_avb_ops mv88e6352_avb_ops = {
125+
.port_ptp_read = mv88e6352_g2_avb_port_ptp_read,
126+
.port_ptp_write = mv88e6352_g2_avb_port_ptp_write,
127+
.ptp_read = mv88e6352_g2_avb_ptp_read,
128+
.ptp_write = mv88e6352_g2_avb_ptp_write,
129+
.tai_read = mv88e6352_g2_avb_tai_read,
130+
.tai_write = mv88e6352_g2_avb_tai_write,
131+
};
132+
133+
static int mv88e6390_g2_avb_port_ptp_read(struct mv88e6xxx_chip *chip,
134+
int port, int addr, u16 *data,
135+
int len)
136+
{
137+
u16 readop = (len == 1 ? MV88E6390_G2_AVB_CMD_OP_READ :
138+
MV88E6390_G2_AVB_CMD_OP_READ_INCR) |
139+
(port << 8) | (MV88E6352_G2_AVB_CMD_BLOCK_PTP << 5) |
140+
addr;
141+
142+
return mv88e6xxx_g2_avb_read(chip, readop, data, len);
143+
}
144+
145+
static int mv88e6390_g2_avb_port_ptp_write(struct mv88e6xxx_chip *chip,
146+
int port, int addr, u16 data)
147+
{
148+
u16 writeop = MV88E6390_G2_AVB_CMD_OP_WRITE | (port << 8) |
149+
(MV88E6352_G2_AVB_CMD_BLOCK_PTP << 5) | addr;
150+
151+
return mv88e6xxx_g2_avb_write(chip, writeop, data);
152+
}
153+
154+
static int mv88e6390_g2_avb_ptp_read(struct mv88e6xxx_chip *chip, int addr,
155+
u16 *data, int len)
156+
{
157+
return mv88e6390_g2_avb_port_ptp_read(chip,
158+
MV88E6390_G2_AVB_CMD_PORT_PTPGLOBAL,
159+
addr, data, len);
160+
}
161+
162+
static int mv88e6390_g2_avb_ptp_write(struct mv88e6xxx_chip *chip, int addr,
163+
u16 data)
164+
{
165+
return mv88e6390_g2_avb_port_ptp_write(chip,
166+
MV88E6390_G2_AVB_CMD_PORT_PTPGLOBAL,
167+
addr, data);
168+
}
169+
170+
static int mv88e6390_g2_avb_tai_read(struct mv88e6xxx_chip *chip, int addr,
171+
u16 *data, int len)
172+
{
173+
return mv88e6390_g2_avb_port_ptp_read(chip,
174+
MV88E6390_G2_AVB_CMD_PORT_TAIGLOBAL,
175+
addr, data, len);
176+
}
177+
178+
static int mv88e6390_g2_avb_tai_write(struct mv88e6xxx_chip *chip, int addr,
179+
u16 data)
180+
{
181+
return mv88e6390_g2_avb_port_ptp_write(chip,
182+
MV88E6390_G2_AVB_CMD_PORT_TAIGLOBAL,
183+
addr, data);
184+
}
185+
186+
const struct mv88e6xxx_avb_ops mv88e6390_avb_ops = {
187+
.port_ptp_read = mv88e6390_g2_avb_port_ptp_read,
188+
.port_ptp_write = mv88e6390_g2_avb_port_ptp_write,
189+
.ptp_read = mv88e6390_g2_avb_ptp_read,
190+
.ptp_write = mv88e6390_g2_avb_ptp_write,
191+
.tai_read = mv88e6390_g2_avb_tai_read,
192+
.tai_write = mv88e6390_g2_avb_tai_write,
193+
};

0 commit comments

Comments
 (0)