blob: 55e5e467facd7f546ba208361ec9fdcfd7a627d9 [file] [log] [blame] [edit]
// SPDX-License-Identifier: (GPL-2.0-or-later OR BSD-3-Clause)
/*
* Copyright (c) 2014-2025, Advanced Micro Devices, Inc.
* Copyright (c) 2014, Synopsys, Inc.
* All rights reserved
*
* Author: Raju Rangoju <Raju.Rangoju@amd.com>
*/
#include <linux/crc32.h>
#include <linux/ip.h>
#include <linux/udp.h>
#include <net/tcp.h>
#include <net/udp.h>
#include <net/checksum.h>
#include <net/selftests.h>
#include "xgbe.h"
#include "xgbe-common.h"
#define XGBE_LOOPBACK_NONE 0
#define XGBE_LOOPBACK_MAC 1
#define XGBE_LOOPBACK_PHY 2
struct xgbe_test {
char name[ETH_GSTRING_LEN];
int lb;
int (*fn)(struct xgbe_prv_data *pdata);
};
static u8 xgbe_test_id;
static int xgbe_test_loopback_validate(struct sk_buff *skb,
struct net_device *ndev,
struct packet_type *pt,
struct net_device *orig_ndev)
{
struct net_test_priv *tdata = pt->af_packet_priv;
const unsigned char *dst = tdata->packet->dst;
const unsigned char *src = tdata->packet->src;
struct netsfhdr *hdr;
struct ethhdr *eh;
struct tcphdr *th;
struct udphdr *uh;
struct iphdr *ih;
int eat;
skb = skb_unshare(skb, GFP_ATOMIC);
if (!skb)
goto out;
eat = (skb->tail + skb->data_len) - skb->end;
if (eat > 0 && skb_shared(skb)) {
skb = skb_share_check(skb, GFP_ATOMIC);
if (!skb)
goto out;
}
if (skb_linearize(skb))
goto out;
if (skb_headlen(skb) < (NET_TEST_PKT_SIZE - ETH_HLEN))
goto out;
eh = (struct ethhdr *)skb_mac_header(skb);
if (dst) {
if (!ether_addr_equal_unaligned(eh->h_dest, dst))
goto out;
}
if (src) {
if (!ether_addr_equal_unaligned(eh->h_source, src))
goto out;
}
ih = ip_hdr(skb);
if (tdata->packet->tcp) {
if (ih->protocol != IPPROTO_TCP)
goto out;
th = (struct tcphdr *)((u8 *)ih + 4 * ih->ihl);
if (th->dest != htons(tdata->packet->dport))
goto out;
hdr = (struct netsfhdr *)((u8 *)th + sizeof(*th));
} else {
if (ih->protocol != IPPROTO_UDP)
goto out;
uh = (struct udphdr *)((u8 *)ih + 4 * ih->ihl);
if (uh->dest != htons(tdata->packet->dport))
goto out;
hdr = (struct netsfhdr *)((u8 *)uh + sizeof(*uh));
}
if (hdr->magic != cpu_to_be64(NET_TEST_PKT_MAGIC))
goto out;
if (tdata->packet->id != hdr->id)
goto out;
tdata->ok = true;
complete(&tdata->comp);
out:
kfree_skb(skb);
return 0;
}
static int __xgbe_test_loopback(struct xgbe_prv_data *pdata,
struct net_packet_attrs *attr)
{
struct net_test_priv *tdata;
struct sk_buff *skb = NULL;
int ret = 0;
tdata = kzalloc(sizeof(*tdata), GFP_KERNEL);
if (!tdata)
return -ENOMEM;
tdata->ok = false;
init_completion(&tdata->comp);
tdata->pt.type = htons(ETH_P_IP);
tdata->pt.func = xgbe_test_loopback_validate;
tdata->pt.dev = pdata->netdev;
tdata->pt.af_packet_priv = tdata;
tdata->packet = attr;
dev_add_pack(&tdata->pt);
skb = net_test_get_skb(pdata->netdev, xgbe_test_id, attr);
if (!skb) {
ret = -ENOMEM;
goto cleanup;
}
xgbe_test_id++;
ret = dev_direct_xmit(skb, attr->queue_mapping);
if (ret)
goto cleanup;
if (!attr->timeout)
attr->timeout = NET_LB_TIMEOUT;
wait_for_completion_timeout(&tdata->comp, attr->timeout);
ret = tdata->ok ? 0 : -ETIMEDOUT;
if (ret)
netdev_err(pdata->netdev, "Response timedout: ret %d\n", ret);
cleanup:
dev_remove_pack(&tdata->pt);
kfree(tdata);
return ret;
}
static int xgbe_test_mac_loopback(struct xgbe_prv_data *pdata)
{
struct net_packet_attrs attr = {};
attr.dst = pdata->netdev->dev_addr;
return __xgbe_test_loopback(pdata, &attr);
}
static int xgbe_test_phy_loopback(struct xgbe_prv_data *pdata)
{
struct net_packet_attrs attr = {};
int ret;
if (!pdata->netdev->phydev) {
netdev_err(pdata->netdev, "phydev not found: cannot start PHY loopback test\n");
return -EOPNOTSUPP;
}
ret = phy_loopback(pdata->netdev->phydev, true, 0);
if (ret)
return ret;
attr.dst = pdata->netdev->dev_addr;
ret = __xgbe_test_loopback(pdata, &attr);
phy_loopback(pdata->netdev->phydev, false, 0);
return ret;
}
static int xgbe_test_sph(struct xgbe_prv_data *pdata)
{
struct net_packet_attrs attr = {};
unsigned long cnt_end, cnt_start;
int ret;
cnt_start = pdata->ext_stats.rx_split_header_packets;
if (!pdata->sph) {
netdev_err(pdata->netdev, "Split Header not enabled\n");
return -EOPNOTSUPP;
}
/* UDP test */
attr.dst = pdata->netdev->dev_addr;
attr.tcp = false;
ret = __xgbe_test_loopback(pdata, &attr);
if (ret)
return ret;
cnt_end = pdata->ext_stats.rx_split_header_packets;
if (cnt_end <= cnt_start)
return -EINVAL;
/* TCP test */
cnt_start = cnt_end;
attr.dst = pdata->netdev->dev_addr;
attr.tcp = true;
ret = __xgbe_test_loopback(pdata, &attr);
if (ret)
return ret;
cnt_end = pdata->ext_stats.rx_split_header_packets;
if (cnt_end <= cnt_start)
return -EINVAL;
return 0;
}
static int xgbe_test_jumbo(struct xgbe_prv_data *pdata)
{
struct net_packet_attrs attr = {};
int size = pdata->rx_buf_size;
attr.dst = pdata->netdev->dev_addr;
attr.max_size = size - ETH_FCS_LEN;
return __xgbe_test_loopback(pdata, &attr);
}
static const struct xgbe_test xgbe_selftests[] = {
{
.name = "MAC Loopback ",
.lb = XGBE_LOOPBACK_MAC,
.fn = xgbe_test_mac_loopback,
}, {
.name = "PHY Loopback ",
.lb = XGBE_LOOPBACK_NONE,
.fn = xgbe_test_phy_loopback,
}, {
.name = "Split Header ",
.lb = XGBE_LOOPBACK_PHY,
.fn = xgbe_test_sph,
}, {
.name = "Jumbo Frame ",
.lb = XGBE_LOOPBACK_PHY,
.fn = xgbe_test_jumbo,
},
};
void xgbe_selftest_run(struct net_device *dev,
struct ethtool_test *etest, u64 *buf)
{
struct xgbe_prv_data *pdata = netdev_priv(dev);
int count = xgbe_selftest_get_count(pdata);
int i, ret;
memset(buf, 0, sizeof(*buf) * count);
xgbe_test_id = 0;
if (etest->flags != ETH_TEST_FL_OFFLINE) {
netdev_err(pdata->netdev, "Only offline tests are supported\n");
etest->flags |= ETH_TEST_FL_FAILED;
return;
} else if (!netif_carrier_ok(dev)) {
netdev_err(pdata->netdev,
"Invalid link, cannot execute tests\n");
etest->flags |= ETH_TEST_FL_FAILED;
return;
}
/* Wait for queues drain */
msleep(200);
for (i = 0; i < count; i++) {
ret = 0;
switch (xgbe_selftests[i].lb) {
case XGBE_LOOPBACK_PHY:
ret = -EOPNOTSUPP;
if (dev->phydev)
ret = phy_loopback(dev->phydev, true, 0);
if (!ret)
break;
fallthrough;
case XGBE_LOOPBACK_MAC:
ret = xgbe_enable_mac_loopback(pdata);
break;
case XGBE_LOOPBACK_NONE:
break;
default:
ret = -EOPNOTSUPP;
break;
}
/*
* First tests will always be MAC / PHY loopback.
* If any of them is not supported we abort earlier.
*/
if (ret) {
netdev_err(pdata->netdev, "Loopback not supported\n");
etest->flags |= ETH_TEST_FL_FAILED;
break;
}
ret = xgbe_selftests[i].fn(pdata);
if (ret && (ret != -EOPNOTSUPP))
etest->flags |= ETH_TEST_FL_FAILED;
buf[i] = ret;
switch (xgbe_selftests[i].lb) {
case XGBE_LOOPBACK_PHY:
ret = -EOPNOTSUPP;
if (dev->phydev)
ret = phy_loopback(dev->phydev, false, 0);
if (!ret)
break;
fallthrough;
case XGBE_LOOPBACK_MAC:
xgbe_disable_mac_loopback(pdata);
break;
default:
break;
}
}
}
void xgbe_selftest_get_strings(struct xgbe_prv_data *pdata, u8 *data)
{
u8 *p = data;
int i;
for (i = 0; i < xgbe_selftest_get_count(pdata); i++)
ethtool_puts(&p, xgbe_selftests[i].name);
}
int xgbe_selftest_get_count(struct xgbe_prv_data *pdata)
{
return ARRAY_SIZE(xgbe_selftests);
}