ocelot.c 41.7 KB
Newer Older
1 2 3 4 5 6 7 8
// SPDX-License-Identifier: (GPL-2.0 OR MIT)
/*
 * Microsemi Ocelot Switch driver
 *
 * Copyright (c) 2017 Microsemi Corporation
 */
#include <linux/if_bridge.h>
#include "ocelot.h"
9
#include "ocelot_vcap.h"
10

11 12 13
#define TABLE_UPDATE_SLEEP_US 10
#define TABLE_UPDATE_TIMEOUT_US 100000

14 15 16 17 18 19
struct ocelot_mact_entry {
	u8 mac[ETH_ALEN];
	u16 vid;
	enum macaccess_entry_type type;
};

20
static inline u32 ocelot_mact_read_macaccess(struct ocelot *ocelot)
21
{
22 23
	return ocelot_read(ocelot, ANA_TABLES_MACACCESS);
}
24

25 26 27
static inline int ocelot_mact_wait_for_completion(struct ocelot *ocelot)
{
	u32 val;
28

29 30 31 32 33
	return readx_poll_timeout(ocelot_mact_read_macaccess,
		ocelot, val,
		(val & ANA_TABLES_MACACCESS_MAC_TABLE_CMD_M) ==
		MACACCESS_CMD_IDLE,
		TABLE_UPDATE_SLEEP_US, TABLE_UPDATE_TIMEOUT_US);
34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
}

static void ocelot_mact_select(struct ocelot *ocelot,
			       const unsigned char mac[ETH_ALEN],
			       unsigned int vid)
{
	u32 macl = 0, mach = 0;

	/* Set the MAC address to handle and the vlan associated in a format
	 * understood by the hardware.
	 */
	mach |= vid    << 16;
	mach |= mac[0] << 8;
	mach |= mac[1] << 0;
	macl |= mac[2] << 24;
	macl |= mac[3] << 16;
	macl |= mac[4] << 8;
	macl |= mac[5] << 0;

	ocelot_write(ocelot, macl, ANA_TABLES_MACLDATA);
	ocelot_write(ocelot, mach, ANA_TABLES_MACHDATA);

}

58 59 60
int ocelot_mact_learn(struct ocelot *ocelot, int port,
		      const unsigned char mac[ETH_ALEN],
		      unsigned int vid, enum macaccess_entry_type type)
61 62 63 64 65 66 67 68 69 70 71 72
{
	ocelot_mact_select(ocelot, mac, vid);

	/* Issue a write command */
	ocelot_write(ocelot, ANA_TABLES_MACACCESS_VALID |
			     ANA_TABLES_MACACCESS_DEST_IDX(port) |
			     ANA_TABLES_MACACCESS_ENTRYTYPE(type) |
			     ANA_TABLES_MACACCESS_MAC_TABLE_CMD(MACACCESS_CMD_LEARN),
			     ANA_TABLES_MACACCESS);

	return ocelot_mact_wait_for_completion(ocelot);
}
73
EXPORT_SYMBOL(ocelot_mact_learn);
74

75 76
int ocelot_mact_forget(struct ocelot *ocelot,
		       const unsigned char mac[ETH_ALEN], unsigned int vid)
77 78 79 80 81 82 83 84 85 86
{
	ocelot_mact_select(ocelot, mac, vid);

	/* Issue a forget command */
	ocelot_write(ocelot,
		     ANA_TABLES_MACACCESS_MAC_TABLE_CMD(MACACCESS_CMD_FORGET),
		     ANA_TABLES_MACACCESS);

	return ocelot_mact_wait_for_completion(ocelot);
}
87
EXPORT_SYMBOL(ocelot_mact_forget);
88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104

static void ocelot_mact_init(struct ocelot *ocelot)
{
	/* Configure the learning mode entries attributes:
	 * - Do not copy the frame to the CPU extraction queues.
	 * - Use the vlan and mac_cpoy for dmac lookup.
	 */
	ocelot_rmw(ocelot, 0,
		   ANA_AGENCTRL_LEARN_CPU_COPY | ANA_AGENCTRL_IGNORE_DMAC_FLAGS
		   | ANA_AGENCTRL_LEARN_FWD_KILL
		   | ANA_AGENCTRL_LEARN_IGNORE_VLAN,
		   ANA_AGENCTRL);

	/* Clear the MAC table */
	ocelot_write(ocelot, MACACCESS_CMD_INIT, ANA_TABLES_MACACCESS);
}

105
static void ocelot_vcap_enable(struct ocelot *ocelot, int port)
106 107 108
{
	ocelot_write_gix(ocelot, ANA_PORT_VCAP_S2_CFG_S2_ENA |
			 ANA_PORT_VCAP_S2_CFG_S2_IP6_CFG(0xa),
109
			 ANA_PORT_VCAP_S2_CFG, port);
110 111
}

112
static inline u32 ocelot_vlant_read_vlanaccess(struct ocelot *ocelot)
113
{
114 115
	return ocelot_read(ocelot, ANA_TABLES_VLANACCESS);
}
116

117 118 119 120 121 122 123 124 125 126
static inline int ocelot_vlant_wait_for_completion(struct ocelot *ocelot)
{
	u32 val;

	return readx_poll_timeout(ocelot_vlant_read_vlanaccess,
		ocelot,
		val,
		(val & ANA_TABLES_VLANACCESS_VLAN_TBL_CMD_M) ==
		ANA_TABLES_VLANACCESS_CMD_IDLE,
		TABLE_UPDATE_SLEEP_US, TABLE_UPDATE_TIMEOUT_US);
127 128
}

129 130 131 132 133 134 135 136 137 138 139 140 141
static int ocelot_vlant_set_mask(struct ocelot *ocelot, u16 vid, u32 mask)
{
	/* Select the VID to configure */
	ocelot_write(ocelot, ANA_TABLES_VLANTIDX_V_INDEX(vid),
		     ANA_TABLES_VLANTIDX);
	/* Set the vlan port members mask and issue a write command */
	ocelot_write(ocelot, ANA_TABLES_VLANACCESS_VLAN_PORT_MASK(mask) |
			     ANA_TABLES_VLANACCESS_CMD_WRITE,
		     ANA_TABLES_VLANACCESS);

	return ocelot_vlant_wait_for_completion(ocelot);
}

142 143
static int ocelot_port_set_native_vlan(struct ocelot *ocelot, int port,
				       u16 vid)
144
{
145
	struct ocelot_port *ocelot_port = ocelot->ports[port];
146
	u32 val = 0;
147

148 149 150 151 152 153 154 155 156 157 158 159 160 161
	if (ocelot_port->vid != vid) {
		/* Always permit deleting the native VLAN (vid = 0) */
		if (ocelot_port->vid && vid) {
			dev_err(ocelot->dev,
				"Port already has a native VLAN: %d\n",
				ocelot_port->vid);
			return -EBUSY;
		}
		ocelot_port->vid = vid;
	}

	ocelot_rmw_gix(ocelot, REW_PORT_VLAN_CFG_PORT_VID(vid),
		       REW_PORT_VLAN_CFG_PORT_VID_M,
		       REW_PORT_VLAN_CFG, port);
162

163
	if (ocelot_port->vlan_aware && !ocelot_port->vid)
164 165 166
		/* If port is vlan-aware and tagged, drop untagged and priority
		 * tagged frames.
		 */
167 168 169 170 171
		val = ANA_PORT_DROP_CFG_DROP_UNTAGGED_ENA |
		      ANA_PORT_DROP_CFG_DROP_PRIO_S_TAGGED_ENA |
		      ANA_PORT_DROP_CFG_DROP_PRIO_C_TAGGED_ENA;
	ocelot_rmw_gix(ocelot, val,
		       ANA_PORT_DROP_CFG_DROP_UNTAGGED_ENA |
172
		       ANA_PORT_DROP_CFG_DROP_PRIO_S_TAGGED_ENA |
173 174
		       ANA_PORT_DROP_CFG_DROP_PRIO_C_TAGGED_ENA,
		       ANA_PORT_DROP_CFG, port);
175

176
	if (ocelot_port->vlan_aware) {
177
		if (ocelot_port->vid)
178
			/* Tag all frames except when VID == DEFAULT_VLAN */
179
			val = REW_TAG_CFG_TAG_CFG(1);
180 181
		else
			/* Tag all frames */
182
			val = REW_TAG_CFG_TAG_CFG(3);
183 184 185
	} else {
		/* Port tagging disabled. */
		val = REW_TAG_CFG_TAG_CFG(0);
186 187 188
	}
	ocelot_rmw_gix(ocelot, val,
		       REW_TAG_CFG_TAG_CFG_M,
189
		       REW_TAG_CFG, port);
190 191

	return 0;
192 193
}

