Commit 95c41842 authored by David S. Miller's avatar David S. Miller

Merge branch 'dsa-ll-fixes'

Vladimir Oltean says:

====================
dsa: Fix mangled link-local MAC DAs with SJA1105 DSA

The SJA1105 hardware tagging protocol is weird and will put DSA
information (source port, switch ID) in the MAC DA of the packets sent
to the CPU, and then send some additional (meta) packets which contain
the original bytes from the previous packet's MAC DA.

The tagging protocol driver contains logic to handle this, but the meta
frames are optional functionality, and there are configurations when
they aren't received (no PTP RX timestamping). Thus, the MAC DA from
packets sent to the stack is not correct in all cases.

Also, during testing it was found that the MAC DA patching procedure was
incorrect.

The investigation comes as a result of this discussion with Paolo:
https://lore.kernel.org/netdev/f494387c8d55d9b1d5a3e88beedeeb448f2e6cc3.camel@redhat.com/
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents ba7bdec3 a372d66a
...@@ -252,6 +252,7 @@ struct sja1105_private { ...@@ -252,6 +252,7 @@ struct sja1105_private {
unsigned long ucast_egress_floods; unsigned long ucast_egress_floods;
unsigned long bcast_egress_floods; unsigned long bcast_egress_floods;
unsigned long hwts_tx_en; unsigned long hwts_tx_en;
unsigned long hwts_rx_en;
const struct sja1105_info *info; const struct sja1105_info *info;
size_t max_xfer_len; size_t max_xfer_len;
struct spi_device *spidev; struct spi_device *spidev;
...@@ -289,7 +290,6 @@ struct sja1105_spi_message { ...@@ -289,7 +290,6 @@ struct sja1105_spi_message {
/* From sja1105_main.c */ /* From sja1105_main.c */
enum sja1105_reset_reason { enum sja1105_reset_reason {
SJA1105_VLAN_FILTERING = 0, SJA1105_VLAN_FILTERING = 0,
SJA1105_RX_HWTSTAMPING,
SJA1105_AGEING_TIME, SJA1105_AGEING_TIME,
SJA1105_SCHEDULING, SJA1105_SCHEDULING,
SJA1105_BEST_EFFORT_POLICING, SJA1105_BEST_EFFORT_POLICING,
......
...@@ -867,11 +867,11 @@ static int sja1105_init_general_params(struct sja1105_private *priv) ...@@ -867,11 +867,11 @@ static int sja1105_init_general_params(struct sja1105_private *priv)
.mac_fltres1 = SJA1105_LINKLOCAL_FILTER_A, .mac_fltres1 = SJA1105_LINKLOCAL_FILTER_A,
.mac_flt1 = SJA1105_LINKLOCAL_FILTER_A_MASK, .mac_flt1 = SJA1105_LINKLOCAL_FILTER_A_MASK,
.incl_srcpt1 = true, .incl_srcpt1 = true,
.send_meta1 = false, .send_meta1 = true,
.mac_fltres0 = SJA1105_LINKLOCAL_FILTER_B, .mac_fltres0 = SJA1105_LINKLOCAL_FILTER_B,
.mac_flt0 = SJA1105_LINKLOCAL_FILTER_B_MASK, .mac_flt0 = SJA1105_LINKLOCAL_FILTER_B_MASK,
.incl_srcpt0 = true, .incl_srcpt0 = true,
.send_meta0 = false, .send_meta0 = true,
/* Default to an invalid value */ /* Default to an invalid value */
.mirr_port = priv->ds->num_ports, .mirr_port = priv->ds->num_ports,
/* No TTEthernet */ /* No TTEthernet */
...@@ -2215,7 +2215,6 @@ static int sja1105_reload_cbs(struct sja1105_private *priv) ...@@ -2215,7 +2215,6 @@ static int sja1105_reload_cbs(struct sja1105_private *priv)
static const char * const sja1105_reset_reasons[] = { static const char * const sja1105_reset_reasons[] = {
[SJA1105_VLAN_FILTERING] = "VLAN filtering", [SJA1105_VLAN_FILTERING] = "VLAN filtering",
[SJA1105_RX_HWTSTAMPING] = "RX timestamping",
[SJA1105_AGEING_TIME] = "Ageing time", [SJA1105_AGEING_TIME] = "Ageing time",
[SJA1105_SCHEDULING] = "Time-aware scheduling", [SJA1105_SCHEDULING] = "Time-aware scheduling",
[SJA1105_BEST_EFFORT_POLICING] = "Best-effort policing", [SJA1105_BEST_EFFORT_POLICING] = "Best-effort policing",
......
...@@ -58,35 +58,10 @@ enum sja1105_ptp_clk_mode { ...@@ -58,35 +58,10 @@ enum sja1105_ptp_clk_mode {
#define ptp_data_to_sja1105(d) \ #define ptp_data_to_sja1105(d) \
container_of((d), struct sja1105_private, ptp_data) container_of((d), struct sja1105_private, ptp_data)
/* Must be called only while the RX timestamping state of the tagger
* is turned off
*/
static int sja1105_change_rxtstamping(struct sja1105_private *priv,
bool on)
{
struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
struct sja1105_general_params_entry *general_params;
struct sja1105_table *table;
table = &priv->static_config.tables[BLK_IDX_GENERAL_PARAMS];
general_params = table->entries;
general_params->send_meta1 = on;
general_params->send_meta0 = on;
ptp_cancel_worker_sync(ptp_data->clock);
skb_queue_purge(&ptp_data->skb_txtstamp_queue);
skb_queue_purge(&ptp_data->skb_rxtstamp_queue);
return sja1105_static_config_reload(priv, SJA1105_RX_HWTSTAMPING);
}
int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr)
{ {
struct sja1105_tagger_data *tagger_data = sja1105_tagger_data(ds);
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct hwtstamp_config config; struct hwtstamp_config config;
bool rx_on;
int rc;
if (copy_from_user(&config, ifr->ifr_data, sizeof(config))) if (copy_from_user(&config, ifr->ifr_data, sizeof(config)))
return -EFAULT; return -EFAULT;
...@@ -104,26 +79,13 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -104,26 +79,13 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr)
switch (config.rx_filter) { switch (config.rx_filter) {
case HWTSTAMP_FILTER_NONE: case HWTSTAMP_FILTER_NONE:
rx_on = false; priv->hwts_rx_en &= ~BIT(port);
break; break;
default: default:
rx_on = true; priv->hwts_rx_en |= BIT(port);
break; break;
} }
if (rx_on != tagger_data->rxtstamp_get_state(ds)) {
tagger_data->rxtstamp_set_state(ds, false);
rc = sja1105_change_rxtstamping(priv, rx_on);
if (rc < 0) {
dev_err(ds->dev,
"Failed to change RX timestamping: %d\n", rc);
return rc;
}
if (rx_on)
tagger_data->rxtstamp_set_state(ds, true);
}
if (copy_to_user(ifr->ifr_data, &config, sizeof(config))) if (copy_to_user(ifr->ifr_data, &config, sizeof(config)))
return -EFAULT; return -EFAULT;
return 0; return 0;
...@@ -131,7 +93,6 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -131,7 +93,6 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr)
int sja1105_hwtstamp_get(struct dsa_switch *ds, int port, struct ifreq *ifr) int sja1105_hwtstamp_get(struct dsa_switch *ds, int port, struct ifreq *ifr)
{ {
struct sja1105_tagger_data *tagger_data = sja1105_tagger_data(ds);
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct hwtstamp_config config; struct hwtstamp_config config;
...@@ -140,7 +101,7 @@ int sja1105_hwtstamp_get(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -140,7 +101,7 @@ int sja1105_hwtstamp_get(struct dsa_switch *ds, int port, struct ifreq *ifr)
config.tx_type = HWTSTAMP_TX_ON; config.tx_type = HWTSTAMP_TX_ON;
else else
config.tx_type = HWTSTAMP_TX_OFF; config.tx_type = HWTSTAMP_TX_OFF;
if (tagger_data->rxtstamp_get_state(ds)) if (priv->hwts_rx_en & BIT(port))
config.rx_filter = HWTSTAMP_FILTER_PTP_V2_L2_EVENT; config.rx_filter = HWTSTAMP_FILTER_PTP_V2_L2_EVENT;
else else
config.rx_filter = HWTSTAMP_FILTER_NONE; config.rx_filter = HWTSTAMP_FILTER_NONE;
...@@ -413,11 +374,10 @@ static long sja1105_rxtstamp_work(struct ptp_clock_info *ptp) ...@@ -413,11 +374,10 @@ static long sja1105_rxtstamp_work(struct ptp_clock_info *ptp)
bool sja1105_rxtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb) bool sja1105_rxtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb)
{ {
struct sja1105_tagger_data *tagger_data = sja1105_tagger_data(ds);
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct sja1105_ptp_data *ptp_data = &priv->ptp_data; struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
if (!tagger_data->rxtstamp_get_state(ds)) if (!(priv->hwts_rx_en & BIT(port)))
return false; return false;
/* We need to read the full PTP clock to reconstruct the Rx /* We need to read the full PTP clock to reconstruct the Rx
......
...@@ -48,13 +48,9 @@ struct sja1105_deferred_xmit_work { ...@@ -48,13 +48,9 @@ struct sja1105_deferred_xmit_work {
/* Global tagger data */ /* Global tagger data */
struct sja1105_tagger_data { struct sja1105_tagger_data {
/* Tagger to switch */
void (*xmit_work_fn)(struct kthread_work *work); void (*xmit_work_fn)(struct kthread_work *work);
void (*meta_tstamp_handler)(struct dsa_switch *ds, int port, u8 ts_id, void (*meta_tstamp_handler)(struct dsa_switch *ds, int port, u8 ts_id,
enum sja1110_meta_tstamp dir, u64 tstamp); enum sja1110_meta_tstamp dir, u64 tstamp);
/* Switch to tagger */
bool (*rxtstamp_get_state)(struct dsa_switch *ds);
void (*rxtstamp_set_state)(struct dsa_switch *ds, bool on);
}; };
struct sja1105_skb_cb { struct sja1105_skb_cb {
......
...@@ -58,11 +58,8 @@ ...@@ -58,11 +58,8 @@
#define SJA1110_TX_TRAILER_LEN 4 #define SJA1110_TX_TRAILER_LEN 4
#define SJA1110_MAX_PADDING_LEN 15 #define SJA1110_MAX_PADDING_LEN 15
#define SJA1105_HWTS_RX_EN 0
struct sja1105_tagger_private { struct sja1105_tagger_private {
struct sja1105_tagger_data data; /* Must be first */ struct sja1105_tagger_data data; /* Must be first */
unsigned long state;
/* Protects concurrent access to the meta state machine /* Protects concurrent access to the meta state machine
* from taggers running on multiple ports on SMP systems * from taggers running on multiple ports on SMP systems
*/ */
...@@ -118,8 +115,8 @@ static void sja1105_meta_unpack(const struct sk_buff *skb, ...@@ -118,8 +115,8 @@ static void sja1105_meta_unpack(const struct sk_buff *skb,
* a unified unpacking command for both device series. * a unified unpacking command for both device series.
*/ */
packing(buf, &meta->tstamp, 31, 0, 4, UNPACK, 0); packing(buf, &meta->tstamp, 31, 0, 4, UNPACK, 0);
packing(buf + 4, &meta->dmac_byte_4, 7, 0, 1, UNPACK, 0); packing(buf + 4, &meta->dmac_byte_3, 7, 0, 1, UNPACK, 0);
packing(buf + 5, &meta->dmac_byte_3, 7, 0, 1, UNPACK, 0); packing(buf + 5, &meta->dmac_byte_4, 7, 0, 1, UNPACK, 0);
packing(buf + 6, &meta->source_port, 7, 0, 1, UNPACK, 0); packing(buf + 6, &meta->source_port, 7, 0, 1, UNPACK, 0);
packing(buf + 7, &meta->switch_id, 7, 0, 1, UNPACK, 0); packing(buf + 7, &meta->switch_id, 7, 0, 1, UNPACK, 0);
} }
...@@ -392,10 +389,6 @@ static struct sk_buff ...@@ -392,10 +389,6 @@ static struct sk_buff
priv = sja1105_tagger_private(ds); priv = sja1105_tagger_private(ds);
if (!test_bit(SJA1105_HWTS_RX_EN, &priv->state))
/* Do normal processing. */
return skb;
spin_lock(&priv->meta_lock); spin_lock(&priv->meta_lock);
/* Was this a link-local frame instead of the meta /* Was this a link-local frame instead of the meta
* that we were expecting? * that we were expecting?
...@@ -431,12 +424,6 @@ static struct sk_buff ...@@ -431,12 +424,6 @@ static struct sk_buff
priv = sja1105_tagger_private(ds); priv = sja1105_tagger_private(ds);
/* Drop the meta frame if we're not in the right state
* to process it.
*/
if (!test_bit(SJA1105_HWTS_RX_EN, &priv->state))
return NULL;
spin_lock(&priv->meta_lock); spin_lock(&priv->meta_lock);
stampable_skb = priv->stampable_skb; stampable_skb = priv->stampable_skb;
...@@ -472,30 +459,6 @@ static struct sk_buff ...@@ -472,30 +459,6 @@ static struct sk_buff
return skb; return skb;
} }
static bool sja1105_rxtstamp_get_state(struct dsa_switch *ds)
{
struct sja1105_tagger_private *priv = sja1105_tagger_private(ds);
return test_bit(SJA1105_HWTS_RX_EN, &priv->state);
}
static void sja1105_rxtstamp_set_state(struct dsa_switch *ds, bool on)
{
struct sja1105_tagger_private *priv = sja1105_tagger_private(ds);
if (on)
set_bit(SJA1105_HWTS_RX_EN, &priv->state);
else
clear_bit(SJA1105_HWTS_RX_EN, &priv->state);
/* Initialize the meta state machine to a known state */
if (!priv->stampable_skb)
return;
kfree_skb(priv->stampable_skb);
priv->stampable_skb = NULL;
}
static bool sja1105_skb_has_tag_8021q(const struct sk_buff *skb) static bool sja1105_skb_has_tag_8021q(const struct sk_buff *skb)
{ {
u16 tpid = ntohs(eth_hdr(skb)->h_proto); u16 tpid = ntohs(eth_hdr(skb)->h_proto);
...@@ -552,9 +515,6 @@ static struct sk_buff *sja1105_rcv(struct sk_buff *skb, ...@@ -552,9 +515,6 @@ static struct sk_buff *sja1105_rcv(struct sk_buff *skb,
*/ */
source_port = hdr->h_dest[3]; source_port = hdr->h_dest[3];
switch_id = hdr->h_dest[4]; switch_id = hdr->h_dest[4];
/* Clear the DMAC bytes that were mangled by the switch */
hdr->h_dest[3] = 0;
hdr->h_dest[4] = 0;
} else if (is_meta) { } else if (is_meta) {
sja1105_meta_unpack(skb, &meta); sja1105_meta_unpack(skb, &meta);
source_port = meta.source_port; source_port = meta.source_port;
...@@ -785,7 +745,6 @@ static void sja1105_disconnect(struct dsa_switch *ds) ...@@ -785,7 +745,6 @@ static void sja1105_disconnect(struct dsa_switch *ds)
static int sja1105_connect(struct dsa_switch *ds) static int sja1105_connect(struct dsa_switch *ds)
{ {
struct sja1105_tagger_data *tagger_data;
struct sja1105_tagger_private *priv; struct sja1105_tagger_private *priv;
struct kthread_worker *xmit_worker; struct kthread_worker *xmit_worker;
int err; int err;
...@@ -805,10 +764,6 @@ static int sja1105_connect(struct dsa_switch *ds) ...@@ -805,10 +764,6 @@ static int sja1105_connect(struct dsa_switch *ds)
} }
priv->xmit_worker = xmit_worker; priv->xmit_worker = xmit_worker;
/* Export functions for switch driver use */
tagger_data = &priv->data;
tagger_data->rxtstamp_get_state = sja1105_rxtstamp_get_state;
tagger_data->rxtstamp_set_state = sja1105_rxtstamp_set_state;
ds->tagger_data = priv; ds->tagger_data = priv;
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment