66fd01fe59
NXP Legal insists that the following are not fine: - Saying "NXP Semiconductors" instead of "NXP", since the company's registered name is "NXP" - Putting a "(c)" sign in the copyright string - Putting a comma in the copyright string The only accepted copyright string format is "Copyright <year-range> NXP". This patch changes the copyright headers in the networking files that were sent by me, or derived from code sent by me. Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com> Reviewed-by: Ramon Fried <rfried.dev@gmail.com>
180 lines
3.7 KiB
C
180 lines
3.7 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Copyright 2019-2021 NXP
|
|
*/
|
|
|
|
#include <asm/eth.h>
|
|
#include <net/dsa.h>
|
|
#include <net.h>
|
|
|
|
#define DSA_SANDBOX_MAGIC 0x00415344
|
|
#define DSA_SANDBOX_TAG_LEN sizeof(struct dsa_sandbox_tag)
|
|
|
|
struct dsa_sandbox_priv {
|
|
struct eth_sandbox_priv *master_priv;
|
|
int port_en_mask;
|
|
};
|
|
|
|
struct dsa_sandbox_tag {
|
|
u32 magic;
|
|
u32 port;
|
|
};
|
|
|
|
static bool sb_dsa_port_enabled(struct udevice *dev, int port)
|
|
{
|
|
struct dsa_sandbox_priv *priv = dev_get_priv(dev);
|
|
|
|
return priv->port_en_mask & BIT(port);
|
|
}
|
|
|
|
static bool sb_dsa_master_enabled(struct udevice *dev)
|
|
{
|
|
struct dsa_sandbox_priv *priv = dev_get_priv(dev);
|
|
|
|
return !priv->master_priv->disabled;
|
|
}
|
|
|
|
static int dsa_sandbox_port_enable(struct udevice *dev, int port,
|
|
struct phy_device *phy)
|
|
{
|
|
struct dsa_sandbox_priv *priv = dev_get_priv(dev);
|
|
|
|
if (!sb_dsa_master_enabled(dev))
|
|
return -EFAULT;
|
|
|
|
priv->port_en_mask |= BIT(port);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void dsa_sandbox_port_disable(struct udevice *dev, int port,
|
|
struct phy_device *phy)
|
|
{
|
|
struct dsa_sandbox_priv *priv = dev_get_priv(dev);
|
|
|
|
priv->port_en_mask &= ~BIT(port);
|
|
}
|
|
|
|
static int dsa_sandbox_xmit(struct udevice *dev, int port, void *packet,
|
|
int length)
|
|
{
|
|
struct dsa_sandbox_tag *tag = packet;
|
|
|
|
if (!sb_dsa_master_enabled(dev))
|
|
return -EFAULT;
|
|
|
|
if (!sb_dsa_port_enabled(dev, port))
|
|
return -EFAULT;
|
|
|
|
tag->magic = DSA_SANDBOX_MAGIC;
|
|
tag->port = port;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int dsa_sandbox_rcv(struct udevice *dev, int *port, void *packet,
|
|
int length)
|
|
{
|
|
struct dsa_sandbox_tag *tag = packet;
|
|
|
|
if (!sb_dsa_master_enabled(dev))
|
|
return -EFAULT;
|
|
|
|
if (tag->magic != DSA_SANDBOX_MAGIC)
|
|
return -EFAULT;
|
|
|
|
*port = tag->port;
|
|
if (!sb_dsa_port_enabled(dev, tag->port))
|
|
return -EFAULT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct dsa_ops dsa_sandbox_ops = {
|
|
.port_enable = dsa_sandbox_port_enable,
|
|
.port_disable = dsa_sandbox_port_disable,
|
|
.xmit = dsa_sandbox_xmit,
|
|
.rcv = dsa_sandbox_rcv,
|
|
};
|
|
|
|
static int sb_dsa_handler(struct udevice *dev, void *packet,
|
|
unsigned int len)
|
|
{
|
|
struct eth_sandbox_priv *master_priv;
|
|
struct dsa_sandbox_tag *tag = packet;
|
|
struct udevice *dsa_dev;
|
|
u32 port_index;
|
|
void *rx_buf;
|
|
int i;
|
|
|
|
/* this emulates the switch hw and the network side */
|
|
if (tag->magic != DSA_SANDBOX_MAGIC)
|
|
return -EFAULT;
|
|
|
|
port_index = tag->port;
|
|
master_priv = dev_get_priv(dev);
|
|
dsa_dev = master_priv->priv;
|
|
if (!sb_dsa_port_enabled(dsa_dev, port_index))
|
|
return -EFAULT;
|
|
|
|
packet += DSA_SANDBOX_TAG_LEN;
|
|
len -= DSA_SANDBOX_TAG_LEN;
|
|
|
|
if (!sandbox_eth_arp_req_to_reply(dev, packet, len))
|
|
goto dsa_tagging;
|
|
if (!sandbox_eth_ping_req_to_reply(dev, packet, len))
|
|
goto dsa_tagging;
|
|
|
|
return 0;
|
|
|
|
dsa_tagging:
|
|
master_priv->recv_packets--;
|
|
i = master_priv->recv_packets;
|
|
rx_buf = master_priv->recv_packet_buffer[i];
|
|
len = master_priv->recv_packet_length[i];
|
|
memmove(rx_buf + DSA_SANDBOX_TAG_LEN, rx_buf, len);
|
|
|
|
tag = rx_buf;
|
|
tag->magic = DSA_SANDBOX_MAGIC;
|
|
tag->port = port_index;
|
|
len += DSA_SANDBOX_TAG_LEN;
|
|
master_priv->recv_packet_length[i] = len;
|
|
master_priv->recv_packets++;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int dsa_sandbox_probe(struct udevice *dev)
|
|
{
|
|
struct dsa_sandbox_priv *priv = dev_get_priv(dev);
|
|
struct udevice *master = dsa_get_master(dev);
|
|
struct eth_sandbox_priv *master_priv;
|
|
|
|
if (!master)
|
|
return -ENODEV;
|
|
|
|
dsa_set_tagging(dev, DSA_SANDBOX_TAG_LEN, 0);
|
|
|
|
master_priv = dev_get_priv(master);
|
|
master_priv->priv = dev;
|
|
master_priv->tx_handler = sb_dsa_handler;
|
|
|
|
priv->master_priv = master_priv;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct udevice_id dsa_sandbox_ids[] = {
|
|
{ .compatible = "sandbox,dsa" },
|
|
{ }
|
|
};
|
|
|
|
U_BOOT_DRIVER(dsa_sandbox) = {
|
|
.name = "dsa_sandbox",
|
|
.id = UCLASS_DSA,
|
|
.of_match = dsa_sandbox_ids,
|
|
.probe = dsa_sandbox_probe,
|
|
.ops = &dsa_sandbox_ops,
|
|
.priv_auto = sizeof(struct dsa_sandbox_priv),
|
|
};
|