194 195
void ocelot_port_vlan_filtering(struct ocelot *ocelot, int port,
				bool vlan_aware)
196 197
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];
198
	u32 val;
199

200
	ocelot_port->vlan_aware = vlan_aware;
201

202 203 204 205 206 207 208 209 210
	if (vlan_aware)
		val = ANA_PORT_VLAN_CFG_VLAN_AWARE_ENA |
		      ANA_PORT_VLAN_CFG_VLAN_POP_CNT(1);
	else
		val = 0;
	ocelot_rmw_gix(ocelot, val,
		       ANA_PORT_VLAN_CFG_VLAN_AWARE_ENA |
		       ANA_PORT_VLAN_CFG_VLAN_POP_CNT_M,
		       ANA_PORT_VLAN_CFG, port);
211

212
	ocelot_port_set_native_vlan(ocelot, port, ocelot_port->vid);
213
}
214
EXPORT_SYMBOL(ocelot_port_vlan_filtering);
215 216 217 218 219 220 221 222 223 224 225 226

/* Default vlan to clasify for untagged frames (may be zero) */
static void ocelot_port_set_pvid(struct ocelot *ocelot, int port, u16 pvid)
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];

	ocelot_rmw_gix(ocelot,
		       ANA_PORT_VLAN_CFG_VLAN_VID(pvid),
		       ANA_PORT_VLAN_CFG_VLAN_VID_M,
		       ANA_PORT_VLAN_CFG, port);

	ocelot_port->pvid = pvid;
227 228
}

229 230
int ocelot_vlan_add(struct ocelot *ocelot, int port, u16 vid, bool pvid,
		    bool untagged)
231 232 233 234
{
	int ret;

	/* Make the port a member of the VLAN */
235
	ocelot->vlan_mask[vid] |= BIT(port);
236 237 238 239 240 241
	ret = ocelot_vlant_set_mask(ocelot, vid, ocelot->vlan_mask[vid]);
	if (ret)
		return ret;

	/* Default ingress vlan classification */
	if (pvid)
242
		ocelot_port_set_pvid(ocelot, port, vid);
243 244

	/* Untagged egress vlan clasification */
245 246 247 248
	if (untagged) {
		ret = ocelot_port_set_native_vlan(ocelot, port, vid);
		if (ret)
			return ret;
249
	}
250 251 252

	return 0;
}
253
EXPORT_SYMBOL(ocelot_vlan_add);
254

255
int ocelot_vlan_del(struct ocelot *ocelot, int port, u16 vid)
256 257 258
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];
	int ret;
259 260

	/* Stop the port from being a member of the vlan */
261
	ocelot->vlan_mask[vid] &= ~BIT(port);
262 263 264 265 266
	ret = ocelot_vlant_set_mask(ocelot, vid, ocelot->vlan_mask[vid]);
	if (ret)
		return ret;

	/* Ingress */
267 268
	if (ocelot_port->pvid == vid)
		ocelot_port_set_pvid(ocelot, port, 0);
269 270

	/* Egress */
271 272
	if (ocelot_port->vid == vid)
		ocelot_port_set_native_vlan(ocelot, port, 0);
273 274 275

	return 0;
}
276
EXPORT_SYMBOL(ocelot_vlan_del);
277

278 279
static void ocelot_vlan_init(struct ocelot *ocelot)
{
280 281
	u16 port, vid;

282 283 284 285
	/* Clear VLAN table, by default all ports are members of all VLANs */
	ocelot_write(ocelot, ANA_TABLES_VLANACCESS_CMD_INIT,
		     ANA_TABLES_VLANACCESS);
	ocelot_vlant_wait_for_completion(ocelot);
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302

	/* Configure the port VLAN memberships */
	for (vid = 1; vid < VLAN_N_VID; vid++) {
		ocelot->vlan_mask[vid] = 0;
		ocelot_vlant_set_mask(ocelot, vid, ocelot->vlan_mask[vid]);
	}

	/* Because VLAN filtering is enabled, we need VID 0 to get untagged
	 * traffic.  It is added automatically if 8021q module is loaded, but
	 * we can't rely on it since module may be not loaded.
	 */
	ocelot->vlan_mask[0] = GENMASK(ocelot->num_phys_ports - 1, 0);
	ocelot_vlant_set_mask(ocelot, 0, ocelot->vlan_mask[0]);

	/* Set vlan ingress filter mask to all ports but the CPU port by
	 * default.
	 */
303 304
	ocelot_write(ocelot, GENMASK(ocelot->num_phys_ports - 1, 0),
		     ANA_VLANMASK);
305 306 307 308 309

	for (port = 0; port < ocelot->num_phys_ports; port++) {
		ocelot_write_gix(ocelot, 0, REW_PORT_VLAN_CFG, port);
		ocelot_write_gix(ocelot, 0, REW_TAG_CFG, port);
	}
310 311
}

312 313
void ocelot_adjust_link(struct ocelot *ocelot, int port,
			struct phy_device *phydev)
314
{
315
	struct ocelot_port *ocelot_port = ocelot->ports[port];
316
	int speed, mode = 0;
317

318
	switch (phydev->speed) {
319 320 321 322 323 324 325 326 327 328 329 330 331 332 333
	case SPEED_10:
		speed = OCELOT_SPEED_10;
		break;
	case SPEED_100:
		speed = OCELOT_SPEED_100;
		break;
	case SPEED_1000:
		speed = OCELOT_SPEED_1000;
		mode = DEV_MAC_MODE_CFG_GIGA_MODE_ENA;
		break;
	case SPEED_2500:
		speed = OCELOT_SPEED_2500;
		mode = DEV_MAC_MODE_CFG_GIGA_MODE_ENA;
		break;
	default:
334 335
		dev_err(ocelot->dev, "Unsupported PHY speed on port %d: %d\n",
			port, phydev->speed);
336 337 338
		return;
	}

339
	phy_print_status(phydev);
340

341
	if (!phydev->link)
342 343 344
		return;

	/* Only full duplex supported for now */
345
	ocelot_port_writel(ocelot_port, DEV_MAC_MODE_CFG_FDX_ENA |
346 347
			   mode, DEV_MAC_MODE_CFG);

348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364
	/* Disable HDX fast control */
	ocelot_port_writel(ocelot_port, DEV_PORT_MISC_HDX_FAST_DIS,
			   DEV_PORT_MISC);

	/* SGMII only for now */
	ocelot_port_writel(ocelot_port, PCS1G_MODE_CFG_SGMII_MODE_ENA,
			   PCS1G_MODE_CFG);
	ocelot_port_writel(ocelot_port, PCS1G_SD_CFG_SD_SEL, PCS1G_SD_CFG);

	/* Enable PCS */
	ocelot_port_writel(ocelot_port, PCS1G_CFG_PCS_ENA, PCS1G_CFG);

	/* No aneg on SGMII */
	ocelot_port_writel(ocelot_port, 0, PCS1G_ANEG_CFG);

	/* No loopback */
	ocelot_port_writel(ocelot_port, 0, PCS1G_LB_CFG);
365 366

	/* Enable MAC module */
367
	ocelot_port_writel(ocelot_port, DEV_MAC_ENA_CFG_RX_ENA |
368 369 370 371
			   DEV_MAC_ENA_CFG_TX_ENA, DEV_MAC_ENA_CFG);

	/* Take MAC, Port, Phy (intern) and PCS (SGMII/Serdes) clock out of
	 * reset */
372
	ocelot_port_writel(ocelot_port, DEV_CLOCK_CFG_LINK_SPEED(speed),
373 374 375 376
			   DEV_CLOCK_CFG);

	/* No PFC */
	ocelot_write_gix(ocelot, ANA_PFC_PFC_CFG_FC_LINK_SPEED(speed),
377
			 ANA_PFC_PFC_CFG, port);
378 379

	/* Core: Enable port for frame transfer */
380 381
	ocelot_fields_write(ocelot, port,
			    QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
382 383 384 385 386 387 388

	/* Flow control */
	ocelot_write_rix(ocelot, SYS_MAC_FC_CFG_PAUSE_VAL_CFG(0xffff) |
			 SYS_MAC_FC_CFG_RX_FC_ENA | SYS_MAC_FC_CFG_TX_FC_ENA |
			 SYS_MAC_FC_CFG_ZERO_PAUSE_ENA |
			 SYS_MAC_FC_CFG_FC_LATENCY_CFG(0x7) |
			 SYS_MAC_FC_CFG_FC_LINK_SPEED(speed),
389 390
			 SYS_MAC_FC_CFG, port);
	ocelot_write_rix(ocelot, 0, ANA_POL_FLOWC, port);
391
}
392
EXPORT_SYMBOL(ocelot_adjust_link);
393

394 395
void ocelot_port_enable(struct ocelot *ocelot, int port,
			struct phy_device *phy)
396 397 398 399 400 401
{
	/* Enable receiving frames on the port, and activate auto-learning of
	 * MAC addresses.
	 */
	ocelot_write_gix(ocelot, ANA_PORT_PORT_CFG_LEARNAUTO |
			 ANA_PORT_PORT_CFG_RECV_ENA |
402 403
			 ANA_PORT_PORT_CFG_PORTID_VAL(port),
			 ANA_PORT_PORT_CFG, port);
404
}
405
EXPORT_SYMBOL(ocelot_port_enable);
406

407
void ocelot_port_disable(struct ocelot *ocelot, int port)
408 409 410 411
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];

	ocelot_port_writel(ocelot_port, 0, DEV_MAC_ENA_CFG);
412
	ocelot_fields_write(ocelot, port, QSYS_SWITCH_PORT_MODE_PORT_ENA, 0);
413
}
414
EXPORT_SYMBOL(ocelot_port_disable);
415

416 417
void ocelot_port_add_txtstamp_skb(struct ocelot *ocelot, int port,
				  struct sk_buff *clone)
418
{
419
	struct ocelot_port *ocelot_port = ocelot->ports[port];
420

421
	spin_lock(&ocelot_port->ts_id_lock);
422

423 424 425 426 427
	skb_shinfo(clone)->tx_flags |= SKBTX_IN_PROGRESS;
	/* Store timestamp ID in cb[0] of sk_buff */
	clone->cb[0] = ocelot_port->ts_id;
	ocelot_port->ts_id = (ocelot_port->ts_id + 1) % 4;
	skb_queue_tail(&ocelot_port->tx_skbs, clone);
428

429
	spin_unlock(&ocelot_port->ts_id_lock);
430 431 432
}
EXPORT_SYMBOL(ocelot_port_add_txtstamp_skb);

433 434
static void ocelot_get_hwtimestamp(struct ocelot *ocelot,
				   struct timespec64 *ts)
435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458
{
	unsigned long flags;
	u32 val;

	spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);

	/* Read current PTP time to get seconds */
	val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);

	val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK | PTP_PIN_CFG_DOM);
	val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_SAVE);
	ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
	ts->tv_sec = ocelot_read_rix(ocelot, PTP_PIN_TOD_SEC_LSB, TOD_ACC_PIN);

	/* Read packet HW timestamp from FIFO */
	val = ocelot_read(ocelot, SYS_PTP_TXSTAMP);
	ts->tv_nsec = SYS_PTP_TXSTAMP_PTP_TXSTAMP(val);

	/* Sec has incremented since the ts was registered */
	if ((ts->tv_sec & 0x1) != !!(val & SYS_PTP_TXSTAMP_PTP_TXSTAMP_SEC))
		ts->tv_sec--;

	spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
}
459 460 461 462 463 464

void ocelot_get_txtstamp(struct ocelot *ocelot)
{
	int budget = OCELOT_PTP_QUEUE_SZ;

	while (budget--) {
465
		struct sk_buff *skb, *skb_tmp, *skb_match = NULL;
466 467 468
		struct skb_shared_hwtstamps shhwtstamps;
		struct ocelot_port *port;
		struct timespec64 ts;
469
		unsigned long flags;
470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486
		u32 val, id, txport;

		val = ocelot_read(ocelot, SYS_PTP_STATUS);

		/* Check if a timestamp can be retrieved */
		if (!(val & SYS_PTP_STATUS_PTP_MESS_VLD))
			break;

		WARN_ON(val & SYS_PTP_STATUS_PTP_OVFL);

		/* Retrieve the ts ID and Tx port */
		id = SYS_PTP_STATUS_PTP_MESS_ID_X(val);
		txport = SYS_PTP_STATUS_PTP_MESS_TXPORT_X(val);

		/* Retrieve its associated skb */
		port = ocelot->ports[txport];

487
		spin_lock_irqsave(&port->tx_skbs.lock, flags);
488

489 490 491 492 493
		skb_queue_walk_safe(&port->tx_skbs, skb, skb_tmp) {
			if (skb->cb[0] != id)
				continue;
			__skb_unlink(skb, &port->tx_skbs);
			skb_match = skb;
494
			break;
495 496
		}

497 498
		spin_unlock_irqrestore(&port->tx_skbs.lock, flags);

499 500
		/* Get the h/w timestamp */
		ocelot_get_hwtimestamp(ocelot, &ts);
501

502
		if (unlikely(!skb_match))
503 504 505 506 507
			continue;

		/* Set the timestamp into the skb */
		memset(&shhwtstamps, 0, sizeof(shhwtstamps));
		shhwtstamps.hwtstamp = ktime_set(ts.tv_sec, ts.tv_nsec);
508
		skb_complete_tx_timestamp(skb_match, &shhwtstamps);
509 510 511

		/* Next ts */
		ocelot_write(ocelot, SYS_PTP_NXT_PTP_NXT, SYS_PTP_NXT);
512 513 514
	}
}
EXPORT_SYMBOL(ocelot_get_txtstamp);
515

516
int ocelot_fdb_add(struct ocelot *ocelot, int port,
517
		   const unsigned char *addr, u16 vid)
518
{
519
	struct ocelot_port *ocelot_port = ocelot->ports[port];
520 521 522 523
	int pgid = port;

	if (port == ocelot->npi)
		pgid = PGID_CPU;
524

525
	if (!vid) {
526
		if (!ocelot_port->vlan_aware)
527 528 529 530
			/* If the bridge is not VLAN aware and no VID was
			 * provided, set it to pvid to ensure the MAC entry
			 * matches incoming untagged packets
			 */
531
			vid = ocelot_port->pvid;
532 533 534 535 536 537 538
		else
			/* If the bridge is VLAN aware a VID must be provided as
			 * otherwise the learnt entry wouldn't match any frame.
			 */
			return -EINVAL;
	}

539
	return ocelot_mact_learn(ocelot, pgid, addr, vid, ENTRYTYPE_LOCKED);
540
}
541
EXPORT_SYMBOL(ocelot_fdb_add);
542

543 544
int ocelot_fdb_del(struct ocelot *ocelot, int port,
		   const unsigned char *addr, u16 vid)
545
{
546 547
	return ocelot_mact_forget(ocelot, addr, vid);
}
548
EXPORT_SYMBOL(ocelot_fdb_del);
549

550 551
int ocelot_port_fdb_do_dump(const unsigned char *addr, u16 vid,
			    bool is_static, void *data)
552
{
553
	struct ocelot_dump_ctx *dump = data;
554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573
	u32 portid = NETLINK_CB(dump->cb->skb).portid;
	u32 seq = dump->cb->nlh->nlmsg_seq;
	struct nlmsghdr *nlh;
	struct ndmsg *ndm;

	if (dump->idx < dump->cb->args[2])
		goto skip;

	nlh = nlmsg_put(dump->skb, portid, seq, RTM_NEWNEIGH,
			sizeof(*ndm), NLM_F_MULTI);
	if (!nlh)
		return -EMSGSIZE;

	ndm = nlmsg_data(nlh);
	ndm->ndm_family  = AF_BRIDGE;
	ndm->ndm_pad1    = 0;
	ndm->ndm_pad2    = 0;
	ndm->ndm_flags   = NTF_SELF;
	ndm->ndm_type    = 0;
	ndm->ndm_ifindex = dump->dev->ifindex;
574
	ndm->ndm_state   = is_static ? NUD_NOARP : NUD_REACHABLE;
575

576
	if (nla_put(dump->skb, NDA_LLADDR, ETH_ALEN, addr))
577 578
		goto nla_put_failure;

579
	if (vid && nla_put_u16(dump->skb, NDA_VLAN, vid))
580 581 582 583 584 585 586 587 588 589 590 591
		goto nla_put_failure;

	nlmsg_end(dump->skb, nlh);

skip:
	dump->idx++;
	return 0;

nla_put_failure:
	nlmsg_cancel(dump->skb, nlh);
	return -EMSGSIZE;
}
592
EXPORT_SYMBOL(ocelot_port_fdb_do_dump);
593

594 595
static int ocelot_mact_read(struct ocelot *ocelot, int port, int row, int col,
			    struct ocelot_mact_entry *entry)
