Commit 9b5bcb19 authored by David S. Miller's avatar David S. Miller

Merge branch 'dsa-tagger-storage'

Vladimir Oltean says:

====================
Replace DSA dp->priv with tagger-owned storage

Ansuel's recent work on qca8k register access over Ethernet:
https://patchwork.kernel.org/project/netdevbpf/cover/20211207145942.7444-1-ansuelsmth@gmail.com/
has triggered me to do something which I should've done for a longer
time:
https://patchwork.kernel.org/project/netdevbpf/patch/20211109095013.27829-7-martin.kaistra@linutronix.de/#24585521
which is to replace dp->priv with something that has less caveats.

The dp->priv was introduced when sja1105 needed to hold stateful
information in the tagging protocol driver. In that design, dp->priv
held memory allocated by the switch driver, because the tagging protocol
driver design was 100% stateless.

Some years have passed and others have started to feel the need for
stateful information kept by the tagger, as well as passing data back
and forth between the tagging protocol driver and the switch driver.
This isn't possible cleanly in DSA due to a circular dependency which
leads to broken module autoloading:
https://lore.kernel.org/netdev/20210908220834.d7gmtnwrorhharna@skbuf/

This patchset introduces a framework that resembles something normal,
which allows data to be passed from the tagging protocol driver (things
like switch management packets, which aren't intended for the network
stack) to the switch driver, while the tagging protocol still remains
more or less stateless. The overall design of the framework was
discussed with Ansuel too and it appears to be flexible enough to cover
the "register access over Ethernet" use case. Additionally, the existing
uses of dp->priv, which have mainly to do with PTP timestamping, have
also been migrated.

Changes in v2:
Fix transient build breakage in patch 5/11 due to a missing parenthesis,
https://patchwork.hopto.org/static/nipa/592567/12665213/build_clang/
and another transient build warning in patch 4/11 that for some reason
doesn't appear in my W=1 C=1 build.
https://patchwork.hopto.org/static/nipa/592567/12665209/build_clang/stderr
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents e0068620 4f3cb343
...@@ -1155,38 +1155,22 @@ static void felix_port_deferred_xmit(struct kthread_work *work) ...@@ -1155,38 +1155,22 @@ static void felix_port_deferred_xmit(struct kthread_work *work)
kfree(xmit_work); kfree(xmit_work);
} }
static int felix_port_setup_tagger_data(struct dsa_switch *ds, int port) static int felix_connect_tag_protocol(struct dsa_switch *ds,
enum dsa_tag_protocol proto)
{ {
struct dsa_port *dp = dsa_to_port(ds, port); struct ocelot_8021q_tagger_data *tagger_data;
struct ocelot *ocelot = ds->priv;
struct felix *felix = ocelot_to_felix(ocelot);
struct felix_port *felix_port;
if (!dsa_port_is_user(dp)) switch (proto) {
case DSA_TAG_PROTO_OCELOT_8021Q:
tagger_data = ocelot_8021q_tagger_data(ds);
tagger_data->xmit_work_fn = felix_port_deferred_xmit;
return 0; return 0;
case DSA_TAG_PROTO_OCELOT:
felix_port = kzalloc(sizeof(*felix_port), GFP_KERNEL); case DSA_TAG_PROTO_SEVILLE:
if (!felix_port)
return -ENOMEM;
felix_port->xmit_worker = felix->xmit_worker;
felix_port->xmit_work_fn = felix_port_deferred_xmit;
dp->priv = felix_port;
return 0; return 0;
} default:
return -EPROTONOSUPPORT;
static void felix_port_teardown_tagger_data(struct dsa_switch *ds, int port) }
{
struct dsa_port *dp = dsa_to_port(ds, port);
struct felix_port *felix_port = dp->priv;
if (!felix_port)
return;
dp->priv = NULL;
kfree(felix_port);
} }
/* Hardware initialization done here so that we can allocate structures with /* Hardware initialization done here so that we can allocate structures with
...@@ -1217,12 +1201,6 @@ static int felix_setup(struct dsa_switch *ds) ...@@ -1217,12 +1201,6 @@ static int felix_setup(struct dsa_switch *ds)
} }
} }
felix->xmit_worker = kthread_create_worker(0, "felix_xmit");
if (IS_ERR(felix->xmit_worker)) {
err = PTR_ERR(felix->xmit_worker);
goto out_deinit_timestamp;
}
for (port = 0; port < ds->num_ports; port++) { for (port = 0; port < ds->num_ports; port++) {
if (dsa_is_unused_port(ds, port)) if (dsa_is_unused_port(ds, port))
continue; continue;
...@@ -1233,14 +1211,6 @@ static int felix_setup(struct dsa_switch *ds) ...@@ -1233,14 +1211,6 @@ static int felix_setup(struct dsa_switch *ds)
* bits of vlan tag. * bits of vlan tag.
*/ */
felix_port_qos_map_init(ocelot, port); felix_port_qos_map_init(ocelot, port);
err = felix_port_setup_tagger_data(ds, port);
if (err) {
dev_err(ds->dev,
"port %d failed to set up tagger data: %pe\n",
port, ERR_PTR(err));
goto out_deinit_ports;
}
} }
err = ocelot_devlink_sb_register(ocelot); err = ocelot_devlink_sb_register(ocelot);
...@@ -1268,13 +1238,9 @@ static int felix_setup(struct dsa_switch *ds) ...@@ -1268,13 +1238,9 @@ static int felix_setup(struct dsa_switch *ds)
if (dsa_is_unused_port(ds, port)) if (dsa_is_unused_port(ds, port))
continue; continue;
felix_port_teardown_tagger_data(ds, port);
ocelot_deinit_port(ocelot, port); ocelot_deinit_port(ocelot, port);
} }
kthread_destroy_worker(felix->xmit_worker);
out_deinit_timestamp:
ocelot_deinit_timestamp(ocelot); ocelot_deinit_timestamp(ocelot);
ocelot_deinit(ocelot); ocelot_deinit(ocelot);
...@@ -1303,12 +1269,9 @@ static void felix_teardown(struct dsa_switch *ds) ...@@ -1303,12 +1269,9 @@ static void felix_teardown(struct dsa_switch *ds)
if (dsa_is_unused_port(ds, port)) if (dsa_is_unused_port(ds, port))
continue; continue;
felix_port_teardown_tagger_data(ds, port);
ocelot_deinit_port(ocelot, port); ocelot_deinit_port(ocelot, port);
} }
kthread_destroy_worker(felix->xmit_worker);
ocelot_devlink_sb_unregister(ocelot); ocelot_devlink_sb_unregister(ocelot);
ocelot_deinit_timestamp(ocelot); ocelot_deinit_timestamp(ocelot);
ocelot_deinit(ocelot); ocelot_deinit(ocelot);
...@@ -1648,6 +1611,7 @@ felix_mrp_del_ring_role(struct dsa_switch *ds, int port, ...@@ -1648,6 +1611,7 @@ felix_mrp_del_ring_role(struct dsa_switch *ds, int port,
const struct dsa_switch_ops felix_switch_ops = { const struct dsa_switch_ops felix_switch_ops = {
.get_tag_protocol = felix_get_tag_protocol, .get_tag_protocol = felix_get_tag_protocol,
.change_tag_protocol = felix_change_tag_protocol, .change_tag_protocol = felix_change_tag_protocol,
.connect_tag_protocol = felix_connect_tag_protocol,
.setup = felix_setup, .setup = felix_setup,
.teardown = felix_teardown, .teardown = felix_teardown,
.set_ageing_time = felix_set_ageing_time, .set_ageing_time = felix_set_ageing_time,
......
...@@ -249,6 +249,7 @@ struct sja1105_private { ...@@ -249,6 +249,7 @@ struct sja1105_private {
bool fixed_link[SJA1105_MAX_NUM_PORTS]; bool fixed_link[SJA1105_MAX_NUM_PORTS];
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;
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;
...@@ -256,11 +257,13 @@ struct sja1105_private { ...@@ -256,11 +257,13 @@ struct sja1105_private {
u16 bridge_pvid[SJA1105_MAX_NUM_PORTS]; u16 bridge_pvid[SJA1105_MAX_NUM_PORTS];
u16 tag_8021q_pvid[SJA1105_MAX_NUM_PORTS]; u16 tag_8021q_pvid[SJA1105_MAX_NUM_PORTS];
struct sja1105_flow_block flow_block; struct sja1105_flow_block flow_block;
struct sja1105_port ports[SJA1105_MAX_NUM_PORTS];
/* Serializes transmission of management frames so that /* Serializes transmission of management frames so that
* the switch doesn't confuse them with one another. * the switch doesn't confuse them with one another.
*/ */
struct mutex mgmt_lock; struct mutex mgmt_lock;
/* PTP two-step TX timestamp ID, and its serialization lock */
spinlock_t ts_id_lock;
u8 ts_id;
/* Serializes access to the dynamic config interface */ /* Serializes access to the dynamic config interface */
struct mutex dynamic_config_lock; struct mutex dynamic_config_lock;
struct devlink_region **regions; struct devlink_region **regions;
...@@ -269,7 +272,6 @@ struct sja1105_private { ...@@ -269,7 +272,6 @@ struct sja1105_private {
struct mii_bus *mdio_base_tx; struct mii_bus *mdio_base_tx;
struct mii_bus *mdio_pcs; struct mii_bus *mdio_pcs;
struct dw_xpcs *xpcs[SJA1105_MAX_NUM_PORTS]; struct dw_xpcs *xpcs[SJA1105_MAX_NUM_PORTS];
struct sja1105_tagger_data tagger_data;
struct sja1105_ptp_data ptp_data; struct sja1105_ptp_data ptp_data;
struct sja1105_tas_data tas_data; struct sja1105_tas_data tas_data;
}; };
......
...@@ -2617,18 +2617,6 @@ static int sja1105_prechangeupper(struct dsa_switch *ds, int port, ...@@ -2617,18 +2617,6 @@ static int sja1105_prechangeupper(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static void sja1105_port_disable(struct dsa_switch *ds, int port)
{
struct sja1105_private *priv = ds->priv;
struct sja1105_port *sp = &priv->ports[port];
if (!dsa_is_user_port(ds, port))
return;
kthread_cancel_work_sync(&sp->xmit_work);
skb_queue_purge(&sp->xmit_queue);
}
static int sja1105_mgmt_xmit(struct dsa_switch *ds, int port, int slot, static int sja1105_mgmt_xmit(struct dsa_switch *ds, int port, int slot,
struct sk_buff *skb, bool takets) struct sk_buff *skb, bool takets)
{ {
...@@ -2687,10 +2675,8 @@ static int sja1105_mgmt_xmit(struct dsa_switch *ds, int port, int slot, ...@@ -2687,10 +2675,8 @@ static int sja1105_mgmt_xmit(struct dsa_switch *ds, int port, int slot,
return NETDEV_TX_OK; return NETDEV_TX_OK;
} }
#define work_to_port(work) \ #define work_to_xmit_work(w) \
container_of((work), struct sja1105_port, xmit_work) container_of((w), struct sja1105_deferred_xmit_work, work)
#define tagger_to_sja1105(t) \
container_of((t), struct sja1105_private, tagger_data)
/* Deferred work is unfortunately necessary because setting up the management /* Deferred work is unfortunately necessary because setting up the management
* route cannot be done from atomit context (SPI transfer takes a sleepable * route cannot be done from atomit context (SPI transfer takes a sleepable
...@@ -2698,24 +2684,40 @@ static int sja1105_mgmt_xmit(struct dsa_switch *ds, int port, int slot, ...@@ -2698,24 +2684,40 @@ static int sja1105_mgmt_xmit(struct dsa_switch *ds, int port, int slot,
*/ */
static void sja1105_port_deferred_xmit(struct kthread_work *work) static void sja1105_port_deferred_xmit(struct kthread_work *work)
{ {
struct sja1105_port *sp = work_to_port(work); struct sja1105_deferred_xmit_work *xmit_work = work_to_xmit_work(work);
struct sja1105_tagger_data *tagger_data = sp->data; struct sk_buff *clone, *skb = xmit_work->skb;
struct sja1105_private *priv = tagger_to_sja1105(tagger_data); struct dsa_switch *ds = xmit_work->dp->ds;
int port = sp - priv->ports; struct sja1105_private *priv = ds->priv;
struct sk_buff *skb; int port = xmit_work->dp->index;
while ((skb = skb_dequeue(&sp->xmit_queue)) != NULL) { clone = SJA1105_SKB_CB(skb)->clone;
struct sk_buff *clone = SJA1105_SKB_CB(skb)->clone;
mutex_lock(&priv->mgmt_lock); mutex_lock(&priv->mgmt_lock);
sja1105_mgmt_xmit(priv->ds, port, 0, skb, !!clone); sja1105_mgmt_xmit(ds, port, 0, skb, !!clone);
/* The clone, if there, was made by dsa_skb_tx_timestamp */ /* The clone, if there, was made by dsa_skb_tx_timestamp */
if (clone) if (clone)
sja1105_ptp_txtstamp_skb(priv->ds, port, clone); sja1105_ptp_txtstamp_skb(ds, port, clone);
mutex_unlock(&priv->mgmt_lock); mutex_unlock(&priv->mgmt_lock);
kfree(xmit_work);
}
static int sja1105_connect_tag_protocol(struct dsa_switch *ds,
enum dsa_tag_protocol proto)
{
struct sja1105_tagger_data *tagger_data;
switch (proto) {
case DSA_TAG_PROTO_SJA1105:
tagger_data = sja1105_tagger_data(ds);
tagger_data->xmit_work_fn = sja1105_port_deferred_xmit;
tagger_data->meta_tstamp_handler = sja1110_process_meta_tstamp;
return 0;
default:
return -EPROTONOSUPPORT;
} }
} }
...@@ -3019,58 +3021,6 @@ static int sja1105_port_bridge_flags(struct dsa_switch *ds, int port, ...@@ -3019,58 +3021,6 @@ static int sja1105_port_bridge_flags(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static void sja1105_teardown_ports(struct sja1105_private *priv)
{
struct dsa_switch *ds = priv->ds;
int port;
for (port = 0; port < ds->num_ports; port++) {
struct sja1105_port *sp = &priv->ports[port];
if (sp->xmit_worker)
kthread_destroy_worker(sp->xmit_worker);
}
}
static int sja1105_setup_ports(struct sja1105_private *priv)
{
struct sja1105_tagger_data *tagger_data = &priv->tagger_data;
struct dsa_switch *ds = priv->ds;
int port, rc;
/* Connections between dsa_port and sja1105_port */
for (port = 0; port < ds->num_ports; port++) {
struct sja1105_port *sp = &priv->ports[port];
struct dsa_port *dp = dsa_to_port(ds, port);
struct kthread_worker *worker;
struct net_device *slave;
if (!dsa_port_is_user(dp))
continue;
dp->priv = sp;
sp->data = tagger_data;
slave = dp->slave;
kthread_init_work(&sp->xmit_work, sja1105_port_deferred_xmit);
worker = kthread_create_worker(0, "%s_xmit", slave->name);
if (IS_ERR(worker)) {
rc = PTR_ERR(worker);
dev_err(ds->dev,
"failed to create deferred xmit thread: %d\n",
rc);
goto out_destroy_workers;
}
sp->xmit_worker = worker;
skb_queue_head_init(&sp->xmit_queue);
}
return 0;
out_destroy_workers:
sja1105_teardown_ports(priv);
return rc;
}
/* The programming model for the SJA1105 switch is "all-at-once" via static /* The programming model for the SJA1105 switch is "all-at-once" via static
* configuration tables. Some of these can be dynamically modified at runtime, * configuration tables. Some of these can be dynamically modified at runtime,
* but not the xMII mode parameters table. * but not the xMII mode parameters table.
...@@ -3116,10 +3066,6 @@ static int sja1105_setup(struct dsa_switch *ds) ...@@ -3116,10 +3066,6 @@ static int sja1105_setup(struct dsa_switch *ds)
} }
} }
rc = sja1105_setup_ports(priv);
if (rc)
goto out_static_config_free;
sja1105_tas_setup(ds); sja1105_tas_setup(ds);
sja1105_flower_setup(ds); sja1105_flower_setup(ds);
...@@ -3176,7 +3122,6 @@ static int sja1105_setup(struct dsa_switch *ds) ...@@ -3176,7 +3122,6 @@ static int sja1105_setup(struct dsa_switch *ds)
out_flower_teardown: out_flower_teardown:
sja1105_flower_teardown(ds); sja1105_flower_teardown(ds);
sja1105_tas_teardown(ds); sja1105_tas_teardown(ds);
sja1105_teardown_ports(priv);
out_static_config_free: out_static_config_free:
sja1105_static_config_free(&priv->static_config); sja1105_static_config_free(&priv->static_config);
...@@ -3196,12 +3141,12 @@ static void sja1105_teardown(struct dsa_switch *ds) ...@@ -3196,12 +3141,12 @@ static void sja1105_teardown(struct dsa_switch *ds)
sja1105_ptp_clock_unregister(ds); sja1105_ptp_clock_unregister(ds);
sja1105_flower_teardown(ds); sja1105_flower_teardown(ds);
sja1105_tas_teardown(ds); sja1105_tas_teardown(ds);
sja1105_teardown_ports(priv);
sja1105_static_config_free(&priv->static_config); sja1105_static_config_free(&priv->static_config);
} }
static const struct dsa_switch_ops sja1105_switch_ops = { static const struct dsa_switch_ops sja1105_switch_ops = {
.get_tag_protocol = sja1105_get_tag_protocol, .get_tag_protocol = sja1105_get_tag_protocol,
.connect_tag_protocol = sja1105_connect_tag_protocol,
.setup = sja1105_setup, .setup = sja1105_setup,
.teardown = sja1105_teardown, .teardown = sja1105_teardown,
.set_ageing_time = sja1105_set_ageing_time, .set_ageing_time = sja1105_set_ageing_time,
...@@ -3215,7 +3160,6 @@ static const struct dsa_switch_ops sja1105_switch_ops = { ...@@ -3215,7 +3160,6 @@ static const struct dsa_switch_ops sja1105_switch_ops = {
.get_ethtool_stats = sja1105_get_ethtool_stats, .get_ethtool_stats = sja1105_get_ethtool_stats,
.get_sset_count = sja1105_get_sset_count, .get_sset_count = sja1105_get_sset_count,
.get_ts_info = sja1105_get_ts_info, .get_ts_info = sja1105_get_ts_info,
.port_disable = sja1105_port_disable,
.port_fdb_dump = sja1105_fdb_dump, .port_fdb_dump = sja1105_fdb_dump,
.port_fdb_add = sja1105_fdb_add, .port_fdb_add = sja1105_fdb_add,
.port_fdb_del = sja1105_fdb_del, .port_fdb_del = sja1105_fdb_del,
...@@ -3383,6 +3327,7 @@ static int sja1105_probe(struct spi_device *spi) ...@@ -3383,6 +3327,7 @@ static int sja1105_probe(struct spi_device *spi)
mutex_init(&priv->ptp_data.lock); mutex_init(&priv->ptp_data.lock);
mutex_init(&priv->dynamic_config_lock); mutex_init(&priv->dynamic_config_lock);
mutex_init(&priv->mgmt_lock); mutex_init(&priv->mgmt_lock);
spin_lock_init(&priv->ts_id_lock);
rc = sja1105_parse_dt(priv); rc = sja1105_parse_dt(priv);
if (rc < 0) { if (rc < 0) {
......
...@@ -58,13 +58,12 @@ enum sja1105_ptp_clk_mode { ...@@ -58,13 +58,12 @@ 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 with priv->tagger_data.state bit /* Must be called only while the RX timestamping state of the tagger
* SJA1105_HWTS_RX_EN cleared * is turned off
*/ */
static int sja1105_change_rxtstamping(struct sja1105_private *priv, static int sja1105_change_rxtstamping(struct sja1105_private *priv,
bool on) bool on)
{ {
struct sja1105_tagger_data *tagger_data = &priv->tagger_data;
struct sja1105_ptp_data *ptp_data = &priv->ptp_data; struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
struct sja1105_general_params_entry *general_params; struct sja1105_general_params_entry *general_params;
struct sja1105_table *table; struct sja1105_table *table;
...@@ -74,13 +73,8 @@ static int sja1105_change_rxtstamping(struct sja1105_private *priv, ...@@ -74,13 +73,8 @@ static int sja1105_change_rxtstamping(struct sja1105_private *priv,
general_params->send_meta1 = on; general_params->send_meta1 = on;
general_params->send_meta0 = on; general_params->send_meta0 = on;
/* Initialize the meta state machine to a known state */
if (priv->tagger_data.stampable_skb) {
kfree_skb(priv->tagger_data.stampable_skb);
priv->tagger_data.stampable_skb = NULL;
}
ptp_cancel_worker_sync(ptp_data->clock); ptp_cancel_worker_sync(ptp_data->clock);
skb_queue_purge(&tagger_data->skb_txtstamp_queue); skb_queue_purge(&ptp_data->skb_txtstamp_queue);
skb_queue_purge(&ptp_data->skb_rxtstamp_queue); skb_queue_purge(&ptp_data->skb_rxtstamp_queue);
return sja1105_static_config_reload(priv, SJA1105_RX_HWTSTAMPING); return sja1105_static_config_reload(priv, SJA1105_RX_HWTSTAMPING);
...@@ -88,6 +82,7 @@ static int sja1105_change_rxtstamping(struct sja1105_private *priv, ...@@ -88,6 +82,7 @@ static int sja1105_change_rxtstamping(struct sja1105_private *priv,
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; bool rx_on;
...@@ -98,10 +93,10 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -98,10 +93,10 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr)
switch (config.tx_type) { switch (config.tx_type) {
case HWTSTAMP_TX_OFF: case HWTSTAMP_TX_OFF:
priv->ports[port].hwts_tx_en = false; priv->hwts_tx_en &= ~BIT(port);
break; break;
case HWTSTAMP_TX_ON: case HWTSTAMP_TX_ON:
priv->ports[port].hwts_tx_en = true; priv->hwts_tx_en |= BIT(port);
break; break;
default: default:
return -ERANGE; return -ERANGE;
...@@ -116,8 +111,8 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -116,8 +111,8 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr)
break; break;
} }
if (rx_on != test_bit(SJA1105_HWTS_RX_EN, &priv->tagger_data.state)) { if (rx_on != tagger_data->rxtstamp_get_state(ds)) {
clear_bit(SJA1105_HWTS_RX_EN, &priv->tagger_data.state); tagger_data->rxtstamp_set_state(ds, false);
rc = sja1105_change_rxtstamping(priv, rx_on); rc = sja1105_change_rxtstamping(priv, rx_on);
if (rc < 0) { if (rc < 0) {
...@@ -126,7 +121,7 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -126,7 +121,7 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr)
return rc; return rc;
} }
if (rx_on) if (rx_on)
set_bit(SJA1105_HWTS_RX_EN, &priv->tagger_data.state); 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)))
...@@ -136,15 +131,16 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr) ...@@ -136,15 +131,16 @@ 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;
config.flags = 0; config.flags = 0;
if (priv->ports[port].hwts_tx_en) if (priv->hwts_tx_en & BIT(port))
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 (test_bit(SJA1105_HWTS_RX_EN, &priv->tagger_data.state)) if (tagger_data->rxtstamp_get_state(ds))
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;
...@@ -417,10 +413,11 @@ static long sja1105_rxtstamp_work(struct ptp_clock_info *ptp) ...@@ -417,10 +413,11 @@ 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 (!test_bit(SJA1105_HWTS_RX_EN, &priv->tagger_data.state)) if (!tagger_data->rxtstamp_get_state(ds))
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
...@@ -453,6 +450,39 @@ bool sja1105_port_rxtstamp(struct dsa_switch *ds, int port, ...@@ -453,6 +450,39 @@ bool sja1105_port_rxtstamp(struct dsa_switch *ds, int port,
return priv->info->rxtstamp(ds, port, skb); return priv->info->rxtstamp(ds, port, skb);
} }
void sja1110_process_meta_tstamp(struct dsa_switch *ds, int port, u8 ts_id,
enum sja1110_meta_tstamp dir, u64 tstamp)
{
struct sja1105_private *priv = ds->priv;
struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
struct sk_buff *skb, *skb_tmp, *skb_match = NULL;
struct skb_shared_hwtstamps shwt = {0};
/* We don't care about RX timestamps on the CPU port */
if (dir == SJA1110_META_TSTAMP_RX)
return;
spin_lock(&ptp_data->skb_txtstamp_queue.lock);
skb_queue_walk_safe(&ptp_data->skb_txtstamp_queue, skb, skb_tmp) {
if (SJA1105_SKB_CB(skb)->ts_id != ts_id)
continue;
__skb_unlink(skb, &ptp_data->skb_txtstamp_queue);
skb_match = skb;
break;
}
spin_unlock(&ptp_data->skb_txtstamp_queue.lock);
if (WARN_ON(!skb_match))
return;
shwt.hwtstamp = ns_to_ktime(sja1105_ticks_to_ns(tstamp));
skb_complete_tx_timestamp(skb_match, &shwt);
}
/* In addition to cloning the skb which is done by the common /* In addition to cloning the skb which is done by the common
* sja1105_port_txtstamp, we need to generate a timestamp ID and save the * sja1105_port_txtstamp, we need to generate a timestamp ID and save the
* packet to the TX timestamping queue. * packet to the TX timestamping queue.
...@@ -461,22 +491,22 @@ void sja1110_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb) ...@@ -461,22 +491,22 @@ void sja1110_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb)
{ {
struct sk_buff *clone = SJA1105_SKB_CB(skb)->clone; struct sk_buff *clone = SJA1105_SKB_CB(skb)->clone;
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct sja1105_port *sp = &priv->ports[port]; struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
u8 ts_id; u8 ts_id;
skb_shinfo(skb)->tx_flags |= SKBTX_IN_PROGRESS; skb_shinfo(skb)->tx_flags |= SKBTX_IN_PROGRESS;
spin_lock(&sp->data->meta_lock); spin_lock(&priv->ts_id_lock);
ts_id = sp->data->ts_id; ts_id = priv->ts_id;
/* Deal automatically with 8-bit wraparound */ /* Deal automatically with 8-bit wraparound */
sp->data->ts_id++; priv->ts_id++;
SJA1105_SKB_CB(clone)->ts_id = ts_id; SJA1105_SKB_CB(clone)->ts_id = ts_id;
spin_unlock(&sp->data->meta_lock); spin_unlock(&priv->ts_id_lock);
skb_queue_tail(&sp->data->skb_txtstamp_queue, clone); skb_queue_tail(&ptp_data->skb_txtstamp_queue, clone);
} }
/* Called from dsa_skb_tx_timestamp. This callback is just to clone /* Called from dsa_skb_tx_timestamp. This callback is just to clone
...@@ -486,10 +516,9 @@ void sja1110_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb) ...@@ -486,10 +516,9 @@ void sja1110_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb)
void sja1105_port_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb) void sja1105_port_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb)
{ {
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct sja1105_port *sp = &priv->ports[port];
struct sk_buff *clone; struct sk_buff *clone;
if (!sp->hwts_tx_en) if (!(priv->hwts_tx_en & BIT(port)))
return; return;
clone = skb_clone_sk(skb); clone = skb_clone_sk(skb);
...@@ -896,7 +925,6 @@ static struct ptp_pin_desc sja1105_ptp_pin = { ...@@ -896,7 +925,6 @@ static struct ptp_pin_desc sja1105_ptp_pin = {
int sja1105_ptp_clock_register(struct dsa_switch *ds) int sja1105_ptp_clock_register(struct dsa_switch *ds)
{ {
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct sja1105_tagger_data *tagger_data = &priv->tagger_data;
struct sja1105_ptp_data *ptp_data = &priv->ptp_data; struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
ptp_data->caps = (struct ptp_clock_info) { ptp_data->caps = (struct ptp_clock_info) {
...@@ -919,8 +947,7 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds) ...@@ -919,8 +947,7 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds)
/* Only used on SJA1105 */ /* Only used on SJA1105 */
skb_queue_head_init(&ptp_data->skb_rxtstamp_queue); skb_queue_head_init(&ptp_data->skb_rxtstamp_queue);
/* Only used on SJA1110 */ /* Only used on SJA1110 */
skb_queue_head_init(&tagger_data->skb_txtstamp_queue); skb_queue_head_init(&ptp_data->skb_txtstamp_queue);
spin_lock_init(&tagger_data->meta_lock);
ptp_data->clock = ptp_clock_register(&ptp_data->caps, ds->dev); ptp_data->clock = ptp_clock_register(&ptp_data->caps, ds->dev);
if (IS_ERR_OR_NULL(ptp_data->clock)) if (IS_ERR_OR_NULL(ptp_data->clock))
...@@ -937,7 +964,6 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds) ...@@ -937,7 +964,6 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds)
void sja1105_ptp_clock_unregister(struct dsa_switch *ds) void sja1105_ptp_clock_unregister(struct dsa_switch *ds)
{ {
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct sja1105_tagger_data *tagger_data = &priv->tagger_data;
struct sja1105_ptp_data *ptp_data = &priv->ptp_data; struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
if (IS_ERR_OR_NULL(ptp_data->clock)) if (IS_ERR_OR_NULL(ptp_data->clock))
...@@ -945,7 +971,7 @@ void sja1105_ptp_clock_unregister(struct dsa_switch *ds) ...@@ -945,7 +971,7 @@ void sja1105_ptp_clock_unregister(struct dsa_switch *ds)
del_timer_sync(&ptp_data->extts_timer); del_timer_sync(&ptp_data->extts_timer);
ptp_cancel_worker_sync(ptp_data->clock); ptp_cancel_worker_sync(ptp_data->clock);
skb_queue_purge(&tagger_data->skb_txtstamp_queue); skb_queue_purge(&ptp_data->skb_txtstamp_queue);
skb_queue_purge(&ptp_data->skb_rxtstamp_queue); skb_queue_purge(&ptp_data->skb_rxtstamp_queue);
ptp_clock_unregister(ptp_data->clock); ptp_clock_unregister(ptp_data->clock);
ptp_data->clock = NULL; ptp_data->clock = NULL;
......
...@@ -8,6 +8,21 @@ ...@@ -8,6 +8,21 @@
#if IS_ENABLED(CONFIG_NET_DSA_SJA1105_PTP) #if IS_ENABLED(CONFIG_NET_DSA_SJA1105_PTP)
/* Timestamps are in units of 8 ns clock ticks (equivalent to
* a fixed 125 MHz clock).
*/
#define SJA1105_TICK_NS 8
static inline s64 ns_to_sja1105_ticks(s64 ns)
{
return ns / SJA1105_TICK_NS;
}
static inline s64 sja1105_ticks_to_ns(s64 ticks)
{
return ticks * SJA1105_TICK_NS;
}
/* Calculate the first base_time in the future that satisfies this /* Calculate the first base_time in the future that satisfies this
* relationship: * relationship:
* *
...@@ -62,6 +77,10 @@ struct sja1105_ptp_data { ...@@ -62,6 +77,10 @@ struct sja1105_ptp_data {
struct timer_list extts_timer; struct timer_list extts_timer;
/* Used only on SJA1105 to reconstruct partial timestamps */ /* Used only on SJA1105 to reconstruct partial timestamps */
struct sk_buff_head skb_rxtstamp_queue; struct sk_buff_head skb_rxtstamp_queue;
/* Used on SJA1110 where meta frames are generated only for
* 2-step TX timestamps
*/
struct sk_buff_head skb_txtstamp_queue;
struct ptp_clock_info caps; struct ptp_clock_info caps;
struct ptp_clock *clock; struct ptp_clock *clock;
struct sja1105_ptp_cmd cmd; struct sja1105_ptp_cmd cmd;
...@@ -112,6 +131,9 @@ bool sja1105_rxtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb); ...@@ -112,6 +131,9 @@ bool sja1105_rxtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb);
bool sja1110_rxtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb); bool sja1110_rxtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb);
void sja1110_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb); void sja1110_txtstamp(struct dsa_switch *ds, int port, struct sk_buff *skb);
void sja1110_process_meta_tstamp(struct dsa_switch *ds, int port, u8 ts_id,
enum sja1110_meta_tstamp dir, u64 tstamp);
#else #else
struct sja1105_ptp_cmd; struct sja1105_ptp_cmd;
...@@ -178,6 +200,8 @@ static inline int sja1105_ptp_commit(struct dsa_switch *ds, ...@@ -178,6 +200,8 @@ static inline int sja1105_ptp_commit(struct dsa_switch *ds,
#define sja1110_rxtstamp NULL #define sja1110_rxtstamp NULL
#define sja1110_txtstamp NULL #define sja1110_txtstamp NULL
#define sja1110_process_meta_tstamp NULL
#endif /* IS_ENABLED(CONFIG_NET_DSA_SJA1105_PTP) */ #endif /* IS_ENABLED(CONFIG_NET_DSA_SJA1105_PTP) */
#endif /* _SJA1105_PTP_H */ #endif /* _SJA1105_PTP_H */
...@@ -8,6 +8,7 @@ ...@@ -8,6 +8,7 @@
#include <linux/kthread.h> #include <linux/kthread.h>
#include <linux/packing.h> #include <linux/packing.h>
#include <linux/skbuff.h> #include <linux/skbuff.h>
#include <net/dsa.h>
struct ocelot_skb_cb { struct ocelot_skb_cb {
struct sk_buff *clone; struct sk_buff *clone;
...@@ -168,11 +169,18 @@ struct felix_deferred_xmit_work { ...@@ -168,11 +169,18 @@ struct felix_deferred_xmit_work {
struct kthread_work work; struct kthread_work work;
}; };
struct felix_port { struct ocelot_8021q_tagger_data {
void (*xmit_work_fn)(struct kthread_work *work); void (*xmit_work_fn)(struct kthread_work *work);
struct kthread_worker *xmit_worker;
}; };
static inline struct ocelot_8021q_tagger_data *
ocelot_8021q_tagger_data(struct dsa_switch *ds)
{
BUG_ON(ds->dst->tag_ops->proto != DSA_TAG_PROTO_OCELOT_8021Q);
return ds->tagger_data;
}
static inline void ocelot_xfh_get_rew_val(void *extraction, u64 *rew_val) static inline void ocelot_xfh_get_rew_val(void *extraction, u64 *rew_val)
{ {
packing(extraction, rew_val, 116, 85, OCELOT_TAG_LEN, UNPACK, 0); packing(extraction, rew_val, 116, 85, OCELOT_TAG_LEN, UNPACK, 0);
......
...@@ -35,23 +35,26 @@ ...@@ -35,23 +35,26 @@
#define SJA1105_META_SMAC 0x222222222222ull #define SJA1105_META_SMAC 0x222222222222ull
#define SJA1105_META_DMAC 0x0180C200000Eull #define SJA1105_META_DMAC 0x0180C200000Eull
#define SJA1105_HWTS_RX_EN 0 enum sja1110_meta_tstamp {
SJA1110_META_TSTAMP_TX = 0,
SJA1110_META_TSTAMP_RX = 1,
};
/* Global tagger data: each struct sja1105_port has a reference to struct sja1105_deferred_xmit_work {
* the structure defined in struct sja1105_private. struct dsa_port *dp;
*/ struct sk_buff *skb;
struct kthread_work work;
};
/* Global tagger data */
struct sja1105_tagger_data { struct sja1105_tagger_data {
struct sk_buff *stampable_skb; /* Tagger to switch */
/* Protects concurrent access to the meta state machine void (*xmit_work_fn)(struct kthread_work *work);
* from taggers running on multiple ports on SMP systems void (*meta_tstamp_handler)(struct dsa_switch *ds, int port, u8 ts_id,
*/ enum sja1110_meta_tstamp dir, u64 tstamp);
spinlock_t meta_lock; /* Switch to tagger */
unsigned long state; bool (*rxtstamp_get_state)(struct dsa_switch *ds);
u8 ts_id; void (*rxtstamp_set_state)(struct dsa_switch *ds, bool on);
/* Used on SJA1110 where meta frames are generated only for
* 2-step TX timestamps
*/
struct sk_buff_head skb_txtstamp_queue;
}; };
struct sja1105_skb_cb { struct sja1105_skb_cb {
...@@ -64,32 +67,12 @@ struct sja1105_skb_cb { ...@@ -64,32 +67,12 @@ struct sja1105_skb_cb {
#define SJA1105_SKB_CB(skb) \ #define SJA1105_SKB_CB(skb) \
((struct sja1105_skb_cb *)((skb)->cb)) ((struct sja1105_skb_cb *)((skb)->cb))
struct sja1105_port { static inline struct sja1105_tagger_data *
struct kthread_worker *xmit_worker; sja1105_tagger_data(struct dsa_switch *ds)
struct kthread_work xmit_work;
struct sk_buff_head xmit_queue;
struct sja1105_tagger_data *data;
bool hwts_tx_en;
};
/* Timestamps are in units of 8 ns clock ticks (equivalent to
* a fixed 125 MHz clock).
*/
#define SJA1105_TICK_NS 8
static inline s64 ns_to_sja1105_ticks(s64 ns)
{
return ns / SJA1105_TICK_NS;
}
static inline s64 sja1105_ticks_to_ns(s64 ticks)
{ {
return ticks * SJA1105_TICK_NS; BUG_ON(ds->dst->tag_ops->proto != DSA_TAG_PROTO_SJA1105);
}
static inline bool dsa_port_is_sja1105(struct dsa_port *dp) return ds->tagger_data;
{
return true;
} }
#endif /* _NET_DSA_SJA1105_H */ #endif /* _NET_DSA_SJA1105_H */
...@@ -82,12 +82,15 @@ enum dsa_tag_protocol { ...@@ -82,12 +82,15 @@ enum dsa_tag_protocol {
}; };
struct dsa_switch; struct dsa_switch;
struct dsa_switch_tree;
struct dsa_device_ops { struct dsa_device_ops {
struct sk_buff *(*xmit)(struct sk_buff *skb, struct net_device *dev); struct sk_buff *(*xmit)(struct sk_buff *skb, struct net_device *dev);
struct sk_buff *(*rcv)(struct sk_buff *skb, struct net_device *dev); struct sk_buff *(*rcv)(struct sk_buff *skb, struct net_device *dev);
void (*flow_dissect)(const struct sk_buff *skb, __be16 *proto, void (*flow_dissect)(const struct sk_buff *skb, __be16 *proto,
int *offset); int *offset);
int (*connect)(struct dsa_switch_tree *dst);
void (*disconnect)(struct dsa_switch_tree *dst);
unsigned int needed_headroom; unsigned int needed_headroom;
unsigned int needed_tailroom; unsigned int needed_tailroom;
const char *name; const char *name;
...@@ -273,12 +276,6 @@ struct dsa_port { ...@@ -273,12 +276,6 @@ struct dsa_port {
struct list_head list; struct list_head list;
/*
* Give the switch driver somewhere to hang its per-port private data
* structures (accessible from the tagger).
*/
void *priv;
/* /*
* Original copy of the master netdev ethtool_ops * Original copy of the master netdev ethtool_ops
*/ */
...@@ -337,6 +334,8 @@ struct dsa_switch { ...@@ -337,6 +334,8 @@ struct dsa_switch {
*/ */
void *priv; void *priv;
void *tagger_data;
/* /*
* Configuration data for this switch. * Configuration data for this switch.
*/ */
...@@ -689,6 +688,13 @@ struct dsa_switch_ops { ...@@ -689,6 +688,13 @@ struct dsa_switch_ops {
enum dsa_tag_protocol mprot); enum dsa_tag_protocol mprot);
int (*change_tag_protocol)(struct dsa_switch *ds, int port, int (*change_tag_protocol)(struct dsa_switch *ds, int port,
enum dsa_tag_protocol proto); enum dsa_tag_protocol proto);
/*
* Method for switch drivers to connect to the tagging protocol driver
* in current use. The switch driver can provide handlers for certain
* types of packets for switch management.
*/
int (*connect_tag_protocol)(struct dsa_switch *ds,
enum dsa_tag_protocol proto);
/* Optional switch-wide initialization and destruction methods */ /* Optional switch-wide initialization and destruction methods */
int (*setup)(struct dsa_switch *ds); int (*setup)(struct dsa_switch *ds);
......
...@@ -248,8 +248,12 @@ static struct dsa_switch_tree *dsa_tree_alloc(int index) ...@@ -248,8 +248,12 @@ static struct dsa_switch_tree *dsa_tree_alloc(int index)
static void dsa_tree_free(struct dsa_switch_tree *dst) static void dsa_tree_free(struct dsa_switch_tree *dst)
{ {
if (dst->tag_ops) if (dst->tag_ops) {
if (dst->tag_ops->disconnect)
dst->tag_ops->disconnect(dst);
dsa_tag_driver_put(dst->tag_ops); dsa_tag_driver_put(dst->tag_ops);
}
list_del(&dst->list); list_del(&dst->list);
kfree(dst); kfree(dst);
} }
...@@ -822,7 +826,7 @@ static int dsa_switch_setup_tag_protocol(struct dsa_switch *ds) ...@@ -822,7 +826,7 @@ static int dsa_switch_setup_tag_protocol(struct dsa_switch *ds)
int err; int err;
if (tag_ops->proto == dst->default_proto) if (tag_ops->proto == dst->default_proto)
return 0; goto connect;
dsa_switch_for_each_cpu_port(cpu_dp, ds) { dsa_switch_for_each_cpu_port(cpu_dp, ds) {
rtnl_lock(); rtnl_lock();
...@@ -836,6 +840,17 @@ static int dsa_switch_setup_tag_protocol(struct dsa_switch *ds) ...@@ -836,6 +840,17 @@ static int dsa_switch_setup_tag_protocol(struct dsa_switch *ds)
} }
} }
connect:
if (ds->ops->connect_tag_protocol) {
err = ds->ops->connect_tag_protocol(ds, tag_ops->proto);
if (err) {
dev_err(ds->dev,
"Unable to connect to tag protocol \"%s\": %pe\n",
tag_ops->name, ERR_PTR(err));
return err;
}
}
return 0; return 0;
} }
...@@ -1136,6 +1151,46 @@ static void dsa_tree_teardown(struct dsa_switch_tree *dst) ...@@ -1136,6 +1151,46 @@ static void dsa_tree_teardown(struct dsa_switch_tree *dst)
dst->setup = false; dst->setup = false;
} }
static int dsa_tree_bind_tag_proto(struct dsa_switch_tree *dst,
const struct dsa_device_ops *tag_ops)
{
const struct dsa_device_ops *old_tag_ops = dst->tag_ops;
struct dsa_notifier_tag_proto_info info;
int err;
dst->tag_ops = tag_ops;
/* Notify the new tagger about the connection to this tree */
if (tag_ops->connect) {
err = tag_ops->connect(dst);
if (err)
goto out_revert;
}
/* Notify the switches from this tree about the connection
* to the new tagger
*/
info.tag_ops = tag_ops;
err = dsa_tree_notify(dst, DSA_NOTIFIER_TAG_PROTO_CONNECT, &info);
if (err && err != -EOPNOTSUPP)
goto out_disconnect;
/* Notify the old tagger about the disconnection from this tree */
if (old_tag_ops->disconnect)
old_tag_ops->disconnect(dst);
return 0;
out_disconnect:
/* Revert the new tagger's connection to this tree */
if (tag_ops->disconnect)
tag_ops->disconnect(dst);
out_revert:
dst->tag_ops = old_tag_ops;
return err;
}
/* Since the dsa/tagging sysfs device attribute is per master, the assumption /* Since the dsa/tagging sysfs device attribute is per master, the assumption
* is that all DSA switches within a tree share the same tagger, otherwise * is that all DSA switches within a tree share the same tagger, otherwise
* they would have formed disjoint trees (different "dsa,member" values). * they would have formed disjoint trees (different "dsa,member" values).
...@@ -1168,12 +1223,15 @@ int dsa_tree_change_tag_proto(struct dsa_switch_tree *dst, ...@@ -1168,12 +1223,15 @@ int dsa_tree_change_tag_proto(struct dsa_switch_tree *dst,
goto out_unlock; goto out_unlock;
} }
/* Notify the tag protocol change */
info.tag_ops = tag_ops; info.tag_ops = tag_ops;
err = dsa_tree_notify(dst, DSA_NOTIFIER_TAG_PROTO, &info); err = dsa_tree_notify(dst, DSA_NOTIFIER_TAG_PROTO, &info);
if (err) if (err)
goto out_unwind_tagger; return err;
dst->tag_ops = tag_ops; err = dsa_tree_bind_tag_proto(dst, tag_ops);
if (err)
goto out_unwind_tagger;
rtnl_unlock(); rtnl_unlock();
...@@ -1260,6 +1318,7 @@ static int dsa_port_parse_cpu(struct dsa_port *dp, struct net_device *master, ...@@ -1260,6 +1318,7 @@ static int dsa_port_parse_cpu(struct dsa_port *dp, struct net_device *master,
struct dsa_switch_tree *dst = ds->dst; struct dsa_switch_tree *dst = ds->dst;
const struct dsa_device_ops *tag_ops; const struct dsa_device_ops *tag_ops;
enum dsa_tag_protocol default_proto; enum dsa_tag_protocol default_proto;
int err;
/* Find out which protocol the switch would prefer. */ /* Find out which protocol the switch would prefer. */
default_proto = dsa_get_tag_protocol(dp, master); default_proto = dsa_get_tag_protocol(dp, master);
...@@ -1307,6 +1366,12 @@ static int dsa_port_parse_cpu(struct dsa_port *dp, struct net_device *master, ...@@ -1307,6 +1366,12 @@ static int dsa_port_parse_cpu(struct dsa_port *dp, struct net_device *master,
*/ */
dsa_tag_driver_put(tag_ops); dsa_tag_driver_put(tag_ops);
} else { } else {
if (tag_ops->connect) {
err = tag_ops->connect(dst);
if (err)
return err;
}
dst->tag_ops = tag_ops; dst->tag_ops = tag_ops;
} }
......
...@@ -37,6 +37,7 @@ enum { ...@@ -37,6 +37,7 @@ enum {
DSA_NOTIFIER_VLAN_DEL, DSA_NOTIFIER_VLAN_DEL,
DSA_NOTIFIER_MTU, DSA_NOTIFIER_MTU,
DSA_NOTIFIER_TAG_PROTO, DSA_NOTIFIER_TAG_PROTO,
DSA_NOTIFIER_TAG_PROTO_CONNECT,
DSA_NOTIFIER_MRP_ADD, DSA_NOTIFIER_MRP_ADD,
DSA_NOTIFIER_MRP_DEL, DSA_NOTIFIER_MRP_DEL,
DSA_NOTIFIER_MRP_ADD_RING_ROLE, DSA_NOTIFIER_MRP_ADD_RING_ROLE,
......
...@@ -647,6 +647,17 @@ static int dsa_switch_change_tag_proto(struct dsa_switch *ds, ...@@ -647,6 +647,17 @@ static int dsa_switch_change_tag_proto(struct dsa_switch *ds,
return 0; return 0;
} }
static int dsa_switch_connect_tag_proto(struct dsa_switch *ds,
struct dsa_notifier_tag_proto_info *info)
{
const struct dsa_device_ops *tag_ops = info->tag_ops;
if (!ds->ops->connect_tag_protocol)
return -EOPNOTSUPP;
return ds->ops->connect_tag_protocol(ds, tag_ops->proto);
}
static int dsa_switch_mrp_add(struct dsa_switch *ds, static int dsa_switch_mrp_add(struct dsa_switch *ds,
struct dsa_notifier_mrp_info *info) struct dsa_notifier_mrp_info *info)
{ {
...@@ -766,6 +777,9 @@ static int dsa_switch_event(struct notifier_block *nb, ...@@ -766,6 +777,9 @@ static int dsa_switch_event(struct notifier_block *nb,
case DSA_NOTIFIER_TAG_PROTO: case DSA_NOTIFIER_TAG_PROTO:
err = dsa_switch_change_tag_proto(ds, info); err = dsa_switch_change_tag_proto(ds, info);
break; break;
case DSA_NOTIFIER_TAG_PROTO_CONNECT:
err = dsa_switch_connect_tag_proto(ds, info);
break;
case DSA_NOTIFIER_MRP_ADD: case DSA_NOTIFIER_MRP_ADD:
err = dsa_switch_mrp_add(ds, info); err = dsa_switch_mrp_add(ds, info);
break; break;
......
...@@ -12,25 +12,39 @@ ...@@ -12,25 +12,39 @@
#include <linux/dsa/ocelot.h> #include <linux/dsa/ocelot.h>
#include "dsa_priv.h" #include "dsa_priv.h"
struct ocelot_8021q_tagger_private {
struct ocelot_8021q_tagger_data data; /* Must be first */
struct kthread_worker *xmit_worker;
};
static struct sk_buff *ocelot_defer_xmit(struct dsa_port *dp, static struct sk_buff *ocelot_defer_xmit(struct dsa_port *dp,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct ocelot_8021q_tagger_private *priv = dp->ds->tagger_data;
struct ocelot_8021q_tagger_data *data = &priv->data;
void (*xmit_work_fn)(struct kthread_work *work);
struct felix_deferred_xmit_work *xmit_work; struct felix_deferred_xmit_work *xmit_work;
struct felix_port *felix_port = dp->priv; struct kthread_worker *xmit_worker;
xmit_work_fn = data->xmit_work_fn;
xmit_worker = priv->xmit_worker;
if (!xmit_work_fn || !xmit_worker)
return NULL;
xmit_work = kzalloc(sizeof(*xmit_work), GFP_ATOMIC); xmit_work = kzalloc(sizeof(*xmit_work), GFP_ATOMIC);
if (!xmit_work) if (!xmit_work)
return NULL; return NULL;
/* Calls felix_port_deferred_xmit in felix.c */ /* Calls felix_port_deferred_xmit in felix.c */
kthread_init_work(&xmit_work->work, felix_port->xmit_work_fn); kthread_init_work(&xmit_work->work, xmit_work_fn);
/* Increase refcount so the kfree_skb in dsa_slave_xmit /* Increase refcount so the kfree_skb in dsa_slave_xmit
* won't really free the packet. * won't really free the packet.
*/ */
xmit_work->dp = dp; xmit_work->dp = dp;
xmit_work->skb = skb_get(skb); xmit_work->skb = skb_get(skb);
kthread_queue_work(felix_port->xmit_worker, &xmit_work->work); kthread_queue_work(xmit_worker, &xmit_work->work);
return NULL; return NULL;
} }
...@@ -67,11 +81,64 @@ static struct sk_buff *ocelot_rcv(struct sk_buff *skb, ...@@ -67,11 +81,64 @@ static struct sk_buff *ocelot_rcv(struct sk_buff *skb,
return skb; return skb;
} }
static void ocelot_disconnect(struct dsa_switch_tree *dst)
{
struct ocelot_8021q_tagger_private *priv;
struct dsa_port *dp;
list_for_each_entry(dp, &dst->ports, list) {
priv = dp->ds->tagger_data;
if (!priv)
continue;
if (priv->xmit_worker)
kthread_destroy_worker(priv->xmit_worker);
kfree(priv);
dp->ds->tagger_data = NULL;
}
}
static int ocelot_connect(struct dsa_switch_tree *dst)
{
struct ocelot_8021q_tagger_private *priv;
struct dsa_port *dp;
int err;
list_for_each_entry(dp, &dst->ports, list) {
if (dp->ds->tagger_data)
continue;
priv = kzalloc(sizeof(*priv), GFP_KERNEL);
if (!priv) {
err = -ENOMEM;
goto out;
}
priv->xmit_worker = kthread_create_worker(0, "felix_xmit");
if (IS_ERR(priv->xmit_worker)) {
err = PTR_ERR(priv->xmit_worker);
goto out;
}
dp->ds->tagger_data = priv;
}
return 0;
out:
ocelot_disconnect(dst);
return err;
}
static const struct dsa_device_ops ocelot_8021q_netdev_ops = { static const struct dsa_device_ops ocelot_8021q_netdev_ops = {
.name = "ocelot-8021q", .name = "ocelot-8021q",
.proto = DSA_TAG_PROTO_OCELOT_8021Q, .proto = DSA_TAG_PROTO_OCELOT_8021Q,
.xmit = ocelot_xmit, .xmit = ocelot_xmit,
.rcv = ocelot_rcv, .rcv = ocelot_rcv,
.connect = ocelot_connect,
.disconnect = ocelot_disconnect,
.needed_headroom = VLAN_HLEN, .needed_headroom = VLAN_HLEN,
.promisc_on_master = true, .promisc_on_master = true,
}; };
......
...@@ -4,7 +4,6 @@ ...@@ -4,7 +4,6 @@
#include <linux/if_vlan.h> #include <linux/if_vlan.h>
#include <linux/dsa/sja1105.h> #include <linux/dsa/sja1105.h>
#include <linux/dsa/8021q.h> #include <linux/dsa/8021q.h>
#include <linux/skbuff.h>
#include <linux/packing.h> #include <linux/packing.h>
#include "dsa_priv.h" #include "dsa_priv.h"
...@@ -54,11 +53,25 @@ ...@@ -54,11 +53,25 @@
#define SJA1110_TX_TRAILER_LEN 4 #define SJA1110_TX_TRAILER_LEN 4
#define SJA1110_MAX_PADDING_LEN 15 #define SJA1110_MAX_PADDING_LEN 15
enum sja1110_meta_tstamp { #define SJA1105_HWTS_RX_EN 0
SJA1110_META_TSTAMP_TX = 0,
SJA1110_META_TSTAMP_RX = 1, 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
*/
spinlock_t meta_lock;
struct sk_buff *stampable_skb;
struct kthread_worker *xmit_worker;
}; };
static struct sja1105_tagger_private *
sja1105_tagger_private(struct dsa_switch *ds)
{
return ds->tagger_data;
}
/* Similar to is_link_local_ether_addr(hdr->h_dest) but also covers PTP */ /* Similar to is_link_local_ether_addr(hdr->h_dest) but also covers PTP */
static inline bool sja1105_is_link_local(const struct sk_buff *skb) static inline bool sja1105_is_link_local(const struct sk_buff *skb)
{ {
...@@ -125,16 +138,30 @@ static inline bool sja1105_is_meta_frame(const struct sk_buff *skb) ...@@ -125,16 +138,30 @@ static inline bool sja1105_is_meta_frame(const struct sk_buff *skb)
static struct sk_buff *sja1105_defer_xmit(struct dsa_port *dp, static struct sk_buff *sja1105_defer_xmit(struct dsa_port *dp,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct sja1105_port *sp = dp->priv; struct sja1105_tagger_data *tagger_data = sja1105_tagger_data(dp->ds);
struct sja1105_tagger_private *priv = sja1105_tagger_private(dp->ds);
void (*xmit_work_fn)(struct kthread_work *work);
struct sja1105_deferred_xmit_work *xmit_work;
struct kthread_worker *xmit_worker;
if (!dsa_port_is_sja1105(dp)) xmit_work_fn = tagger_data->xmit_work_fn;
return skb; xmit_worker = priv->xmit_worker;
if (!xmit_work_fn || !xmit_worker)
return NULL;
xmit_work = kzalloc(sizeof(*xmit_work), GFP_ATOMIC);
if (!xmit_work)
return NULL;
kthread_init_work(&xmit_work->work, xmit_work_fn);
/* Increase refcount so the kfree_skb in dsa_slave_xmit /* Increase refcount so the kfree_skb in dsa_slave_xmit
* won't really free the packet. * won't really free the packet.
*/ */
skb_queue_tail(&sp->xmit_queue, skb_get(skb)); xmit_work->dp = dp;
kthread_queue_work(sp->xmit_worker, &sp->xmit_work); xmit_work->skb = skb_get(skb);
kthread_queue_work(xmit_worker, &xmit_work->work);
return NULL; return NULL;
} }
...@@ -355,32 +382,32 @@ static struct sk_buff ...@@ -355,32 +382,32 @@ static struct sk_buff
*/ */
if (is_link_local) { if (is_link_local) {
struct dsa_port *dp = dsa_slave_to_port(skb->dev); struct dsa_port *dp = dsa_slave_to_port(skb->dev);
struct sja1105_port *sp = dp->priv; struct sja1105_tagger_private *priv;
struct dsa_switch *ds = dp->ds;
if (unlikely(!dsa_port_is_sja1105(dp))) priv = sja1105_tagger_private(ds);
return skb;
if (!test_bit(SJA1105_HWTS_RX_EN, &sp->data->state)) if (!test_bit(SJA1105_HWTS_RX_EN, &priv->state))
/* Do normal processing. */ /* Do normal processing. */
return skb; return skb;
spin_lock(&sp->data->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?
*/ */
if (sp->data->stampable_skb) { if (priv->stampable_skb) {
dev_err_ratelimited(dp->ds->dev, dev_err_ratelimited(ds->dev,
"Expected meta frame, is %12llx " "Expected meta frame, is %12llx "
"in the DSA master multicast filter?\n", "in the DSA master multicast filter?\n",
SJA1105_META_DMAC); SJA1105_META_DMAC);
kfree_skb(sp->data->stampable_skb); kfree_skb(priv->stampable_skb);
} }
/* Hold a reference to avoid dsa_switch_rcv /* Hold a reference to avoid dsa_switch_rcv
* from freeing the skb. * from freeing the skb.
*/ */
sp->data->stampable_skb = skb_get(skb); priv->stampable_skb = skb_get(skb);
spin_unlock(&sp->data->meta_lock); spin_unlock(&priv->meta_lock);
/* Tell DSA we got nothing */ /* Tell DSA we got nothing */
return NULL; return NULL;
...@@ -393,37 +420,37 @@ static struct sk_buff ...@@ -393,37 +420,37 @@ static struct sk_buff
*/ */
} else if (is_meta) { } else if (is_meta) {
struct dsa_port *dp = dsa_slave_to_port(skb->dev); struct dsa_port *dp = dsa_slave_to_port(skb->dev);
struct sja1105_port *sp = dp->priv; struct sja1105_tagger_private *priv;
struct dsa_switch *ds = dp->ds;
struct sk_buff *stampable_skb; struct sk_buff *stampable_skb;
if (unlikely(!dsa_port_is_sja1105(dp))) priv = sja1105_tagger_private(ds);
return skb;
/* Drop the meta frame if we're not in the right state /* Drop the meta frame if we're not in the right state
* to process it. * to process it.
*/ */
if (!test_bit(SJA1105_HWTS_RX_EN, &sp->data->state)) if (!test_bit(SJA1105_HWTS_RX_EN, &priv->state))
return NULL; return NULL;
spin_lock(&sp->data->meta_lock); spin_lock(&priv->meta_lock);
stampable_skb = sp->data->stampable_skb; stampable_skb = priv->stampable_skb;
sp->data->stampable_skb = NULL; priv->stampable_skb = NULL;
/* Was this a meta frame instead of the link-local /* Was this a meta frame instead of the link-local
* that we were expecting? * that we were expecting?
*/ */
if (!stampable_skb) { if (!stampable_skb) {
dev_err_ratelimited(dp->ds->dev, dev_err_ratelimited(ds->dev,
"Unexpected meta frame\n"); "Unexpected meta frame\n");
spin_unlock(&sp->data->meta_lock); spin_unlock(&priv->meta_lock);
return NULL; return NULL;
} }
if (stampable_skb->dev != skb->dev) { if (stampable_skb->dev != skb->dev) {
dev_err_ratelimited(dp->ds->dev, dev_err_ratelimited(ds->dev,
"Meta frame on wrong port\n"); "Meta frame on wrong port\n");
spin_unlock(&sp->data->meta_lock); spin_unlock(&priv->meta_lock);
return NULL; return NULL;
} }
...@@ -434,12 +461,36 @@ static struct sk_buff ...@@ -434,12 +461,36 @@ static struct sk_buff
skb = stampable_skb; skb = stampable_skb;
sja1105_transfer_meta(skb, meta); sja1105_transfer_meta(skb, meta);
spin_unlock(&sp->data->meta_lock); spin_unlock(&priv->meta_lock);
} }
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);
...@@ -526,48 +577,12 @@ static struct sk_buff *sja1105_rcv(struct sk_buff *skb, ...@@ -526,48 +577,12 @@ static struct sk_buff *sja1105_rcv(struct sk_buff *skb,
is_meta); is_meta);
} }
static void sja1110_process_meta_tstamp(struct dsa_switch *ds, int port,
u8 ts_id, enum sja1110_meta_tstamp dir,
u64 tstamp)
{
struct sk_buff *skb, *skb_tmp, *skb_match = NULL;
struct dsa_port *dp = dsa_to_port(ds, port);
struct skb_shared_hwtstamps shwt = {0};
struct sja1105_port *sp = dp->priv;
if (!dsa_port_is_sja1105(dp))
return;
/* We don't care about RX timestamps on the CPU port */
if (dir == SJA1110_META_TSTAMP_RX)
return;
spin_lock(&sp->data->skb_txtstamp_queue.lock);
skb_queue_walk_safe(&sp->data->skb_txtstamp_queue, skb, skb_tmp) {
if (SJA1105_SKB_CB(skb)->ts_id != ts_id)
continue;
__skb_unlink(skb, &sp->data->skb_txtstamp_queue);
skb_match = skb;
break;
}
spin_unlock(&sp->data->skb_txtstamp_queue.lock);
if (WARN_ON(!skb_match))
return;
shwt.hwtstamp = ns_to_ktime(sja1105_ticks_to_ns(tstamp));
skb_complete_tx_timestamp(skb_match, &shwt);
}
static struct sk_buff *sja1110_rcv_meta(struct sk_buff *skb, u16 rx_header) static struct sk_buff *sja1110_rcv_meta(struct sk_buff *skb, u16 rx_header)
{ {
u8 *buf = dsa_etype_header_pos_rx(skb) + SJA1110_HEADER_LEN; u8 *buf = dsa_etype_header_pos_rx(skb) + SJA1110_HEADER_LEN;
int switch_id = SJA1110_RX_HEADER_SWITCH_ID(rx_header); int switch_id = SJA1110_RX_HEADER_SWITCH_ID(rx_header);
int n_ts = SJA1110_RX_HEADER_N_TS(rx_header); int n_ts = SJA1110_RX_HEADER_N_TS(rx_header);
struct sja1105_tagger_data *tagger_data;
struct net_device *master = skb->dev; struct net_device *master = skb->dev;
struct dsa_port *cpu_dp; struct dsa_port *cpu_dp;
struct dsa_switch *ds; struct dsa_switch *ds;
...@@ -581,6 +596,10 @@ static struct sk_buff *sja1110_rcv_meta(struct sk_buff *skb, u16 rx_header) ...@@ -581,6 +596,10 @@ static struct sk_buff *sja1110_rcv_meta(struct sk_buff *skb, u16 rx_header)
return NULL; return NULL;
} }
tagger_data = sja1105_tagger_data(ds);
if (!tagger_data->meta_tstamp_handler)
return NULL;
for (i = 0; i <= n_ts; i++) { for (i = 0; i <= n_ts; i++) {
u8 ts_id, source_port, dir; u8 ts_id, source_port, dir;
u64 tstamp; u64 tstamp;
...@@ -590,7 +609,7 @@ static struct sk_buff *sja1110_rcv_meta(struct sk_buff *skb, u16 rx_header) ...@@ -590,7 +609,7 @@ static struct sk_buff *sja1110_rcv_meta(struct sk_buff *skb, u16 rx_header)
dir = (buf[1] & BIT(3)) >> 3; dir = (buf[1] & BIT(3)) >> 3;
tstamp = be64_to_cpu(*(__be64 *)(buf + 2)); tstamp = be64_to_cpu(*(__be64 *)(buf + 2));
sja1110_process_meta_tstamp(ds, source_port, ts_id, dir, tagger_data->meta_tstamp_handler(ds, source_port, ts_id, dir,
tstamp); tstamp);
buf += SJA1110_META_TSTAMP_SIZE; buf += SJA1110_META_TSTAMP_SIZE;
...@@ -722,11 +741,74 @@ static void sja1110_flow_dissect(const struct sk_buff *skb, __be16 *proto, ...@@ -722,11 +741,74 @@ static void sja1110_flow_dissect(const struct sk_buff *skb, __be16 *proto,
*proto = ((__be16 *)skb->data)[(VLAN_HLEN / 2) - 1]; *proto = ((__be16 *)skb->data)[(VLAN_HLEN / 2) - 1];
} }
static void sja1105_disconnect(struct dsa_switch_tree *dst)
{
struct sja1105_tagger_private *priv;
struct dsa_port *dp;
list_for_each_entry(dp, &dst->ports, list) {
priv = dp->ds->tagger_data;
if (!priv)
continue;
if (priv->xmit_worker)
kthread_destroy_worker(priv->xmit_worker);
kfree(priv);
dp->ds->priv = NULL;
}
}
static int sja1105_connect(struct dsa_switch_tree *dst)
{
struct sja1105_tagger_data *tagger_data;
struct sja1105_tagger_private *priv;
struct kthread_worker *xmit_worker;
struct dsa_port *dp;
int err;
list_for_each_entry(dp, &dst->ports, list) {
if (dp->ds->tagger_data)
continue;
priv = kzalloc(sizeof(*priv), GFP_KERNEL);
if (!priv) {
err = -ENOMEM;
goto out;
}
spin_lock_init(&priv->meta_lock);
xmit_worker = kthread_create_worker(0, "dsa%d:%d_xmit",
dst->index, dp->ds->index);
if (IS_ERR(xmit_worker)) {
err = PTR_ERR(xmit_worker);
goto out;
}
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;
dp->ds->tagger_data = priv;
}
return 0;
out:
sja1105_disconnect(dst);
return err;
}
static const struct dsa_device_ops sja1105_netdev_ops = { static const struct dsa_device_ops sja1105_netdev_ops = {
.name = "sja1105", .name = "sja1105",
.proto = DSA_TAG_PROTO_SJA1105, .proto = DSA_TAG_PROTO_SJA1105,
.xmit = sja1105_xmit, .xmit = sja1105_xmit,
.rcv = sja1105_rcv, .rcv = sja1105_rcv,
.connect = sja1105_connect,
.disconnect = sja1105_disconnect,
.needed_headroom = VLAN_HLEN, .needed_headroom = VLAN_HLEN,
.flow_dissect = sja1105_flow_dissect, .flow_dissect = sja1105_flow_dissect,
.promisc_on_master = true, .promisc_on_master = true,
...@@ -740,6 +822,8 @@ static const struct dsa_device_ops sja1110_netdev_ops = { ...@@ -740,6 +822,8 @@ static const struct dsa_device_ops sja1110_netdev_ops = {
.proto = DSA_TAG_PROTO_SJA1110, .proto = DSA_TAG_PROTO_SJA1110,
.xmit = sja1110_xmit, .xmit = sja1110_xmit,
.rcv = sja1110_rcv, .rcv = sja1110_rcv,
.connect = sja1105_connect,
.disconnect = sja1105_disconnect,
.flow_dissect = sja1110_flow_dissect, .flow_dissect = sja1110_flow_dissect,
.needed_headroom = SJA1110_HEADER_LEN + VLAN_HLEN, .needed_headroom = SJA1110_HEADER_LEN + VLAN_HLEN,
.needed_tailroom = SJA1110_RX_TRAILER_LEN + SJA1110_MAX_PADDING_LEN, .needed_tailroom = SJA1110_RX_TRAILER_LEN + SJA1110_MAX_PADDING_LEN,
......
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