ddcebda08b
Add kernel tag that introduced the patch on backport patch. Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
364 lines
9.6 KiB
Diff
364 lines
9.6 KiB
Diff
From 5950c7c0a68c915b336c70f79388626e2d576ab7 Mon Sep 17 00:00:00 2001
|
|
From: Ansuel Smith <ansuelsmth@gmail.com>
|
|
Date: Wed, 2 Feb 2022 01:03:29 +0100
|
|
Subject: [PATCH 10/16] net: dsa: qca8k: add support for mgmt read/write in
|
|
Ethernet packet
|
|
|
|
Add qca8k side support for mgmt read/write in Ethernet packet.
|
|
qca8k supports some specially crafted Ethernet packet that can be used
|
|
for mgmt read/write instead of the legacy method uart/internal mdio.
|
|
This add support for the qca8k side to craft the packet and enqueue it.
|
|
Each port and the qca8k_priv have a special struct to put data in it.
|
|
The completion API is used to wait for the packet to be received back
|
|
with the requested data.
|
|
|
|
The various steps are:
|
|
1. Craft the special packet with the qca hdr set to mgmt read/write
|
|
mode.
|
|
2. Set the lock in the dedicated mgmt struct.
|
|
3. Increment the seq number and set it in the mgmt pkt
|
|
4. Reinit the completion.
|
|
5. Enqueue the packet.
|
|
6. Wait the packet to be received.
|
|
7. Use the data set by the tagger to complete the mdio operation.
|
|
|
|
If the completion timeouts or the ack value is not true, the legacy
|
|
mdio way is used.
|
|
|
|
It has to be considered that in the initial setup mdio is still used and
|
|
mdio is still used until DSA is ready to accept and tag packet.
|
|
|
|
tag_proto_connect() is used to fill the required handler for the tagger
|
|
to correctly parse and elaborate the special Ethernet mdio packet.
|
|
|
|
Locking is added to qca8k_master_change() to make sure no mgmt Ethernet
|
|
are in progress.
|
|
|
|
Signed-off-by: Ansuel Smith <ansuelsmth@gmail.com>
|
|
Signed-off-by: David S. Miller <davem@davemloft.net>
|
|
---
|
|
drivers/net/dsa/qca8k.c | 225 ++++++++++++++++++++++++++++++++++++++++
|
|
drivers/net/dsa/qca8k.h | 13 +++
|
|
2 files changed, 238 insertions(+)
|
|
|
|
--- a/drivers/net/dsa/qca8k.c
|
|
+++ b/drivers/net/dsa/qca8k.c
|
|
@@ -20,6 +20,7 @@
|
|
#include <linux/phylink.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/etherdevice.h>
|
|
+#include <linux/dsa/tag_qca.h>
|
|
|
|
#include "qca8k.h"
|
|
|
|
@@ -170,6 +171,194 @@ qca8k_rmw(struct qca8k_priv *priv, u32 r
|
|
return regmap_update_bits(priv->regmap, reg, mask, write_val);
|
|
}
|
|
|
|
+static void qca8k_rw_reg_ack_handler(struct dsa_switch *ds, struct sk_buff *skb)
|
|
+{
|
|
+ struct qca8k_mgmt_eth_data *mgmt_eth_data;
|
|
+ struct qca8k_priv *priv = ds->priv;
|
|
+ struct qca_mgmt_ethhdr *mgmt_ethhdr;
|
|
+ u8 len, cmd;
|
|
+
|
|
+ mgmt_ethhdr = (struct qca_mgmt_ethhdr *)skb_mac_header(skb);
|
|
+ mgmt_eth_data = &priv->mgmt_eth_data;
|
|
+
|
|
+ cmd = FIELD_GET(QCA_HDR_MGMT_CMD, mgmt_ethhdr->command);
|
|
+ len = FIELD_GET(QCA_HDR_MGMT_LENGTH, mgmt_ethhdr->command);
|
|
+
|
|
+ /* Make sure the seq match the requested packet */
|
|
+ if (mgmt_ethhdr->seq == mgmt_eth_data->seq)
|
|
+ mgmt_eth_data->ack = true;
|
|
+
|
|
+ if (cmd == MDIO_READ) {
|
|
+ mgmt_eth_data->data[0] = mgmt_ethhdr->mdio_data;
|
|
+
|
|
+ /* Get the rest of the 12 byte of data */
|
|
+ if (len > QCA_HDR_MGMT_DATA1_LEN)
|
|
+ memcpy(mgmt_eth_data->data + 1, skb->data,
|
|
+ QCA_HDR_MGMT_DATA2_LEN);
|
|
+ }
|
|
+
|
|
+ complete(&mgmt_eth_data->rw_done);
|
|
+}
|
|
+
|
|
+static struct sk_buff *qca8k_alloc_mdio_header(enum mdio_cmd cmd, u32 reg, u32 *val,
|
|
+ int priority)
|
|
+{
|
|
+ struct qca_mgmt_ethhdr *mgmt_ethhdr;
|
|
+ struct sk_buff *skb;
|
|
+ u16 hdr;
|
|
+
|
|
+ skb = dev_alloc_skb(QCA_HDR_MGMT_PKT_LEN);
|
|
+ if (!skb)
|
|
+ return NULL;
|
|
+
|
|
+ skb_reset_mac_header(skb);
|
|
+ skb_set_network_header(skb, skb->len);
|
|
+
|
|
+ mgmt_ethhdr = skb_push(skb, QCA_HDR_MGMT_HEADER_LEN + QCA_HDR_LEN);
|
|
+
|
|
+ hdr = FIELD_PREP(QCA_HDR_XMIT_VERSION, QCA_HDR_VERSION);
|
|
+ hdr |= FIELD_PREP(QCA_HDR_XMIT_PRIORITY, priority);
|
|
+ hdr |= QCA_HDR_XMIT_FROM_CPU;
|
|
+ hdr |= FIELD_PREP(QCA_HDR_XMIT_DP_BIT, BIT(0));
|
|
+ hdr |= FIELD_PREP(QCA_HDR_XMIT_CONTROL, QCA_HDR_XMIT_TYPE_RW_REG);
|
|
+
|
|
+ mgmt_ethhdr->command = FIELD_PREP(QCA_HDR_MGMT_ADDR, reg);
|
|
+ mgmt_ethhdr->command |= FIELD_PREP(QCA_HDR_MGMT_LENGTH, 4);
|
|
+ mgmt_ethhdr->command |= FIELD_PREP(QCA_HDR_MGMT_CMD, cmd);
|
|
+ mgmt_ethhdr->command |= FIELD_PREP(QCA_HDR_MGMT_CHECK_CODE,
|
|
+ QCA_HDR_MGMT_CHECK_CODE_VAL);
|
|
+
|
|
+ if (cmd == MDIO_WRITE)
|
|
+ mgmt_ethhdr->mdio_data = *val;
|
|
+
|
|
+ mgmt_ethhdr->hdr = htons(hdr);
|
|
+
|
|
+ skb_put_zero(skb, QCA_HDR_MGMT_DATA2_LEN + QCA_HDR_MGMT_PADDING_LEN);
|
|
+
|
|
+ return skb;
|
|
+}
|
|
+
|
|
+static void qca8k_mdio_header_fill_seq_num(struct sk_buff *skb, u32 seq_num)
|
|
+{
|
|
+ struct qca_mgmt_ethhdr *mgmt_ethhdr;
|
|
+
|
|
+ mgmt_ethhdr = (struct qca_mgmt_ethhdr *)skb->data;
|
|
+ mgmt_ethhdr->seq = FIELD_PREP(QCA_HDR_MGMT_SEQ_NUM, seq_num);
|
|
+}
|
|
+
|
|
+static int qca8k_read_eth(struct qca8k_priv *priv, u32 reg, u32 *val)
|
|
+{
|
|
+ struct qca8k_mgmt_eth_data *mgmt_eth_data = &priv->mgmt_eth_data;
|
|
+ struct sk_buff *skb;
|
|
+ bool ack;
|
|
+ int ret;
|
|
+
|
|
+ skb = qca8k_alloc_mdio_header(MDIO_READ, reg, NULL,
|
|
+ QCA8K_ETHERNET_MDIO_PRIORITY);
|
|
+ if (!skb)
|
|
+ return -ENOMEM;
|
|
+
|
|
+ mutex_lock(&mgmt_eth_data->mutex);
|
|
+
|
|
+ /* Check mgmt_master if is operational */
|
|
+ if (!priv->mgmt_master) {
|
|
+ kfree_skb(skb);
|
|
+ mutex_unlock(&mgmt_eth_data->mutex);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ skb->dev = priv->mgmt_master;
|
|
+
|
|
+ reinit_completion(&mgmt_eth_data->rw_done);
|
|
+
|
|
+ /* Increment seq_num and set it in the mdio pkt */
|
|
+ mgmt_eth_data->seq++;
|
|
+ qca8k_mdio_header_fill_seq_num(skb, mgmt_eth_data->seq);
|
|
+ mgmt_eth_data->ack = false;
|
|
+
|
|
+ dev_queue_xmit(skb);
|
|
+
|
|
+ ret = wait_for_completion_timeout(&mgmt_eth_data->rw_done,
|
|
+ msecs_to_jiffies(QCA8K_ETHERNET_TIMEOUT));
|
|
+
|
|
+ *val = mgmt_eth_data->data[0];
|
|
+ ack = mgmt_eth_data->ack;
|
|
+
|
|
+ mutex_unlock(&mgmt_eth_data->mutex);
|
|
+
|
|
+ if (ret <= 0)
|
|
+ return -ETIMEDOUT;
|
|
+
|
|
+ if (!ack)
|
|
+ return -EINVAL;
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int qca8k_write_eth(struct qca8k_priv *priv, u32 reg, u32 val)
|
|
+{
|
|
+ struct qca8k_mgmt_eth_data *mgmt_eth_data = &priv->mgmt_eth_data;
|
|
+ struct sk_buff *skb;
|
|
+ bool ack;
|
|
+ int ret;
|
|
+
|
|
+ skb = qca8k_alloc_mdio_header(MDIO_WRITE, reg, &val,
|
|
+ QCA8K_ETHERNET_MDIO_PRIORITY);
|
|
+ if (!skb)
|
|
+ return -ENOMEM;
|
|
+
|
|
+ mutex_lock(&mgmt_eth_data->mutex);
|
|
+
|
|
+ /* Check mgmt_master if is operational */
|
|
+ if (!priv->mgmt_master) {
|
|
+ kfree_skb(skb);
|
|
+ mutex_unlock(&mgmt_eth_data->mutex);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ skb->dev = priv->mgmt_master;
|
|
+
|
|
+ reinit_completion(&mgmt_eth_data->rw_done);
|
|
+
|
|
+ /* Increment seq_num and set it in the mdio pkt */
|
|
+ mgmt_eth_data->seq++;
|
|
+ qca8k_mdio_header_fill_seq_num(skb, mgmt_eth_data->seq);
|
|
+ mgmt_eth_data->ack = false;
|
|
+
|
|
+ dev_queue_xmit(skb);
|
|
+
|
|
+ ret = wait_for_completion_timeout(&mgmt_eth_data->rw_done,
|
|
+ msecs_to_jiffies(QCA8K_ETHERNET_TIMEOUT));
|
|
+
|
|
+ ack = mgmt_eth_data->ack;
|
|
+
|
|
+ mutex_unlock(&mgmt_eth_data->mutex);
|
|
+
|
|
+ if (ret <= 0)
|
|
+ return -ETIMEDOUT;
|
|
+
|
|
+ if (!ack)
|
|
+ return -EINVAL;
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+static int
|
|
+qca8k_regmap_update_bits_eth(struct qca8k_priv *priv, u32 reg, u32 mask, u32 write_val)
|
|
+{
|
|
+ u32 val = 0;
|
|
+ int ret;
|
|
+
|
|
+ ret = qca8k_read_eth(priv, reg, &val);
|
|
+ if (ret)
|
|
+ return ret;
|
|
+
|
|
+ val &= ~mask;
|
|
+ val |= write_val;
|
|
+
|
|
+ return qca8k_write_eth(priv, reg, val);
|
|
+}
|
|
+
|
|
static int
|
|
qca8k_regmap_read(void *ctx, uint32_t reg, uint32_t *val)
|
|
{
|
|
@@ -178,6 +367,9 @@ qca8k_regmap_read(void *ctx, uint32_t re
|
|
u16 r1, r2, page;
|
|
int ret;
|
|
|
|
+ if (!qca8k_read_eth(priv, reg, val))
|
|
+ return 0;
|
|
+
|
|
qca8k_split_addr(reg, &r1, &r2, &page);
|
|
|
|
mutex_lock_nested(&bus->mdio_lock, MDIO_MUTEX_NESTED);
|
|
@@ -201,6 +393,9 @@ qca8k_regmap_write(void *ctx, uint32_t r
|
|
u16 r1, r2, page;
|
|
int ret;
|
|
|
|
+ if (!qca8k_write_eth(priv, reg, val))
|
|
+ return 0;
|
|
+
|
|
qca8k_split_addr(reg, &r1, &r2, &page);
|
|
|
|
mutex_lock_nested(&bus->mdio_lock, MDIO_MUTEX_NESTED);
|
|
@@ -225,6 +420,9 @@ qca8k_regmap_update_bits(void *ctx, uint
|
|
u32 val;
|
|
int ret;
|
|
|
|
+ if (!qca8k_regmap_update_bits_eth(priv, reg, mask, write_val))
|
|
+ return 0;
|
|
+
|
|
qca8k_split_addr(reg, &r1, &r2, &page);
|
|
|
|
mutex_lock_nested(&bus->mdio_lock, MDIO_MUTEX_NESTED);
|
|
@@ -2412,7 +2610,30 @@ qca8k_master_change(struct dsa_switch *d
|
|
if (dp->index != 0)
|
|
return;
|
|
|
|
+ mutex_lock(&priv->mgmt_eth_data.mutex);
|
|
+
|
|
priv->mgmt_master = operational ? (struct net_device *)master : NULL;
|
|
+
|
|
+ mutex_unlock(&priv->mgmt_eth_data.mutex);
|
|
+}
|
|
+
|
|
+static int qca8k_connect_tag_protocol(struct dsa_switch *ds,
|
|
+ enum dsa_tag_protocol proto)
|
|
+{
|
|
+ struct qca_tagger_data *tagger_data;
|
|
+
|
|
+ switch (proto) {
|
|
+ case DSA_TAG_PROTO_QCA:
|
|
+ tagger_data = ds->tagger_data;
|
|
+
|
|
+ tagger_data->rw_reg_ack_handler = qca8k_rw_reg_ack_handler;
|
|
+
|
|
+ break;
|
|
+ default:
|
|
+ return -EOPNOTSUPP;
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
}
|
|
|
|
static const struct dsa_switch_ops qca8k_switch_ops = {
|
|
@@ -2451,6 +2672,7 @@ static const struct dsa_switch_ops qca8k
|
|
.port_lag_join = qca8k_port_lag_join,
|
|
.port_lag_leave = qca8k_port_lag_leave,
|
|
.master_state_change = qca8k_master_change,
|
|
+ .connect_tag_protocol = qca8k_connect_tag_protocol,
|
|
};
|
|
|
|
static int qca8k_read_switch_id(struct qca8k_priv *priv)
|
|
@@ -2530,6 +2752,9 @@ qca8k_sw_probe(struct mdio_device *mdiod
|
|
if (!priv->ds)
|
|
return -ENOMEM;
|
|
|
|
+ mutex_init(&priv->mgmt_eth_data.mutex);
|
|
+ init_completion(&priv->mgmt_eth_data.rw_done);
|
|
+
|
|
priv->ds->dev = &mdiodev->dev;
|
|
priv->ds->num_ports = QCA8K_NUM_PORTS;
|
|
priv->ds->priv = priv;
|
|
--- a/drivers/net/dsa/qca8k.h
|
|
+++ b/drivers/net/dsa/qca8k.h
|
|
@@ -11,6 +11,10 @@
|
|
#include <linux/delay.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/gpio.h>
|
|
+#include <linux/dsa/tag_qca.h>
|
|
+
|
|
+#define QCA8K_ETHERNET_MDIO_PRIORITY 7
|
|
+#define QCA8K_ETHERNET_TIMEOUT 100
|
|
|
|
#define QCA8K_NUM_PORTS 7
|
|
#define QCA8K_NUM_CPU_PORTS 2
|
|
@@ -328,6 +332,14 @@ enum {
|
|
QCA8K_CPU_PORT6,
|
|
};
|
|
|
|
+struct qca8k_mgmt_eth_data {
|
|
+ struct completion rw_done;
|
|
+ struct mutex mutex; /* Enforce one mdio read/write at time */
|
|
+ bool ack;
|
|
+ u32 seq;
|
|
+ u32 data[4];
|
|
+};
|
|
+
|
|
struct qca8k_ports_config {
|
|
bool sgmii_rx_clk_falling_edge;
|
|
bool sgmii_tx_clk_falling_edge;
|
|
@@ -354,6 +366,7 @@ struct qca8k_priv {
|
|
struct gpio_desc *reset_gpio;
|
|
unsigned int port_mtu[QCA8K_NUM_PORTS];
|
|
struct net_device *mgmt_master; /* Track if mdio/mib Ethernet is available */
|
|
+ struct qca8k_mgmt_eth_data mgmt_eth_data;
|
|
};
|
|
|
|
struct qca8k_mib_desc {
|