596 597
{
	u32 val, dst, macl, mach;
598
	char mac[ETH_ALEN];
599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620

	/* Set row and column to read from */
	ocelot_field_write(ocelot, ANA_TABLES_MACTINDX_M_INDEX, row);
	ocelot_field_write(ocelot, ANA_TABLES_MACTINDX_BUCKET, col);

	/* Issue a read command */
	ocelot_write(ocelot,
		     ANA_TABLES_MACACCESS_MAC_TABLE_CMD(MACACCESS_CMD_READ),
		     ANA_TABLES_MACACCESS);

	if (ocelot_mact_wait_for_completion(ocelot))
		return -ETIMEDOUT;

	/* Read the entry flags */
	val = ocelot_read(ocelot, ANA_TABLES_MACACCESS);
	if (!(val & ANA_TABLES_MACACCESS_VALID))
		return -EINVAL;

	/* If the entry read has another port configured as its destination,
	 * do not report it.
	 */
	dst = (val & ANA_TABLES_MACACCESS_DEST_IDX_M) >> 3;
621
	if (dst != port)
622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640
		return -EINVAL;

	/* Get the entry's MAC address and VLAN id */
	macl = ocelot_read(ocelot, ANA_TABLES_MACLDATA);
	mach = ocelot_read(ocelot, ANA_TABLES_MACHDATA);

	mac[0] = (mach >> 8)  & 0xff;
	mac[1] = (mach >> 0)  & 0xff;
	mac[2] = (macl >> 24) & 0xff;
	mac[3] = (macl >> 16) & 0xff;
	mac[4] = (macl >> 8)  & 0xff;
	mac[5] = (macl >> 0)  & 0xff;

	entry->vid = (mach >> 16) & 0xfff;
	ether_addr_copy(entry->mac, mac);

	return 0;
}

641 642
int ocelot_fdb_dump(struct ocelot *ocelot, int port,
		    dsa_fdb_dump_cb_t *cb, void *data)
643
{
644
	int i, j;
645

646 647
	/* Loop through all the mac tables entries. */
	for (i = 0; i < ocelot->num_mact_rows; i++) {
648
		for (j = 0; j < 4; j++) {
649 650 651 652 653
			struct ocelot_mact_entry entry;
			bool is_static;
			int ret;

			ret = ocelot_mact_read(ocelot, port, i, j, &entry);
654 655 656 657 658 659
			/* If the entry is invalid (wrong port, invalid...),
			 * skip it.
			 */
			if (ret == -EINVAL)
				continue;
			else if (ret)
660 661 662
				return ret;

			is_static = (entry.type == ENTRYTYPE_LOCKED);
663

664
			ret = cb(entry.mac, entry.vid, is_static, data);
665
			if (ret)
666
				return ret;
667 668 669
		}
	}

670 671
	return 0;
}
672
EXPORT_SYMBOL(ocelot_fdb_dump);
673

674
int ocelot_hwstamp_get(struct ocelot *ocelot, int port, struct ifreq *ifr)
675 676 677 678
{
	return copy_to_user(ifr->ifr_data, &ocelot->hwtstamp_config,
			    sizeof(ocelot->hwtstamp_config)) ? -EFAULT : 0;
}
679
EXPORT_SYMBOL(ocelot_hwstamp_get);
680

681
int ocelot_hwstamp_set(struct ocelot *ocelot, int port, struct ifreq *ifr)
682
{
683
	struct ocelot_port *ocelot_port = ocelot->ports[port];
684 685 686 687 688 689 690 691 692 693 694 695
	struct hwtstamp_config cfg;

	if (copy_from_user(&cfg, ifr->ifr_data, sizeof(cfg)))
		return -EFAULT;

	/* reserved for future extensions */
	if (cfg.flags)
		return -EINVAL;

	/* Tx type sanity check */
	switch (cfg.tx_type) {
	case HWTSTAMP_TX_ON:
696
		ocelot_port->ptp_cmd = IFH_REW_OP_TWO_STEP_PTP;
697 698 699 700 701
		break;
	case HWTSTAMP_TX_ONESTEP_SYNC:
		/* IFH_REW_OP_ONE_STEP_PTP updates the correctional field, we
		 * need to update the origin time.
		 */
702
		ocelot_port->ptp_cmd = IFH_REW_OP_ORIGIN_PTP;
703 704
		break;
	case HWTSTAMP_TX_OFF:
705
		ocelot_port->ptp_cmd = 0;
706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743
		break;
	default:
		return -ERANGE;
	}

	mutex_lock(&ocelot->ptp_lock);

	switch (cfg.rx_filter) {
	case HWTSTAMP_FILTER_NONE:
		break;
	case HWTSTAMP_FILTER_ALL:
	case HWTSTAMP_FILTER_SOME:
	case HWTSTAMP_FILTER_PTP_V1_L4_EVENT:
	case HWTSTAMP_FILTER_PTP_V1_L4_SYNC:
	case HWTSTAMP_FILTER_PTP_V1_L4_DELAY_REQ:
	case HWTSTAMP_FILTER_NTP_ALL:
	case HWTSTAMP_FILTER_PTP_V2_L4_EVENT:
	case HWTSTAMP_FILTER_PTP_V2_L4_SYNC:
	case HWTSTAMP_FILTER_PTP_V2_L4_DELAY_REQ:
	case HWTSTAMP_FILTER_PTP_V2_L2_EVENT:
	case HWTSTAMP_FILTER_PTP_V2_L2_SYNC:
	case HWTSTAMP_FILTER_PTP_V2_L2_DELAY_REQ:
	case HWTSTAMP_FILTER_PTP_V2_EVENT:
	case HWTSTAMP_FILTER_PTP_V2_SYNC:
	case HWTSTAMP_FILTER_PTP_V2_DELAY_REQ:
		cfg.rx_filter = HWTSTAMP_FILTER_PTP_V2_EVENT;
		break;
	default:
		mutex_unlock(&ocelot->ptp_lock);
		return -ERANGE;
	}

	/* Commit back the result & save it */
	memcpy(&ocelot->hwtstamp_config, &cfg, sizeof(cfg));
	mutex_unlock(&ocelot->ptp_lock);

	return copy_to_user(ifr->ifr_data, &cfg, sizeof(cfg)) ? -EFAULT : 0;
}
744
EXPORT_SYMBOL(ocelot_hwstamp_set);
745

746
void ocelot_get_strings(struct ocelot *ocelot, int port, u32 sset, u8 *data)
747 748 749 750 751 752 753 754 755 756
{
	int i;

	if (sset != ETH_SS_STATS)
		return;

	for (i = 0; i < ocelot->num_stats; i++)
		memcpy(data + i * ETH_GSTRING_LEN, ocelot->stats_layout[i].name,
		       ETH_GSTRING_LEN);
}
757
EXPORT_SYMBOL(ocelot_get_strings);
758

759
static void ocelot_update_stats(struct ocelot *ocelot)
760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783
{
	int i, j;

	mutex_lock(&ocelot->stats_lock);

	for (i = 0; i < ocelot->num_phys_ports; i++) {
		/* Configure the port to read the stats from */
		ocelot_write(ocelot, SYS_STAT_CFG_STAT_VIEW(i), SYS_STAT_CFG);

		for (j = 0; j < ocelot->num_stats; j++) {
			u32 val;
			unsigned int idx = i * ocelot->num_stats + j;

			val = ocelot_read_rix(ocelot, SYS_COUNT_RX_OCTETS,
					      ocelot->stats_layout[j].offset);

			if (val < (ocelot->stats[idx] & U32_MAX))
				ocelot->stats[idx] += (u64)1 << 32;

			ocelot->stats[idx] = (ocelot->stats[idx] &
					      ~(u64)U32_MAX) + val;
		}
	}

784 785 786 787 788 789 790 791 792 793 794
	mutex_unlock(&ocelot->stats_lock);
}

static void ocelot_check_stats_work(struct work_struct *work)
{
	struct delayed_work *del_work = to_delayed_work(work);
	struct ocelot *ocelot = container_of(del_work, struct ocelot,
					     stats_work);

	ocelot_update_stats(ocelot);

795 796 797 798
	queue_delayed_work(ocelot->stats_queue, &ocelot->stats_work,
			   OCELOT_STATS_CHECK_DELAY);
}

799
void ocelot_get_ethtool_stats(struct ocelot *ocelot, int port, u64 *data)
800 801 802 803
{
	int i;

	/* check and update now */
804
	ocelot_update_stats(ocelot);
805 806 807

	/* Copy all counters */
	for (i = 0; i < ocelot->num_stats; i++)
808
		*data++ = ocelot->stats[port * ocelot->num_stats + i];
809
}
810
EXPORT_SYMBOL(ocelot_get_ethtool_stats);
811

812
int ocelot_get_sset_count(struct ocelot *ocelot, int port, int sset)
813
{
814 815
	if (sset != ETH_SS_STATS)
		return -EOPNOTSUPP;
816

817 818
	return ocelot->num_stats;
}
819
EXPORT_SYMBOL(ocelot_get_sset_count);
820

