162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Broadcom STB ASP 2.0 Driver
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (c) 2023 Broadcom
662306a36Sopenharmony_ci */
762306a36Sopenharmony_ci#include <linux/etherdevice.h>
862306a36Sopenharmony_ci#include <linux/if_vlan.h>
962306a36Sopenharmony_ci#include <linux/init.h>
1062306a36Sopenharmony_ci#include <linux/interrupt.h>
1162306a36Sopenharmony_ci#include <linux/module.h>
1262306a36Sopenharmony_ci#include <linux/kernel.h>
1362306a36Sopenharmony_ci#include <linux/platform_device.h>
1462306a36Sopenharmony_ci#include <linux/of.h>
1562306a36Sopenharmony_ci#include <linux/of_address.h>
1662306a36Sopenharmony_ci#include <linux/of_platform.h>
1762306a36Sopenharmony_ci#include <linux/clk.h>
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci#include "bcmasp.h"
2062306a36Sopenharmony_ci#include "bcmasp_intf_defs.h"
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic void _intr2_mask_clear(struct bcmasp_priv *priv, u32 mask)
2362306a36Sopenharmony_ci{
2462306a36Sopenharmony_ci	intr2_core_wl(priv, mask, ASP_INTR2_MASK_CLEAR);
2562306a36Sopenharmony_ci	priv->irq_mask &= ~mask;
2662306a36Sopenharmony_ci}
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_cistatic void _intr2_mask_set(struct bcmasp_priv *priv, u32 mask)
2962306a36Sopenharmony_ci{
3062306a36Sopenharmony_ci	intr2_core_wl(priv, mask, ASP_INTR2_MASK_SET);
3162306a36Sopenharmony_ci	priv->irq_mask |= mask;
3262306a36Sopenharmony_ci}
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_civoid bcmasp_enable_tx_irq(struct bcmasp_intf *intf, int en)
3562306a36Sopenharmony_ci{
3662306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci	if (en)
3962306a36Sopenharmony_ci		_intr2_mask_clear(priv, ASP_INTR2_TX_DESC(intf->channel));
4062306a36Sopenharmony_ci	else
4162306a36Sopenharmony_ci		_intr2_mask_set(priv, ASP_INTR2_TX_DESC(intf->channel));
4262306a36Sopenharmony_ci}
4362306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(bcmasp_enable_tx_irq);
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_civoid bcmasp_enable_rx_irq(struct bcmasp_intf *intf, int en)
4662306a36Sopenharmony_ci{
4762306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	if (en)
5062306a36Sopenharmony_ci		_intr2_mask_clear(priv, ASP_INTR2_RX_ECH(intf->channel));
5162306a36Sopenharmony_ci	else
5262306a36Sopenharmony_ci		_intr2_mask_set(priv, ASP_INTR2_RX_ECH(intf->channel));
5362306a36Sopenharmony_ci}
5462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(bcmasp_enable_rx_irq);
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_cistatic void bcmasp_intr2_mask_set_all(struct bcmasp_priv *priv)
5762306a36Sopenharmony_ci{
5862306a36Sopenharmony_ci	_intr2_mask_set(priv, 0xffffffff);
5962306a36Sopenharmony_ci	priv->irq_mask = 0xffffffff;
6062306a36Sopenharmony_ci}
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_cistatic void bcmasp_intr2_clear_all(struct bcmasp_priv *priv)
6362306a36Sopenharmony_ci{
6462306a36Sopenharmony_ci	intr2_core_wl(priv, 0xffffffff, ASP_INTR2_CLEAR);
6562306a36Sopenharmony_ci}
6662306a36Sopenharmony_ci
6762306a36Sopenharmony_cistatic void bcmasp_intr2_handling(struct bcmasp_intf *intf, u32 status)
6862306a36Sopenharmony_ci{
6962306a36Sopenharmony_ci	if (status & ASP_INTR2_RX_ECH(intf->channel)) {
7062306a36Sopenharmony_ci		if (likely(napi_schedule_prep(&intf->rx_napi))) {
7162306a36Sopenharmony_ci			bcmasp_enable_rx_irq(intf, 0);
7262306a36Sopenharmony_ci			__napi_schedule_irqoff(&intf->rx_napi);
7362306a36Sopenharmony_ci		}
7462306a36Sopenharmony_ci	}
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_ci	if (status & ASP_INTR2_TX_DESC(intf->channel)) {
7762306a36Sopenharmony_ci		if (likely(napi_schedule_prep(&intf->tx_napi))) {
7862306a36Sopenharmony_ci			bcmasp_enable_tx_irq(intf, 0);
7962306a36Sopenharmony_ci			__napi_schedule_irqoff(&intf->tx_napi);
8062306a36Sopenharmony_ci		}
8162306a36Sopenharmony_ci	}
8262306a36Sopenharmony_ci}
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_cistatic irqreturn_t bcmasp_isr(int irq, void *data)
8562306a36Sopenharmony_ci{
8662306a36Sopenharmony_ci	struct bcmasp_priv *priv = data;
8762306a36Sopenharmony_ci	struct bcmasp_intf *intf;
8862306a36Sopenharmony_ci	u32 status;
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	status = intr2_core_rl(priv, ASP_INTR2_STATUS) &
9162306a36Sopenharmony_ci		~intr2_core_rl(priv, ASP_INTR2_MASK_STATUS);
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci	intr2_core_wl(priv, status, ASP_INTR2_CLEAR);
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_ci	if (unlikely(status == 0)) {
9662306a36Sopenharmony_ci		dev_warn(&priv->pdev->dev, "l2 spurious interrupt\n");
9762306a36Sopenharmony_ci		return IRQ_NONE;
9862306a36Sopenharmony_ci	}
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_ci	/* Handle intferfaces */
10162306a36Sopenharmony_ci	list_for_each_entry(intf, &priv->intfs, list)
10262306a36Sopenharmony_ci		bcmasp_intr2_handling(intf, status);
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	return IRQ_HANDLED;
10562306a36Sopenharmony_ci}
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_civoid bcmasp_flush_rx_port(struct bcmasp_intf *intf)
10862306a36Sopenharmony_ci{
10962306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
11062306a36Sopenharmony_ci	u32 mask;
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	switch (intf->port) {
11362306a36Sopenharmony_ci	case 0:
11462306a36Sopenharmony_ci		mask = ASP_CTRL_UMAC0_FLUSH_MASK;
11562306a36Sopenharmony_ci		break;
11662306a36Sopenharmony_ci	case 1:
11762306a36Sopenharmony_ci		mask = ASP_CTRL_UMAC1_FLUSH_MASK;
11862306a36Sopenharmony_ci		break;
11962306a36Sopenharmony_ci	case 2:
12062306a36Sopenharmony_ci		mask = ASP_CTRL_SPB_FLUSH_MASK;
12162306a36Sopenharmony_ci		break;
12262306a36Sopenharmony_ci	default:
12362306a36Sopenharmony_ci		/* Not valid port */
12462306a36Sopenharmony_ci		return;
12562306a36Sopenharmony_ci	}
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ci	rx_ctrl_core_wl(priv, mask, priv->hw_info->rx_ctrl_flush);
12862306a36Sopenharmony_ci}
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_cistatic void bcmasp_netfilt_hw_en_wake(struct bcmasp_priv *priv,
13162306a36Sopenharmony_ci				      struct bcmasp_net_filter *nfilt)
13262306a36Sopenharmony_ci{
13362306a36Sopenharmony_ci	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_OFFSET_L3_1(64),
13462306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_OFFSET(nfilt->hw_index));
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_OFFSET_L2(32) |
13762306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_OFFSET_L3_0(32) |
13862306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_OFFSET_L3_1(96) |
13962306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_OFFSET_L4(32),
14062306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_OFFSET(nfilt->hw_index + 1));
14162306a36Sopenharmony_ci
14262306a36Sopenharmony_ci	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_CFG_CH(nfilt->port + 8) |
14362306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_EN |
14462306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L2_EN |
14562306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L3_EN |
14662306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L4_EN |
14762306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L3_FRM(2) |
14862306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L4_FRM(2) |
14962306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_UMC(nfilt->port),
15062306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG(nfilt->hw_index));
15162306a36Sopenharmony_ci
15262306a36Sopenharmony_ci	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_CFG_CH(nfilt->port + 8) |
15362306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_EN |
15462306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L2_EN |
15562306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L3_EN |
15662306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L4_EN |
15762306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L3_FRM(2) |
15862306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_L4_FRM(2) |
15962306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG_UMC(nfilt->port),
16062306a36Sopenharmony_ci			  ASP_RX_FILTER_NET_CFG(nfilt->hw_index + 1));
16162306a36Sopenharmony_ci}
16262306a36Sopenharmony_ci
16362306a36Sopenharmony_ci#define MAX_WAKE_FILTER_SIZE		256
16462306a36Sopenharmony_cienum asp_netfilt_reg_type {
16562306a36Sopenharmony_ci	ASP_NETFILT_MATCH = 0,
16662306a36Sopenharmony_ci	ASP_NETFILT_MASK,
16762306a36Sopenharmony_ci	ASP_NETFILT_MAX
16862306a36Sopenharmony_ci};
16962306a36Sopenharmony_ci
17062306a36Sopenharmony_cistatic int bcmasp_netfilt_get_reg_offset(struct bcmasp_priv *priv,
17162306a36Sopenharmony_ci					 struct bcmasp_net_filter *nfilt,
17262306a36Sopenharmony_ci					 enum asp_netfilt_reg_type reg_type,
17362306a36Sopenharmony_ci					 u32 offset)
17462306a36Sopenharmony_ci{
17562306a36Sopenharmony_ci	u32 block_index, filter_sel;
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ci	if (offset < 32) {
17862306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L2;
17962306a36Sopenharmony_ci		filter_sel = nfilt->hw_index;
18062306a36Sopenharmony_ci	} else if (offset < 64) {
18162306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L2;
18262306a36Sopenharmony_ci		filter_sel = nfilt->hw_index + 1;
18362306a36Sopenharmony_ci	} else if (offset < 96) {
18462306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L3_0;
18562306a36Sopenharmony_ci		filter_sel = nfilt->hw_index;
18662306a36Sopenharmony_ci	} else if (offset < 128) {
18762306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L3_0;
18862306a36Sopenharmony_ci		filter_sel = nfilt->hw_index + 1;
18962306a36Sopenharmony_ci	} else if (offset < 160) {
19062306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L3_1;
19162306a36Sopenharmony_ci		filter_sel = nfilt->hw_index;
19262306a36Sopenharmony_ci	} else if (offset < 192) {
19362306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L3_1;
19462306a36Sopenharmony_ci		filter_sel = nfilt->hw_index + 1;
19562306a36Sopenharmony_ci	} else if (offset < 224) {
19662306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L4;
19762306a36Sopenharmony_ci		filter_sel = nfilt->hw_index;
19862306a36Sopenharmony_ci	} else if (offset < 256) {
19962306a36Sopenharmony_ci		block_index = ASP_RX_FILTER_NET_L4;
20062306a36Sopenharmony_ci		filter_sel = nfilt->hw_index + 1;
20162306a36Sopenharmony_ci	} else {
20262306a36Sopenharmony_ci		return -EINVAL;
20362306a36Sopenharmony_ci	}
20462306a36Sopenharmony_ci
20562306a36Sopenharmony_ci	switch (reg_type) {
20662306a36Sopenharmony_ci	case ASP_NETFILT_MATCH:
20762306a36Sopenharmony_ci		return ASP_RX_FILTER_NET_PAT(filter_sel, block_index,
20862306a36Sopenharmony_ci					     (offset % 32));
20962306a36Sopenharmony_ci	case ASP_NETFILT_MASK:
21062306a36Sopenharmony_ci		return ASP_RX_FILTER_NET_MASK(filter_sel, block_index,
21162306a36Sopenharmony_ci					      (offset % 32));
21262306a36Sopenharmony_ci	default:
21362306a36Sopenharmony_ci		return -EINVAL;
21462306a36Sopenharmony_ci	}
21562306a36Sopenharmony_ci}
21662306a36Sopenharmony_ci
21762306a36Sopenharmony_cistatic void bcmasp_netfilt_wr(struct bcmasp_priv *priv,
21862306a36Sopenharmony_ci			      struct bcmasp_net_filter *nfilt,
21962306a36Sopenharmony_ci			      enum asp_netfilt_reg_type reg_type,
22062306a36Sopenharmony_ci			      u32 val, u32 offset)
22162306a36Sopenharmony_ci{
22262306a36Sopenharmony_ci	int reg_offset;
22362306a36Sopenharmony_ci
22462306a36Sopenharmony_ci	/* HW only accepts 4 byte aligned writes */
22562306a36Sopenharmony_ci	if (!IS_ALIGNED(offset, 4) || offset > MAX_WAKE_FILTER_SIZE)
22662306a36Sopenharmony_ci		return;
22762306a36Sopenharmony_ci
22862306a36Sopenharmony_ci	reg_offset = bcmasp_netfilt_get_reg_offset(priv, nfilt, reg_type,
22962306a36Sopenharmony_ci						   offset);
23062306a36Sopenharmony_ci
23162306a36Sopenharmony_ci	rx_filter_core_wl(priv, val, reg_offset);
23262306a36Sopenharmony_ci}
23362306a36Sopenharmony_ci
23462306a36Sopenharmony_cistatic u32 bcmasp_netfilt_rd(struct bcmasp_priv *priv,
23562306a36Sopenharmony_ci			     struct bcmasp_net_filter *nfilt,
23662306a36Sopenharmony_ci			     enum asp_netfilt_reg_type reg_type,
23762306a36Sopenharmony_ci			     u32 offset)
23862306a36Sopenharmony_ci{
23962306a36Sopenharmony_ci	int reg_offset;
24062306a36Sopenharmony_ci
24162306a36Sopenharmony_ci	/* HW only accepts 4 byte aligned writes */
24262306a36Sopenharmony_ci	if (!IS_ALIGNED(offset, 4) || offset > MAX_WAKE_FILTER_SIZE)
24362306a36Sopenharmony_ci		return 0;
24462306a36Sopenharmony_ci
24562306a36Sopenharmony_ci	reg_offset = bcmasp_netfilt_get_reg_offset(priv, nfilt, reg_type,
24662306a36Sopenharmony_ci						   offset);
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_ci	return rx_filter_core_rl(priv, reg_offset);
24962306a36Sopenharmony_ci}
25062306a36Sopenharmony_ci
25162306a36Sopenharmony_cistatic int bcmasp_netfilt_wr_m_wake(struct bcmasp_priv *priv,
25262306a36Sopenharmony_ci				    struct bcmasp_net_filter *nfilt,
25362306a36Sopenharmony_ci				    u32 offset, void *match, void *mask,
25462306a36Sopenharmony_ci				    size_t size)
25562306a36Sopenharmony_ci{
25662306a36Sopenharmony_ci	u32 shift, mask_val = 0, match_val = 0;
25762306a36Sopenharmony_ci	bool first_byte = true;
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ci	if ((offset + size) > MAX_WAKE_FILTER_SIZE)
26062306a36Sopenharmony_ci		return -EINVAL;
26162306a36Sopenharmony_ci
26262306a36Sopenharmony_ci	while (size--) {
26362306a36Sopenharmony_ci		/* The HW only accepts 4 byte aligned writes, so if we
26462306a36Sopenharmony_ci		 * begin unaligned or if remaining bytes less than 4,
26562306a36Sopenharmony_ci		 * we need to read then write to avoid losing current
26662306a36Sopenharmony_ci		 * register state
26762306a36Sopenharmony_ci		 */
26862306a36Sopenharmony_ci		if (first_byte && (!IS_ALIGNED(offset, 4) || size < 3)) {
26962306a36Sopenharmony_ci			match_val = bcmasp_netfilt_rd(priv, nfilt,
27062306a36Sopenharmony_ci						      ASP_NETFILT_MATCH,
27162306a36Sopenharmony_ci						      ALIGN_DOWN(offset, 4));
27262306a36Sopenharmony_ci			mask_val = bcmasp_netfilt_rd(priv, nfilt,
27362306a36Sopenharmony_ci						     ASP_NETFILT_MASK,
27462306a36Sopenharmony_ci						     ALIGN_DOWN(offset, 4));
27562306a36Sopenharmony_ci		}
27662306a36Sopenharmony_ci
27762306a36Sopenharmony_ci		shift = (3 - (offset % 4)) * 8;
27862306a36Sopenharmony_ci		match_val &= ~GENMASK(shift + 7, shift);
27962306a36Sopenharmony_ci		mask_val &= ~GENMASK(shift + 7, shift);
28062306a36Sopenharmony_ci		match_val |= (u32)(*((u8 *)match) << shift);
28162306a36Sopenharmony_ci		mask_val |= (u32)(*((u8 *)mask) << shift);
28262306a36Sopenharmony_ci
28362306a36Sopenharmony_ci		/* If last byte or last byte of word, write to reg */
28462306a36Sopenharmony_ci		if (!size || ((offset % 4) == 3)) {
28562306a36Sopenharmony_ci			bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MATCH,
28662306a36Sopenharmony_ci					  match_val, ALIGN_DOWN(offset, 4));
28762306a36Sopenharmony_ci			bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MASK,
28862306a36Sopenharmony_ci					  mask_val, ALIGN_DOWN(offset, 4));
28962306a36Sopenharmony_ci			first_byte = true;
29062306a36Sopenharmony_ci		} else {
29162306a36Sopenharmony_ci			first_byte = false;
29262306a36Sopenharmony_ci		}
29362306a36Sopenharmony_ci
29462306a36Sopenharmony_ci		offset++;
29562306a36Sopenharmony_ci		match++;
29662306a36Sopenharmony_ci		mask++;
29762306a36Sopenharmony_ci	}
29862306a36Sopenharmony_ci
29962306a36Sopenharmony_ci	return 0;
30062306a36Sopenharmony_ci}
30162306a36Sopenharmony_ci
30262306a36Sopenharmony_cistatic void bcmasp_netfilt_reset_hw(struct bcmasp_priv *priv,
30362306a36Sopenharmony_ci				    struct bcmasp_net_filter *nfilt)
30462306a36Sopenharmony_ci{
30562306a36Sopenharmony_ci	int i;
30662306a36Sopenharmony_ci
30762306a36Sopenharmony_ci	for (i = 0; i < MAX_WAKE_FILTER_SIZE; i += 4) {
30862306a36Sopenharmony_ci		bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MATCH, 0, i);
30962306a36Sopenharmony_ci		bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MASK, 0, i);
31062306a36Sopenharmony_ci	}
31162306a36Sopenharmony_ci}
31262306a36Sopenharmony_ci
31362306a36Sopenharmony_cistatic void bcmasp_netfilt_tcpip4_wr(struct bcmasp_priv *priv,
31462306a36Sopenharmony_ci				     struct bcmasp_net_filter *nfilt,
31562306a36Sopenharmony_ci				     struct ethtool_tcpip4_spec *match,
31662306a36Sopenharmony_ci				     struct ethtool_tcpip4_spec *mask,
31762306a36Sopenharmony_ci				     u32 offset)
31862306a36Sopenharmony_ci{
31962306a36Sopenharmony_ci	__be16 val_16, mask_16;
32062306a36Sopenharmony_ci
32162306a36Sopenharmony_ci	val_16 = htons(ETH_P_IP);
32262306a36Sopenharmony_ci	mask_16 = htons(0xFFFF);
32362306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
32462306a36Sopenharmony_ci				 &val_16, &mask_16, sizeof(val_16));
32562306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 1,
32662306a36Sopenharmony_ci				 &match->tos, &mask->tos,
32762306a36Sopenharmony_ci				 sizeof(match->tos));
32862306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 12,
32962306a36Sopenharmony_ci				 &match->ip4src, &mask->ip4src,
33062306a36Sopenharmony_ci				 sizeof(match->ip4src));
33162306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 16,
33262306a36Sopenharmony_ci				 &match->ip4dst, &mask->ip4dst,
33362306a36Sopenharmony_ci				 sizeof(match->ip4dst));
33462306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 20,
33562306a36Sopenharmony_ci				 &match->psrc, &mask->psrc,
33662306a36Sopenharmony_ci				 sizeof(match->psrc));
33762306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 22,
33862306a36Sopenharmony_ci				 &match->pdst, &mask->pdst,
33962306a36Sopenharmony_ci				 sizeof(match->pdst));
34062306a36Sopenharmony_ci}
34162306a36Sopenharmony_ci
34262306a36Sopenharmony_cistatic void bcmasp_netfilt_tcpip6_wr(struct bcmasp_priv *priv,
34362306a36Sopenharmony_ci				     struct bcmasp_net_filter *nfilt,
34462306a36Sopenharmony_ci				     struct ethtool_tcpip6_spec *match,
34562306a36Sopenharmony_ci				     struct ethtool_tcpip6_spec *mask,
34662306a36Sopenharmony_ci				     u32 offset)
34762306a36Sopenharmony_ci{
34862306a36Sopenharmony_ci	__be16 val_16, mask_16;
34962306a36Sopenharmony_ci
35062306a36Sopenharmony_ci	val_16 = htons(ETH_P_IPV6);
35162306a36Sopenharmony_ci	mask_16 = htons(0xFFFF);
35262306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
35362306a36Sopenharmony_ci				 &val_16, &mask_16, sizeof(val_16));
35462306a36Sopenharmony_ci	val_16 = htons(match->tclass << 4);
35562306a36Sopenharmony_ci	mask_16 = htons(mask->tclass << 4);
35662306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset,
35762306a36Sopenharmony_ci				 &val_16, &mask_16, sizeof(val_16));
35862306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 8,
35962306a36Sopenharmony_ci				 &match->ip6src, &mask->ip6src,
36062306a36Sopenharmony_ci				 sizeof(match->ip6src));
36162306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 24,
36262306a36Sopenharmony_ci				 &match->ip6dst, &mask->ip6dst,
36362306a36Sopenharmony_ci				 sizeof(match->ip6dst));
36462306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 40,
36562306a36Sopenharmony_ci				 &match->psrc, &mask->psrc,
36662306a36Sopenharmony_ci				 sizeof(match->psrc));
36762306a36Sopenharmony_ci	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 42,
36862306a36Sopenharmony_ci				 &match->pdst, &mask->pdst,
36962306a36Sopenharmony_ci				 sizeof(match->pdst));
37062306a36Sopenharmony_ci}
37162306a36Sopenharmony_ci
37262306a36Sopenharmony_cistatic int bcmasp_netfilt_wr_to_hw(struct bcmasp_priv *priv,
37362306a36Sopenharmony_ci				   struct bcmasp_net_filter *nfilt)
37462306a36Sopenharmony_ci{
37562306a36Sopenharmony_ci	struct ethtool_rx_flow_spec *fs = &nfilt->fs;
37662306a36Sopenharmony_ci	unsigned int offset = 0;
37762306a36Sopenharmony_ci	__be16 val_16, mask_16;
37862306a36Sopenharmony_ci	u8 val_8, mask_8;
37962306a36Sopenharmony_ci
38062306a36Sopenharmony_ci	/* Currently only supports wake filters */
38162306a36Sopenharmony_ci	if (!nfilt->wake_filter)
38262306a36Sopenharmony_ci		return -EINVAL;
38362306a36Sopenharmony_ci
38462306a36Sopenharmony_ci	bcmasp_netfilt_reset_hw(priv, nfilt);
38562306a36Sopenharmony_ci
38662306a36Sopenharmony_ci	if (fs->flow_type & FLOW_MAC_EXT) {
38762306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, 0, &fs->h_ext.h_dest,
38862306a36Sopenharmony_ci					 &fs->m_ext.h_dest,
38962306a36Sopenharmony_ci					 sizeof(fs->h_ext.h_dest));
39062306a36Sopenharmony_ci	}
39162306a36Sopenharmony_ci
39262306a36Sopenharmony_ci	if ((fs->flow_type & FLOW_EXT) &&
39362306a36Sopenharmony_ci	    (fs->m_ext.vlan_etype || fs->m_ext.vlan_tci)) {
39462306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2),
39562306a36Sopenharmony_ci					 &fs->h_ext.vlan_etype,
39662306a36Sopenharmony_ci					 &fs->m_ext.vlan_etype,
39762306a36Sopenharmony_ci					 sizeof(fs->h_ext.vlan_etype));
39862306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ((ETH_ALEN * 2) + 2),
39962306a36Sopenharmony_ci					 &fs->h_ext.vlan_tci,
40062306a36Sopenharmony_ci					 &fs->m_ext.vlan_tci,
40162306a36Sopenharmony_ci					 sizeof(fs->h_ext.vlan_tci));
40262306a36Sopenharmony_ci		offset += VLAN_HLEN;
40362306a36Sopenharmony_ci	}
40462306a36Sopenharmony_ci
40562306a36Sopenharmony_ci	switch (fs->flow_type & ~(FLOW_EXT | FLOW_MAC_EXT)) {
40662306a36Sopenharmony_ci	case ETHER_FLOW:
40762306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, 0,
40862306a36Sopenharmony_ci					 &fs->h_u.ether_spec.h_dest,
40962306a36Sopenharmony_ci					 &fs->m_u.ether_spec.h_dest,
41062306a36Sopenharmony_ci					 sizeof(fs->h_u.ether_spec.h_dest));
41162306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_ALEN,
41262306a36Sopenharmony_ci					 &fs->h_u.ether_spec.h_source,
41362306a36Sopenharmony_ci					 &fs->m_u.ether_spec.h_source,
41462306a36Sopenharmony_ci					 sizeof(fs->h_u.ether_spec.h_source));
41562306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
41662306a36Sopenharmony_ci					 &fs->h_u.ether_spec.h_proto,
41762306a36Sopenharmony_ci					 &fs->m_u.ether_spec.h_proto,
41862306a36Sopenharmony_ci					 sizeof(fs->h_u.ether_spec.h_proto));
41962306a36Sopenharmony_ci
42062306a36Sopenharmony_ci		break;
42162306a36Sopenharmony_ci	case IP_USER_FLOW:
42262306a36Sopenharmony_ci		val_16 = htons(ETH_P_IP);
42362306a36Sopenharmony_ci		mask_16 = htons(0xFFFF);
42462306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
42562306a36Sopenharmony_ci					 &val_16, &mask_16, sizeof(val_16));
42662306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 1,
42762306a36Sopenharmony_ci					 &fs->h_u.usr_ip4_spec.tos,
42862306a36Sopenharmony_ci					 &fs->m_u.usr_ip4_spec.tos,
42962306a36Sopenharmony_ci					 sizeof(fs->h_u.usr_ip4_spec.tos));
43062306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 9,
43162306a36Sopenharmony_ci					 &fs->h_u.usr_ip4_spec.proto,
43262306a36Sopenharmony_ci					 &fs->m_u.usr_ip4_spec.proto,
43362306a36Sopenharmony_ci					 sizeof(fs->h_u.usr_ip4_spec.proto));
43462306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 12,
43562306a36Sopenharmony_ci					 &fs->h_u.usr_ip4_spec.ip4src,
43662306a36Sopenharmony_ci					 &fs->m_u.usr_ip4_spec.ip4src,
43762306a36Sopenharmony_ci					 sizeof(fs->h_u.usr_ip4_spec.ip4src));
43862306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 16,
43962306a36Sopenharmony_ci					 &fs->h_u.usr_ip4_spec.ip4dst,
44062306a36Sopenharmony_ci					 &fs->m_u.usr_ip4_spec.ip4dst,
44162306a36Sopenharmony_ci					 sizeof(fs->h_u.usr_ip4_spec.ip4dst));
44262306a36Sopenharmony_ci		if (!fs->m_u.usr_ip4_spec.l4_4_bytes)
44362306a36Sopenharmony_ci			break;
44462306a36Sopenharmony_ci
44562306a36Sopenharmony_ci		/* Only supports 20 byte IPv4 header */
44662306a36Sopenharmony_ci		val_8 = 0x45;
44762306a36Sopenharmony_ci		mask_8 = 0xFF;
44862306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset,
44962306a36Sopenharmony_ci					 &val_8, &mask_8, sizeof(val_8));
45062306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt,
45162306a36Sopenharmony_ci					 ETH_HLEN + 20 + offset,
45262306a36Sopenharmony_ci					 &fs->h_u.usr_ip4_spec.l4_4_bytes,
45362306a36Sopenharmony_ci					 &fs->m_u.usr_ip4_spec.l4_4_bytes,
45462306a36Sopenharmony_ci					 sizeof(fs->h_u.usr_ip4_spec.l4_4_bytes)
45562306a36Sopenharmony_ci					 );
45662306a36Sopenharmony_ci		break;
45762306a36Sopenharmony_ci	case TCP_V4_FLOW:
45862306a36Sopenharmony_ci		val_8 = IPPROTO_TCP;
45962306a36Sopenharmony_ci		mask_8 = 0xFF;
46062306a36Sopenharmony_ci		bcmasp_netfilt_tcpip4_wr(priv, nfilt, &fs->h_u.tcp_ip4_spec,
46162306a36Sopenharmony_ci					 &fs->m_u.tcp_ip4_spec, offset);
46262306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 9,
46362306a36Sopenharmony_ci					 &val_8, &mask_8, sizeof(val_8));
46462306a36Sopenharmony_ci		break;
46562306a36Sopenharmony_ci	case UDP_V4_FLOW:
46662306a36Sopenharmony_ci		val_8 = IPPROTO_UDP;
46762306a36Sopenharmony_ci		mask_8 = 0xFF;
46862306a36Sopenharmony_ci		bcmasp_netfilt_tcpip4_wr(priv, nfilt, &fs->h_u.udp_ip4_spec,
46962306a36Sopenharmony_ci					 &fs->m_u.udp_ip4_spec, offset);
47062306a36Sopenharmony_ci
47162306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 9,
47262306a36Sopenharmony_ci					 &val_8, &mask_8, sizeof(val_8));
47362306a36Sopenharmony_ci		break;
47462306a36Sopenharmony_ci	case TCP_V6_FLOW:
47562306a36Sopenharmony_ci		val_8 = IPPROTO_TCP;
47662306a36Sopenharmony_ci		mask_8 = 0xFF;
47762306a36Sopenharmony_ci		bcmasp_netfilt_tcpip6_wr(priv, nfilt, &fs->h_u.tcp_ip6_spec,
47862306a36Sopenharmony_ci					 &fs->m_u.tcp_ip6_spec, offset);
47962306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 6,
48062306a36Sopenharmony_ci					 &val_8, &mask_8, sizeof(val_8));
48162306a36Sopenharmony_ci		break;
48262306a36Sopenharmony_ci	case UDP_V6_FLOW:
48362306a36Sopenharmony_ci		val_8 = IPPROTO_UDP;
48462306a36Sopenharmony_ci		mask_8 = 0xFF;
48562306a36Sopenharmony_ci		bcmasp_netfilt_tcpip6_wr(priv, nfilt, &fs->h_u.udp_ip6_spec,
48662306a36Sopenharmony_ci					 &fs->m_u.udp_ip6_spec, offset);
48762306a36Sopenharmony_ci		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 6,
48862306a36Sopenharmony_ci					 &val_8, &mask_8, sizeof(val_8));
48962306a36Sopenharmony_ci		break;
49062306a36Sopenharmony_ci	}
49162306a36Sopenharmony_ci
49262306a36Sopenharmony_ci	bcmasp_netfilt_hw_en_wake(priv, nfilt);
49362306a36Sopenharmony_ci
49462306a36Sopenharmony_ci	return 0;
49562306a36Sopenharmony_ci}
49662306a36Sopenharmony_ci
49762306a36Sopenharmony_civoid bcmasp_netfilt_suspend(struct bcmasp_intf *intf)
49862306a36Sopenharmony_ci{
49962306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
50062306a36Sopenharmony_ci	bool write = false;
50162306a36Sopenharmony_ci	int ret, i;
50262306a36Sopenharmony_ci
50362306a36Sopenharmony_ci	/* Write all filters to HW */
50462306a36Sopenharmony_ci	for (i = 0; i < NUM_NET_FILTERS; i++) {
50562306a36Sopenharmony_ci		/* If the filter does not match the port, skip programming. */
50662306a36Sopenharmony_ci		if (!priv->net_filters[i].claimed ||
50762306a36Sopenharmony_ci		    priv->net_filters[i].port != intf->port)
50862306a36Sopenharmony_ci			continue;
50962306a36Sopenharmony_ci
51062306a36Sopenharmony_ci		if (i > 0 && (i % 2) &&
51162306a36Sopenharmony_ci		    priv->net_filters[i].wake_filter &&
51262306a36Sopenharmony_ci		    priv->net_filters[i - 1].wake_filter)
51362306a36Sopenharmony_ci			continue;
51462306a36Sopenharmony_ci
51562306a36Sopenharmony_ci		ret = bcmasp_netfilt_wr_to_hw(priv, &priv->net_filters[i]);
51662306a36Sopenharmony_ci		if (!ret)
51762306a36Sopenharmony_ci			write = true;
51862306a36Sopenharmony_ci	}
51962306a36Sopenharmony_ci
52062306a36Sopenharmony_ci	/* Successfully programmed at least one wake filter
52162306a36Sopenharmony_ci	 * so enable top level wake config
52262306a36Sopenharmony_ci	 */
52362306a36Sopenharmony_ci	if (write)
52462306a36Sopenharmony_ci		rx_filter_core_wl(priv, (ASP_RX_FILTER_OPUT_EN |
52562306a36Sopenharmony_ci				  ASP_RX_FILTER_LNR_MD |
52662306a36Sopenharmony_ci				  ASP_RX_FILTER_GEN_WK_EN |
52762306a36Sopenharmony_ci				  ASP_RX_FILTER_NT_FLT_EN),
52862306a36Sopenharmony_ci				  ASP_RX_FILTER_BLK_CTRL);
52962306a36Sopenharmony_ci}
53062306a36Sopenharmony_ci
53162306a36Sopenharmony_ciint bcmasp_netfilt_get_all_active(struct bcmasp_intf *intf, u32 *rule_locs,
53262306a36Sopenharmony_ci				  u32 *rule_cnt)
53362306a36Sopenharmony_ci{
53462306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
53562306a36Sopenharmony_ci	int j = 0, i;
53662306a36Sopenharmony_ci
53762306a36Sopenharmony_ci	for (i = 0; i < NUM_NET_FILTERS; i++) {
53862306a36Sopenharmony_ci		if (!priv->net_filters[i].claimed ||
53962306a36Sopenharmony_ci		    priv->net_filters[i].port != intf->port)
54062306a36Sopenharmony_ci			continue;
54162306a36Sopenharmony_ci
54262306a36Sopenharmony_ci		if (i > 0 && (i % 2) &&
54362306a36Sopenharmony_ci		    priv->net_filters[i].wake_filter &&
54462306a36Sopenharmony_ci		    priv->net_filters[i - 1].wake_filter)
54562306a36Sopenharmony_ci			continue;
54662306a36Sopenharmony_ci
54762306a36Sopenharmony_ci		if (j == *rule_cnt)
54862306a36Sopenharmony_ci			return -EMSGSIZE;
54962306a36Sopenharmony_ci
55062306a36Sopenharmony_ci		rule_locs[j++] = priv->net_filters[i].fs.location;
55162306a36Sopenharmony_ci	}
55262306a36Sopenharmony_ci
55362306a36Sopenharmony_ci	*rule_cnt = j;
55462306a36Sopenharmony_ci
55562306a36Sopenharmony_ci	return 0;
55662306a36Sopenharmony_ci}
55762306a36Sopenharmony_ci
55862306a36Sopenharmony_ciint bcmasp_netfilt_get_active(struct bcmasp_intf *intf)
55962306a36Sopenharmony_ci{
56062306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
56162306a36Sopenharmony_ci	int cnt = 0, i;
56262306a36Sopenharmony_ci
56362306a36Sopenharmony_ci	for (i = 0; i < NUM_NET_FILTERS; i++) {
56462306a36Sopenharmony_ci		if (!priv->net_filters[i].claimed ||
56562306a36Sopenharmony_ci		    priv->net_filters[i].port != intf->port)
56662306a36Sopenharmony_ci			continue;
56762306a36Sopenharmony_ci
56862306a36Sopenharmony_ci		/* Skip over a wake filter pair */
56962306a36Sopenharmony_ci		if (i > 0 && (i % 2) &&
57062306a36Sopenharmony_ci		    priv->net_filters[i].wake_filter &&
57162306a36Sopenharmony_ci		    priv->net_filters[i - 1].wake_filter)
57262306a36Sopenharmony_ci			continue;
57362306a36Sopenharmony_ci
57462306a36Sopenharmony_ci		cnt++;
57562306a36Sopenharmony_ci	}
57662306a36Sopenharmony_ci
57762306a36Sopenharmony_ci	return cnt;
57862306a36Sopenharmony_ci}
57962306a36Sopenharmony_ci
58062306a36Sopenharmony_cibool bcmasp_netfilt_check_dup(struct bcmasp_intf *intf,
58162306a36Sopenharmony_ci			      struct ethtool_rx_flow_spec *fs)
58262306a36Sopenharmony_ci{
58362306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
58462306a36Sopenharmony_ci	struct ethtool_rx_flow_spec *cur;
58562306a36Sopenharmony_ci	size_t fs_size = 0;
58662306a36Sopenharmony_ci	int i;
58762306a36Sopenharmony_ci
58862306a36Sopenharmony_ci	for (i = 0; i < NUM_NET_FILTERS; i++) {
58962306a36Sopenharmony_ci		if (!priv->net_filters[i].claimed ||
59062306a36Sopenharmony_ci		    priv->net_filters[i].port != intf->port)
59162306a36Sopenharmony_ci			continue;
59262306a36Sopenharmony_ci
59362306a36Sopenharmony_ci		cur = &priv->net_filters[i].fs;
59462306a36Sopenharmony_ci
59562306a36Sopenharmony_ci		if (cur->flow_type != fs->flow_type ||
59662306a36Sopenharmony_ci		    cur->ring_cookie != fs->ring_cookie)
59762306a36Sopenharmony_ci			continue;
59862306a36Sopenharmony_ci
59962306a36Sopenharmony_ci		switch (fs->flow_type & ~(FLOW_EXT | FLOW_MAC_EXT)) {
60062306a36Sopenharmony_ci		case ETHER_FLOW:
60162306a36Sopenharmony_ci			fs_size = sizeof(struct ethhdr);
60262306a36Sopenharmony_ci			break;
60362306a36Sopenharmony_ci		case IP_USER_FLOW:
60462306a36Sopenharmony_ci			fs_size = sizeof(struct ethtool_usrip4_spec);
60562306a36Sopenharmony_ci			break;
60662306a36Sopenharmony_ci		case TCP_V6_FLOW:
60762306a36Sopenharmony_ci		case UDP_V6_FLOW:
60862306a36Sopenharmony_ci			fs_size = sizeof(struct ethtool_tcpip6_spec);
60962306a36Sopenharmony_ci			break;
61062306a36Sopenharmony_ci		case TCP_V4_FLOW:
61162306a36Sopenharmony_ci		case UDP_V4_FLOW:
61262306a36Sopenharmony_ci			fs_size = sizeof(struct ethtool_tcpip4_spec);
61362306a36Sopenharmony_ci			break;
61462306a36Sopenharmony_ci		default:
61562306a36Sopenharmony_ci			continue;
61662306a36Sopenharmony_ci		}
61762306a36Sopenharmony_ci
61862306a36Sopenharmony_ci		if (memcmp(&cur->h_u, &fs->h_u, fs_size) ||
61962306a36Sopenharmony_ci		    memcmp(&cur->m_u, &fs->m_u, fs_size))
62062306a36Sopenharmony_ci			continue;
62162306a36Sopenharmony_ci
62262306a36Sopenharmony_ci		if (cur->flow_type & FLOW_EXT) {
62362306a36Sopenharmony_ci			if (cur->h_ext.vlan_etype != fs->h_ext.vlan_etype ||
62462306a36Sopenharmony_ci			    cur->m_ext.vlan_etype != fs->m_ext.vlan_etype ||
62562306a36Sopenharmony_ci			    cur->h_ext.vlan_tci != fs->h_ext.vlan_tci ||
62662306a36Sopenharmony_ci			    cur->m_ext.vlan_tci != fs->m_ext.vlan_tci ||
62762306a36Sopenharmony_ci			    cur->h_ext.data[0] != fs->h_ext.data[0])
62862306a36Sopenharmony_ci				continue;
62962306a36Sopenharmony_ci		}
63062306a36Sopenharmony_ci		if (cur->flow_type & FLOW_MAC_EXT) {
63162306a36Sopenharmony_ci			if (memcmp(&cur->h_ext.h_dest,
63262306a36Sopenharmony_ci				   &fs->h_ext.h_dest, ETH_ALEN) ||
63362306a36Sopenharmony_ci			    memcmp(&cur->m_ext.h_dest,
63462306a36Sopenharmony_ci				   &fs->m_ext.h_dest, ETH_ALEN))
63562306a36Sopenharmony_ci				continue;
63662306a36Sopenharmony_ci		}
63762306a36Sopenharmony_ci
63862306a36Sopenharmony_ci		return true;
63962306a36Sopenharmony_ci	}
64062306a36Sopenharmony_ci
64162306a36Sopenharmony_ci	return false;
64262306a36Sopenharmony_ci}
64362306a36Sopenharmony_ci
64462306a36Sopenharmony_ci/* If no network filter found, return open filter.
64562306a36Sopenharmony_ci * If no more open filters return NULL
64662306a36Sopenharmony_ci */
64762306a36Sopenharmony_cistruct bcmasp_net_filter *bcmasp_netfilt_get_init(struct bcmasp_intf *intf,
64862306a36Sopenharmony_ci						  u32 loc, bool wake_filter,
64962306a36Sopenharmony_ci						  bool init)
65062306a36Sopenharmony_ci{
65162306a36Sopenharmony_ci	struct bcmasp_net_filter *nfilter = NULL;
65262306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
65362306a36Sopenharmony_ci	int i, open_index = -1;
65462306a36Sopenharmony_ci
65562306a36Sopenharmony_ci	/* Check whether we exceed the filter table capacity */
65662306a36Sopenharmony_ci	if (loc != RX_CLS_LOC_ANY && loc >= NUM_NET_FILTERS)
65762306a36Sopenharmony_ci		return ERR_PTR(-EINVAL);
65862306a36Sopenharmony_ci
65962306a36Sopenharmony_ci	/* If the filter location is busy (already claimed) and we are initializing
66062306a36Sopenharmony_ci	 * the filter (insertion), return a busy error code.
66162306a36Sopenharmony_ci	 */
66262306a36Sopenharmony_ci	if (loc != RX_CLS_LOC_ANY && init && priv->net_filters[loc].claimed)
66362306a36Sopenharmony_ci		return ERR_PTR(-EBUSY);
66462306a36Sopenharmony_ci
66562306a36Sopenharmony_ci	/* We need two filters for wake-up, so we cannot use an odd filter */
66662306a36Sopenharmony_ci	if (wake_filter && loc != RX_CLS_LOC_ANY && (loc % 2))
66762306a36Sopenharmony_ci		return ERR_PTR(-EINVAL);
66862306a36Sopenharmony_ci
66962306a36Sopenharmony_ci	/* Initialize the loop index based on the desired location or from 0 */
67062306a36Sopenharmony_ci	i = loc == RX_CLS_LOC_ANY ? 0 : loc;
67162306a36Sopenharmony_ci
67262306a36Sopenharmony_ci	for ( ; i < NUM_NET_FILTERS; i++) {
67362306a36Sopenharmony_ci		/* Found matching network filter */
67462306a36Sopenharmony_ci		if (!init &&
67562306a36Sopenharmony_ci		    priv->net_filters[i].claimed &&
67662306a36Sopenharmony_ci		    priv->net_filters[i].hw_index == i &&
67762306a36Sopenharmony_ci		    priv->net_filters[i].port == intf->port)
67862306a36Sopenharmony_ci			return &priv->net_filters[i];
67962306a36Sopenharmony_ci
68062306a36Sopenharmony_ci		/* If we don't need a new filter or new filter already found */
68162306a36Sopenharmony_ci		if (!init || open_index >= 0)
68262306a36Sopenharmony_ci			continue;
68362306a36Sopenharmony_ci
68462306a36Sopenharmony_ci		/* Wake filter conslidates two filters to cover more bytes
68562306a36Sopenharmony_ci		 * Wake filter is open if...
68662306a36Sopenharmony_ci		 * 1. It is an even filter
68762306a36Sopenharmony_ci		 * 2. The current and next filter is not claimed
68862306a36Sopenharmony_ci		 */
68962306a36Sopenharmony_ci		if (wake_filter && !(i % 2) && !priv->net_filters[i].claimed &&
69062306a36Sopenharmony_ci		    !priv->net_filters[i + 1].claimed)
69162306a36Sopenharmony_ci			open_index = i;
69262306a36Sopenharmony_ci		else if (!priv->net_filters[i].claimed)
69362306a36Sopenharmony_ci			open_index = i;
69462306a36Sopenharmony_ci	}
69562306a36Sopenharmony_ci
69662306a36Sopenharmony_ci	if (open_index >= 0) {
69762306a36Sopenharmony_ci		nfilter = &priv->net_filters[open_index];
69862306a36Sopenharmony_ci		nfilter->claimed = true;
69962306a36Sopenharmony_ci		nfilter->port = intf->port;
70062306a36Sopenharmony_ci		nfilter->hw_index = open_index;
70162306a36Sopenharmony_ci	}
70262306a36Sopenharmony_ci
70362306a36Sopenharmony_ci	if (wake_filter && open_index >= 0) {
70462306a36Sopenharmony_ci		/* Claim next filter */
70562306a36Sopenharmony_ci		priv->net_filters[open_index + 1].claimed = true;
70662306a36Sopenharmony_ci		priv->net_filters[open_index + 1].wake_filter = true;
70762306a36Sopenharmony_ci		nfilter->wake_filter = true;
70862306a36Sopenharmony_ci	}
70962306a36Sopenharmony_ci
71062306a36Sopenharmony_ci	return nfilter ? nfilter : ERR_PTR(-EINVAL);
71162306a36Sopenharmony_ci}
71262306a36Sopenharmony_ci
71362306a36Sopenharmony_civoid bcmasp_netfilt_release(struct bcmasp_intf *intf,
71462306a36Sopenharmony_ci			    struct bcmasp_net_filter *nfilt)
71562306a36Sopenharmony_ci{
71662306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
71762306a36Sopenharmony_ci
71862306a36Sopenharmony_ci	if (nfilt->wake_filter) {
71962306a36Sopenharmony_ci		memset(&priv->net_filters[nfilt->hw_index + 1], 0,
72062306a36Sopenharmony_ci		       sizeof(struct bcmasp_net_filter));
72162306a36Sopenharmony_ci	}
72262306a36Sopenharmony_ci
72362306a36Sopenharmony_ci	memset(nfilt, 0, sizeof(struct bcmasp_net_filter));
72462306a36Sopenharmony_ci}
72562306a36Sopenharmony_ci
72662306a36Sopenharmony_cistatic void bcmasp_addr_to_uint(unsigned char *addr, u32 *high, u32 *low)
72762306a36Sopenharmony_ci{
72862306a36Sopenharmony_ci	*high = (u32)(addr[0] << 8 | addr[1]);
72962306a36Sopenharmony_ci	*low = (u32)(addr[2] << 24 | addr[3] << 16 | addr[4] << 8 |
73062306a36Sopenharmony_ci		     addr[5]);
73162306a36Sopenharmony_ci}
73262306a36Sopenharmony_ci
73362306a36Sopenharmony_cistatic void bcmasp_set_mda_filter(struct bcmasp_intf *intf,
73462306a36Sopenharmony_ci				  const unsigned char *addr,
73562306a36Sopenharmony_ci				  unsigned char *mask,
73662306a36Sopenharmony_ci				  unsigned int i)
73762306a36Sopenharmony_ci{
73862306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
73962306a36Sopenharmony_ci	u32 addr_h, addr_l, mask_h, mask_l;
74062306a36Sopenharmony_ci
74162306a36Sopenharmony_ci	/* Set local copy */
74262306a36Sopenharmony_ci	ether_addr_copy(priv->mda_filters[i].mask, mask);
74362306a36Sopenharmony_ci	ether_addr_copy(priv->mda_filters[i].addr, addr);
74462306a36Sopenharmony_ci
74562306a36Sopenharmony_ci	/* Write to HW */
74662306a36Sopenharmony_ci	bcmasp_addr_to_uint(priv->mda_filters[i].mask, &mask_h, &mask_l);
74762306a36Sopenharmony_ci	bcmasp_addr_to_uint(priv->mda_filters[i].addr, &addr_h, &addr_l);
74862306a36Sopenharmony_ci	rx_filter_core_wl(priv, addr_h, ASP_RX_FILTER_MDA_PAT_H(i));
74962306a36Sopenharmony_ci	rx_filter_core_wl(priv, addr_l, ASP_RX_FILTER_MDA_PAT_L(i));
75062306a36Sopenharmony_ci	rx_filter_core_wl(priv, mask_h, ASP_RX_FILTER_MDA_MSK_H(i));
75162306a36Sopenharmony_ci	rx_filter_core_wl(priv, mask_l, ASP_RX_FILTER_MDA_MSK_L(i));
75262306a36Sopenharmony_ci}
75362306a36Sopenharmony_ci
75462306a36Sopenharmony_cistatic void bcmasp_en_mda_filter(struct bcmasp_intf *intf, bool en,
75562306a36Sopenharmony_ci				 unsigned int i)
75662306a36Sopenharmony_ci{
75762306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
75862306a36Sopenharmony_ci
75962306a36Sopenharmony_ci	if (priv->mda_filters[i].en == en)
76062306a36Sopenharmony_ci		return;
76162306a36Sopenharmony_ci
76262306a36Sopenharmony_ci	priv->mda_filters[i].en = en;
76362306a36Sopenharmony_ci	priv->mda_filters[i].port = intf->port;
76462306a36Sopenharmony_ci
76562306a36Sopenharmony_ci	rx_filter_core_wl(priv, ((intf->channel + 8) |
76662306a36Sopenharmony_ci			  (en << ASP_RX_FILTER_MDA_CFG_EN_SHIFT) |
76762306a36Sopenharmony_ci			  ASP_RX_FILTER_MDA_CFG_UMC_SEL(intf->port)),
76862306a36Sopenharmony_ci			  ASP_RX_FILTER_MDA_CFG(i));
76962306a36Sopenharmony_ci}
77062306a36Sopenharmony_ci
77162306a36Sopenharmony_ci/* There are 32 MDA filters shared between all ports, we reserve 4 filters per
77262306a36Sopenharmony_ci * port for the following.
77362306a36Sopenharmony_ci * - Promisc: Filter to allow all packets when promisc is enabled
77462306a36Sopenharmony_ci * - All Multicast
77562306a36Sopenharmony_ci * - Broadcast
77662306a36Sopenharmony_ci * - Own address
77762306a36Sopenharmony_ci *
77862306a36Sopenharmony_ci * The reserved filters are identified as so.
77962306a36Sopenharmony_ci * - Promisc: (index * 4) + 0
78062306a36Sopenharmony_ci * - All Multicast: (index * 4) + 1
78162306a36Sopenharmony_ci * - Broadcast: (index * 4) + 2
78262306a36Sopenharmony_ci * - Own address: (index * 4) + 3
78362306a36Sopenharmony_ci */
78462306a36Sopenharmony_cienum asp_rx_filter_id {
78562306a36Sopenharmony_ci	ASP_RX_FILTER_MDA_PROMISC = 0,
78662306a36Sopenharmony_ci	ASP_RX_FILTER_MDA_ALLMULTI,
78762306a36Sopenharmony_ci	ASP_RX_FILTER_MDA_BROADCAST,
78862306a36Sopenharmony_ci	ASP_RX_FILTER_MDA_OWN_ADDR,
78962306a36Sopenharmony_ci	ASP_RX_FILTER_MDA_RES_MAX,
79062306a36Sopenharmony_ci};
79162306a36Sopenharmony_ci
79262306a36Sopenharmony_ci#define ASP_RX_FILT_MDA(intf, name)	(((intf)->index * \
79362306a36Sopenharmony_ci					  ASP_RX_FILTER_MDA_RES_MAX) \
79462306a36Sopenharmony_ci					 + ASP_RX_FILTER_MDA_##name)
79562306a36Sopenharmony_ci
79662306a36Sopenharmony_cistatic int bcmasp_total_res_mda_cnt(struct bcmasp_priv *priv)
79762306a36Sopenharmony_ci{
79862306a36Sopenharmony_ci	return list_count_nodes(&priv->intfs) * ASP_RX_FILTER_MDA_RES_MAX;
79962306a36Sopenharmony_ci}
80062306a36Sopenharmony_ci
80162306a36Sopenharmony_civoid bcmasp_set_promisc(struct bcmasp_intf *intf, bool en)
80262306a36Sopenharmony_ci{
80362306a36Sopenharmony_ci	unsigned int i = ASP_RX_FILT_MDA(intf, PROMISC);
80462306a36Sopenharmony_ci	unsigned char promisc[ETH_ALEN];
80562306a36Sopenharmony_ci
80662306a36Sopenharmony_ci	eth_zero_addr(promisc);
80762306a36Sopenharmony_ci	/* Set mask to 00:00:00:00:00:00 to match all packets */
80862306a36Sopenharmony_ci	bcmasp_set_mda_filter(intf, promisc, promisc, i);
80962306a36Sopenharmony_ci	bcmasp_en_mda_filter(intf, en, i);
81062306a36Sopenharmony_ci}
81162306a36Sopenharmony_ci
81262306a36Sopenharmony_civoid bcmasp_set_allmulti(struct bcmasp_intf *intf, bool en)
81362306a36Sopenharmony_ci{
81462306a36Sopenharmony_ci	unsigned char allmulti[] = {0x01, 0x00, 0x00, 0x00, 0x00, 0x00};
81562306a36Sopenharmony_ci	unsigned int i = ASP_RX_FILT_MDA(intf, ALLMULTI);
81662306a36Sopenharmony_ci
81762306a36Sopenharmony_ci	/* Set mask to 01:00:00:00:00:00 to match all multicast */
81862306a36Sopenharmony_ci	bcmasp_set_mda_filter(intf, allmulti, allmulti, i);
81962306a36Sopenharmony_ci	bcmasp_en_mda_filter(intf, en, i);
82062306a36Sopenharmony_ci}
82162306a36Sopenharmony_ci
82262306a36Sopenharmony_civoid bcmasp_set_broad(struct bcmasp_intf *intf, bool en)
82362306a36Sopenharmony_ci{
82462306a36Sopenharmony_ci	unsigned int i = ASP_RX_FILT_MDA(intf, BROADCAST);
82562306a36Sopenharmony_ci	unsigned char addr[ETH_ALEN];
82662306a36Sopenharmony_ci
82762306a36Sopenharmony_ci	eth_broadcast_addr(addr);
82862306a36Sopenharmony_ci	bcmasp_set_mda_filter(intf, addr, addr, i);
82962306a36Sopenharmony_ci	bcmasp_en_mda_filter(intf, en, i);
83062306a36Sopenharmony_ci}
83162306a36Sopenharmony_ci
83262306a36Sopenharmony_civoid bcmasp_set_oaddr(struct bcmasp_intf *intf, const unsigned char *addr,
83362306a36Sopenharmony_ci		      bool en)
83462306a36Sopenharmony_ci{
83562306a36Sopenharmony_ci	unsigned char mask[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
83662306a36Sopenharmony_ci	unsigned int i = ASP_RX_FILT_MDA(intf, OWN_ADDR);
83762306a36Sopenharmony_ci
83862306a36Sopenharmony_ci	bcmasp_set_mda_filter(intf, addr, mask, i);
83962306a36Sopenharmony_ci	bcmasp_en_mda_filter(intf, en, i);
84062306a36Sopenharmony_ci}
84162306a36Sopenharmony_ci
84262306a36Sopenharmony_civoid bcmasp_disable_all_filters(struct bcmasp_intf *intf)
84362306a36Sopenharmony_ci{
84462306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
84562306a36Sopenharmony_ci	unsigned int i;
84662306a36Sopenharmony_ci	int res_count;
84762306a36Sopenharmony_ci
84862306a36Sopenharmony_ci	res_count = bcmasp_total_res_mda_cnt(intf->parent);
84962306a36Sopenharmony_ci
85062306a36Sopenharmony_ci	/* Disable all filters held by this port */
85162306a36Sopenharmony_ci	for (i = res_count; i < NUM_MDA_FILTERS; i++) {
85262306a36Sopenharmony_ci		if (priv->mda_filters[i].en &&
85362306a36Sopenharmony_ci		    priv->mda_filters[i].port == intf->port)
85462306a36Sopenharmony_ci			bcmasp_en_mda_filter(intf, 0, i);
85562306a36Sopenharmony_ci	}
85662306a36Sopenharmony_ci}
85762306a36Sopenharmony_ci
85862306a36Sopenharmony_cistatic int bcmasp_combine_set_filter(struct bcmasp_intf *intf,
85962306a36Sopenharmony_ci				     unsigned char *addr, unsigned char *mask,
86062306a36Sopenharmony_ci				     int i)
86162306a36Sopenharmony_ci{
86262306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
86362306a36Sopenharmony_ci	u64 addr1, addr2, mask1, mask2, mask3;
86462306a36Sopenharmony_ci
86562306a36Sopenharmony_ci	/* Switch to u64 to help with the calculations */
86662306a36Sopenharmony_ci	addr1 = ether_addr_to_u64(priv->mda_filters[i].addr);
86762306a36Sopenharmony_ci	mask1 = ether_addr_to_u64(priv->mda_filters[i].mask);
86862306a36Sopenharmony_ci	addr2 = ether_addr_to_u64(addr);
86962306a36Sopenharmony_ci	mask2 = ether_addr_to_u64(mask);
87062306a36Sopenharmony_ci
87162306a36Sopenharmony_ci	/* Check if one filter resides within the other */
87262306a36Sopenharmony_ci	mask3 = mask1 & mask2;
87362306a36Sopenharmony_ci	if (mask3 == mask1 && ((addr1 & mask1) == (addr2 & mask1))) {
87462306a36Sopenharmony_ci		/* Filter 2 resides within filter 1, so everything is good */
87562306a36Sopenharmony_ci		return 0;
87662306a36Sopenharmony_ci	} else if (mask3 == mask2 && ((addr1 & mask2) == (addr2 & mask2))) {
87762306a36Sopenharmony_ci		/* Filter 1 resides within filter 2, so swap filters */
87862306a36Sopenharmony_ci		bcmasp_set_mda_filter(intf, addr, mask, i);
87962306a36Sopenharmony_ci		return 0;
88062306a36Sopenharmony_ci	}
88162306a36Sopenharmony_ci
88262306a36Sopenharmony_ci	/* Unable to combine */
88362306a36Sopenharmony_ci	return -EINVAL;
88462306a36Sopenharmony_ci}
88562306a36Sopenharmony_ci
88662306a36Sopenharmony_ciint bcmasp_set_en_mda_filter(struct bcmasp_intf *intf, unsigned char *addr,
88762306a36Sopenharmony_ci			     unsigned char *mask)
88862306a36Sopenharmony_ci{
88962306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
89062306a36Sopenharmony_ci	int ret, res_count;
89162306a36Sopenharmony_ci	unsigned int i;
89262306a36Sopenharmony_ci
89362306a36Sopenharmony_ci	res_count = bcmasp_total_res_mda_cnt(intf->parent);
89462306a36Sopenharmony_ci
89562306a36Sopenharmony_ci	for (i = res_count; i < NUM_MDA_FILTERS; i++) {
89662306a36Sopenharmony_ci		/* If filter not enabled or belongs to another port skip */
89762306a36Sopenharmony_ci		if (!priv->mda_filters[i].en ||
89862306a36Sopenharmony_ci		    priv->mda_filters[i].port != intf->port)
89962306a36Sopenharmony_ci			continue;
90062306a36Sopenharmony_ci
90162306a36Sopenharmony_ci		/* Attempt to combine filters */
90262306a36Sopenharmony_ci		ret = bcmasp_combine_set_filter(intf, addr, mask, i);
90362306a36Sopenharmony_ci		if (!ret) {
90462306a36Sopenharmony_ci			intf->mib.filters_combine_cnt++;
90562306a36Sopenharmony_ci			return 0;
90662306a36Sopenharmony_ci		}
90762306a36Sopenharmony_ci	}
90862306a36Sopenharmony_ci
90962306a36Sopenharmony_ci	/* Create new filter if possible */
91062306a36Sopenharmony_ci	for (i = res_count; i < NUM_MDA_FILTERS; i++) {
91162306a36Sopenharmony_ci		if (priv->mda_filters[i].en)
91262306a36Sopenharmony_ci			continue;
91362306a36Sopenharmony_ci
91462306a36Sopenharmony_ci		bcmasp_set_mda_filter(intf, addr, mask, i);
91562306a36Sopenharmony_ci		bcmasp_en_mda_filter(intf, 1, i);
91662306a36Sopenharmony_ci		return 0;
91762306a36Sopenharmony_ci	}
91862306a36Sopenharmony_ci
91962306a36Sopenharmony_ci	/* No room for new filter */
92062306a36Sopenharmony_ci	return -EINVAL;
92162306a36Sopenharmony_ci}
92262306a36Sopenharmony_ci
92362306a36Sopenharmony_cistatic void bcmasp_core_init_filters(struct bcmasp_priv *priv)
92462306a36Sopenharmony_ci{
92562306a36Sopenharmony_ci	unsigned int i;
92662306a36Sopenharmony_ci
92762306a36Sopenharmony_ci	/* Disable all filters and reset software view since the HW
92862306a36Sopenharmony_ci	 * can lose context while in deep sleep suspend states
92962306a36Sopenharmony_ci	 */
93062306a36Sopenharmony_ci	for (i = 0; i < NUM_MDA_FILTERS; i++) {
93162306a36Sopenharmony_ci		rx_filter_core_wl(priv, 0x0, ASP_RX_FILTER_MDA_CFG(i));
93262306a36Sopenharmony_ci		priv->mda_filters[i].en = 0;
93362306a36Sopenharmony_ci	}
93462306a36Sopenharmony_ci
93562306a36Sopenharmony_ci	for (i = 0; i < NUM_NET_FILTERS; i++)
93662306a36Sopenharmony_ci		rx_filter_core_wl(priv, 0x0, ASP_RX_FILTER_NET_CFG(i));
93762306a36Sopenharmony_ci
93862306a36Sopenharmony_ci	/* Top level filter enable bit should be enabled at all times, set
93962306a36Sopenharmony_ci	 * GEN_WAKE_CLEAR to clear the network filter wake-up which would
94062306a36Sopenharmony_ci	 * otherwise be sticky
94162306a36Sopenharmony_ci	 */
94262306a36Sopenharmony_ci	rx_filter_core_wl(priv, (ASP_RX_FILTER_OPUT_EN |
94362306a36Sopenharmony_ci			  ASP_RX_FILTER_MDA_EN |
94462306a36Sopenharmony_ci			  ASP_RX_FILTER_GEN_WK_CLR |
94562306a36Sopenharmony_ci			  ASP_RX_FILTER_NT_FLT_EN),
94662306a36Sopenharmony_ci			  ASP_RX_FILTER_BLK_CTRL);
94762306a36Sopenharmony_ci}
94862306a36Sopenharmony_ci
94962306a36Sopenharmony_ci/* ASP core initialization */
95062306a36Sopenharmony_cistatic void bcmasp_core_init(struct bcmasp_priv *priv)
95162306a36Sopenharmony_ci{
95262306a36Sopenharmony_ci	tx_analytics_core_wl(priv, 0x0, ASP_TX_ANALYTICS_CTRL);
95362306a36Sopenharmony_ci	rx_analytics_core_wl(priv, 0x4, ASP_RX_ANALYTICS_CTRL);
95462306a36Sopenharmony_ci
95562306a36Sopenharmony_ci	rx_edpkt_core_wl(priv, (ASP_EDPKT_HDR_SZ_128 << ASP_EDPKT_HDR_SZ_SHIFT),
95662306a36Sopenharmony_ci			 ASP_EDPKT_HDR_CFG);
95762306a36Sopenharmony_ci	rx_edpkt_core_wl(priv,
95862306a36Sopenharmony_ci			 (ASP_EDPKT_ENDI_BT_SWP_WD << ASP_EDPKT_ENDI_DESC_SHIFT),
95962306a36Sopenharmony_ci			 ASP_EDPKT_ENDI);
96062306a36Sopenharmony_ci
96162306a36Sopenharmony_ci	rx_edpkt_core_wl(priv, 0x1b, ASP_EDPKT_BURST_BUF_PSCAL_TOUT);
96262306a36Sopenharmony_ci	rx_edpkt_core_wl(priv, 0x3e8, ASP_EDPKT_BURST_BUF_WRITE_TOUT);
96362306a36Sopenharmony_ci	rx_edpkt_core_wl(priv, 0x3e8, ASP_EDPKT_BURST_BUF_READ_TOUT);
96462306a36Sopenharmony_ci
96562306a36Sopenharmony_ci	rx_edpkt_core_wl(priv, ASP_EDPKT_ENABLE_EN, ASP_EDPKT_ENABLE);
96662306a36Sopenharmony_ci
96762306a36Sopenharmony_ci	/* Disable and clear both UniMAC's wake-up interrupts to avoid
96862306a36Sopenharmony_ci	 * sticky interrupts.
96962306a36Sopenharmony_ci	 */
97062306a36Sopenharmony_ci	_intr2_mask_set(priv, ASP_INTR2_UMC0_WAKE | ASP_INTR2_UMC1_WAKE);
97162306a36Sopenharmony_ci	intr2_core_wl(priv, ASP_INTR2_UMC0_WAKE | ASP_INTR2_UMC1_WAKE,
97262306a36Sopenharmony_ci		      ASP_INTR2_CLEAR);
97362306a36Sopenharmony_ci}
97462306a36Sopenharmony_ci
97562306a36Sopenharmony_cistatic void bcmasp_core_clock_select(struct bcmasp_priv *priv, bool slow)
97662306a36Sopenharmony_ci{
97762306a36Sopenharmony_ci	u32 reg;
97862306a36Sopenharmony_ci
97962306a36Sopenharmony_ci	reg = ctrl_core_rl(priv, ASP_CTRL_CORE_CLOCK_SELECT);
98062306a36Sopenharmony_ci	if (slow)
98162306a36Sopenharmony_ci		reg &= ~ASP_CTRL_CORE_CLOCK_SELECT_MAIN;
98262306a36Sopenharmony_ci	else
98362306a36Sopenharmony_ci		reg |= ASP_CTRL_CORE_CLOCK_SELECT_MAIN;
98462306a36Sopenharmony_ci	ctrl_core_wl(priv, reg, ASP_CTRL_CORE_CLOCK_SELECT);
98562306a36Sopenharmony_ci}
98662306a36Sopenharmony_ci
98762306a36Sopenharmony_cistatic void bcmasp_core_clock_set_ll(struct bcmasp_priv *priv, u32 clr, u32 set)
98862306a36Sopenharmony_ci{
98962306a36Sopenharmony_ci	u32 reg;
99062306a36Sopenharmony_ci
99162306a36Sopenharmony_ci	reg = ctrl_core_rl(priv, ASP_CTRL_CLOCK_CTRL);
99262306a36Sopenharmony_ci	reg &= ~clr;
99362306a36Sopenharmony_ci	reg |= set;
99462306a36Sopenharmony_ci	ctrl_core_wl(priv, reg, ASP_CTRL_CLOCK_CTRL);
99562306a36Sopenharmony_ci
99662306a36Sopenharmony_ci	reg = ctrl_core_rl(priv, ASP_CTRL_SCRATCH_0);
99762306a36Sopenharmony_ci	reg &= ~clr;
99862306a36Sopenharmony_ci	reg |= set;
99962306a36Sopenharmony_ci	ctrl_core_wl(priv, reg, ASP_CTRL_SCRATCH_0);
100062306a36Sopenharmony_ci}
100162306a36Sopenharmony_ci
100262306a36Sopenharmony_cistatic void bcmasp_core_clock_set(struct bcmasp_priv *priv, u32 clr, u32 set)
100362306a36Sopenharmony_ci{
100462306a36Sopenharmony_ci	unsigned long flags;
100562306a36Sopenharmony_ci
100662306a36Sopenharmony_ci	spin_lock_irqsave(&priv->clk_lock, flags);
100762306a36Sopenharmony_ci	bcmasp_core_clock_set_ll(priv, clr, set);
100862306a36Sopenharmony_ci	spin_unlock_irqrestore(&priv->clk_lock, flags);
100962306a36Sopenharmony_ci}
101062306a36Sopenharmony_ci
101162306a36Sopenharmony_civoid bcmasp_core_clock_set_intf(struct bcmasp_intf *intf, bool en)
101262306a36Sopenharmony_ci{
101362306a36Sopenharmony_ci	u32 intf_mask = ASP_CTRL_CLOCK_CTRL_ASP_RGMII_DIS(intf->port);
101462306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
101562306a36Sopenharmony_ci	unsigned long flags;
101662306a36Sopenharmony_ci	u32 reg;
101762306a36Sopenharmony_ci
101862306a36Sopenharmony_ci	/* When enabling an interface, if the RX or TX clocks were not enabled,
101962306a36Sopenharmony_ci	 * enable them. Conversely, while disabling an interface, if this is
102062306a36Sopenharmony_ci	 * the last one enabled, we can turn off the shared RX and TX clocks as
102162306a36Sopenharmony_ci	 * well. We control enable bits which is why we test for equality on
102262306a36Sopenharmony_ci	 * the RGMII clock bit mask.
102362306a36Sopenharmony_ci	 */
102462306a36Sopenharmony_ci	spin_lock_irqsave(&priv->clk_lock, flags);
102562306a36Sopenharmony_ci	if (en) {
102662306a36Sopenharmony_ci		intf_mask |= ASP_CTRL_CLOCK_CTRL_ASP_TX_DISABLE |
102762306a36Sopenharmony_ci			     ASP_CTRL_CLOCK_CTRL_ASP_RX_DISABLE;
102862306a36Sopenharmony_ci		bcmasp_core_clock_set_ll(priv, intf_mask, 0);
102962306a36Sopenharmony_ci	} else {
103062306a36Sopenharmony_ci		reg = ctrl_core_rl(priv, ASP_CTRL_SCRATCH_0) | intf_mask;
103162306a36Sopenharmony_ci		if ((reg & ASP_CTRL_CLOCK_CTRL_ASP_RGMII_MASK) ==
103262306a36Sopenharmony_ci		    ASP_CTRL_CLOCK_CTRL_ASP_RGMII_MASK)
103362306a36Sopenharmony_ci			intf_mask |= ASP_CTRL_CLOCK_CTRL_ASP_TX_DISABLE |
103462306a36Sopenharmony_ci				     ASP_CTRL_CLOCK_CTRL_ASP_RX_DISABLE;
103562306a36Sopenharmony_ci		bcmasp_core_clock_set_ll(priv, 0, intf_mask);
103662306a36Sopenharmony_ci	}
103762306a36Sopenharmony_ci	spin_unlock_irqrestore(&priv->clk_lock, flags);
103862306a36Sopenharmony_ci}
103962306a36Sopenharmony_ci
104062306a36Sopenharmony_cistatic irqreturn_t bcmasp_isr_wol(int irq, void *data)
104162306a36Sopenharmony_ci{
104262306a36Sopenharmony_ci	struct bcmasp_priv *priv = data;
104362306a36Sopenharmony_ci	u32 status;
104462306a36Sopenharmony_ci
104562306a36Sopenharmony_ci	/* No L3 IRQ, so we good */
104662306a36Sopenharmony_ci	if (priv->wol_irq <= 0)
104762306a36Sopenharmony_ci		goto irq_handled;
104862306a36Sopenharmony_ci
104962306a36Sopenharmony_ci	status = wakeup_intr2_core_rl(priv, ASP_WAKEUP_INTR2_STATUS) &
105062306a36Sopenharmony_ci		~wakeup_intr2_core_rl(priv, ASP_WAKEUP_INTR2_MASK_STATUS);
105162306a36Sopenharmony_ci	wakeup_intr2_core_wl(priv, status, ASP_WAKEUP_INTR2_CLEAR);
105262306a36Sopenharmony_ci
105362306a36Sopenharmony_ciirq_handled:
105462306a36Sopenharmony_ci	pm_wakeup_event(&priv->pdev->dev, 0);
105562306a36Sopenharmony_ci	return IRQ_HANDLED;
105662306a36Sopenharmony_ci}
105762306a36Sopenharmony_ci
105862306a36Sopenharmony_cistatic int bcmasp_get_and_request_irq(struct bcmasp_priv *priv, int i)
105962306a36Sopenharmony_ci{
106062306a36Sopenharmony_ci	struct platform_device *pdev = priv->pdev;
106162306a36Sopenharmony_ci	int irq, ret;
106262306a36Sopenharmony_ci
106362306a36Sopenharmony_ci	irq = platform_get_irq_optional(pdev, i);
106462306a36Sopenharmony_ci	if (irq < 0)
106562306a36Sopenharmony_ci		return irq;
106662306a36Sopenharmony_ci
106762306a36Sopenharmony_ci	ret = devm_request_irq(&pdev->dev, irq, bcmasp_isr_wol, 0,
106862306a36Sopenharmony_ci			       pdev->name, priv);
106962306a36Sopenharmony_ci	if (ret)
107062306a36Sopenharmony_ci		return ret;
107162306a36Sopenharmony_ci
107262306a36Sopenharmony_ci	return irq;
107362306a36Sopenharmony_ci}
107462306a36Sopenharmony_ci
107562306a36Sopenharmony_cistatic void bcmasp_init_wol_shared(struct bcmasp_priv *priv)
107662306a36Sopenharmony_ci{
107762306a36Sopenharmony_ci	struct platform_device *pdev = priv->pdev;
107862306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
107962306a36Sopenharmony_ci	int irq;
108062306a36Sopenharmony_ci
108162306a36Sopenharmony_ci	irq = bcmasp_get_and_request_irq(priv, 1);
108262306a36Sopenharmony_ci	if (irq < 0) {
108362306a36Sopenharmony_ci		dev_warn(dev, "Failed to init WoL irq: %d\n", irq);
108462306a36Sopenharmony_ci		return;
108562306a36Sopenharmony_ci	}
108662306a36Sopenharmony_ci
108762306a36Sopenharmony_ci	priv->wol_irq = irq;
108862306a36Sopenharmony_ci	priv->wol_irq_enabled_mask = 0;
108962306a36Sopenharmony_ci	device_set_wakeup_capable(&pdev->dev, 1);
109062306a36Sopenharmony_ci}
109162306a36Sopenharmony_ci
109262306a36Sopenharmony_cistatic void bcmasp_enable_wol_shared(struct bcmasp_intf *intf, bool en)
109362306a36Sopenharmony_ci{
109462306a36Sopenharmony_ci	struct bcmasp_priv *priv = intf->parent;
109562306a36Sopenharmony_ci	struct device *dev = &priv->pdev->dev;
109662306a36Sopenharmony_ci
109762306a36Sopenharmony_ci	if (en) {
109862306a36Sopenharmony_ci		if (priv->wol_irq_enabled_mask) {
109962306a36Sopenharmony_ci			set_bit(intf->port, &priv->wol_irq_enabled_mask);
110062306a36Sopenharmony_ci			return;
110162306a36Sopenharmony_ci		}
110262306a36Sopenharmony_ci
110362306a36Sopenharmony_ci		/* First enable */
110462306a36Sopenharmony_ci		set_bit(intf->port, &priv->wol_irq_enabled_mask);
110562306a36Sopenharmony_ci		enable_irq_wake(priv->wol_irq);
110662306a36Sopenharmony_ci		device_set_wakeup_enable(dev, 1);
110762306a36Sopenharmony_ci	} else {
110862306a36Sopenharmony_ci		if (!priv->wol_irq_enabled_mask)
110962306a36Sopenharmony_ci			return;
111062306a36Sopenharmony_ci
111162306a36Sopenharmony_ci		clear_bit(intf->port, &priv->wol_irq_enabled_mask);
111262306a36Sopenharmony_ci		if (priv->wol_irq_enabled_mask)
111362306a36Sopenharmony_ci			return;
111462306a36Sopenharmony_ci
111562306a36Sopenharmony_ci		/* Last disable */
111662306a36Sopenharmony_ci		disable_irq_wake(priv->wol_irq);
111762306a36Sopenharmony_ci		device_set_wakeup_enable(dev, 0);
111862306a36Sopenharmony_ci	}
111962306a36Sopenharmony_ci}
112062306a36Sopenharmony_ci
112162306a36Sopenharmony_cistatic void bcmasp_wol_irq_destroy_shared(struct bcmasp_priv *priv)
112262306a36Sopenharmony_ci{
112362306a36Sopenharmony_ci	if (priv->wol_irq > 0)
112462306a36Sopenharmony_ci		free_irq(priv->wol_irq, priv);
112562306a36Sopenharmony_ci}
112662306a36Sopenharmony_ci
112762306a36Sopenharmony_cistatic void bcmasp_init_wol_per_intf(struct bcmasp_priv *priv)
112862306a36Sopenharmony_ci{
112962306a36Sopenharmony_ci	struct platform_device *pdev = priv->pdev;
113062306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
113162306a36Sopenharmony_ci	struct bcmasp_intf *intf;
113262306a36Sopenharmony_ci	int irq;
113362306a36Sopenharmony_ci
113462306a36Sopenharmony_ci	list_for_each_entry(intf, &priv->intfs, list) {
113562306a36Sopenharmony_ci		irq = bcmasp_get_and_request_irq(priv, intf->port + 1);
113662306a36Sopenharmony_ci		if (irq < 0) {
113762306a36Sopenharmony_ci			dev_warn(dev, "Failed to init WoL irq(port %d): %d\n",
113862306a36Sopenharmony_ci				 intf->port, irq);
113962306a36Sopenharmony_ci			continue;
114062306a36Sopenharmony_ci		}
114162306a36Sopenharmony_ci
114262306a36Sopenharmony_ci		intf->wol_irq = irq;
114362306a36Sopenharmony_ci		intf->wol_irq_enabled = false;
114462306a36Sopenharmony_ci		device_set_wakeup_capable(&pdev->dev, 1);
114562306a36Sopenharmony_ci	}
114662306a36Sopenharmony_ci}
114762306a36Sopenharmony_ci
114862306a36Sopenharmony_cistatic void bcmasp_enable_wol_per_intf(struct bcmasp_intf *intf, bool en)
114962306a36Sopenharmony_ci{
115062306a36Sopenharmony_ci	struct device *dev = &intf->parent->pdev->dev;
115162306a36Sopenharmony_ci
115262306a36Sopenharmony_ci	if (en ^ intf->wol_irq_enabled)
115362306a36Sopenharmony_ci		irq_set_irq_wake(intf->wol_irq, en);
115462306a36Sopenharmony_ci
115562306a36Sopenharmony_ci	intf->wol_irq_enabled = en;
115662306a36Sopenharmony_ci	device_set_wakeup_enable(dev, en);
115762306a36Sopenharmony_ci}
115862306a36Sopenharmony_ci
115962306a36Sopenharmony_cistatic void bcmasp_wol_irq_destroy_per_intf(struct bcmasp_priv *priv)
116062306a36Sopenharmony_ci{
116162306a36Sopenharmony_ci	struct bcmasp_intf *intf;
116262306a36Sopenharmony_ci
116362306a36Sopenharmony_ci	list_for_each_entry(intf, &priv->intfs, list) {
116462306a36Sopenharmony_ci		if (intf->wol_irq > 0)
116562306a36Sopenharmony_ci			free_irq(intf->wol_irq, priv);
116662306a36Sopenharmony_ci	}
116762306a36Sopenharmony_ci}
116862306a36Sopenharmony_ci
116962306a36Sopenharmony_cistatic struct bcmasp_hw_info v20_hw_info = {
117062306a36Sopenharmony_ci	.rx_ctrl_flush = ASP_RX_CTRL_FLUSH,
117162306a36Sopenharmony_ci	.umac2fb = UMAC2FB_OFFSET,
117262306a36Sopenharmony_ci	.rx_ctrl_fb_out_frame_count = ASP_RX_CTRL_FB_OUT_FRAME_COUNT,
117362306a36Sopenharmony_ci	.rx_ctrl_fb_filt_out_frame_count = ASP_RX_CTRL_FB_FILT_OUT_FRAME_COUNT,
117462306a36Sopenharmony_ci	.rx_ctrl_fb_rx_fifo_depth = ASP_RX_CTRL_FB_RX_FIFO_DEPTH,
117562306a36Sopenharmony_ci};
117662306a36Sopenharmony_ci
117762306a36Sopenharmony_cistatic const struct bcmasp_plat_data v20_plat_data = {
117862306a36Sopenharmony_ci	.init_wol = bcmasp_init_wol_per_intf,
117962306a36Sopenharmony_ci	.enable_wol = bcmasp_enable_wol_per_intf,
118062306a36Sopenharmony_ci	.destroy_wol = bcmasp_wol_irq_destroy_per_intf,
118162306a36Sopenharmony_ci	.hw_info = &v20_hw_info,
118262306a36Sopenharmony_ci};
118362306a36Sopenharmony_ci
118462306a36Sopenharmony_cistatic struct bcmasp_hw_info v21_hw_info = {
118562306a36Sopenharmony_ci	.rx_ctrl_flush = ASP_RX_CTRL_FLUSH_2_1,
118662306a36Sopenharmony_ci	.umac2fb = UMAC2FB_OFFSET_2_1,
118762306a36Sopenharmony_ci	.rx_ctrl_fb_out_frame_count = ASP_RX_CTRL_FB_OUT_FRAME_COUNT_2_1,
118862306a36Sopenharmony_ci	.rx_ctrl_fb_filt_out_frame_count =
118962306a36Sopenharmony_ci		ASP_RX_CTRL_FB_FILT_OUT_FRAME_COUNT_2_1,
119062306a36Sopenharmony_ci	.rx_ctrl_fb_rx_fifo_depth = ASP_RX_CTRL_FB_RX_FIFO_DEPTH_2_1,
119162306a36Sopenharmony_ci};
119262306a36Sopenharmony_ci
119362306a36Sopenharmony_cistatic const struct bcmasp_plat_data v21_plat_data = {
119462306a36Sopenharmony_ci	.init_wol = bcmasp_init_wol_shared,
119562306a36Sopenharmony_ci	.enable_wol = bcmasp_enable_wol_shared,
119662306a36Sopenharmony_ci	.destroy_wol = bcmasp_wol_irq_destroy_shared,
119762306a36Sopenharmony_ci	.hw_info = &v21_hw_info,
119862306a36Sopenharmony_ci};
119962306a36Sopenharmony_ci
120062306a36Sopenharmony_cistatic const struct of_device_id bcmasp_of_match[] = {
120162306a36Sopenharmony_ci	{ .compatible = "brcm,asp-v2.0", .data = &v20_plat_data },
120262306a36Sopenharmony_ci	{ .compatible = "brcm,asp-v2.1", .data = &v21_plat_data },
120362306a36Sopenharmony_ci	{ /* sentinel */ },
120462306a36Sopenharmony_ci};
120562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, bcmasp_of_match);
120662306a36Sopenharmony_ci
120762306a36Sopenharmony_cistatic const struct of_device_id bcmasp_mdio_of_match[] = {
120862306a36Sopenharmony_ci	{ .compatible = "brcm,asp-v2.1-mdio", },
120962306a36Sopenharmony_ci	{ .compatible = "brcm,asp-v2.0-mdio", },
121062306a36Sopenharmony_ci	{ /* sentinel */ },
121162306a36Sopenharmony_ci};
121262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, bcmasp_mdio_of_match);
121362306a36Sopenharmony_ci
121462306a36Sopenharmony_cistatic void bcmasp_remove_intfs(struct bcmasp_priv *priv)
121562306a36Sopenharmony_ci{
121662306a36Sopenharmony_ci	struct bcmasp_intf *intf, *n;
121762306a36Sopenharmony_ci
121862306a36Sopenharmony_ci	list_for_each_entry_safe(intf, n, &priv->intfs, list) {
121962306a36Sopenharmony_ci		list_del(&intf->list);
122062306a36Sopenharmony_ci		bcmasp_interface_destroy(intf);
122162306a36Sopenharmony_ci	}
122262306a36Sopenharmony_ci}
122362306a36Sopenharmony_ci
122462306a36Sopenharmony_cistatic int bcmasp_probe(struct platform_device *pdev)
122562306a36Sopenharmony_ci{
122662306a36Sopenharmony_ci	struct device_node *ports_node, *intf_node;
122762306a36Sopenharmony_ci	const struct bcmasp_plat_data *pdata;
122862306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
122962306a36Sopenharmony_ci	struct bcmasp_priv *priv;
123062306a36Sopenharmony_ci	struct bcmasp_intf *intf;
123162306a36Sopenharmony_ci	int ret = 0, count = 0;
123262306a36Sopenharmony_ci	unsigned int i;
123362306a36Sopenharmony_ci
123462306a36Sopenharmony_ci	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
123562306a36Sopenharmony_ci	if (!priv)
123662306a36Sopenharmony_ci		return -ENOMEM;
123762306a36Sopenharmony_ci
123862306a36Sopenharmony_ci	priv->irq = platform_get_irq(pdev, 0);
123962306a36Sopenharmony_ci	if (priv->irq <= 0)
124062306a36Sopenharmony_ci		return -EINVAL;
124162306a36Sopenharmony_ci
124262306a36Sopenharmony_ci	priv->clk = devm_clk_get_optional_enabled(dev, "sw_asp");
124362306a36Sopenharmony_ci	if (IS_ERR(priv->clk))
124462306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(priv->clk),
124562306a36Sopenharmony_ci				     "failed to request clock\n");
124662306a36Sopenharmony_ci
124762306a36Sopenharmony_ci	/* Base from parent node */
124862306a36Sopenharmony_ci	priv->base = devm_platform_ioremap_resource(pdev, 0);
124962306a36Sopenharmony_ci	if (IS_ERR(priv->base))
125062306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(priv->base), "failed to iomap\n");
125162306a36Sopenharmony_ci
125262306a36Sopenharmony_ci	ret = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(40));
125362306a36Sopenharmony_ci	if (ret)
125462306a36Sopenharmony_ci		return dev_err_probe(dev, ret, "unable to set DMA mask: %d\n", ret);
125562306a36Sopenharmony_ci
125662306a36Sopenharmony_ci	dev_set_drvdata(&pdev->dev, priv);
125762306a36Sopenharmony_ci	priv->pdev = pdev;
125862306a36Sopenharmony_ci	spin_lock_init(&priv->mda_lock);
125962306a36Sopenharmony_ci	spin_lock_init(&priv->clk_lock);
126062306a36Sopenharmony_ci	mutex_init(&priv->wol_lock);
126162306a36Sopenharmony_ci	mutex_init(&priv->net_lock);
126262306a36Sopenharmony_ci	INIT_LIST_HEAD(&priv->intfs);
126362306a36Sopenharmony_ci
126462306a36Sopenharmony_ci	pdata = device_get_match_data(&pdev->dev);
126562306a36Sopenharmony_ci	if (!pdata)
126662306a36Sopenharmony_ci		return dev_err_probe(dev, -EINVAL, "unable to find platform data\n");
126762306a36Sopenharmony_ci
126862306a36Sopenharmony_ci	priv->init_wol = pdata->init_wol;
126962306a36Sopenharmony_ci	priv->enable_wol = pdata->enable_wol;
127062306a36Sopenharmony_ci	priv->destroy_wol = pdata->destroy_wol;
127162306a36Sopenharmony_ci	priv->hw_info = pdata->hw_info;
127262306a36Sopenharmony_ci
127362306a36Sopenharmony_ci	/* Enable all clocks to ensure successful probing */
127462306a36Sopenharmony_ci	bcmasp_core_clock_set(priv, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE, 0);
127562306a36Sopenharmony_ci
127662306a36Sopenharmony_ci	/* Switch to the main clock */
127762306a36Sopenharmony_ci	bcmasp_core_clock_select(priv, false);
127862306a36Sopenharmony_ci
127962306a36Sopenharmony_ci	bcmasp_intr2_mask_set_all(priv);
128062306a36Sopenharmony_ci	bcmasp_intr2_clear_all(priv);
128162306a36Sopenharmony_ci
128262306a36Sopenharmony_ci	ret = devm_request_irq(&pdev->dev, priv->irq, bcmasp_isr, 0,
128362306a36Sopenharmony_ci			       pdev->name, priv);
128462306a36Sopenharmony_ci	if (ret)
128562306a36Sopenharmony_ci		return dev_err_probe(dev, ret, "failed to request ASP interrupt: %d", ret);
128662306a36Sopenharmony_ci
128762306a36Sopenharmony_ci	/* Register mdio child nodes */
128862306a36Sopenharmony_ci	of_platform_populate(dev->of_node, bcmasp_mdio_of_match, NULL, dev);
128962306a36Sopenharmony_ci
129062306a36Sopenharmony_ci	/* ASP specific initialization, Needs to be done regardless of
129162306a36Sopenharmony_ci	 * how many interfaces come up.
129262306a36Sopenharmony_ci	 */
129362306a36Sopenharmony_ci	bcmasp_core_init(priv);
129462306a36Sopenharmony_ci	bcmasp_core_init_filters(priv);
129562306a36Sopenharmony_ci
129662306a36Sopenharmony_ci	ports_node = of_find_node_by_name(dev->of_node, "ethernet-ports");
129762306a36Sopenharmony_ci	if (!ports_node) {
129862306a36Sopenharmony_ci		dev_warn(dev, "No ports found\n");
129962306a36Sopenharmony_ci		return -EINVAL;
130062306a36Sopenharmony_ci	}
130162306a36Sopenharmony_ci
130262306a36Sopenharmony_ci	i = 0;
130362306a36Sopenharmony_ci	for_each_available_child_of_node(ports_node, intf_node) {
130462306a36Sopenharmony_ci		intf = bcmasp_interface_create(priv, intf_node, i);
130562306a36Sopenharmony_ci		if (!intf) {
130662306a36Sopenharmony_ci			dev_err(dev, "Cannot create eth interface %d\n", i);
130762306a36Sopenharmony_ci			bcmasp_remove_intfs(priv);
130862306a36Sopenharmony_ci			of_node_put(intf_node);
130962306a36Sopenharmony_ci			goto of_put_exit;
131062306a36Sopenharmony_ci		}
131162306a36Sopenharmony_ci		list_add_tail(&intf->list, &priv->intfs);
131262306a36Sopenharmony_ci		i++;
131362306a36Sopenharmony_ci	}
131462306a36Sopenharmony_ci
131562306a36Sopenharmony_ci	/* Check and enable WoL */
131662306a36Sopenharmony_ci	priv->init_wol(priv);
131762306a36Sopenharmony_ci
131862306a36Sopenharmony_ci	/* Drop the clock reference count now and let ndo_open()/ndo_close()
131962306a36Sopenharmony_ci	 * manage it for us from now on.
132062306a36Sopenharmony_ci	 */
132162306a36Sopenharmony_ci	bcmasp_core_clock_set(priv, 0, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE);
132262306a36Sopenharmony_ci
132362306a36Sopenharmony_ci	clk_disable_unprepare(priv->clk);
132462306a36Sopenharmony_ci
132562306a36Sopenharmony_ci	/* Now do the registration of the network ports which will take care
132662306a36Sopenharmony_ci	 * of managing the clock properly.
132762306a36Sopenharmony_ci	 */
132862306a36Sopenharmony_ci	list_for_each_entry(intf, &priv->intfs, list) {
132962306a36Sopenharmony_ci		ret = register_netdev(intf->ndev);
133062306a36Sopenharmony_ci		if (ret) {
133162306a36Sopenharmony_ci			netdev_err(intf->ndev,
133262306a36Sopenharmony_ci				   "failed to register net_device: %d\n", ret);
133362306a36Sopenharmony_ci			priv->destroy_wol(priv);
133462306a36Sopenharmony_ci			bcmasp_remove_intfs(priv);
133562306a36Sopenharmony_ci			goto of_put_exit;
133662306a36Sopenharmony_ci		}
133762306a36Sopenharmony_ci		count++;
133862306a36Sopenharmony_ci	}
133962306a36Sopenharmony_ci
134062306a36Sopenharmony_ci	dev_info(dev, "Initialized %d port(s)\n", count);
134162306a36Sopenharmony_ci
134262306a36Sopenharmony_ciof_put_exit:
134362306a36Sopenharmony_ci	of_node_put(ports_node);
134462306a36Sopenharmony_ci	return ret;
134562306a36Sopenharmony_ci}
134662306a36Sopenharmony_ci
134762306a36Sopenharmony_cistatic int bcmasp_remove(struct platform_device *pdev)
134862306a36Sopenharmony_ci{
134962306a36Sopenharmony_ci	struct bcmasp_priv *priv = dev_get_drvdata(&pdev->dev);
135062306a36Sopenharmony_ci
135162306a36Sopenharmony_ci	if (!priv)
135262306a36Sopenharmony_ci		return 0;
135362306a36Sopenharmony_ci
135462306a36Sopenharmony_ci	priv->destroy_wol(priv);
135562306a36Sopenharmony_ci	bcmasp_remove_intfs(priv);
135662306a36Sopenharmony_ci
135762306a36Sopenharmony_ci	return 0;
135862306a36Sopenharmony_ci}
135962306a36Sopenharmony_ci
136062306a36Sopenharmony_cistatic void bcmasp_shutdown(struct platform_device *pdev)
136162306a36Sopenharmony_ci{
136262306a36Sopenharmony_ci	bcmasp_remove(pdev);
136362306a36Sopenharmony_ci}
136462306a36Sopenharmony_ci
136562306a36Sopenharmony_cistatic int __maybe_unused bcmasp_suspend(struct device *d)
136662306a36Sopenharmony_ci{
136762306a36Sopenharmony_ci	struct bcmasp_priv *priv = dev_get_drvdata(d);
136862306a36Sopenharmony_ci	struct bcmasp_intf *intf;
136962306a36Sopenharmony_ci	int ret;
137062306a36Sopenharmony_ci
137162306a36Sopenharmony_ci	list_for_each_entry(intf, &priv->intfs, list) {
137262306a36Sopenharmony_ci		ret = bcmasp_interface_suspend(intf);
137362306a36Sopenharmony_ci		if (ret)
137462306a36Sopenharmony_ci			break;
137562306a36Sopenharmony_ci	}
137662306a36Sopenharmony_ci
137762306a36Sopenharmony_ci	ret = clk_prepare_enable(priv->clk);
137862306a36Sopenharmony_ci	if (ret)
137962306a36Sopenharmony_ci		return ret;
138062306a36Sopenharmony_ci
138162306a36Sopenharmony_ci	/* Whether Wake-on-LAN is enabled or not, we can always disable
138262306a36Sopenharmony_ci	 * the shared TX clock
138362306a36Sopenharmony_ci	 */
138462306a36Sopenharmony_ci	bcmasp_core_clock_set(priv, 0, ASP_CTRL_CLOCK_CTRL_ASP_TX_DISABLE);
138562306a36Sopenharmony_ci
138662306a36Sopenharmony_ci	bcmasp_core_clock_select(priv, true);
138762306a36Sopenharmony_ci
138862306a36Sopenharmony_ci	clk_disable_unprepare(priv->clk);
138962306a36Sopenharmony_ci
139062306a36Sopenharmony_ci	return ret;
139162306a36Sopenharmony_ci}
139262306a36Sopenharmony_ci
139362306a36Sopenharmony_cistatic int __maybe_unused bcmasp_resume(struct device *d)
139462306a36Sopenharmony_ci{
139562306a36Sopenharmony_ci	struct bcmasp_priv *priv = dev_get_drvdata(d);
139662306a36Sopenharmony_ci	struct bcmasp_intf *intf;
139762306a36Sopenharmony_ci	int ret;
139862306a36Sopenharmony_ci
139962306a36Sopenharmony_ci	ret = clk_prepare_enable(priv->clk);
140062306a36Sopenharmony_ci	if (ret)
140162306a36Sopenharmony_ci		return ret;
140262306a36Sopenharmony_ci
140362306a36Sopenharmony_ci	/* Switch to the main clock domain */
140462306a36Sopenharmony_ci	bcmasp_core_clock_select(priv, false);
140562306a36Sopenharmony_ci
140662306a36Sopenharmony_ci	/* Re-enable all clocks for re-initialization */
140762306a36Sopenharmony_ci	bcmasp_core_clock_set(priv, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE, 0);
140862306a36Sopenharmony_ci
140962306a36Sopenharmony_ci	bcmasp_core_init(priv);
141062306a36Sopenharmony_ci	bcmasp_core_init_filters(priv);
141162306a36Sopenharmony_ci
141262306a36Sopenharmony_ci	/* And disable them to let the network devices take care of them */
141362306a36Sopenharmony_ci	bcmasp_core_clock_set(priv, 0, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE);
141462306a36Sopenharmony_ci
141562306a36Sopenharmony_ci	clk_disable_unprepare(priv->clk);
141662306a36Sopenharmony_ci
141762306a36Sopenharmony_ci	list_for_each_entry(intf, &priv->intfs, list) {
141862306a36Sopenharmony_ci		ret = bcmasp_interface_resume(intf);
141962306a36Sopenharmony_ci		if (ret)
142062306a36Sopenharmony_ci			break;
142162306a36Sopenharmony_ci	}
142262306a36Sopenharmony_ci
142362306a36Sopenharmony_ci	return ret;
142462306a36Sopenharmony_ci}
142562306a36Sopenharmony_ci
142662306a36Sopenharmony_cistatic SIMPLE_DEV_PM_OPS(bcmasp_pm_ops,
142762306a36Sopenharmony_ci			 bcmasp_suspend, bcmasp_resume);
142862306a36Sopenharmony_ci
142962306a36Sopenharmony_cistatic struct platform_driver bcmasp_driver = {
143062306a36Sopenharmony_ci	.probe = bcmasp_probe,
143162306a36Sopenharmony_ci	.remove = bcmasp_remove,
143262306a36Sopenharmony_ci	.shutdown = bcmasp_shutdown,
143362306a36Sopenharmony_ci	.driver = {
143462306a36Sopenharmony_ci		.name = "brcm,asp-v2",
143562306a36Sopenharmony_ci		.of_match_table = bcmasp_of_match,
143662306a36Sopenharmony_ci		.pm = &bcmasp_pm_ops,
143762306a36Sopenharmony_ci	},
143862306a36Sopenharmony_ci};
143962306a36Sopenharmony_cimodule_platform_driver(bcmasp_driver);
144062306a36Sopenharmony_ci
144162306a36Sopenharmony_ciMODULE_DESCRIPTION("Broadcom ASP 2.0 Ethernet controller driver");
144262306a36Sopenharmony_ciMODULE_ALIAS("platform:brcm,asp-v2");
144362306a36Sopenharmony_ciMODULE_LICENSE("GPL");
1444