@@ -138,6 +138,7 @@ static void qca8k_rw_reg_ack_handler(struct dsa_switch *ds, struct sk_buff *skb)
struct qca8k_priv *priv = ds->priv;
struct qca_mgmt_ethhdr *mgmt_ethhdr;
u8 len, cmd;
+ int err = 0;
mgmt_ethhdr = (struct qca_mgmt_ethhdr *)skb_mac_header(skb);
mgmt_eth_data = &priv->mgmt_eth_data;
@@ -146,10 +147,8 @@ static void qca8k_rw_reg_ack_handler(struct dsa_switch *ds, struct sk_buff *skb)
len = FIELD_GET(QCA_HDR_MGMT_LENGTH, mgmt_ethhdr->command);
/* Make sure the seq match the requested packet */
- if (mgmt_ethhdr->seq == dsa_inband_seqno(&mgmt_eth_data->inband))
- mgmt_eth_data->err = 0;
- else
- mgmt_eth_data->err = -EINVAL;
+ if (mgmt_ethhdr->seq != dsa_inband_seqno(&mgmt_eth_data->inband))
+ err = -EPROTO;
if (cmd == MDIO_READ) {
mgmt_eth_data->data[0] = mgmt_ethhdr->mdio_data;
@@ -162,7 +161,7 @@ static void qca8k_rw_reg_ack_handler(struct dsa_switch *ds, struct sk_buff *skb)
QCA_HDR_MGMT_DATA2_LEN);
}
- dsa_inband_complete(&mgmt_eth_data->inband);
+ dsa_inband_complete(&mgmt_eth_data->inband, err);
}
static struct sk_buff *qca8k_alloc_mdio_header(enum mdio_cmd cmd, u32 reg, u32 *val,
@@ -231,7 +230,6 @@ static int qca8k_read_eth(struct qca8k_priv *priv, u32 reg, u32 *val, int len)
{
struct qca8k_mgmt_eth_data *mgmt_eth_data = &priv->mgmt_eth_data;
struct sk_buff *skb;
- int err;
int ret;
skb = qca8k_alloc_mdio_header(MDIO_READ, reg, NULL,
@@ -258,24 +256,15 @@ static int qca8k_read_eth(struct qca8k_priv *priv, u32 reg, u32 *val, int len)
if (len > QCA_HDR_MGMT_DATA1_LEN)
memcpy(val + 1, mgmt_eth_data->data + 1, len - QCA_HDR_MGMT_DATA1_LEN);
- err = mgmt_eth_data->err;
-
mutex_unlock(&mgmt_eth_data->mutex);
- if (ret)
- return ret;
-
- if (err)
- return -ret;
-
- return 0;
+ return ret;
}
static int qca8k_write_eth(struct qca8k_priv *priv, u32 reg, u32 *val, int len)
{
struct qca8k_mgmt_eth_data *mgmt_eth_data = &priv->mgmt_eth_data;
struct sk_buff *skb;
- int err;
int ret;
skb = qca8k_alloc_mdio_header(MDIO_WRITE, reg, val,
@@ -298,17 +287,9 @@ static int qca8k_write_eth(struct qca8k_priv *priv, u32 reg, u32 *val, int len)
qca8k_mdio_header_fill_seq_num,
QCA8K_ETHERNET_TIMEOUT);
- err = mgmt_eth_data->err;
-
mutex_unlock(&mgmt_eth_data->mutex);
- if (ret)
- return ret;
-
- if (err)
- return err;
-
- return 0;
+ return ret;
}
static int
@@ -431,21 +412,15 @@ qca8k_phy_eth_busy_wait(struct qca8k_mgmt_eth_data *mgmt_eth_data,
struct sk_buff *read_skb, u32 *val)
{
struct sk_buff *skb = skb_copy(read_skb, GFP_KERNEL);
- int err;
int ret;
ret = dsa_inband_request(&mgmt_eth_data->inband, skb,
qca8k_mdio_header_fill_seq_num,
QCA8K_ETHERNET_TIMEOUT);
- err = mgmt_eth_data->err;
-
if (ret)
return ret;
- if (err)
- return err;
-
*val = mgmt_eth_data->data[0];
return 0;
@@ -460,7 +435,6 @@ qca8k_phy_eth_command(struct qca8k_priv *priv, bool read, int phy,
u32 write_val, clear_val = 0, val;
struct net_device *mgmt_master;
int ret, ret1;
- int err;
if (regnum >= QCA8K_MDIO_MASTER_MAX_REG)
return -EINVAL;
@@ -522,19 +496,11 @@ qca8k_phy_eth_command(struct qca8k_priv *priv, bool read, int phy,
qca8k_mdio_header_fill_seq_num,
QCA8K_ETHERNET_TIMEOUT);
- err = mgmt_eth_data->err;
-
if (ret) {
kfree_skb(read_skb);
goto exit;
}
- if (err) {
- ret = err;
- kfree_skb(read_skb);
- goto exit;
- }
-
ret = read_poll_timeout(qca8k_phy_eth_busy_wait, ret1,
!(val & QCA8K_MDIO_MASTER_BUSY), 0,
QCA8K_BUSY_WAIT_TIMEOUT * USEC_PER_MSEC, false,
@@ -550,16 +516,9 @@ qca8k_phy_eth_command(struct qca8k_priv *priv, bool read, int phy,
qca8k_mdio_header_fill_seq_num,
QCA8K_ETHERNET_TIMEOUT);
- err = mgmt_eth_data->err;
-
if (ret)
goto exit;
- if (err) {
- ret = err;
- goto exit;
- }
-
ret = mgmt_eth_data->data[0] & QCA8K_MDIO_MASTER_DATA_MASK;
} else {
kfree_skb(read_skb);
@@ -1440,6 +1399,7 @@ static void qca8k_mib_autocast_handler(struct dsa_switch *ds, struct sk_buff *sk
const struct qca8k_mib_desc *mib;
struct mib_ethhdr *mib_ethhdr;
int i, mib_len, offset = 0;
+ int err = 0;
u64 *data;
u8 port;
@@ -1450,8 +1410,10 @@ static void qca8k_mib_autocast_handler(struct dsa_switch *ds, struct sk_buff *sk
* parse only the requested one.
*/
port = FIELD_GET(QCA_HDR_RECV_SOURCE_PORT, ntohs(mib_ethhdr->hdr));
- if (port != mib_eth_data->req_port)
+ if (port != mib_eth_data->req_port) {
+ err = -EPROTO;
goto exit;
+ }
data = mib_eth_data->data;
@@ -1480,7 +1442,7 @@ static void qca8k_mib_autocast_handler(struct dsa_switch *ds, struct sk_buff *sk
exit:
/* Complete on receiving all the mib packet */
if (refcount_dec_and_test(&mib_eth_data->port_parsed))
- dsa_inband_complete(&mib_eth_data->inband);
+ dsa_inband_complete(&mib_eth_data->inband, err);
}
static int
@@ -348,7 +348,6 @@ enum {
struct qca8k_mgmt_eth_data {
struct dsa_inband inband;
struct mutex mutex; /* Enforce one mdio read/write at time */
- int err;
u32 data[4];
};
@@ -1284,10 +1284,11 @@ struct dsa_inband {
struct completion completion;
u32 seqno;
u32 seqno_mask;
+ int err;
};
void dsa_inband_init(struct dsa_inband *inband, u32 seqno_mask);
-void dsa_inband_complete(struct dsa_inband *inband);
+void dsa_inband_complete(struct dsa_inband *inband, int err);
int dsa_inband_request(struct dsa_inband *inband, struct sk_buff *skb,
void (* insert_seqno)(struct sk_buff *skb, u32 seqno),
int timeout_ms);
@@ -526,8 +526,9 @@ void dsa_inband_init(struct dsa_inband *inband, u32 seqno_mask)
}
EXPORT_SYMBOL_GPL(dsa_inband_init);
-void dsa_inband_complete(struct dsa_inband *inband)
+void dsa_inband_complete(struct dsa_inband *inband, int err)
{
+ inband->err = err;
complete(&inband->completion);
}
EXPORT_SYMBOL_GPL(dsa_inband_complete);
@@ -553,6 +554,7 @@ int dsa_inband_request(struct dsa_inband *inband, struct sk_buff *skb,
int ret;
reinit_completion(&inband->completion);
+ inband->err = 0;
if (insert_seqno) {
inband->seqno++;
@@ -566,7 +568,8 @@ int dsa_inband_request(struct dsa_inband *inband, struct sk_buff *skb,
return ret;
if (ret == 0)
return -ETIMEDOUT;
- return 0;
+
+ return inband->err;
}
EXPORT_SYMBOL_GPL(dsa_inband_request);
The code which decodes the frame and signals the complete can experience error, such as wrong sequence number. Pass an error code between the completer and the function waiting on the complete. This simplifies the error handling, since all errors are combined into one place. At the same time, return -EPROTO if the sequence numbers don't match. This is more appropriate than EINVAL. Signed-off-by: Andrew Lunn <andrew@lunn.ch> --- drivers/net/dsa/qca/qca8k-8xxx.c | 60 ++++++-------------------------- drivers/net/dsa/qca/qca8k.h | 1 - include/net/dsa.h | 3 +- net/dsa/dsa.c | 7 ++-- 4 files changed, 18 insertions(+), 53 deletions(-)