821 822
int ocelot_get_ts_info(struct ocelot *ocelot, int port,
		       struct ethtool_ts_info *info)
823
{
824 825
	info->phc_index = ocelot->ptp_clock ?
			  ptp_clock_index(ocelot->ptp_clock) : -1;
826 827 828 829 830 831
	if (info->phc_index == -1) {
		info->so_timestamping |= SOF_TIMESTAMPING_TX_SOFTWARE |
					 SOF_TIMESTAMPING_RX_SOFTWARE |
					 SOF_TIMESTAMPING_SOFTWARE;
		return 0;
	}
832 833 834 835 836 837 838 839 840 841 842 843
	info->so_timestamping |= SOF_TIMESTAMPING_TX_SOFTWARE |
				 SOF_TIMESTAMPING_RX_SOFTWARE |
				 SOF_TIMESTAMPING_SOFTWARE |
				 SOF_TIMESTAMPING_TX_HARDWARE |
				 SOF_TIMESTAMPING_RX_HARDWARE |
				 SOF_TIMESTAMPING_RAW_HARDWARE;
	info->tx_types = BIT(HWTSTAMP_TX_OFF) | BIT(HWTSTAMP_TX_ON) |
			 BIT(HWTSTAMP_TX_ONESTEP_SYNC);
	info->rx_filters = BIT(HWTSTAMP_FILTER_NONE) | BIT(HWTSTAMP_FILTER_ALL);

	return 0;
}
844
EXPORT_SYMBOL(ocelot_get_ts_info);
845

846
void ocelot_bridge_stp_state_set(struct ocelot *ocelot, int port, u8 state)
847 848
{
	u32 port_cfg;
849
	int p, i;
850

851 852
	if (!(BIT(port) & ocelot->bridge_mask))
		return;
853

854
	port_cfg = ocelot_read_gix(ocelot, ANA_PORT_PORT_CFG, port);
855 856 857

	switch (state) {
	case BR_STATE_FORWARDING:
858
		ocelot->bridge_fwd_mask |= BIT(port);
859
		fallthrough;
860 861 862 863 864 865
	case BR_STATE_LEARNING:
		port_cfg |= ANA_PORT_PORT_CFG_LEARN_ENA;
		break;

	default:
		port_cfg &= ~ANA_PORT_PORT_CFG_LEARN_ENA;
866
		ocelot->bridge_fwd_mask &= ~BIT(port);
867 868 869
		break;
	}

870
	ocelot_write_gix(ocelot, port_cfg, ANA_PORT_PORT_CFG, port);
871 872 873 874

	/* Apply FWD mask. The loop is needed to add/remove the current port as
	 * a source for the other ports.
	 */
875
	for (p = 0; p < ocelot->num_phys_ports; p++) {
876
		if (ocelot->bridge_fwd_mask & BIT(p)) {
877
			unsigned long mask = ocelot->bridge_fwd_mask & ~BIT(p);
878 879 880 881 882 883 884

			for (i = 0; i < ocelot->num_phys_ports; i++) {
				unsigned long bond_mask = ocelot->lags[i];

				if (!bond_mask)
					continue;

885
				if (bond_mask & BIT(p)) {
886 887 888 889 890
					mask &= ~bond_mask;
					break;
				}
			}

891
			ocelot_write_rix(ocelot, mask,
892
					 ANA_PGID_PGID, PGID_SRC + p);
893
		} else {
894
			ocelot_write_rix(ocelot, 0,
895
					 ANA_PGID_PGID, PGID_SRC + p);
896 897
		}
	}
898
}
899
EXPORT_SYMBOL(ocelot_bridge_stp_state_set);
900

901
void ocelot_set_ageing_time(struct ocelot *ocelot, unsigned int msecs)
902
{
903 904 905 906 907 908 909 910 911
	unsigned int age_period = ANA_AUTOAGE_AGE_PERIOD(msecs / 2000);

	/* Setting AGE_PERIOD to zero effectively disables automatic aging,
	 * which is clearly not what our intention is. So avoid that.
	 */
	if (!age_period)
		age_period = 1;

	ocelot_rmw(ocelot, age_period, ANA_AUTOAGE_AGE_PERIOD_M, ANA_AUTOAGE);
912
}
913
EXPORT_SYMBOL(ocelot_set_ageing_time);
914 915 916 917 918 919 920 921 922 923 924 925 926 927 928

static struct ocelot_multicast *ocelot_multicast_get(struct ocelot *ocelot,
						     const unsigned char *addr,
						     u16 vid)
{
	struct ocelot_multicast *mc;

	list_for_each_entry(mc, &ocelot->multicast, list) {
		if (ether_addr_equal(mc->addr, addr) && mc->vid == vid)
			return mc;
	}

	return NULL;
}

929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985
static enum macaccess_entry_type ocelot_classify_mdb(const unsigned char *addr)
{
	if (addr[0] == 0x01 && addr[1] == 0x00 && addr[2] == 0x5e)
		return ENTRYTYPE_MACv4;
	if (addr[0] == 0x33 && addr[1] == 0x33)
		return ENTRYTYPE_MACv6;
	return ENTRYTYPE_NORMAL;
}

static int ocelot_mdb_get_pgid(struct ocelot *ocelot,
			       enum macaccess_entry_type entry_type)
{
	int pgid;

	/* According to VSC7514 datasheet 3.9.1.5 IPv4 Multicast Entries and
	 * 3.9.1.6 IPv6 Multicast Entries, "Instead of a lookup in the
	 * destination mask table (PGID), the destination set is programmed as
	 * part of the entry MAC address.", and the DEST_IDX is set to 0.
	 */
	if (entry_type == ENTRYTYPE_MACv4 ||
	    entry_type == ENTRYTYPE_MACv6)
		return 0;

	for_each_nonreserved_multicast_dest_pgid(ocelot, pgid) {
		struct ocelot_multicast *mc;
		bool used = false;

		list_for_each_entry(mc, &ocelot->multicast, list) {
			if (mc->pgid == pgid) {
				used = true;
				break;
			}
		}

		if (!used)
			return pgid;
	}

	return -1;
}

static void ocelot_encode_ports_to_mdb(unsigned char *addr,
				       struct ocelot_multicast *mc,
				       enum macaccess_entry_type entry_type)
{
	memcpy(addr, mc->addr, ETH_ALEN);

	if (entry_type == ENTRYTYPE_MACv4) {
		addr[0] = 0;
		addr[1] = mc->ports >> 8;
		addr[2] = mc->ports & 0xff;
	} else if (entry_type == ENTRYTYPE_MACv6) {
		addr[0] = mc->ports >> 8;
		addr[1] = mc->ports & 0xff;
	}
}

986 987
int ocelot_port_mdb_add(struct ocelot *ocelot, int port,
			const struct switchdev_obj_port_mdb *mdb)
988
{
989
	struct ocelot_port *ocelot_port = ocelot->ports[port];
990
	enum macaccess_entry_type entry_type;
991
	unsigned char addr[ETH_ALEN];
992
	struct ocelot_multicast *mc;
993 994 995
	u16 vid = mdb->vid;
	bool new = false;

996 997 998
	if (port == ocelot->npi)
		port = ocelot->num_phys_ports;

999
	if (!vid)
1000
		vid = ocelot_port->pvid;
1001

1002 1003
	entry_type = ocelot_classify_mdb(mdb->addr);

1004 1005
	mc = ocelot_multicast_get(ocelot, mdb->addr, vid);
	if (!mc) {
1006 1007 1008 1009 1010 1011 1012 1013 1014
		int pgid = ocelot_mdb_get_pgid(ocelot, entry_type);

		if (pgid < 0) {
			dev_err(ocelot->dev,
				"No more PGIDs available for mdb %pM vid %d\n",
				mdb->addr, vid);
			return -ENOSPC;
		}

1015 1016 1017 1018 1019 1020
		mc = devm_kzalloc(ocelot->dev, sizeof(*mc), GFP_KERNEL);
		if (!mc)
			return -ENOMEM;

		memcpy(mc->addr, mdb->addr, ETH_ALEN);
		mc->vid = vid;
1021
		mc->pgid = pgid;
1022 1023 1024 1025 1026 1027

		list_add_tail(&mc->list, &ocelot->multicast);
		new = true;
	}

	if (!new) {
1028
		ocelot_encode_ports_to_mdb(addr, mc, entry_type);
1029 1030 1031
		ocelot_mact_forget(ocelot, addr, vid);
	}

1032
	mc->ports |= BIT(port);
1033
	ocelot_encode_ports_to_mdb(addr, mc, entry_type);
1034

1035
	return ocelot_mact_learn(ocelot, mc->pgid, addr, vid, entry_type);
1036
}
1037
EXPORT_SYMBOL(ocelot_port_mdb_add);
1038

1039 1040
int ocelot_port_mdb_del(struct ocelot *ocelot, int port,
			const struct switchdev_obj_port_mdb *mdb)
1041
{
1042
	struct ocelot_port *ocelot_port = ocelot->ports[port];
1043
	enum macaccess_entry_type entry_type;
1044
	unsigned char addr[ETH_ALEN];
1045
	struct ocelot_multicast *mc;
1046 1047
	u16 vid = mdb->vid;

1048 1049 1050
	if (port == ocelot->npi)
		port = ocelot->num_phys_ports;

1051
	if (!vid)
1052
		vid = ocelot_port->pvid;
1053 1054 1055 1056 1057

	mc = ocelot_multicast_get(ocelot, mdb->addr, vid);
	if (!mc)
		return -ENOENT;

1058 1059 1060
	entry_type = ocelot_classify_mdb(mdb->addr);

	ocelot_encode_ports_to_mdb(addr, mc, entry_type);
1061 1062
	ocelot_mact_forget(ocelot, addr, vid);

1063
	mc->ports &= ~BIT(port);
1064 1065 1066 1067 1068 1069
	if (!mc->ports) {
		list_del(&mc->list);
		devm_kfree(ocelot->dev, mc);
		return 0;
	}

1070
	ocelot_encode_ports_to_mdb(addr, mc, entry_type);
1071

1072
	return ocelot_mact_learn(ocelot, mc->pgid, addr, vid, entry_type);
1073
}
1074
EXPORT_SYMBOL(ocelot_port_mdb_del);
1075

1076 1077
int ocelot_port_bridge_join(struct ocelot *ocelot, int port,
			    struct net_device *bridge)
1078 1079 1080 1081 1082 1083 1084 1085 1086 1087
{
	if (!ocelot->bridge_mask) {
		ocelot->hw_bridge_dev = bridge;
	} else {
		if (ocelot->hw_bridge_dev != bridge)
			/* This is adding the port to a second bridge, this is
			 * unsupported */
			return -ENODEV;
	}

1088
	ocelot->bridge_mask |= BIT(port);
1089 1090 1091

	return 0;
}
1092
EXPORT_SYMBOL(ocelot_port_bridge_join);
1093

1094 1095
int ocelot_port_bridge_leave(struct ocelot *ocelot, int port,
			     struct net_device *bridge)
1096
{
1097
	ocelot->bridge_mask &= ~BIT(port);
1098 1099 1100

	if (!ocelot->bridge_mask)
		ocelot->hw_bridge_dev = NULL;
1101

1102 1103 1104
	ocelot_port_vlan_filtering(ocelot, port, 0);
	ocelot_port_set_pvid(ocelot, port, 0);
	return ocelot_port_set_native_vlan(ocelot, port, 0);
1105
}
1106
EXPORT_SYMBOL(ocelot_port_bridge_leave);
1107

1108 1109 1110 1111 1112
static void ocelot_set_aggr_pgids(struct ocelot *ocelot)
{
	int i, port, lag;

	/* Reset destination and aggregation PGIDS */
1113
	for_each_unicast_dest_pgid(ocelot, port)
1114 1115
		ocelot_write_rix(ocelot, BIT(port), ANA_PGID_PGID, port);

1116
	for_each_aggr_pgid(ocelot, i)
1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137
		ocelot_write_rix(ocelot, GENMASK(ocelot->num_phys_ports - 1, 0),
				 ANA_PGID_PGID, i);

	/* Now, set PGIDs for each LAG */
	for (lag = 0; lag < ocelot->num_phys_ports; lag++) {
		unsigned long bond_mask;
		int aggr_count = 0;
		u8 aggr_idx[16];

		bond_mask = ocelot->lags[lag];
		if (!bond_mask)
			continue;

		for_each_set_bit(port, &bond_mask, ocelot->num_phys_ports) {
			// Destination mask
			ocelot_write_rix(ocelot, bond_mask,
					 ANA_PGID_PGID, port);
			aggr_idx[aggr_count] = port;
			aggr_count++;
		}

1138
		for_each_aggr_pgid(ocelot, i) {
1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165
			u32 ac;

			ac = ocelot_read_rix(ocelot, ANA_PGID_PGID, i);
			ac &= ~bond_mask;
			ac |= BIT(aggr_idx[i % aggr_count]);
			ocelot_write_rix(ocelot, ac, ANA_PGID_PGID, i);
		}
	}
}

static void ocelot_setup_lag(struct ocelot *ocelot, int lag)
{
	unsigned long bond_mask = ocelot->lags[lag];
	unsigned int p;

	for_each_set_bit(p, &bond_mask, ocelot->num_phys_ports) {
		u32 port_cfg = ocelot_read_gix(ocelot, ANA_PORT_PORT_CFG, p);

		port_cfg &= ~ANA_PORT_PORT_CFG_PORTID_VAL_M;

		/* Use lag port as logical port for port i */
		ocelot_write_gix(ocelot, port_cfg |
				 ANA_PORT_PORT_CFG_PORTID_VAL(lag),
				 ANA_PORT_PORT_CFG, p);
	}
}

1166 1167
int ocelot_port_lag_join(struct ocelot *ocelot, int port,
			 struct net_device *bond)
1168 1169 1170
{
	struct net_device *ndev;
	u32 bond_mask = 0;
1171
	int lag, lp;
1172 1173 1174

	rcu_read_lock();
	for_each_netdev_in_bond_rcu(bond, ndev) {
1175
		struct ocelot_port_private *priv = netdev_priv(ndev);
1176

1177
		bond_mask |= BIT(priv->chip_port);
1178 1179 1180 1181 1182 1183 1184 1185
	}
	rcu_read_unlock();

	lp = __ffs(bond_mask);

	/* If the new port is the lowest one, use it as the logical port from
	 * now on
	 */
1186 1187 1188 1189
	if (port == lp) {
		lag = port;
		ocelot->lags[port] = bond_mask;
		bond_mask &= ~BIT(port);
1190 1191 1192 1193 1194 1195
		if (bond_mask) {
			lp = __ffs(bond_mask);
			ocelot->lags[lp] = 0;
		}
	} else {
		lag = lp;
1196
		ocelot->lags[lp] |= BIT(port);
1197 1198 1199 1200 1201 1202 1203
	}

	ocelot_setup_lag(ocelot, lag);
	ocelot_set_aggr_pgids(ocelot);

	return 0;
}
1204
EXPORT_SYMBOL(ocelot_port_lag_join);
1205

1206 1207
void ocelot_port_lag_leave(struct ocelot *ocelot, int port,
			   struct net_device *bond)
1208 1209 1210 1211 1212 1213
{
	u32 port_cfg;
	int i;

	/* Remove port from any lag */
	for (i = 0; i < ocelot->num_phys_ports; i++)
1214
		ocelot->lags[i] &= ~BIT(port);
1215 1216 1217 1218

	/* if it was the logical port of the lag, move the lag config to the
	 * next port
	 */
1219 1220
	if (ocelot->lags[port]) {
		int n = __ffs(ocelot->lags[port]);
1221

1222 1223
		ocelot->lags[n] = ocelot->lags[port];
		ocelot->lags[port] = 0;
1224 1225 1226 1227

		ocelot_setup_lag(ocelot, n);
	}

1228
	port_cfg = ocelot_read_gix(ocelot, ANA_PORT_PORT_CFG, port);
1229
	port_cfg &= ~ANA_PORT_PORT_CFG_PORTID_VAL_M;
1230 1231
	ocelot_write_gix(ocelot, port_cfg | ANA_PORT_PORT_CFG_PORTID_VAL(port),
			 ANA_PORT_PORT_CFG, port);
1232 1233 1234

	ocelot_set_aggr_pgids(ocelot);
}
1235
EXPORT_SYMBOL(ocelot_port_lag_leave);
1236

1237 1238
/* Configure the maximum SDU (L2 payload) on RX to the value specified in @sdu.
 * The length of VLAN tags is accounted for automatically via DEV_MAC_TAGS_CFG.
1239 1240 1241
 * In the special case that it's the NPI port that we're configuring, the
 * length of the tag and optional prefix needs to be accounted for privately,
 * in order to be able to sustain communication at the requested @sdu.
1242
 */
1243
void ocelot_port_set_maxlen(struct ocelot *ocelot, int port, size_t sdu)
1244 1245
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];
1246
	int maxlen = sdu + ETH_HLEN + ETH_FCS_LEN;
1247
	int pause_start, pause_stop;
1248
	int atop_wm;
1249

