incl_srcpt has the limitation, mentioned in commit 
b4638af8885a ("net:
dsa: sja1105: always enable the INCL_SRCPT option"), that frames with a
MAC DA of 01:80:c2:xx:yy:zz will be received as 01:80:c2:00:00:zz unless
PTP RX timestamping is enabled.
The incl_srcpt option was initially unconditionally enabled, then that
changed with commit 
42824463d38d ("net: dsa: sja1105: Limit use of
incl_srcpt to bridge+vlan mode"), then again with 
b4638af8885a ("net:
dsa: sja1105: always enable the INCL_SRCPT option"). Bottom line is that
it now needs to be always enabled, otherwise the driver does not have a
reliable source of information regarding source_port and switch_id for
link-local traffic (tag_8021q VLANs may be imprecise since now they
identify an entire bridging domain when ports are not standalone).
If we accept that PTP RX timestamping (and therefore, meta frame
generation) is always enabled in hardware, then that limitation could be
avoided and packets with any MAC DA can be properly received, because
meta frames do contain the original bytes from the MAC DA of their
associated link-local packet.
This change enables meta frame generation unconditionally, which also
has the nice side effects of simplifying the switch control path
(a switch reset is no longer required on hwtstamping settings change)
and the tagger data path (it no longer needs to be informed whether to
expect meta frames or not - it always does).
Fixes: 227d07a07ef1 ("net: dsa: sja1105: Add support for traffic through standalone ports")
Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com>
Reviewed-by: Simon Horman <simon.horman@corigine.com>
Reviewed-by: Florian Fainelli <florian.fainelli@broadcom.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
        unsigned long ucast_egress_floods;
        unsigned long bcast_egress_floods;
        unsigned long hwts_tx_en;
+       unsigned long hwts_rx_en;
        const struct sja1105_info *info;
        size_t max_xfer_len;
        struct spi_device *spidev;
 /* From sja1105_main.c */
 enum sja1105_reset_reason {
        SJA1105_VLAN_FILTERING = 0,
-       SJA1105_RX_HWTSTAMPING,
        SJA1105_AGEING_TIME,
        SJA1105_SCHEDULING,
        SJA1105_BEST_EFFORT_POLICING,
 
                .mac_fltres1 = SJA1105_LINKLOCAL_FILTER_A,
                .mac_flt1    = SJA1105_LINKLOCAL_FILTER_A_MASK,
                .incl_srcpt1 = true,
-               .send_meta1  = false,
+               .send_meta1  = true,
                .mac_fltres0 = SJA1105_LINKLOCAL_FILTER_B,
                .mac_flt0    = SJA1105_LINKLOCAL_FILTER_B_MASK,
                .incl_srcpt0 = true,
-               .send_meta0  = false,
+               .send_meta0  = true,
                /* Default to an invalid value */
                .mirr_port = priv->ds->num_ports,
                /* No TTEthernet */
 
 static const char * const sja1105_reset_reasons[] = {
        [SJA1105_VLAN_FILTERING] = "VLAN filtering",
-       [SJA1105_RX_HWTSTAMPING] = "RX timestamping",
        [SJA1105_AGEING_TIME] = "Ageing time",
        [SJA1105_SCHEDULING] = "Time-aware scheduling",
        [SJA1105_BEST_EFFORT_POLICING] = "Best-effort policing",
 
 #define ptp_data_to_sja1105(d) \
                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)
 {
-       struct sja1105_tagger_data *tagger_data = sja1105_tagger_data(ds);
        struct sja1105_private *priv = ds->priv;
        struct hwtstamp_config config;
-       bool rx_on;
-       int rc;
 
        if (copy_from_user(&config, ifr->ifr_data, sizeof(config)))
                return -EFAULT;
 
        switch (config.rx_filter) {
        case HWTSTAMP_FILTER_NONE:
-               rx_on = false;
+               priv->hwts_rx_en &= ~BIT(port);
                break;
        default:
-               rx_on = true;
+               priv->hwts_rx_en |= BIT(port);
                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)))
                return -EFAULT;
        return 0;
 
 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 hwtstamp_config config;
 
                config.tx_type = HWTSTAMP_TX_ON;
        else
                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;
        else
                config.rx_filter = HWTSTAMP_FILTER_NONE;
 
 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_ptp_data *ptp_data = &priv->ptp_data;
 
-       if (!tagger_data->rxtstamp_get_state(ds))
+       if (!(priv->hwts_rx_en & BIT(port)))
                return false;
 
        /* We need to read the full PTP clock to reconstruct the Rx
 
 
 /* Global tagger data */
 struct sja1105_tagger_data {
-       /* Tagger to switch */
        void (*xmit_work_fn)(struct kthread_work *work);
        void (*meta_tstamp_handler)(struct dsa_switch *ds, int port, u8 ts_id,
                                    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 {
 
 #define SJA1110_TX_TRAILER_LEN                 4
 #define SJA1110_MAX_PADDING_LEN                        15
 
-#define SJA1105_HWTS_RX_EN                     0
-
 struct sja1105_tagger_private {
        struct sja1105_tagger_data data; /* Must be first */
-       unsigned long state;
        /* Protects concurrent access to the meta state machine
         * from taggers running on multiple ports on SMP systems
         */
 
                priv = sja1105_tagger_private(ds);
 
-               if (!test_bit(SJA1105_HWTS_RX_EN, &priv->state))
-                       /* Do normal processing. */
-                       return skb;
-
                spin_lock(&priv->meta_lock);
                /* Was this a link-local frame instead of the meta
                 * that we were expecting?
 
                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);
 
                stampable_skb = priv->stampable_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)
 {
        u16 tpid = ntohs(eth_hdr(skb)->h_proto);
                 */
                source_port = hdr->h_dest[3];
                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) {
                sja1105_meta_unpack(skb, &meta);
                source_port = meta.source_port;
 
 static int sja1105_connect(struct dsa_switch *ds)
 {
-       struct sja1105_tagger_data *tagger_data;
        struct sja1105_tagger_private *priv;
        struct kthread_worker *xmit_worker;
        int err;
        }
 
        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;
 
        return 0;