1250 1251 1252 1253 1254 1255 1256 1257 1258
	if (port == ocelot->npi) {
		maxlen += OCELOT_TAG_LEN;

		if (ocelot->inj_prefix == OCELOT_TAG_PREFIX_SHORT)
			maxlen += OCELOT_SHORT_PREFIX_LEN;
		else if (ocelot->inj_prefix == OCELOT_TAG_PREFIX_LONG)
			maxlen += OCELOT_LONG_PREFIX_LEN;
	}

1259
	ocelot_port_writel(ocelot_port, maxlen, DEV_MAC_MAXLEN_CFG);
1260

1261 1262 1263
	/* Set Pause watermark hysteresis */
	pause_start = 6 * maxlen / OCELOT_BUFFER_CELL_SZ;
	pause_stop = 4 * maxlen / OCELOT_BUFFER_CELL_SZ;
1264 1265 1266 1267
	ocelot_fields_write(ocelot, port, SYS_PAUSE_CFG_PAUSE_START,
			    pause_start);
	ocelot_fields_write(ocelot, port, SYS_PAUSE_CFG_PAUSE_STOP,
			    pause_stop);
1268 1269

	/* Tail dropping watermark */
1270 1271
	atop_wm = (ocelot->shared_queue_sz - 9 * maxlen) /
		   OCELOT_BUFFER_CELL_SZ;
1272
	ocelot_write_rix(ocelot, ocelot->ops->wm_enc(9 * maxlen),
1273
			 SYS_ATOP, port);
1274
	ocelot_write(ocelot, ocelot->ops->wm_enc(atop_wm), SYS_ATOP_TOT_CFG);
1275
}
1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293
EXPORT_SYMBOL(ocelot_port_set_maxlen);

int ocelot_get_max_mtu(struct ocelot *ocelot, int port)
{
	int max_mtu = 65535 - ETH_HLEN - ETH_FCS_LEN;

	if (port == ocelot->npi) {
		max_mtu -= OCELOT_TAG_LEN;

		if (ocelot->inj_prefix == OCELOT_TAG_PREFIX_SHORT)
			max_mtu -= OCELOT_SHORT_PREFIX_LEN;
		else if (ocelot->inj_prefix == OCELOT_TAG_PREFIX_LONG)
			max_mtu -= OCELOT_LONG_PREFIX_LEN;
	}

	return max_mtu;
}
EXPORT_SYMBOL(ocelot_get_max_mtu);
1294

1295
void ocelot_init_port(struct ocelot *ocelot, int port)
1296 1297 1298
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];

1299
	skb_queue_head_init(&ocelot_port->tx_skbs);
1300
	spin_lock_init(&ocelot_port->ts_id_lock);
1301 1302 1303

	/* Basic L2 initialization */

1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319
	/* Set MAC IFG Gaps
	 * FDX: TX_IFG = 5, RX_IFG1 = RX_IFG2 = 0
	 * !FDX: TX_IFG = 5, RX_IFG1 = RX_IFG2 = 5
	 */
	ocelot_port_writel(ocelot_port, DEV_MAC_IFG_CFG_TX_IFG(5),
			   DEV_MAC_IFG_CFG);

	/* Load seed (0) and set MAC HDX late collision  */
	ocelot_port_writel(ocelot_port, DEV_MAC_HDX_CFG_LATE_COL_POS(67) |
			   DEV_MAC_HDX_CFG_SEED_LOAD,
			   DEV_MAC_HDX_CFG);
	mdelay(1);
	ocelot_port_writel(ocelot_port, DEV_MAC_HDX_CFG_LATE_COL_POS(67),
			   DEV_MAC_HDX_CFG);

	/* Set Max Length and maximum tags allowed */
1320
	ocelot_port_set_maxlen(ocelot, port, ETH_DATA_LEN);
1321 1322
	ocelot_port_writel(ocelot_port, DEV_MAC_TAGS_CFG_TAG_ID(ETH_P_8021AD) |
			   DEV_MAC_TAGS_CFG_VLAN_AWR_ENA |
1323
			   DEV_MAC_TAGS_CFG_VLAN_DBL_AWR_ENA |
1324 1325 1326 1327 1328 1329 1330
			   DEV_MAC_TAGS_CFG_VLAN_LEN_AWR_ENA,
			   DEV_MAC_TAGS_CFG);

	/* Set SMAC of Pause frame (00:00:00:00:00:00) */
	ocelot_port_writel(ocelot_port, 0, DEV_MAC_FC_MAC_HIGH_CFG);
	ocelot_port_writel(ocelot_port, 0, DEV_MAC_FC_MAC_LOW_CFG);

1331
	/* Enable transmission of pause frames */
1332
	ocelot_fields_write(ocelot, port, SYS_PAUSE_CFG_PAUSE_ENA, 1);
1333

1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346
	/* Drop frames with multicast source address */
	ocelot_rmw_gix(ocelot, ANA_PORT_DROP_CFG_DROP_MC_SMAC_ENA,
		       ANA_PORT_DROP_CFG_DROP_MC_SMAC_ENA,
		       ANA_PORT_DROP_CFG, port);

	/* Set default VLAN and tag type to 8021Q. */
	ocelot_rmw_gix(ocelot, REW_PORT_VLAN_CFG_PORT_TPID(ETH_P_8021Q),
		       REW_PORT_VLAN_CFG_PORT_TPID_M,
		       REW_PORT_VLAN_CFG, port);

	/* Enable vcap lookups */
	ocelot_vcap_enable(ocelot, port);
}
1347
EXPORT_SYMBOL(ocelot_init_port);
1348

1349 1350 1351 1352 1353 1354 1355 1356 1357
/* Configure and enable the CPU port module, which is a set of queues.
 * If @npi contains a valid port index, the CPU port module is connected
 * to the Node Processor Interface (NPI). This is the mode through which
 * frames can be injected from and extracted to an external CPU,
 * over Ethernet.
 */
void ocelot_configure_cpu(struct ocelot *ocelot, int npi,
			  enum ocelot_tag_prefix injection,
			  enum ocelot_tag_prefix extraction)
1358
{
1359 1360
	int cpu = ocelot->num_phys_ports;

1361 1362 1363 1364
	ocelot->npi = npi;
	ocelot->inj_prefix = injection;
	ocelot->xtr_prefix = extraction;

1365
	/* The unicast destination PGID for the CPU port module is unused */
1366
	ocelot_write_rix(ocelot, 0, ANA_PGID_PGID, cpu);
1367 1368 1369 1370
	/* Instead set up a multicast destination PGID for traffic copied to
	 * the CPU. Whitelisted MAC addresses like the port netdevice MAC
	 * addresses will be copied to the CPU via this PGID.
	 */
1371 1372 1373 1374 1375
	ocelot_write_rix(ocelot, BIT(cpu), ANA_PGID_PGID, PGID_CPU);
	ocelot_write_gix(ocelot, ANA_PORT_PORT_CFG_RECV_ENA |
			 ANA_PORT_PORT_CFG_PORTID_VAL(cpu),
			 ANA_PORT_PORT_CFG, cpu);

1376
	if (npi >= 0 && npi < ocelot->num_phys_ports) {
1377
		ocelot_write(ocelot, QSYS_EXT_CPU_CFG_EXT_CPUQ_MSK_M |
1378
			     QSYS_EXT_CPU_CFG_EXT_CPU_PORT(npi),
1379
			     QSYS_EXT_CPU_CFG);
1380

1381
		/* Enable NPI port */
1382 1383
		ocelot_fields_write(ocelot, npi,
				    QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
1384
		/* NPI port Injection/Extraction configuration */
1385 1386 1387 1388
		ocelot_fields_write(ocelot, npi, SYS_PORT_MODE_INCL_XTR_HDR,
				    extraction);
		ocelot_fields_write(ocelot, npi, SYS_PORT_MODE_INCL_INJ_HDR,
				    injection);
1389 1390

		/* Disable transmission of pause frames */
1391
		ocelot_fields_write(ocelot, npi, SYS_PAUSE_CFG_PAUSE_ENA, 0);
1392 1393
	}

1394
	/* Enable CPU port module */
1395
	ocelot_fields_write(ocelot, cpu, QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
1396
	/* CPU port Injection/Extraction configuration */
1397 1398 1399 1400
	ocelot_fields_write(ocelot, cpu, SYS_PORT_MODE_INCL_XTR_HDR,
			    extraction);
	ocelot_fields_write(ocelot, cpu, SYS_PORT_MODE_INCL_INJ_HDR,
			    injection);
1401 1402 1403 1404 1405 1406 1407

	/* Configure the CPU port to be VLAN aware */
	ocelot_write_gix(ocelot, ANA_PORT_VLAN_CFG_VLAN_VID(0) |
				 ANA_PORT_VLAN_CFG_VLAN_AWARE_ENA |
				 ANA_PORT_VLAN_CFG_VLAN_POP_CNT(1),
			 ANA_PORT_VLAN_CFG, cpu);
}
1408
EXPORT_SYMBOL(ocelot_configure_cpu);
1409

1410 1411 1412
int ocelot_init(struct ocelot *ocelot)
{
	char queue_name[32];
1413 1414
	int i, ret;
	u32 port;
1415

1416 1417 1418 1419 1420 1421 1422 1423
	if (ocelot->ops->reset) {
		ret = ocelot->ops->reset(ocelot);
		if (ret) {
			dev_err(ocelot->dev, "Switch reset failed\n");
			return ret;
		}
	}

1424 1425 1426 1427 1428
	ocelot->lags = devm_kcalloc(ocelot->dev, ocelot->num_phys_ports,
				    sizeof(u32), GFP_KERNEL);
	if (!ocelot->lags)
		return -ENOMEM;

1429 1430 1431 1432 1433 1434 1435
	ocelot->stats = devm_kcalloc(ocelot->dev,
				     ocelot->num_phys_ports * ocelot->num_stats,
				     sizeof(u64), GFP_KERNEL);
	if (!ocelot->stats)
		return -ENOMEM;

	mutex_init(&ocelot->stats_lock);
1436 1437
	mutex_init(&ocelot->ptp_lock);
	spin_lock_init(&ocelot->ptp_clock_lock);
1438 1439 1440 1441 1442 1443
	snprintf(queue_name, sizeof(queue_name), "%s-stats",
		 dev_name(ocelot->dev));
	ocelot->stats_queue = create_singlethread_workqueue(queue_name);
	if (!ocelot->stats_queue)
		return -ENOMEM;

1444
	INIT_LIST_HEAD(&ocelot->multicast);
1445 1446
	ocelot_mact_init(ocelot);
	ocelot_vlan_init(ocelot);
1447
	ocelot_vcap_init(ocelot);
1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500 1501 1502

	for (port = 0; port < ocelot->num_phys_ports; port++) {
		/* Clear all counters (5 groups) */
		ocelot_write(ocelot, SYS_STAT_CFG_STAT_VIEW(port) |
				     SYS_STAT_CFG_STAT_CLEAR_SHOT(0x7f),
			     SYS_STAT_CFG);
	}

	/* Only use S-Tag */
	ocelot_write(ocelot, ETH_P_8021AD, SYS_VLAN_ETYPE_CFG);

	/* Aggregation mode */
	ocelot_write(ocelot, ANA_AGGR_CFG_AC_SMAC_ENA |
			     ANA_AGGR_CFG_AC_DMAC_ENA |
			     ANA_AGGR_CFG_AC_IP4_SIPDIP_ENA |
			     ANA_AGGR_CFG_AC_IP4_TCPUDP_ENA, ANA_AGGR_CFG);

	/* Set MAC age time to default value. The entry is aged after
	 * 2*AGE_PERIOD
	 */
	ocelot_write(ocelot,
		     ANA_AUTOAGE_AGE_PERIOD(BR_DEFAULT_AGEING_TIME / 2 / HZ),
		     ANA_AUTOAGE);

	/* Disable learning for frames discarded by VLAN ingress filtering */
	regmap_field_write(ocelot->regfields[ANA_ADVLEARN_VLAN_CHK], 1);

	/* Setup frame ageing - fixed value "2 sec" - in 6.5 us units */
	ocelot_write(ocelot, SYS_FRM_AGING_AGE_TX_ENA |
		     SYS_FRM_AGING_MAX_AGE(307692), SYS_FRM_AGING);

	/* Setup flooding PGIDs */
	ocelot_write_rix(ocelot, ANA_FLOODING_FLD_MULTICAST(PGID_MC) |
			 ANA_FLOODING_FLD_BROADCAST(PGID_MC) |
			 ANA_FLOODING_FLD_UNICAST(PGID_UC),
			 ANA_FLOODING, 0);
	ocelot_write(ocelot, ANA_FLOODING_IPMC_FLD_MC6_DATA(PGID_MCIPV6) |
		     ANA_FLOODING_IPMC_FLD_MC6_CTRL(PGID_MC) |
		     ANA_FLOODING_IPMC_FLD_MC4_DATA(PGID_MCIPV4) |
		     ANA_FLOODING_IPMC_FLD_MC4_CTRL(PGID_MC),
		     ANA_FLOODING_IPMC);

	for (port = 0; port < ocelot->num_phys_ports; port++) {
		/* Transmit the frame to the local port. */
		ocelot_write_rix(ocelot, BIT(port), ANA_PGID_PGID, port);
		/* Do not forward BPDU frames to the front ports. */
		ocelot_write_gix(ocelot,
				 ANA_PORT_CPU_FWD_BPDU_CFG_BPDU_REDIR_ENA(0xffff),
				 ANA_PORT_CPU_FWD_BPDU_CFG,
				 port);
		/* Ensure bridging is disabled */
		ocelot_write_rix(ocelot, 0, ANA_PGID_PGID, PGID_SRC + port);
	}

	/* Allow broadcast MAC frames. */
1503
	for_each_nonreserved_multicast_dest_pgid(ocelot, i) {
1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532 1533 1534
		u32 val = ANA_PGID_PGID_PGID(GENMASK(ocelot->num_phys_ports - 1, 0));

		ocelot_write_rix(ocelot, val, ANA_PGID_PGID, i);
	}
	ocelot_write_rix(ocelot,
			 ANA_PGID_PGID_PGID(GENMASK(ocelot->num_phys_ports, 0)),
			 ANA_PGID_PGID, PGID_MC);
	ocelot_write_rix(ocelot, 0, ANA_PGID_PGID, PGID_MCIPV4);
	ocelot_write_rix(ocelot, 0, ANA_PGID_PGID, PGID_MCIPV6);

	/* Allow manual injection via DEVCPU_QS registers, and byte swap these
	 * registers endianness.
	 */
	ocelot_write_rix(ocelot, QS_INJ_GRP_CFG_BYTE_SWAP |
			 QS_INJ_GRP_CFG_MODE(1), QS_INJ_GRP_CFG, 0);
	ocelot_write_rix(ocelot, QS_XTR_GRP_CFG_BYTE_SWAP |
			 QS_XTR_GRP_CFG_MODE(1), QS_XTR_GRP_CFG, 0);
	ocelot_write(ocelot, ANA_CPUQ_CFG_CPUQ_MIRROR(2) |
		     ANA_CPUQ_CFG_CPUQ_LRN(2) |
		     ANA_CPUQ_CFG_CPUQ_MAC_COPY(2) |
		     ANA_CPUQ_CFG_CPUQ_SRC_COPY(2) |
		     ANA_CPUQ_CFG_CPUQ_LOCKED_PORTMOVE(2) |
		     ANA_CPUQ_CFG_CPUQ_ALLBRIDGE(6) |
		     ANA_CPUQ_CFG_CPUQ_IPMC_CTRL(6) |
		     ANA_CPUQ_CFG_CPUQ_IGMP(6) |
		     ANA_CPUQ_CFG_CPUQ_MLD(6), ANA_CPUQ_CFG);
	for (i = 0; i < 16; i++)
		ocelot_write_rix(ocelot, ANA_CPUQ_8021_CFG_CPUQ_GARP_VAL(6) |
				 ANA_CPUQ_8021_CFG_CPUQ_BPDU_VAL(6),
				 ANA_CPUQ_8021_CFG, i);

1535
	INIT_DELAYED_WORK(&ocelot->stats_work, ocelot_check_stats_work);
1536 1537
	queue_delayed_work(ocelot->stats_queue, &ocelot->stats_work,
			   OCELOT_STATS_CHECK_DELAY);
1538

1539 1540 1541 1542 1543 1544
	return 0;
}
EXPORT_SYMBOL(ocelot_init);

void ocelot_deinit(struct ocelot *ocelot)
{
1545
	cancel_delayed_work(&ocelot->stats_work);
1546 1547 1548 1549 1550
	destroy_workqueue(ocelot->stats_queue);
	mutex_destroy(&ocelot->stats_lock);
}
EXPORT_SYMBOL(ocelot_deinit);

1551 1552 1553 1554 1555 1556 1557 1558
void ocelot_deinit_port(struct ocelot *ocelot, int port)
{
	struct ocelot_port *ocelot_port = ocelot->ports[port];

	skb_queue_purge(&ocelot_port->tx_skbs);
}
EXPORT_SYMBOL(ocelot_deinit_port);

1559
MODULE_LICENSE("Dual MIT/GPL");