mirror of https://gitee.com/openkylin/linux.git
Merge branch 'New-DSA-driver-for-VSC9953-Seville-switch'
Vladimir Oltean says: ==================== New DSA driver for VSC9953 Seville switch Looking at the Felix and Ocelot drivers, Maxim asked if it would be possible to use them as a base for a new driver for the Seville switch inside NXP T1040. Turns out, it is! The result is that the mscc_felix driver was extended to probe on Seville. The biggest challenge seems to be getting register read/write API generic enough to cover such wild bitfield variations between hardware generations. Currently, both felix and seville are built under the same kernel config option (NET_DSA_MSCC_FELIX). This has both some advantages (no need to duplicate the Lynx PCS code from felix_vsc9959.c) and some disadvantages (Seville needs to depend on PCI and on ENETC_MDIO). This will be further refined as time progresses. The driver has been completely reviewed. Previous submission was here, it wasn't accepted due to a conflict with Mark Brown's tree, very late in the release cycle: https://patchwork.ozlabs.org/project/netdev/cover/20200531122640.1375715-1-olteanv@gmail.com/ So this is more of a repost, with the only changes being related to rebasing on top of the cleanup I had to do in Ocelot. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
35c891e161
|
@ -4,10 +4,15 @@ Microchip Ocelot switch driver family
|
|||
Felix
|
||||
-----
|
||||
|
||||
The VSC9959 core is currently the only switch supported by the driver, and is
|
||||
found in the NXP LS1028A. It is a PCI device, part of the larger ENETC root
|
||||
complex. As a result, the ethernet-switch node is a sub-node of the PCIe root
|
||||
complex node and its "reg" property conforms to the parent node bindings:
|
||||
Currently the switches supported by the felix driver are:
|
||||
|
||||
- VSC9959 (Felix)
|
||||
- VSC9953 (Seville)
|
||||
|
||||
The VSC9959 switch is found in the NXP LS1028A. It is a PCI device, part of the
|
||||
larger ENETC root complex. As a result, the ethernet-switch node is a sub-node
|
||||
of the PCIe root complex node and its "reg" property conforms to the parent
|
||||
node bindings:
|
||||
|
||||
* reg: Specifies PCIe Device Number and Function Number of the endpoint device,
|
||||
in this case for the Ethernet L2Switch it is PF5 (of device 0, bus 0).
|
||||
|
@ -114,3 +119,95 @@ Example:
|
|||
};
|
||||
};
|
||||
};
|
||||
|
||||
The VSC9953 switch is found inside NXP T1040. It is a platform device with the
|
||||
following required properties:
|
||||
|
||||
- compatible:
|
||||
Must be "mscc,vsc9953-switch".
|
||||
|
||||
Supported PHY interface types (appropriate SerDes protocol setting changes are
|
||||
needed in the RCW binary):
|
||||
|
||||
* phy_mode = "internal": on ports 8 and 9
|
||||
* phy_mode = "sgmii": on ports 0, 1, 2, 3, 4, 5, 6, 7
|
||||
* phy_mode = "qsgmii": on ports 0, 1, 2, 3, 4, 5, 6, 7
|
||||
|
||||
Example:
|
||||
|
||||
&soc {
|
||||
ethernet-switch@800000 {
|
||||
#address-cells = <0x1>;
|
||||
#size-cells = <0x0>;
|
||||
compatible = "mscc,vsc9953-switch";
|
||||
little-endian;
|
||||
reg = <0x800000 0x290000>;
|
||||
|
||||
ports {
|
||||
#address-cells = <0x1>;
|
||||
#size-cells = <0x0>;
|
||||
|
||||
port@0 {
|
||||
reg = <0x0>;
|
||||
label = "swp0";
|
||||
};
|
||||
|
||||
port@1 {
|
||||
reg = <0x1>;
|
||||
label = "swp1";
|
||||
};
|
||||
|
||||
port@2 {
|
||||
reg = <0x2>;
|
||||
label = "swp2";
|
||||
};
|
||||
|
||||
port@3 {
|
||||
reg = <0x3>;
|
||||
label = "swp3";
|
||||
};
|
||||
|
||||
port@4 {
|
||||
reg = <0x4>;
|
||||
label = "swp4";
|
||||
};
|
||||
|
||||
port@5 {
|
||||
reg = <0x5>;
|
||||
label = "swp5";
|
||||
};
|
||||
|
||||
port@6 {
|
||||
reg = <0x6>;
|
||||
label = "swp6";
|
||||
};
|
||||
|
||||
port@7 {
|
||||
reg = <0x7>;
|
||||
label = "swp7";
|
||||
};
|
||||
|
||||
port@8 {
|
||||
reg = <0x8>;
|
||||
phy-mode = "internal";
|
||||
ethernet = <&enet0>;
|
||||
|
||||
fixed-link {
|
||||
speed = <2500>;
|
||||
full-duplex;
|
||||
};
|
||||
};
|
||||
|
||||
port@9 {
|
||||
reg = <0x9>;
|
||||
phy-mode = "internal";
|
||||
status = "disabled";
|
||||
|
||||
fixed-link {
|
||||
speed = <2500>;
|
||||
full-duplex;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
||||
|
|
|
@ -10,7 +10,11 @@ config NET_DSA_MSCC_FELIX
|
|||
select NET_DSA_TAG_OCELOT
|
||||
select FSL_ENETC_MDIO
|
||||
help
|
||||
This driver supports the VSC9959 network switch, which is a member of
|
||||
the Vitesse / Microsemi / Microchip Ocelot family of switching cores.
|
||||
It is embedded as a PCIe function of the NXP LS1028A ENETC integrated
|
||||
endpoint.
|
||||
This driver supports network switches from the the Vitesse /
|
||||
Microsemi / Microchip Ocelot family of switching cores that are
|
||||
connected to their host CPU via Ethernet.
|
||||
The following switches are supported:
|
||||
- VSC9959 (Felix): embedded as a PCIe function of the NXP LS1028A
|
||||
ENETC integrated endpoint.
|
||||
- VSC9953 (Seville): embedded as a platform device on the
|
||||
NXP T1040 SoC.
|
||||
|
|
|
@ -3,4 +3,5 @@ obj-$(CONFIG_NET_DSA_MSCC_FELIX) += mscc_felix.o
|
|||
|
||||
mscc_felix-objs := \
|
||||
felix.o \
|
||||
felix_vsc9959.o
|
||||
felix_vsc9959.o \
|
||||
seville_vsc9953.o
|
||||
|
|
|
@ -1,5 +1,9 @@
|
|||
// SPDX-License-Identifier: GPL-2.0
|
||||
/* Copyright 2019 NXP Semiconductors
|
||||
*
|
||||
* This is an umbrella module for all network switches that are
|
||||
* register-compatible with Ocelot and that perform I/O to their host CPU
|
||||
* through an NPI (Node Processor Interface) Ethernet port.
|
||||
*/
|
||||
#include <uapi/linux/if_bridge.h>
|
||||
#include <soc/mscc/ocelot_vcap.h>
|
||||
|
@ -9,6 +13,7 @@
|
|||
#include <soc/mscc/ocelot_ana.h>
|
||||
#include <soc/mscc/ocelot_ptp.h>
|
||||
#include <soc/mscc/ocelot.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/packing.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of_net.h>
|
||||
|
@ -185,37 +190,10 @@ static void felix_phylink_validate(struct dsa_switch *ds, int port,
|
|||
struct phylink_link_state *state)
|
||||
{
|
||||
struct ocelot *ocelot = ds->priv;
|
||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
__ETHTOOL_DECLARE_LINK_MODE_MASK(mask) = { 0, };
|
||||
struct felix *felix = ocelot_to_felix(ocelot);
|
||||
|
||||
if (state->interface != PHY_INTERFACE_MODE_NA &&
|
||||
state->interface != ocelot_port->phy_mode) {
|
||||
bitmap_zero(supported, __ETHTOOL_LINK_MODE_MASK_NBITS);
|
||||
return;
|
||||
}
|
||||
|
||||
phylink_set_port_modes(mask);
|
||||
phylink_set(mask, Autoneg);
|
||||
phylink_set(mask, Pause);
|
||||
phylink_set(mask, Asym_Pause);
|
||||
phylink_set(mask, 10baseT_Half);
|
||||
phylink_set(mask, 10baseT_Full);
|
||||
phylink_set(mask, 100baseT_Half);
|
||||
phylink_set(mask, 100baseT_Full);
|
||||
phylink_set(mask, 1000baseT_Half);
|
||||
phylink_set(mask, 1000baseT_Full);
|
||||
|
||||
if (state->interface == PHY_INTERFACE_MODE_INTERNAL ||
|
||||
state->interface == PHY_INTERFACE_MODE_2500BASEX ||
|
||||
state->interface == PHY_INTERFACE_MODE_USXGMII) {
|
||||
phylink_set(mask, 2500baseT_Full);
|
||||
phylink_set(mask, 2500baseX_Full);
|
||||
}
|
||||
|
||||
bitmap_and(supported, supported, mask,
|
||||
__ETHTOOL_LINK_MODE_MASK_NBITS);
|
||||
bitmap_and(state->advertising, state->advertising, mask,
|
||||
__ETHTOOL_LINK_MODE_MASK_NBITS);
|
||||
if (felix->info->phylink_validate)
|
||||
felix->info->phylink_validate(ocelot, port, supported, state);
|
||||
}
|
||||
|
||||
static int felix_phylink_mac_pcs_get_state(struct dsa_switch *ds, int port,
|
||||
|
@ -249,8 +227,7 @@ static void felix_phylink_mac_link_down(struct dsa_switch *ds, int port,
|
|||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
|
||||
ocelot_port_writel(ocelot_port, 0, DEV_MAC_ENA_CFG);
|
||||
ocelot_rmw_rix(ocelot, 0, QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE, port);
|
||||
ocelot_fields_write(ocelot, port, QSYS_SWITCH_PORT_MODE_PORT_ENA, 0);
|
||||
}
|
||||
|
||||
static void felix_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
||||
|
@ -326,10 +303,8 @@ static void felix_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
|||
ANA_PORT_PORT_CFG, port);
|
||||
|
||||
/* Core: Enable port for frame transfer */
|
||||
ocelot_write_rix(ocelot, QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE |
|
||||
QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG(1) |
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE, port);
|
||||
ocelot_fields_write(ocelot, port,
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
|
||||
|
||||
if (felix->info->pcs_link_up)
|
||||
felix->info->pcs_link_up(ocelot, port, link_an_mode, interface,
|
||||
|
@ -459,7 +434,6 @@ static int felix_init_structs(struct felix *felix, int num_phys_ports)
|
|||
{
|
||||
struct ocelot *ocelot = &felix->ocelot;
|
||||
phy_interface_t *port_phy_modes;
|
||||
resource_size_t switch_base;
|
||||
struct resource res;
|
||||
int port, i, err;
|
||||
|
||||
|
@ -490,9 +464,6 @@ static int felix_init_structs(struct felix *felix, int num_phys_ports)
|
|||
return err;
|
||||
}
|
||||
|
||||
switch_base = pci_resource_start(felix->pdev,
|
||||
felix->info->switch_pci_bar);
|
||||
|
||||
for (i = 0; i < TARGET_MAX; i++) {
|
||||
struct regmap *target;
|
||||
|
||||
|
@ -501,8 +472,8 @@ static int felix_init_structs(struct felix *felix, int num_phys_ports)
|
|||
|
||||
memcpy(&res, &felix->info->target_io_res[i], sizeof(res));
|
||||
res.flags = IORESOURCE_MEM;
|
||||
res.start += switch_base;
|
||||
res.end += switch_base;
|
||||
res.start += felix->switch_base;
|
||||
res.end += felix->switch_base;
|
||||
|
||||
target = ocelot_regmap_init(ocelot, &res);
|
||||
if (IS_ERR(target)) {
|
||||
|
@ -524,7 +495,8 @@ static int felix_init_structs(struct felix *felix, int num_phys_ports)
|
|||
|
||||
for (port = 0; port < num_phys_ports; port++) {
|
||||
struct ocelot_port *ocelot_port;
|
||||
void __iomem *port_regs;
|
||||
struct regmap *target;
|
||||
u8 *template;
|
||||
|
||||
ocelot_port = devm_kzalloc(ocelot->dev,
|
||||
sizeof(struct ocelot_port),
|
||||
|
@ -538,21 +510,34 @@ static int felix_init_structs(struct felix *felix, int num_phys_ports)
|
|||
|
||||
memcpy(&res, &felix->info->port_io_res[port], sizeof(res));
|
||||
res.flags = IORESOURCE_MEM;
|
||||
res.start += switch_base;
|
||||
res.end += switch_base;
|
||||
res.start += felix->switch_base;
|
||||
res.end += felix->switch_base;
|
||||
|
||||
port_regs = devm_ioremap_resource(ocelot->dev, &res);
|
||||
if (IS_ERR(port_regs)) {
|
||||
target = ocelot_regmap_init(ocelot, &res);
|
||||
if (IS_ERR(target)) {
|
||||
dev_err(ocelot->dev,
|
||||
"failed to map registers for port %d\n", port);
|
||||
"Failed to map memory space for port %d\n",
|
||||
port);
|
||||
kfree(port_phy_modes);
|
||||
return PTR_ERR(port_regs);
|
||||
return PTR_ERR(target);
|
||||
}
|
||||
|
||||
template = devm_kzalloc(ocelot->dev, OCELOT_TAG_LEN,
|
||||
GFP_KERNEL);
|
||||
if (!template) {
|
||||
dev_err(ocelot->dev,
|
||||
"Failed to allocate memory for DSA tag\n");
|
||||
kfree(port_phy_modes);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
ocelot_port->phy_mode = port_phy_modes[port];
|
||||
ocelot_port->ocelot = ocelot;
|
||||
ocelot_port->regs = port_regs;
|
||||
ocelot_port->target = target;
|
||||
ocelot_port->xmit_template = template;
|
||||
ocelot->ports[port] = ocelot_port;
|
||||
|
||||
felix->info->xmit_template_populate(ocelot, port);
|
||||
}
|
||||
|
||||
kfree(port_phy_modes);
|
||||
|
@ -791,7 +776,7 @@ static int felix_port_setup_tc(struct dsa_switch *ds, int port,
|
|||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
static const struct dsa_switch_ops felix_switch_ops = {
|
||||
const struct dsa_switch_ops felix_switch_ops = {
|
||||
.get_tag_protocol = felix_get_tag_protocol,
|
||||
.setup = felix_setup,
|
||||
.teardown = felix_teardown,
|
||||
|
@ -834,149 +819,28 @@ static const struct dsa_switch_ops felix_switch_ops = {
|
|||
.port_setup_tc = felix_port_setup_tc,
|
||||
};
|
||||
|
||||
static struct felix_info *felix_instance_tbl[] = {
|
||||
[FELIX_INSTANCE_VSC9959] = &felix_info_vsc9959,
|
||||
};
|
||||
|
||||
static irqreturn_t felix_irq_handler(int irq, void *data)
|
||||
static int __init felix_init(void)
|
||||
{
|
||||
struct ocelot *ocelot = (struct ocelot *)data;
|
||||
|
||||
/* The INTB interrupt is used for both PTP TX timestamp interrupt
|
||||
* and preemption status change interrupt on each port.
|
||||
*
|
||||
* - Get txtstamp if have
|
||||
* - TODO: handle preemption. Without handling it, driver may get
|
||||
* interrupt storm.
|
||||
*/
|
||||
|
||||
ocelot_get_txtstamp(ocelot);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static int felix_pci_probe(struct pci_dev *pdev,
|
||||
const struct pci_device_id *id)
|
||||
{
|
||||
enum felix_instance instance = id->driver_data;
|
||||
struct dsa_switch *ds;
|
||||
struct ocelot *ocelot;
|
||||
struct felix *felix;
|
||||
int err;
|
||||
|
||||
if (pdev->dev.of_node && !of_device_is_available(pdev->dev.of_node)) {
|
||||
dev_info(&pdev->dev, "device is disabled, skipping\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
err = pci_register_driver(&felix_vsc9959_pci_driver);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = pci_enable_device(pdev);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "device enable failed\n");
|
||||
goto err_pci_enable;
|
||||
}
|
||||
|
||||
/* set up for high or low dma */
|
||||
err = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(64));
|
||||
if (err) {
|
||||
err = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(32));
|
||||
if (err) {
|
||||
dev_err(&pdev->dev,
|
||||
"DMA configuration failed: 0x%x\n", err);
|
||||
goto err_dma;
|
||||
}
|
||||
}
|
||||
|
||||
felix = kzalloc(sizeof(struct felix), GFP_KERNEL);
|
||||
if (!felix) {
|
||||
err = -ENOMEM;
|
||||
dev_err(&pdev->dev, "Failed to allocate driver memory\n");
|
||||
goto err_alloc_felix;
|
||||
}
|
||||
|
||||
pci_set_drvdata(pdev, felix);
|
||||
ocelot = &felix->ocelot;
|
||||
ocelot->dev = &pdev->dev;
|
||||
felix->pdev = pdev;
|
||||
felix->info = felix_instance_tbl[instance];
|
||||
|
||||
pci_set_master(pdev);
|
||||
|
||||
err = devm_request_threaded_irq(&pdev->dev, pdev->irq, NULL,
|
||||
&felix_irq_handler, IRQF_ONESHOT,
|
||||
"felix-intb", ocelot);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to request irq\n");
|
||||
goto err_alloc_irq;
|
||||
}
|
||||
|
||||
ocelot->ptp = 1;
|
||||
|
||||
ds = kzalloc(sizeof(struct dsa_switch), GFP_KERNEL);
|
||||
if (!ds) {
|
||||
err = -ENOMEM;
|
||||
dev_err(&pdev->dev, "Failed to allocate DSA switch\n");
|
||||
goto err_alloc_ds;
|
||||
}
|
||||
|
||||
ds->dev = &pdev->dev;
|
||||
ds->num_ports = felix->info->num_ports;
|
||||
ds->num_tx_queues = felix->info->num_tx_queues;
|
||||
ds->ops = &felix_switch_ops;
|
||||
ds->priv = ocelot;
|
||||
felix->ds = ds;
|
||||
|
||||
err = dsa_register_switch(ds);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to register DSA switch: %d\n", err);
|
||||
goto err_register_ds;
|
||||
}
|
||||
err = platform_driver_register(&seville_vsc9953_driver);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
return 0;
|
||||
|
||||
err_register_ds:
|
||||
kfree(ds);
|
||||
err_alloc_ds:
|
||||
err_alloc_irq:
|
||||
err_alloc_felix:
|
||||
kfree(felix);
|
||||
err_dma:
|
||||
pci_disable_device(pdev);
|
||||
err_pci_enable:
|
||||
return err;
|
||||
}
|
||||
module_init(felix_init);
|
||||
|
||||
static void felix_pci_remove(struct pci_dev *pdev)
|
||||
static void __exit felix_exit(void)
|
||||
{
|
||||
struct felix *felix;
|
||||
|
||||
felix = pci_get_drvdata(pdev);
|
||||
|
||||
dsa_unregister_switch(felix->ds);
|
||||
|
||||
kfree(felix->ds);
|
||||
kfree(felix);
|
||||
|
||||
pci_disable_device(pdev);
|
||||
pci_unregister_driver(&felix_vsc9959_pci_driver);
|
||||
platform_driver_unregister(&seville_vsc9953_driver);
|
||||
}
|
||||
|
||||
static struct pci_device_id felix_ids[] = {
|
||||
{
|
||||
/* NXP LS1028A */
|
||||
PCI_DEVICE(PCI_VENDOR_ID_FREESCALE, 0xEEF0),
|
||||
.driver_data = FELIX_INSTANCE_VSC9959,
|
||||
},
|
||||
{ 0, }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(pci, felix_ids);
|
||||
|
||||
static struct pci_driver felix_pci_driver = {
|
||||
.name = KBUILD_MODNAME,
|
||||
.id_table = felix_ids,
|
||||
.probe = felix_pci_probe,
|
||||
.remove = felix_pci_remove,
|
||||
};
|
||||
|
||||
module_pci_driver(felix_pci_driver);
|
||||
module_exit(felix_exit);
|
||||
|
||||
MODULE_DESCRIPTION("Felix Switch driver");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
|
|
|
@ -37,28 +37,42 @@ struct felix_info {
|
|||
int speed, int duplex);
|
||||
void (*pcs_link_state)(struct ocelot *ocelot, int port,
|
||||
struct phylink_link_state *state);
|
||||
void (*phylink_validate)(struct ocelot *ocelot, int port,
|
||||
unsigned long *supported,
|
||||
struct phylink_link_state *state);
|
||||
int (*prevalidate_phy_mode)(struct ocelot *ocelot, int port,
|
||||
phy_interface_t phy_mode);
|
||||
int (*port_setup_tc)(struct dsa_switch *ds, int port,
|
||||
enum tc_setup_type type, void *type_data);
|
||||
void (*port_sched_speed_set)(struct ocelot *ocelot, int port,
|
||||
u32 speed);
|
||||
void (*xmit_template_populate)(struct ocelot *ocelot, int port);
|
||||
};
|
||||
|
||||
extern struct felix_info felix_info_vsc9959;
|
||||
|
||||
enum felix_instance {
|
||||
FELIX_INSTANCE_VSC9959 = 0,
|
||||
};
|
||||
extern const struct dsa_switch_ops felix_switch_ops;
|
||||
extern struct pci_driver felix_vsc9959_pci_driver;
|
||||
extern struct platform_driver seville_vsc9953_driver;
|
||||
|
||||
/* DSA glue / front-end for struct ocelot */
|
||||
struct felix {
|
||||
struct dsa_switch *ds;
|
||||
struct pci_dev *pdev;
|
||||
struct felix_info *info;
|
||||
const struct felix_info *info;
|
||||
struct ocelot ocelot;
|
||||
struct mii_bus *imdio;
|
||||
struct phy_device **pcs;
|
||||
resource_size_t switch_base;
|
||||
resource_size_t imdio_base;
|
||||
};
|
||||
|
||||
void vsc9959_pcs_link_state(struct ocelot *ocelot, int port,
|
||||
struct phylink_link_state *state);
|
||||
void vsc9959_pcs_config(struct ocelot *ocelot, int port,
|
||||
unsigned int link_an_mode,
|
||||
const struct phylink_link_state *state);
|
||||
void vsc9959_pcs_link_up(struct ocelot *ocelot, int port,
|
||||
unsigned int link_an_mode,
|
||||
phy_interface_t interface,
|
||||
int speed, int duplex);
|
||||
void vsc9959_mdio_bus_free(struct ocelot *ocelot);
|
||||
|
||||
#endif
|
||||
|
|
|
@ -8,6 +8,7 @@
|
|||
#include <soc/mscc/ocelot_ptp.h>
|
||||
#include <soc/mscc/ocelot_sys.h>
|
||||
#include <soc/mscc/ocelot.h>
|
||||
#include <linux/packing.h>
|
||||
#include <net/pkt_sched.h>
|
||||
#include <linux/iopoll.h>
|
||||
#include <linux/pci.h>
|
||||
|
@ -329,7 +330,49 @@ static const u32 vsc9959_gcb_regmap[] = {
|
|||
REG(GCB_SOFT_RST, 0x000004),
|
||||
};
|
||||
|
||||
static const u32 *vsc9959_regmap[] = {
|
||||
static const u32 vsc9959_dev_gmii_regmap[] = {
|
||||
REG(DEV_CLOCK_CFG, 0x0),
|
||||
REG(DEV_PORT_MISC, 0x4),
|
||||
REG(DEV_EVENTS, 0x8),
|
||||
REG(DEV_EEE_CFG, 0xc),
|
||||
REG(DEV_RX_PATH_DELAY, 0x10),
|
||||
REG(DEV_TX_PATH_DELAY, 0x14),
|
||||
REG(DEV_PTP_PREDICT_CFG, 0x18),
|
||||
REG(DEV_MAC_ENA_CFG, 0x1c),
|
||||
REG(DEV_MAC_MODE_CFG, 0x20),
|
||||
REG(DEV_MAC_MAXLEN_CFG, 0x24),
|
||||
REG(DEV_MAC_TAGS_CFG, 0x28),
|
||||
REG(DEV_MAC_ADV_CHK_CFG, 0x2c),
|
||||
REG(DEV_MAC_IFG_CFG, 0x30),
|
||||
REG(DEV_MAC_HDX_CFG, 0x34),
|
||||
REG(DEV_MAC_DBG_CFG, 0x38),
|
||||
REG(DEV_MAC_FC_MAC_LOW_CFG, 0x3c),
|
||||
REG(DEV_MAC_FC_MAC_HIGH_CFG, 0x40),
|
||||
REG(DEV_MAC_STICKY, 0x44),
|
||||
REG_RESERVED(PCS1G_CFG),
|
||||
REG_RESERVED(PCS1G_MODE_CFG),
|
||||
REG_RESERVED(PCS1G_SD_CFG),
|
||||
REG_RESERVED(PCS1G_ANEG_CFG),
|
||||
REG_RESERVED(PCS1G_ANEG_NP_CFG),
|
||||
REG_RESERVED(PCS1G_LB_CFG),
|
||||
REG_RESERVED(PCS1G_DBG_CFG),
|
||||
REG_RESERVED(PCS1G_CDET_CFG),
|
||||
REG_RESERVED(PCS1G_ANEG_STATUS),
|
||||
REG_RESERVED(PCS1G_ANEG_NP_STATUS),
|
||||
REG_RESERVED(PCS1G_LINK_STATUS),
|
||||
REG_RESERVED(PCS1G_LINK_DOWN_CNT),
|
||||
REG_RESERVED(PCS1G_STICKY),
|
||||
REG_RESERVED(PCS1G_DEBUG_STATUS),
|
||||
REG_RESERVED(PCS1G_LPI_CFG),
|
||||
REG_RESERVED(PCS1G_LPI_WAKE_ERROR_CNT),
|
||||
REG_RESERVED(PCS1G_LPI_STATUS),
|
||||
REG_RESERVED(PCS1G_TSTPAT_MODE_CFG),
|
||||
REG_RESERVED(PCS1G_TSTPAT_STATUS),
|
||||
REG_RESERVED(DEV_PCS_FX100_CFG),
|
||||
REG_RESERVED(DEV_PCS_FX100_STATUS),
|
||||
};
|
||||
|
||||
static const u32 *vsc9959_regmap[TARGET_MAX] = {
|
||||
[ANA] = vsc9959_ana_regmap,
|
||||
[QS] = vsc9959_qs_regmap,
|
||||
[QSYS] = vsc9959_qsys_regmap,
|
||||
|
@ -338,10 +381,11 @@ static const u32 *vsc9959_regmap[] = {
|
|||
[S2] = vsc9959_s2_regmap,
|
||||
[PTP] = vsc9959_ptp_regmap,
|
||||
[GCB] = vsc9959_gcb_regmap,
|
||||
[DEV_GMII] = vsc9959_dev_gmii_regmap,
|
||||
};
|
||||
|
||||
/* Addresses are relative to the PCI device's base address */
|
||||
static const struct resource vsc9959_target_io_res[] = {
|
||||
static const struct resource vsc9959_target_io_res[TARGET_MAX] = {
|
||||
[ANA] = {
|
||||
.start = 0x0280000,
|
||||
.end = 0x028ffff,
|
||||
|
@ -426,7 +470,7 @@ static const struct resource vsc9959_imdio_res = {
|
|||
.name = "imdio",
|
||||
};
|
||||
|
||||
static const struct reg_field vsc9959_regfields[] = {
|
||||
static const struct reg_field vsc9959_regfields[REGFIELD_MAX] = {
|
||||
[ANA_ADVLEARN_VLAN_CHK] = REG_FIELD(ANA_ADVLEARN, 6, 6),
|
||||
[ANA_ADVLEARN_LEARN_MIRROR] = REG_FIELD(ANA_ADVLEARN, 0, 5),
|
||||
[ANA_ANEVENTS_FLOOD_DISCARD] = REG_FIELD(ANA_ANEVENTS, 30, 30),
|
||||
|
@ -460,6 +504,20 @@ static const struct reg_field vsc9959_regfields[] = {
|
|||
[ANA_TABLES_MACTINDX_M_INDEX] = REG_FIELD(ANA_TABLES_MACTINDX, 0, 10),
|
||||
[SYS_RESET_CFG_CORE_ENA] = REG_FIELD(SYS_RESET_CFG, 0, 0),
|
||||
[GCB_SOFT_RST_SWC_RST] = REG_FIELD(GCB_SOFT_RST, 0, 0),
|
||||
/* Replicated per number of ports (7), register size 4 per port */
|
||||
[QSYS_SWITCH_PORT_MODE_PORT_ENA] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 14, 14, 7, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 11, 13, 7, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_YEL_RSRVD] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 10, 10, 7, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 9, 9, 7, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_TX_PFC_ENA] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 1, 8, 7, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_TX_PFC_MODE] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 0, 0, 7, 4),
|
||||
[SYS_PORT_MODE_DATA_WO_TS] = REG_FIELD_ID(SYS_PORT_MODE, 5, 6, 7, 4),
|
||||
[SYS_PORT_MODE_INCL_INJ_HDR] = REG_FIELD_ID(SYS_PORT_MODE, 3, 4, 7, 4),
|
||||
[SYS_PORT_MODE_INCL_XTR_HDR] = REG_FIELD_ID(SYS_PORT_MODE, 1, 2, 7, 4),
|
||||
[SYS_PORT_MODE_INCL_HDR_ERR] = REG_FIELD_ID(SYS_PORT_MODE, 0, 0, 7, 4),
|
||||
[SYS_PAUSE_CFG_PAUSE_START] = REG_FIELD_ID(SYS_PAUSE_CFG, 10, 18, 7, 4),
|
||||
[SYS_PAUSE_CFG_PAUSE_STOP] = REG_FIELD_ID(SYS_PAUSE_CFG, 1, 9, 7, 4),
|
||||
[SYS_PAUSE_CFG_PAUSE_ENA] = REG_FIELD_ID(SYS_PAUSE_CFG, 0, 1, 7, 4),
|
||||
};
|
||||
|
||||
static const struct ocelot_stat_layout vsc9959_stats_layout[] = {
|
||||
|
@ -794,7 +852,7 @@ static void vsc9959_pcs_config_usxgmii(struct phy_device *pcs,
|
|||
USXGMII_ADVERTISE_FDX);
|
||||
}
|
||||
|
||||
static void vsc9959_pcs_config(struct ocelot *ocelot, int port,
|
||||
void vsc9959_pcs_config(struct ocelot *ocelot, int port,
|
||||
unsigned int link_an_mode,
|
||||
const struct phylink_link_state *state)
|
||||
{
|
||||
|
@ -907,7 +965,7 @@ static void vsc9959_pcs_link_up_2500basex(struct phy_device *pcs,
|
|||
phy_clear_bits(pcs, MII_BMCR, BMCR_ANENABLE);
|
||||
}
|
||||
|
||||
static void vsc9959_pcs_link_up(struct ocelot *ocelot, int port,
|
||||
void vsc9959_pcs_link_up(struct ocelot *ocelot, int port,
|
||||
unsigned int link_an_mode,
|
||||
phy_interface_t interface,
|
||||
int speed, int duplex)
|
||||
|
@ -1038,7 +1096,7 @@ static void vsc9959_pcs_link_state_usxgmii(struct phy_device *pcs,
|
|||
pcs->duplex = DUPLEX_HALF;
|
||||
}
|
||||
|
||||
static void vsc9959_pcs_link_state(struct ocelot *ocelot, int port,
|
||||
void vsc9959_pcs_link_state(struct ocelot *ocelot, int port,
|
||||
struct phylink_link_state *state)
|
||||
{
|
||||
struct felix *felix = ocelot_to_felix(ocelot);
|
||||
|
@ -1070,6 +1128,43 @@ static void vsc9959_pcs_link_state(struct ocelot *ocelot, int port,
|
|||
vsc9959_pcs_link_state_resolve(pcs, state);
|
||||
}
|
||||
|
||||
static void vsc9959_phylink_validate(struct ocelot *ocelot, int port,
|
||||
unsigned long *supported,
|
||||
struct phylink_link_state *state)
|
||||
{
|
||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
__ETHTOOL_DECLARE_LINK_MODE_MASK(mask) = { 0, };
|
||||
|
||||
if (state->interface != PHY_INTERFACE_MODE_NA &&
|
||||
state->interface != ocelot_port->phy_mode) {
|
||||
bitmap_zero(supported, __ETHTOOL_LINK_MODE_MASK_NBITS);
|
||||
return;
|
||||
}
|
||||
|
||||
phylink_set_port_modes(mask);
|
||||
phylink_set(mask, Autoneg);
|
||||
phylink_set(mask, Pause);
|
||||
phylink_set(mask, Asym_Pause);
|
||||
phylink_set(mask, 10baseT_Half);
|
||||
phylink_set(mask, 10baseT_Full);
|
||||
phylink_set(mask, 100baseT_Half);
|
||||
phylink_set(mask, 100baseT_Full);
|
||||
phylink_set(mask, 1000baseT_Half);
|
||||
phylink_set(mask, 1000baseT_Full);
|
||||
|
||||
if (state->interface == PHY_INTERFACE_MODE_INTERNAL ||
|
||||
state->interface == PHY_INTERFACE_MODE_2500BASEX ||
|
||||
state->interface == PHY_INTERFACE_MODE_USXGMII) {
|
||||
phylink_set(mask, 2500baseT_Full);
|
||||
phylink_set(mask, 2500baseX_Full);
|
||||
}
|
||||
|
||||
bitmap_and(supported, supported, mask,
|
||||
__ETHTOOL_LINK_MODE_MASK_NBITS);
|
||||
bitmap_and(state->advertising, state->advertising, mask,
|
||||
__ETHTOOL_LINK_MODE_MASK_NBITS);
|
||||
}
|
||||
|
||||
static int vsc9959_prevalidate_phy_mode(struct ocelot *ocelot, int port,
|
||||
phy_interface_t phy_mode)
|
||||
{
|
||||
|
@ -1091,8 +1186,21 @@ static int vsc9959_prevalidate_phy_mode(struct ocelot *ocelot, int port,
|
|||
}
|
||||
}
|
||||
|
||||
/* Watermark encode
|
||||
* Bit 8: Unit; 0:1, 1:16
|
||||
* Bit 7-0: Value to be multiplied with unit
|
||||
*/
|
||||
static u16 vsc9959_wm_enc(u16 value)
|
||||
{
|
||||
if (value >= BIT(8))
|
||||
return BIT(8) | (value / 16);
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
static const struct ocelot_ops vsc9959_ops = {
|
||||
.reset = vsc9959_reset,
|
||||
.wm_enc = vsc9959_wm_enc,
|
||||
};
|
||||
|
||||
static int vsc9959_mdio_bus_alloc(struct ocelot *ocelot)
|
||||
|
@ -1100,7 +1208,6 @@ static int vsc9959_mdio_bus_alloc(struct ocelot *ocelot)
|
|||
struct felix *felix = ocelot_to_felix(ocelot);
|
||||
struct enetc_mdio_priv *mdio_priv;
|
||||
struct device *dev = ocelot->dev;
|
||||
resource_size_t imdio_base;
|
||||
void __iomem *imdio_regs;
|
||||
struct resource res;
|
||||
struct enetc_hw *hw;
|
||||
|
@ -1116,13 +1223,10 @@ static int vsc9959_mdio_bus_alloc(struct ocelot *ocelot)
|
|||
return -ENOMEM;
|
||||
}
|
||||
|
||||
imdio_base = pci_resource_start(felix->pdev,
|
||||
felix->info->imdio_pci_bar);
|
||||
|
||||
memcpy(&res, felix->info->imdio_res, sizeof(res));
|
||||
res.flags = IORESOURCE_MEM;
|
||||
res.start += imdio_base;
|
||||
res.end += imdio_base;
|
||||
res.start += felix->imdio_base;
|
||||
res.end += felix->imdio_base;
|
||||
|
||||
imdio_regs = devm_ioremap_resource(dev, &res);
|
||||
if (IS_ERR(imdio_regs)) {
|
||||
|
@ -1182,7 +1286,7 @@ static int vsc9959_mdio_bus_alloc(struct ocelot *ocelot)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static void vsc9959_mdio_bus_free(struct ocelot *ocelot)
|
||||
void vsc9959_mdio_bus_free(struct ocelot *ocelot)
|
||||
{
|
||||
struct felix *felix = ocelot_to_felix(ocelot);
|
||||
int port;
|
||||
|
@ -1378,7 +1482,25 @@ static int vsc9959_port_setup_tc(struct dsa_switch *ds, int port,
|
|||
}
|
||||
}
|
||||
|
||||
struct felix_info felix_info_vsc9959 = {
|
||||
static void vsc9959_xmit_template_populate(struct ocelot *ocelot, int port)
|
||||
{
|
||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
u8 *template = ocelot_port->xmit_template;
|
||||
u64 bypass, dest, src;
|
||||
|
||||
/* Set the source port as the CPU port module and not the
|
||||
* NPI port
|
||||
*/
|
||||
src = ocelot->num_phys_ports;
|
||||
dest = BIT(port);
|
||||
bypass = true;
|
||||
|
||||
packing(template, &bypass, 127, 127, OCELOT_TAG_LEN, PACK, 0);
|
||||
packing(template, &dest, 68, 56, OCELOT_TAG_LEN, PACK, 0);
|
||||
packing(template, &src, 46, 43, OCELOT_TAG_LEN, PACK, 0);
|
||||
}
|
||||
|
||||
static const struct felix_info felix_info_vsc9959 = {
|
||||
.target_io_res = vsc9959_target_io_res,
|
||||
.port_io_res = vsc9959_port_io_res,
|
||||
.imdio_res = &vsc9959_imdio_res,
|
||||
|
@ -1401,7 +1523,148 @@ struct felix_info felix_info_vsc9959 = {
|
|||
.pcs_config = vsc9959_pcs_config,
|
||||
.pcs_link_up = vsc9959_pcs_link_up,
|
||||
.pcs_link_state = vsc9959_pcs_link_state,
|
||||
.phylink_validate = vsc9959_phylink_validate,
|
||||
.prevalidate_phy_mode = vsc9959_prevalidate_phy_mode,
|
||||
.port_setup_tc = vsc9959_port_setup_tc,
|
||||
.port_sched_speed_set = vsc9959_sched_speed_set,
|
||||
.xmit_template_populate = vsc9959_xmit_template_populate,
|
||||
};
|
||||
|
||||
static irqreturn_t felix_irq_handler(int irq, void *data)
|
||||
{
|
||||
struct ocelot *ocelot = (struct ocelot *)data;
|
||||
|
||||
/* The INTB interrupt is used for both PTP TX timestamp interrupt
|
||||
* and preemption status change interrupt on each port.
|
||||
*
|
||||
* - Get txtstamp if have
|
||||
* - TODO: handle preemption. Without handling it, driver may get
|
||||
* interrupt storm.
|
||||
*/
|
||||
|
||||
ocelot_get_txtstamp(ocelot);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static int felix_pci_probe(struct pci_dev *pdev,
|
||||
const struct pci_device_id *id)
|
||||
{
|
||||
struct dsa_switch *ds;
|
||||
struct ocelot *ocelot;
|
||||
struct felix *felix;
|
||||
int err;
|
||||
|
||||
if (pdev->dev.of_node && !of_device_is_available(pdev->dev.of_node)) {
|
||||
dev_info(&pdev->dev, "device is disabled, skipping\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
err = pci_enable_device(pdev);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "device enable failed\n");
|
||||
goto err_pci_enable;
|
||||
}
|
||||
|
||||
/* set up for high or low dma */
|
||||
err = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(64));
|
||||
if (err) {
|
||||
err = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(32));
|
||||
if (err) {
|
||||
dev_err(&pdev->dev,
|
||||
"DMA configuration failed: 0x%x\n", err);
|
||||
goto err_dma;
|
||||
}
|
||||
}
|
||||
|
||||
felix = kzalloc(sizeof(struct felix), GFP_KERNEL);
|
||||
if (!felix) {
|
||||
err = -ENOMEM;
|
||||
dev_err(&pdev->dev, "Failed to allocate driver memory\n");
|
||||
goto err_alloc_felix;
|
||||
}
|
||||
|
||||
pci_set_drvdata(pdev, felix);
|
||||
ocelot = &felix->ocelot;
|
||||
ocelot->dev = &pdev->dev;
|
||||
felix->info = &felix_info_vsc9959;
|
||||
felix->switch_base = pci_resource_start(pdev,
|
||||
felix->info->switch_pci_bar);
|
||||
felix->imdio_base = pci_resource_start(pdev,
|
||||
felix->info->imdio_pci_bar);
|
||||
|
||||
pci_set_master(pdev);
|
||||
|
||||
err = devm_request_threaded_irq(&pdev->dev, pdev->irq, NULL,
|
||||
&felix_irq_handler, IRQF_ONESHOT,
|
||||
"felix-intb", ocelot);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to request irq\n");
|
||||
goto err_alloc_irq;
|
||||
}
|
||||
|
||||
ocelot->ptp = 1;
|
||||
|
||||
ds = kzalloc(sizeof(struct dsa_switch), GFP_KERNEL);
|
||||
if (!ds) {
|
||||
err = -ENOMEM;
|
||||
dev_err(&pdev->dev, "Failed to allocate DSA switch\n");
|
||||
goto err_alloc_ds;
|
||||
}
|
||||
|
||||
ds->dev = &pdev->dev;
|
||||
ds->num_ports = felix->info->num_ports;
|
||||
ds->num_tx_queues = felix->info->num_tx_queues;
|
||||
ds->ops = &felix_switch_ops;
|
||||
ds->priv = ocelot;
|
||||
felix->ds = ds;
|
||||
|
||||
err = dsa_register_switch(ds);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to register DSA switch: %d\n", err);
|
||||
goto err_register_ds;
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
err_register_ds:
|
||||
kfree(ds);
|
||||
err_alloc_ds:
|
||||
err_alloc_irq:
|
||||
err_alloc_felix:
|
||||
kfree(felix);
|
||||
err_dma:
|
||||
pci_disable_device(pdev);
|
||||
err_pci_enable:
|
||||
return err;
|
||||
}
|
||||
|
||||
static void felix_pci_remove(struct pci_dev *pdev)
|
||||
{
|
||||
struct felix *felix;
|
||||
|
||||
felix = pci_get_drvdata(pdev);
|
||||
|
||||
dsa_unregister_switch(felix->ds);
|
||||
|
||||
kfree(felix->ds);
|
||||
kfree(felix);
|
||||
|
||||
pci_disable_device(pdev);
|
||||
}
|
||||
|
||||
static struct pci_device_id felix_ids[] = {
|
||||
{
|
||||
/* NXP LS1028A */
|
||||
PCI_DEVICE(PCI_VENDOR_ID_FREESCALE, 0xEEF0),
|
||||
},
|
||||
{ 0, }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(pci, felix_ids);
|
||||
|
||||
struct pci_driver felix_vsc9959_pci_driver = {
|
||||
.name = "mscc_felix",
|
||||
.id_table = felix_ids,
|
||||
.probe = felix_pci_probe,
|
||||
.remove = felix_pci_remove,
|
||||
};
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -309,18 +309,6 @@ static void ocelot_vlan_init(struct ocelot *ocelot)
|
|||
}
|
||||
}
|
||||
|
||||
/* Watermark encode
|
||||
* Bit 8: Unit; 0:1, 1:16
|
||||
* Bit 7-0: Value to be multiplied with unit
|
||||
*/
|
||||
static u16 ocelot_wm_enc(u16 value)
|
||||
{
|
||||
if (value >= BIT(8))
|
||||
return BIT(8) | (value / 16);
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
void ocelot_adjust_link(struct ocelot *ocelot, int port,
|
||||
struct phy_device *phydev)
|
||||
{
|
||||
|
@ -389,10 +377,8 @@ void ocelot_adjust_link(struct ocelot *ocelot, int port,
|
|||
ANA_PFC_PFC_CFG, port);
|
||||
|
||||
/* Core: Enable port for frame transfer */
|
||||
ocelot_write_rix(ocelot, QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE |
|
||||
QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG(1) |
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE, port);
|
||||
ocelot_fields_write(ocelot, port,
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
|
||||
|
||||
/* Flow control */
|
||||
ocelot_write_rix(ocelot, SYS_MAC_FC_CFG_PAUSE_VAL_CFG(0xffff) |
|
||||
|
@ -423,8 +409,7 @@ void ocelot_port_disable(struct ocelot *ocelot, int port)
|
|||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
|
||||
ocelot_port_writel(ocelot_port, 0, DEV_MAC_ENA_CFG);
|
||||
ocelot_rmw_rix(ocelot, 0, QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE, port);
|
||||
ocelot_fields_write(ocelot, port, QSYS_SWITCH_PORT_MODE_PORT_ENA, 0);
|
||||
}
|
||||
EXPORT_SYMBOL(ocelot_port_disable);
|
||||
|
||||
|
@ -1262,6 +1247,7 @@ void ocelot_port_set_maxlen(struct ocelot *ocelot, int port, size_t sdu)
|
|||
{
|
||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
int maxlen = sdu + ETH_HLEN + ETH_FCS_LEN;
|
||||
int pause_start, pause_stop;
|
||||
int atop_wm;
|
||||
|
||||
if (port == ocelot->npi) {
|
||||
|
@ -1275,20 +1261,20 @@ void ocelot_port_set_maxlen(struct ocelot *ocelot, int port, size_t sdu)
|
|||
|
||||
ocelot_port_writel(ocelot_port, maxlen, DEV_MAC_MAXLEN_CFG);
|
||||
|
||||
/* Set Pause WM hysteresis
|
||||
* 152 = 6 * maxlen / OCELOT_BUFFER_CELL_SZ
|
||||
* 101 = 4 * maxlen / OCELOT_BUFFER_CELL_SZ
|
||||
*/
|
||||
ocelot_write_rix(ocelot, SYS_PAUSE_CFG_PAUSE_ENA |
|
||||
SYS_PAUSE_CFG_PAUSE_STOP(101) |
|
||||
SYS_PAUSE_CFG_PAUSE_START(152), SYS_PAUSE_CFG, port);
|
||||
/* Set Pause watermark hysteresis */
|
||||
pause_start = 6 * maxlen / OCELOT_BUFFER_CELL_SZ;
|
||||
pause_stop = 4 * maxlen / OCELOT_BUFFER_CELL_SZ;
|
||||
ocelot_fields_write(ocelot, port, SYS_PAUSE_CFG_PAUSE_START,
|
||||
pause_start);
|
||||
ocelot_fields_write(ocelot, port, SYS_PAUSE_CFG_PAUSE_STOP,
|
||||
pause_stop);
|
||||
|
||||
/* Tail dropping watermark */
|
||||
atop_wm = (ocelot->shared_queue_sz - 9 * maxlen) /
|
||||
OCELOT_BUFFER_CELL_SZ;
|
||||
ocelot_write_rix(ocelot, ocelot_wm_enc(9 * maxlen),
|
||||
ocelot_write_rix(ocelot, ocelot->ops->wm_enc(9 * maxlen),
|
||||
SYS_ATOP, port);
|
||||
ocelot_write(ocelot, ocelot_wm_enc(atop_wm), SYS_ATOP_TOT_CFG);
|
||||
ocelot_write(ocelot, ocelot->ops->wm_enc(atop_wm), SYS_ATOP_TOT_CFG);
|
||||
}
|
||||
EXPORT_SYMBOL(ocelot_port_set_maxlen);
|
||||
|
||||
|
@ -1344,6 +1330,9 @@ void ocelot_init_port(struct ocelot *ocelot, int port)
|
|||
ocelot_port_writel(ocelot_port, 0, DEV_MAC_FC_MAC_HIGH_CFG);
|
||||
ocelot_port_writel(ocelot_port, 0, DEV_MAC_FC_MAC_LOW_CFG);
|
||||
|
||||
/* Enable transmission of pause frames */
|
||||
ocelot_fields_write(ocelot, port, SYS_PAUSE_CFG_PAUSE_ENA, 1);
|
||||
|
||||
/* Drop frames with multicast source address */
|
||||
ocelot_rmw_gix(ocelot, ANA_PORT_DROP_CFG_DROP_MC_SMAC_ENA,
|
||||
ANA_PORT_DROP_CFG_DROP_MC_SMAC_ENA,
|
||||
|
@ -1392,27 +1381,25 @@ void ocelot_configure_cpu(struct ocelot *ocelot, int npi,
|
|||
QSYS_EXT_CPU_CFG);
|
||||
|
||||
/* Enable NPI port */
|
||||
ocelot_write_rix(ocelot,
|
||||
QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE |
|
||||
QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG(1) |
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE, npi);
|
||||
ocelot_fields_write(ocelot, npi,
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
|
||||
/* NPI port Injection/Extraction configuration */
|
||||
ocelot_write_rix(ocelot,
|
||||
SYS_PORT_MODE_INCL_XTR_HDR(extraction) |
|
||||
SYS_PORT_MODE_INCL_INJ_HDR(injection),
|
||||
SYS_PORT_MODE, npi);
|
||||
ocelot_fields_write(ocelot, npi, SYS_PORT_MODE_INCL_XTR_HDR,
|
||||
extraction);
|
||||
ocelot_fields_write(ocelot, npi, SYS_PORT_MODE_INCL_INJ_HDR,
|
||||
injection);
|
||||
|
||||
/* Disable transmission of pause frames */
|
||||
ocelot_fields_write(ocelot, npi, SYS_PAUSE_CFG_PAUSE_ENA, 0);
|
||||
}
|
||||
|
||||
/* Enable CPU port module */
|
||||
ocelot_write_rix(ocelot, QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE |
|
||||
QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG(1) |
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE, cpu);
|
||||
ocelot_fields_write(ocelot, cpu, QSYS_SWITCH_PORT_MODE_PORT_ENA, 1);
|
||||
/* CPU port Injection/Extraction configuration */
|
||||
ocelot_write_rix(ocelot, SYS_PORT_MODE_INCL_XTR_HDR(extraction) |
|
||||
SYS_PORT_MODE_INCL_INJ_HDR(injection),
|
||||
SYS_PORT_MODE, cpu);
|
||||
ocelot_fields_write(ocelot, cpu, SYS_PORT_MODE_INCL_XTR_HDR,
|
||||
extraction);
|
||||
ocelot_fields_write(ocelot, cpu, SYS_PORT_MODE_INCL_INJ_HDR,
|
||||
injection);
|
||||
|
||||
/* Configure the CPU port to be VLAN aware */
|
||||
ocelot_write_gix(ocelot, ANA_PORT_VLAN_CFG_VLAN_VID(0) |
|
||||
|
|
|
@ -102,11 +102,7 @@ void ocelot_port_lag_leave(struct ocelot *ocelot, int port,
|
|||
u32 ocelot_port_readl(struct ocelot_port *port, u32 reg);
|
||||
void ocelot_port_writel(struct ocelot_port *port, u32 val, u32 reg);
|
||||
|
||||
#define ocelot_field_write(ocelot, reg, val) regmap_field_write((ocelot)->regfields[(reg)], (val))
|
||||
#define ocelot_field_read(ocelot, reg, val) regmap_field_read((ocelot)->regfields[(reg)], (val))
|
||||
|
||||
int ocelot_probe_port(struct ocelot *ocelot, u8 port,
|
||||
void __iomem *regs,
|
||||
int ocelot_probe_port(struct ocelot *ocelot, int port, struct regmap *target,
|
||||
struct phy_device *phy);
|
||||
|
||||
void ocelot_set_cpu_port(struct ocelot *ocelot, int cpu,
|
||||
|
@ -117,7 +113,4 @@ extern struct notifier_block ocelot_netdevice_nb;
|
|||
extern struct notifier_block ocelot_switchdev_nb;
|
||||
extern struct notifier_block ocelot_switchdev_blocking_nb;
|
||||
|
||||
#define ocelot_field_write(ocelot, reg, val) regmap_field_write((ocelot)->regfields[(reg)], (val))
|
||||
#define ocelot_field_read(ocelot, reg, val) regmap_field_read((ocelot)->regfields[(reg)], (val))
|
||||
|
||||
#endif
|
||||
|
|
|
@ -49,13 +49,25 @@ EXPORT_SYMBOL(__ocelot_rmw_ix);
|
|||
|
||||
u32 ocelot_port_readl(struct ocelot_port *port, u32 reg)
|
||||
{
|
||||
return readl(port->regs + reg);
|
||||
struct ocelot *ocelot = port->ocelot;
|
||||
u16 target = reg >> TARGET_OFFSET;
|
||||
u32 val;
|
||||
|
||||
WARN_ON(!target);
|
||||
|
||||
regmap_read(port->target, ocelot->map[target][reg & REG_MASK], &val);
|
||||
return val;
|
||||
}
|
||||
EXPORT_SYMBOL(ocelot_port_readl);
|
||||
|
||||
void ocelot_port_writel(struct ocelot_port *port, u32 val, u32 reg)
|
||||
{
|
||||
writel(val, port->regs + reg);
|
||||
struct ocelot *ocelot = port->ocelot;
|
||||
u16 target = reg >> TARGET_OFFSET;
|
||||
|
||||
WARN_ON(!target);
|
||||
|
||||
regmap_write(port->target, ocelot->map[target][reg & REG_MASK], val);
|
||||
}
|
||||
EXPORT_SYMBOL(ocelot_port_writel);
|
||||
|
||||
|
@ -77,6 +89,8 @@ int ocelot_regfields_init(struct ocelot *ocelot,
|
|||
regfield.reg = ocelot->map[target][reg & REG_MASK];
|
||||
regfield.lsb = regfields[i].lsb;
|
||||
regfield.msb = regfields[i].msb;
|
||||
regfield.id_size = regfields[i].id_size;
|
||||
regfield.id_offset = regfields[i].id_offset;
|
||||
|
||||
ocelot->regfields[i] =
|
||||
devm_regmap_field_alloc(ocelot->dev,
|
||||
|
|
|
@ -1005,8 +1005,7 @@ struct notifier_block ocelot_switchdev_blocking_nb __read_mostly = {
|
|||
.notifier_call = ocelot_switchdev_blocking_event,
|
||||
};
|
||||
|
||||
int ocelot_probe_port(struct ocelot *ocelot, u8 port,
|
||||
void __iomem *regs,
|
||||
int ocelot_probe_port(struct ocelot *ocelot, int port, struct regmap *target,
|
||||
struct phy_device *phy)
|
||||
{
|
||||
struct ocelot_port_private *priv;
|
||||
|
@ -1024,7 +1023,7 @@ int ocelot_probe_port(struct ocelot *ocelot, u8 port,
|
|||
priv->chip_port = port;
|
||||
ocelot_port = &priv->port;
|
||||
ocelot_port->ocelot = ocelot;
|
||||
ocelot_port->regs = regs;
|
||||
ocelot_port->target = target;
|
||||
ocelot->ports[port] = ocelot_port;
|
||||
|
||||
dev->netdev_ops = &ocelot_port_netdev_ops;
|
||||
|
|
|
@ -263,7 +263,49 @@ static const u32 ocelot_ptp_regmap[] = {
|
|||
REG(PTP_CLK_CFG_ADJ_FREQ, 0x0000a8),
|
||||
};
|
||||
|
||||
static const u32 *ocelot_regmap[] = {
|
||||
static const u32 ocelot_dev_gmii_regmap[] = {
|
||||
REG(DEV_CLOCK_CFG, 0x0),
|
||||
REG(DEV_PORT_MISC, 0x4),
|
||||
REG(DEV_EVENTS, 0x8),
|
||||
REG(DEV_EEE_CFG, 0xc),
|
||||
REG(DEV_RX_PATH_DELAY, 0x10),
|
||||
REG(DEV_TX_PATH_DELAY, 0x14),
|
||||
REG(DEV_PTP_PREDICT_CFG, 0x18),
|
||||
REG(DEV_MAC_ENA_CFG, 0x1c),
|
||||
REG(DEV_MAC_MODE_CFG, 0x20),
|
||||
REG(DEV_MAC_MAXLEN_CFG, 0x24),
|
||||
REG(DEV_MAC_TAGS_CFG, 0x28),
|
||||
REG(DEV_MAC_ADV_CHK_CFG, 0x2c),
|
||||
REG(DEV_MAC_IFG_CFG, 0x30),
|
||||
REG(DEV_MAC_HDX_CFG, 0x34),
|
||||
REG(DEV_MAC_DBG_CFG, 0x38),
|
||||
REG(DEV_MAC_FC_MAC_LOW_CFG, 0x3c),
|
||||
REG(DEV_MAC_FC_MAC_HIGH_CFG, 0x40),
|
||||
REG(DEV_MAC_STICKY, 0x44),
|
||||
REG(PCS1G_CFG, 0x48),
|
||||
REG(PCS1G_MODE_CFG, 0x4c),
|
||||
REG(PCS1G_SD_CFG, 0x50),
|
||||
REG(PCS1G_ANEG_CFG, 0x54),
|
||||
REG(PCS1G_ANEG_NP_CFG, 0x58),
|
||||
REG(PCS1G_LB_CFG, 0x5c),
|
||||
REG(PCS1G_DBG_CFG, 0x60),
|
||||
REG(PCS1G_CDET_CFG, 0x64),
|
||||
REG(PCS1G_ANEG_STATUS, 0x68),
|
||||
REG(PCS1G_ANEG_NP_STATUS, 0x6c),
|
||||
REG(PCS1G_LINK_STATUS, 0x70),
|
||||
REG(PCS1G_LINK_DOWN_CNT, 0x74),
|
||||
REG(PCS1G_STICKY, 0x78),
|
||||
REG(PCS1G_DEBUG_STATUS, 0x7c),
|
||||
REG(PCS1G_LPI_CFG, 0x80),
|
||||
REG(PCS1G_LPI_WAKE_ERROR_CNT, 0x84),
|
||||
REG(PCS1G_LPI_STATUS, 0x88),
|
||||
REG(PCS1G_TSTPAT_MODE_CFG, 0x8c),
|
||||
REG(PCS1G_TSTPAT_STATUS, 0x90),
|
||||
REG(DEV_PCS_FX100_CFG, 0x94),
|
||||
REG(DEV_PCS_FX100_STATUS, 0x98),
|
||||
};
|
||||
|
||||
static const u32 *ocelot_regmap[TARGET_MAX] = {
|
||||
[ANA] = ocelot_ana_regmap,
|
||||
[QS] = ocelot_qs_regmap,
|
||||
[QSYS] = ocelot_qsys_regmap,
|
||||
|
@ -271,9 +313,10 @@ static const u32 *ocelot_regmap[] = {
|
|||
[SYS] = ocelot_sys_regmap,
|
||||
[S2] = ocelot_s2_regmap,
|
||||
[PTP] = ocelot_ptp_regmap,
|
||||
[DEV_GMII] = ocelot_dev_gmii_regmap,
|
||||
};
|
||||
|
||||
static const struct reg_field ocelot_regfields[] = {
|
||||
static const struct reg_field ocelot_regfields[REGFIELD_MAX] = {
|
||||
[ANA_ADVLEARN_VLAN_CHK] = REG_FIELD(ANA_ADVLEARN, 11, 11),
|
||||
[ANA_ADVLEARN_LEARN_MIRROR] = REG_FIELD(ANA_ADVLEARN, 0, 10),
|
||||
[ANA_ANEVENTS_MSTI_DROP] = REG_FIELD(ANA_ANEVENTS, 27, 27),
|
||||
|
@ -315,6 +358,20 @@ static const struct reg_field ocelot_regfields[] = {
|
|||
[SYS_RESET_CFG_CORE_ENA] = REG_FIELD(SYS_RESET_CFG, 2, 2),
|
||||
[SYS_RESET_CFG_MEM_ENA] = REG_FIELD(SYS_RESET_CFG, 1, 1),
|
||||
[SYS_RESET_CFG_MEM_INIT] = REG_FIELD(SYS_RESET_CFG, 0, 0),
|
||||
/* Replicated per number of ports (11), register size 4 per port */
|
||||
[QSYS_SWITCH_PORT_MODE_PORT_ENA] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 14, 14, 11, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 11, 13, 11, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_YEL_RSRVD] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 10, 10, 11, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 9, 9, 11, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_TX_PFC_ENA] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 1, 8, 11, 4),
|
||||
[QSYS_SWITCH_PORT_MODE_TX_PFC_MODE] = REG_FIELD_ID(QSYS_SWITCH_PORT_MODE, 0, 0, 11, 4),
|
||||
[SYS_PORT_MODE_DATA_WO_TS] = REG_FIELD_ID(SYS_PORT_MODE, 5, 6, 11, 4),
|
||||
[SYS_PORT_MODE_INCL_INJ_HDR] = REG_FIELD_ID(SYS_PORT_MODE, 3, 4, 11, 4),
|
||||
[SYS_PORT_MODE_INCL_XTR_HDR] = REG_FIELD_ID(SYS_PORT_MODE, 1, 2, 11, 4),
|
||||
[SYS_PORT_MODE_INCL_HDR_ERR] = REG_FIELD_ID(SYS_PORT_MODE, 0, 0, 11, 4),
|
||||
[SYS_PAUSE_CFG_PAUSE_START] = REG_FIELD_ID(SYS_PAUSE_CFG, 10, 18, 11, 4),
|
||||
[SYS_PAUSE_CFG_PAUSE_STOP] = REG_FIELD_ID(SYS_PAUSE_CFG, 1, 9, 11, 4),
|
||||
[SYS_PAUSE_CFG_PAUSE_ENA] = REG_FIELD_ID(SYS_PAUSE_CFG, 0, 1, 11, 4),
|
||||
};
|
||||
|
||||
static const struct ocelot_stat_layout ocelot_stats_layout[] = {
|
||||
|
@ -682,8 +739,21 @@ static int ocelot_reset(struct ocelot *ocelot)
|
|||
return 0;
|
||||
}
|
||||
|
||||
/* Watermark encode
|
||||
* Bit 8: Unit; 0:1, 1:16
|
||||
* Bit 7-0: Value to be multiplied with unit
|
||||
*/
|
||||
static u16 ocelot_wm_enc(u16 value)
|
||||
{
|
||||
if (value >= BIT(8))
|
||||
return BIT(8) | (value / 16);
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
static const struct ocelot_ops ocelot_ops = {
|
||||
.reset = ocelot_reset,
|
||||
.wm_enc = ocelot_wm_enc,
|
||||
};
|
||||
|
||||
static const struct vcap_field vsc7514_vcap_is2_keys[] = {
|
||||
|
@ -948,9 +1018,9 @@ static int mscc_ocelot_probe(struct platform_device *pdev)
|
|||
struct device_node *phy_node;
|
||||
phy_interface_t phy_mode;
|
||||
struct phy_device *phy;
|
||||
struct regmap *target;
|
||||
struct resource *res;
|
||||
struct phy *serdes;
|
||||
void __iomem *regs;
|
||||
char res_name[8];
|
||||
u32 port;
|
||||
|
||||
|
@ -961,8 +1031,8 @@ static int mscc_ocelot_probe(struct platform_device *pdev)
|
|||
|
||||
res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
|
||||
res_name);
|
||||
regs = devm_ioremap_resource(&pdev->dev, res);
|
||||
if (IS_ERR(regs))
|
||||
target = ocelot_regmap_init(ocelot, res);
|
||||
if (IS_ERR(target))
|
||||
continue;
|
||||
|
||||
phy_node = of_parse_phandle(portnp, "phy-handle", 0);
|
||||
|
@ -974,7 +1044,7 @@ static int mscc_ocelot_probe(struct platform_device *pdev)
|
|||
if (!phy)
|
||||
continue;
|
||||
|
||||
err = ocelot_probe_port(ocelot, port, regs, phy);
|
||||
err = ocelot_probe_port(ocelot, port, target, phy);
|
||||
if (err) {
|
||||
of_node_put(portnp);
|
||||
goto out_put_ports;
|
||||
|
|
|
@ -126,6 +126,7 @@ enum ocelot_target {
|
|||
HSIO,
|
||||
PTP,
|
||||
GCB,
|
||||
DEV_GMII,
|
||||
TARGET_MAX,
|
||||
};
|
||||
|
||||
|
@ -408,6 +409,48 @@ enum ocelot_reg {
|
|||
PTP_CLK_CFG_ADJ_CFG,
|
||||
PTP_CLK_CFG_ADJ_FREQ,
|
||||
GCB_SOFT_RST = GCB << TARGET_OFFSET,
|
||||
GCB_MIIM_MII_STATUS,
|
||||
GCB_MIIM_MII_CMD,
|
||||
GCB_MIIM_MII_DATA,
|
||||
DEV_CLOCK_CFG = DEV_GMII << TARGET_OFFSET,
|
||||
DEV_PORT_MISC,
|
||||
DEV_EVENTS,
|
||||
DEV_EEE_CFG,
|
||||
DEV_RX_PATH_DELAY,
|
||||
DEV_TX_PATH_DELAY,
|
||||
DEV_PTP_PREDICT_CFG,
|
||||
DEV_MAC_ENA_CFG,
|
||||
DEV_MAC_MODE_CFG,
|
||||
DEV_MAC_MAXLEN_CFG,
|
||||
DEV_MAC_TAGS_CFG,
|
||||
DEV_MAC_ADV_CHK_CFG,
|
||||
DEV_MAC_IFG_CFG,
|
||||
DEV_MAC_HDX_CFG,
|
||||
DEV_MAC_DBG_CFG,
|
||||
DEV_MAC_FC_MAC_LOW_CFG,
|
||||
DEV_MAC_FC_MAC_HIGH_CFG,
|
||||
DEV_MAC_STICKY,
|
||||
PCS1G_CFG,
|
||||
PCS1G_MODE_CFG,
|
||||
PCS1G_SD_CFG,
|
||||
PCS1G_ANEG_CFG,
|
||||
PCS1G_ANEG_NP_CFG,
|
||||
PCS1G_LB_CFG,
|
||||
PCS1G_DBG_CFG,
|
||||
PCS1G_CDET_CFG,
|
||||
PCS1G_ANEG_STATUS,
|
||||
PCS1G_ANEG_NP_STATUS,
|
||||
PCS1G_LINK_STATUS,
|
||||
PCS1G_LINK_DOWN_CNT,
|
||||
PCS1G_STICKY,
|
||||
PCS1G_DEBUG_STATUS,
|
||||
PCS1G_LPI_CFG,
|
||||
PCS1G_LPI_WAKE_ERROR_CNT,
|
||||
PCS1G_LPI_STATUS,
|
||||
PCS1G_TSTPAT_MODE_CFG,
|
||||
PCS1G_TSTPAT_STATUS,
|
||||
DEV_PCS_FX100_CFG,
|
||||
DEV_PCS_FX100_STATUS,
|
||||
};
|
||||
|
||||
enum ocelot_regfield {
|
||||
|
@ -447,15 +490,30 @@ enum ocelot_regfield {
|
|||
ANA_TABLES_MACACCESS_B_DOM,
|
||||
ANA_TABLES_MACTINDX_BUCKET,
|
||||
ANA_TABLES_MACTINDX_M_INDEX,
|
||||
QSYS_SWITCH_PORT_MODE_PORT_ENA,
|
||||
QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG,
|
||||
QSYS_SWITCH_PORT_MODE_YEL_RSRVD,
|
||||
QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE,
|
||||
QSYS_SWITCH_PORT_MODE_TX_PFC_ENA,
|
||||
QSYS_SWITCH_PORT_MODE_TX_PFC_MODE,
|
||||
QSYS_TIMED_FRAME_ENTRY_TFRM_VLD,
|
||||
QSYS_TIMED_FRAME_ENTRY_TFRM_FP,
|
||||
QSYS_TIMED_FRAME_ENTRY_TFRM_PORTNO,
|
||||
QSYS_TIMED_FRAME_ENTRY_TFRM_TM_SEL,
|
||||
QSYS_TIMED_FRAME_ENTRY_TFRM_TM_T,
|
||||
SYS_PORT_MODE_DATA_WO_TS,
|
||||
SYS_PORT_MODE_INCL_INJ_HDR,
|
||||
SYS_PORT_MODE_INCL_XTR_HDR,
|
||||
SYS_PORT_MODE_INCL_HDR_ERR,
|
||||
SYS_RESET_CFG_CORE_ENA,
|
||||
SYS_RESET_CFG_MEM_ENA,
|
||||
SYS_RESET_CFG_MEM_INIT,
|
||||
GCB_SOFT_RST_SWC_RST,
|
||||
GCB_MIIM_MII_STATUS_PENDING,
|
||||
GCB_MIIM_MII_STATUS_BUSY,
|
||||
SYS_PAUSE_CFG_PAUSE_START,
|
||||
SYS_PAUSE_CFG_PAUSE_STOP,
|
||||
SYS_PAUSE_CFG_PAUSE_ENA,
|
||||
REGFIELD_MAX
|
||||
};
|
||||
|
||||
|
@ -483,6 +541,7 @@ struct ocelot;
|
|||
|
||||
struct ocelot_ops {
|
||||
int (*reset)(struct ocelot *ocelot);
|
||||
u16 (*wm_enc)(u16 value);
|
||||
};
|
||||
|
||||
struct ocelot_vcap_block {
|
||||
|
@ -494,7 +553,7 @@ struct ocelot_vcap_block {
|
|||
struct ocelot_port {
|
||||
struct ocelot *ocelot;
|
||||
|
||||
void __iomem *regs;
|
||||
struct regmap *target;
|
||||
|
||||
bool vlan_aware;
|
||||
|
||||
|
@ -509,6 +568,8 @@ struct ocelot_port {
|
|||
u8 ts_id;
|
||||
|
||||
phy_interface_t phy_mode;
|
||||
|
||||
u8 *xmit_template;
|
||||
};
|
||||
|
||||
struct ocelot {
|
||||
|
@ -593,6 +654,11 @@ struct ocelot_policer {
|
|||
#define ocelot_rmw_rix(ocelot, val, m, reg, ri) __ocelot_rmw_ix(ocelot, val, m, reg, reg##_RSZ * (ri))
|
||||
#define ocelot_rmw(ocelot, val, m, reg) __ocelot_rmw_ix(ocelot, val, m, reg, 0)
|
||||
|
||||
#define ocelot_field_write(ocelot, reg, val) regmap_field_write((ocelot)->regfields[(reg)], (val))
|
||||
#define ocelot_field_read(ocelot, reg, val) regmap_field_read((ocelot)->regfields[(reg)], (val))
|
||||
#define ocelot_fields_write(ocelot, id, reg, val) regmap_fields_write((ocelot)->regfields[(reg)], (id), (val))
|
||||
#define ocelot_fields_read(ocelot, id, reg, val) regmap_fields_read((ocelot)->regfields[(reg)], (id), (val))
|
||||
|
||||
/* I/O */
|
||||
u32 ocelot_port_readl(struct ocelot_port *port, u32 reg);
|
||||
void ocelot_port_writel(struct ocelot_port *port, u32 val, u32 reg);
|
||||
|
|
|
@ -8,8 +8,6 @@
|
|||
#ifndef _MSCC_OCELOT_DEV_H_
|
||||
#define _MSCC_OCELOT_DEV_H_
|
||||
|
||||
#define DEV_CLOCK_CFG 0x0
|
||||
|
||||
#define DEV_CLOCK_CFG_MAC_TX_RST BIT(7)
|
||||
#define DEV_CLOCK_CFG_MAC_RX_RST BIT(6)
|
||||
#define DEV_CLOCK_CFG_PCS_TX_RST BIT(5)
|
||||
|
@ -19,18 +17,12 @@
|
|||
#define DEV_CLOCK_CFG_LINK_SPEED(x) ((x) & GENMASK(1, 0))
|
||||
#define DEV_CLOCK_CFG_LINK_SPEED_M GENMASK(1, 0)
|
||||
|
||||
#define DEV_PORT_MISC 0x4
|
||||
|
||||
#define DEV_PORT_MISC_FWD_ERROR_ENA BIT(4)
|
||||
#define DEV_PORT_MISC_FWD_PAUSE_ENA BIT(3)
|
||||
#define DEV_PORT_MISC_FWD_CTRL_ENA BIT(2)
|
||||
#define DEV_PORT_MISC_DEV_LOOP_ENA BIT(1)
|
||||
#define DEV_PORT_MISC_HDX_FAST_DIS BIT(0)
|
||||
|
||||
#define DEV_EVENTS 0x8
|
||||
|
||||
#define DEV_EEE_CFG 0xc
|
||||
|
||||
#define DEV_EEE_CFG_EEE_ENA BIT(22)
|
||||
#define DEV_EEE_CFG_EEE_TIMER_AGE(x) (((x) << 15) & GENMASK(21, 15))
|
||||
#define DEV_EEE_CFG_EEE_TIMER_AGE_M GENMASK(21, 15)
|
||||
|
@ -43,33 +35,19 @@
|
|||
#define DEV_EEE_CFG_EEE_TIMER_HOLDOFF_X(x) (((x) & GENMASK(7, 1)) >> 1)
|
||||
#define DEV_EEE_CFG_PORT_LPI BIT(0)
|
||||
|
||||
#define DEV_RX_PATH_DELAY 0x10
|
||||
|
||||
#define DEV_TX_PATH_DELAY 0x14
|
||||
|
||||
#define DEV_PTP_PREDICT_CFG 0x18
|
||||
|
||||
#define DEV_PTP_PREDICT_CFG_PTP_PHY_PREDICT_CFG(x) (((x) << 4) & GENMASK(11, 4))
|
||||
#define DEV_PTP_PREDICT_CFG_PTP_PHY_PREDICT_CFG_M GENMASK(11, 4)
|
||||
#define DEV_PTP_PREDICT_CFG_PTP_PHY_PREDICT_CFG_X(x) (((x) & GENMASK(11, 4)) >> 4)
|
||||
#define DEV_PTP_PREDICT_CFG_PTP_PHASE_PREDICT_CFG(x) ((x) & GENMASK(3, 0))
|
||||
#define DEV_PTP_PREDICT_CFG_PTP_PHASE_PREDICT_CFG_M GENMASK(3, 0)
|
||||
|
||||
#define DEV_MAC_ENA_CFG 0x1c
|
||||
|
||||
#define DEV_MAC_ENA_CFG_RX_ENA BIT(4)
|
||||
#define DEV_MAC_ENA_CFG_TX_ENA BIT(0)
|
||||
|
||||
#define DEV_MAC_MODE_CFG 0x20
|
||||
|
||||
#define DEV_MAC_MODE_CFG_FC_WORD_SYNC_ENA BIT(8)
|
||||
#define DEV_MAC_MODE_CFG_GIGA_MODE_ENA BIT(4)
|
||||
#define DEV_MAC_MODE_CFG_FDX_ENA BIT(0)
|
||||
|
||||
#define DEV_MAC_MAXLEN_CFG 0x24
|
||||
|
||||
#define DEV_MAC_TAGS_CFG 0x28
|
||||
|
||||
#define DEV_MAC_TAGS_CFG_TAG_ID(x) (((x) << 16) & GENMASK(31, 16))
|
||||
#define DEV_MAC_TAGS_CFG_TAG_ID_M GENMASK(31, 16)
|
||||
#define DEV_MAC_TAGS_CFG_TAG_ID_X(x) (((x) & GENMASK(31, 16)) >> 16)
|
||||
|
@ -77,12 +55,8 @@
|
|||
#define DEV_MAC_TAGS_CFG_VLAN_DBL_AWR_ENA BIT(1)
|
||||
#define DEV_MAC_TAGS_CFG_VLAN_AWR_ENA BIT(0)
|
||||
|
||||
#define DEV_MAC_ADV_CHK_CFG 0x2c
|
||||
|
||||
#define DEV_MAC_ADV_CHK_CFG_LEN_DROP_ENA BIT(0)
|
||||
|
||||
#define DEV_MAC_IFG_CFG 0x30
|
||||
|
||||
#define DEV_MAC_IFG_CFG_RESTORE_OLD_IPG_CHECK BIT(17)
|
||||
#define DEV_MAC_IFG_CFG_REDUCED_TX_IFG BIT(16)
|
||||
#define DEV_MAC_IFG_CFG_TX_IFG(x) (((x) << 8) & GENMASK(12, 8))
|
||||
|
@ -94,8 +68,6 @@
|
|||
#define DEV_MAC_IFG_CFG_RX_IFG1(x) ((x) & GENMASK(3, 0))
|
||||
#define DEV_MAC_IFG_CFG_RX_IFG1_M GENMASK(3, 0)
|
||||
|
||||
#define DEV_MAC_HDX_CFG 0x34
|
||||
|
||||
#define DEV_MAC_HDX_CFG_BYPASS_COL_SYNC BIT(26)
|
||||
#define DEV_MAC_HDX_CFG_OB_ENA BIT(25)
|
||||
#define DEV_MAC_HDX_CFG_WEXC_DIS BIT(24)
|
||||
|
@ -107,17 +79,9 @@
|
|||
#define DEV_MAC_HDX_CFG_LATE_COL_POS(x) ((x) & GENMASK(6, 0))
|
||||
#define DEV_MAC_HDX_CFG_LATE_COL_POS_M GENMASK(6, 0)
|
||||
|
||||
#define DEV_MAC_DBG_CFG 0x38
|
||||
|
||||
#define DEV_MAC_DBG_CFG_TBI_MODE BIT(4)
|
||||
#define DEV_MAC_DBG_CFG_IFG_CRS_EXT_CHK_ENA BIT(0)
|
||||
|
||||
#define DEV_MAC_FC_MAC_LOW_CFG 0x3c
|
||||
|
||||
#define DEV_MAC_FC_MAC_HIGH_CFG 0x40
|
||||
|
||||
#define DEV_MAC_STICKY 0x44
|
||||
|
||||
#define DEV_MAC_STICKY_RX_IPG_SHRINK_STICKY BIT(9)
|
||||
#define DEV_MAC_STICKY_RX_PREAM_SHRINK_STICKY BIT(8)
|
||||
#define DEV_MAC_STICKY_RX_CARRIER_EXT_STICKY BIT(7)
|
||||
|
@ -129,25 +93,17 @@
|
|||
#define DEV_MAC_STICKY_TX_FRM_LEN_OVR_STICKY BIT(1)
|
||||
#define DEV_MAC_STICKY_TX_ABORT_STICKY BIT(0)
|
||||
|
||||
#define PCS1G_CFG 0x48
|
||||
|
||||
#define PCS1G_CFG_LINK_STATUS_TYPE BIT(4)
|
||||
#define PCS1G_CFG_AN_LINK_CTRL_ENA BIT(1)
|
||||
#define PCS1G_CFG_PCS_ENA BIT(0)
|
||||
|
||||
#define PCS1G_MODE_CFG 0x4c
|
||||
|
||||
#define PCS1G_MODE_CFG_UNIDIR_MODE_ENA BIT(4)
|
||||
#define PCS1G_MODE_CFG_SGMII_MODE_ENA BIT(0)
|
||||
|
||||
#define PCS1G_SD_CFG 0x50
|
||||
|
||||
#define PCS1G_SD_CFG_SD_SEL BIT(8)
|
||||
#define PCS1G_SD_CFG_SD_POL BIT(4)
|
||||
#define PCS1G_SD_CFG_SD_ENA BIT(0)
|
||||
|
||||
#define PCS1G_ANEG_CFG 0x54
|
||||
|
||||
#define PCS1G_ANEG_CFG_ADV_ABILITY(x) (((x) << 16) & GENMASK(31, 16))
|
||||
#define PCS1G_ANEG_CFG_ADV_ABILITY_M GENMASK(31, 16)
|
||||
#define PCS1G_ANEG_CFG_ADV_ABILITY_X(x) (((x) & GENMASK(31, 16)) >> 16)
|
||||
|
@ -155,29 +111,19 @@
|
|||
#define PCS1G_ANEG_CFG_ANEG_RESTART_ONE_SHOT BIT(1)
|
||||
#define PCS1G_ANEG_CFG_ANEG_ENA BIT(0)
|
||||
|
||||
#define PCS1G_ANEG_NP_CFG 0x58
|
||||
|
||||
#define PCS1G_ANEG_NP_CFG_NP_TX(x) (((x) << 16) & GENMASK(31, 16))
|
||||
#define PCS1G_ANEG_NP_CFG_NP_TX_M GENMASK(31, 16)
|
||||
#define PCS1G_ANEG_NP_CFG_NP_TX_X(x) (((x) & GENMASK(31, 16)) >> 16)
|
||||
#define PCS1G_ANEG_NP_CFG_NP_LOADED_ONE_SHOT BIT(0)
|
||||
|
||||
#define PCS1G_LB_CFG 0x5c
|
||||
|
||||
#define PCS1G_LB_CFG_RA_ENA BIT(4)
|
||||
#define PCS1G_LB_CFG_GMII_PHY_LB_ENA BIT(1)
|
||||
#define PCS1G_LB_CFG_TBI_HOST_LB_ENA BIT(0)
|
||||
|
||||
#define PCS1G_DBG_CFG 0x60
|
||||
|
||||
#define PCS1G_DBG_CFG_UDLT BIT(0)
|
||||
|
||||
#define PCS1G_CDET_CFG 0x64
|
||||
|
||||
#define PCS1G_CDET_CFG_CDET_ENA BIT(0)
|
||||
|
||||
#define PCS1G_ANEG_STATUS 0x68
|
||||
|
||||
#define PCS1G_ANEG_STATUS_LP_ADV_ABILITY(x) (((x) << 16) & GENMASK(31, 16))
|
||||
#define PCS1G_ANEG_STATUS_LP_ADV_ABILITY_M GENMASK(31, 16)
|
||||
#define PCS1G_ANEG_STATUS_LP_ADV_ABILITY_X(x) (((x) & GENMASK(31, 16)) >> 16)
|
||||
|
@ -185,10 +131,6 @@
|
|||
#define PCS1G_ANEG_STATUS_PAGE_RX_STICKY BIT(3)
|
||||
#define PCS1G_ANEG_STATUS_ANEG_COMPLETE BIT(0)
|
||||
|
||||
#define PCS1G_ANEG_NP_STATUS 0x6c
|
||||
|
||||
#define PCS1G_LINK_STATUS 0x70
|
||||
|
||||
#define PCS1G_LINK_STATUS_DELAY_VAR(x) (((x) << 12) & GENMASK(15, 12))
|
||||
#define PCS1G_LINK_STATUS_DELAY_VAR_M GENMASK(15, 12)
|
||||
#define PCS1G_LINK_STATUS_DELAY_VAR_X(x) (((x) & GENMASK(15, 12)) >> 12)
|
||||
|
@ -196,17 +138,9 @@
|
|||
#define PCS1G_LINK_STATUS_LINK_STATUS BIT(4)
|
||||
#define PCS1G_LINK_STATUS_SYNC_STATUS BIT(0)
|
||||
|
||||
#define PCS1G_LINK_DOWN_CNT 0x74
|
||||
|
||||
#define PCS1G_STICKY 0x78
|
||||
|
||||
#define PCS1G_STICKY_LINK_DOWN_STICKY BIT(4)
|
||||
#define PCS1G_STICKY_OUT_OF_SYNC_STICKY BIT(0)
|
||||
|
||||
#define PCS1G_DEBUG_STATUS 0x7c
|
||||
|
||||
#define PCS1G_LPI_CFG 0x80
|
||||
|
||||
#define PCS1G_LPI_CFG_QSGMII_MS_SEL BIT(20)
|
||||
#define PCS1G_LPI_CFG_RX_LPI_OUT_DIS BIT(17)
|
||||
#define PCS1G_LPI_CFG_LPI_TESTMODE BIT(16)
|
||||
|
@ -215,10 +149,6 @@
|
|||
#define PCS1G_LPI_CFG_LPI_RX_WTIM_X(x) (((x) & GENMASK(5, 4)) >> 4)
|
||||
#define PCS1G_LPI_CFG_TX_ASSERT_LPIDLE BIT(0)
|
||||
|
||||
#define PCS1G_LPI_WAKE_ERROR_CNT 0x84
|
||||
|
||||
#define PCS1G_LPI_STATUS 0x88
|
||||
|
||||
#define PCS1G_LPI_STATUS_RX_LPI_FAIL BIT(16)
|
||||
#define PCS1G_LPI_STATUS_RX_LPI_EVENT_STICKY BIT(12)
|
||||
#define PCS1G_LPI_STATUS_RX_QUIET BIT(9)
|
||||
|
@ -227,18 +157,12 @@
|
|||
#define PCS1G_LPI_STATUS_TX_QUIET BIT(1)
|
||||
#define PCS1G_LPI_STATUS_TX_LPI_MODE BIT(0)
|
||||
|
||||
#define PCS1G_TSTPAT_MODE_CFG 0x8c
|
||||
|
||||
#define PCS1G_TSTPAT_STATUS 0x90
|
||||
|
||||
#define PCS1G_TSTPAT_STATUS_JTP_ERR_CNT(x) (((x) << 8) & GENMASK(15, 8))
|
||||
#define PCS1G_TSTPAT_STATUS_JTP_ERR_CNT_M GENMASK(15, 8)
|
||||
#define PCS1G_TSTPAT_STATUS_JTP_ERR_CNT_X(x) (((x) & GENMASK(15, 8)) >> 8)
|
||||
#define PCS1G_TSTPAT_STATUS_JTP_ERR BIT(4)
|
||||
#define PCS1G_TSTPAT_STATUS_JTP_LOCK BIT(0)
|
||||
|
||||
#define DEV_PCS_FX100_CFG 0x94
|
||||
|
||||
#define DEV_PCS_FX100_CFG_SD_SEL BIT(26)
|
||||
#define DEV_PCS_FX100_CFG_SD_POL BIT(25)
|
||||
#define DEV_PCS_FX100_CFG_SD_ENA BIT(24)
|
||||
|
@ -259,8 +183,6 @@
|
|||
#define DEV_PCS_FX100_CFG_FEFGEN_ENA BIT(1)
|
||||
#define DEV_PCS_FX100_CFG_PCS_ENA BIT(0)
|
||||
|
||||
#define DEV_PCS_FX100_STATUS 0x98
|
||||
|
||||
#define DEV_PCS_FX100_STATUS_EDGE_POS_PTP(x) (((x) << 8) & GENMASK(11, 8))
|
||||
#define DEV_PCS_FX100_STATUS_EDGE_POS_PTP_M GENMASK(11, 8)
|
||||
#define DEV_PCS_FX100_STATUS_EDGE_POS_PTP_X(x) (((x) & GENMASK(11, 8)) >> 8)
|
||||
|
|
|
@ -13,19 +13,6 @@
|
|||
#define QSYS_PORT_MODE_DEQUEUE_DIS BIT(1)
|
||||
#define QSYS_PORT_MODE_DEQUEUE_LATE BIT(0)
|
||||
|
||||
#define QSYS_SWITCH_PORT_MODE_RSZ 0x4
|
||||
|
||||
#define QSYS_SWITCH_PORT_MODE_PORT_ENA BIT(14)
|
||||
#define QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG(x) (((x) << 11) & GENMASK(13, 11))
|
||||
#define QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG_M GENMASK(13, 11)
|
||||
#define QSYS_SWITCH_PORT_MODE_SCH_NEXT_CFG_X(x) (((x) & GENMASK(13, 11)) >> 11)
|
||||
#define QSYS_SWITCH_PORT_MODE_YEL_RSRVD BIT(10)
|
||||
#define QSYS_SWITCH_PORT_MODE_INGRESS_DROP_MODE BIT(9)
|
||||
#define QSYS_SWITCH_PORT_MODE_TX_PFC_ENA(x) (((x) << 1) & GENMASK(8, 1))
|
||||
#define QSYS_SWITCH_PORT_MODE_TX_PFC_ENA_M GENMASK(8, 1)
|
||||
#define QSYS_SWITCH_PORT_MODE_TX_PFC_ENA_X(x) (((x) & GENMASK(8, 1)) >> 1)
|
||||
#define QSYS_SWITCH_PORT_MODE_TX_PFC_MODE BIT(0)
|
||||
|
||||
#define QSYS_STAT_CNT_CFG_TX_GREEN_CNT_MODE BIT(5)
|
||||
#define QSYS_STAT_CNT_CFG_TX_YELLOW_CNT_MODE BIT(4)
|
||||
#define QSYS_STAT_CNT_CFG_DROP_GREEN_CNT_MODE BIT(3)
|
||||
|
|
|
@ -12,19 +12,6 @@
|
|||
|
||||
#define SYS_COUNT_TX_OCTETS_RSZ 0x4
|
||||
|
||||
#define SYS_PORT_MODE_RSZ 0x4
|
||||
|
||||
#define SYS_PORT_MODE_DATA_WO_TS(x) (((x) << 5) & GENMASK(6, 5))
|
||||
#define SYS_PORT_MODE_DATA_WO_TS_M GENMASK(6, 5)
|
||||
#define SYS_PORT_MODE_DATA_WO_TS_X(x) (((x) & GENMASK(6, 5)) >> 5)
|
||||
#define SYS_PORT_MODE_INCL_INJ_HDR(x) (((x) << 3) & GENMASK(4, 3))
|
||||
#define SYS_PORT_MODE_INCL_INJ_HDR_M GENMASK(4, 3)
|
||||
#define SYS_PORT_MODE_INCL_INJ_HDR_X(x) (((x) & GENMASK(4, 3)) >> 3)
|
||||
#define SYS_PORT_MODE_INCL_XTR_HDR(x) (((x) << 1) & GENMASK(2, 1))
|
||||
#define SYS_PORT_MODE_INCL_XTR_HDR_M GENMASK(2, 1)
|
||||
#define SYS_PORT_MODE_INCL_XTR_HDR_X(x) (((x) & GENMASK(2, 1)) >> 1)
|
||||
#define SYS_PORT_MODE_INJ_HDR_ERR BIT(0)
|
||||
|
||||
#define SYS_FRONT_PORT_MODE_RSZ 0x4
|
||||
|
||||
#define SYS_FRONT_PORT_MODE_HDX_MODE BIT(0)
|
||||
|
@ -56,16 +43,6 @@
|
|||
#define SYS_TIMESTAMP_OFFSET_TIMESTAMP_OFFSET(x) ((x) & GENMASK(5, 0))
|
||||
#define SYS_TIMESTAMP_OFFSET_TIMESTAMP_OFFSET_M GENMASK(5, 0)
|
||||
|
||||
#define SYS_PAUSE_CFG_RSZ 0x4
|
||||
|
||||
#define SYS_PAUSE_CFG_PAUSE_START(x) (((x) << 10) & GENMASK(18, 10))
|
||||
#define SYS_PAUSE_CFG_PAUSE_START_M GENMASK(18, 10)
|
||||
#define SYS_PAUSE_CFG_PAUSE_START_X(x) (((x) & GENMASK(18, 10)) >> 10)
|
||||
#define SYS_PAUSE_CFG_PAUSE_STOP(x) (((x) << 1) & GENMASK(9, 1))
|
||||
#define SYS_PAUSE_CFG_PAUSE_STOP_M GENMASK(9, 1)
|
||||
#define SYS_PAUSE_CFG_PAUSE_STOP_X(x) (((x) & GENMASK(9, 1)) >> 1)
|
||||
#define SYS_PAUSE_CFG_PAUSE_ENA BIT(0)
|
||||
|
||||
#define SYS_PAUSE_TOT_CFG_PAUSE_TOT_START(x) (((x) << 9) & GENMASK(17, 9))
|
||||
#define SYS_PAUSE_TOT_CFG_PAUSE_TOT_START_M GENMASK(17, 9)
|
||||
#define SYS_PAUSE_TOT_CFG_PAUSE_TOT_START_X(x) (((x) & GENMASK(17, 9)) >> 9)
|
||||
|
|
|
@ -137,11 +137,10 @@ static struct sk_buff *ocelot_xmit(struct sk_buff *skb,
|
|||
struct net_device *netdev)
|
||||
{
|
||||
struct dsa_port *dp = dsa_slave_to_port(netdev);
|
||||
u64 bypass, dest, src, qos_class, rew_op;
|
||||
struct dsa_switch *ds = dp->ds;
|
||||
int port = dp->index;
|
||||
struct ocelot *ocelot = ds->priv;
|
||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||
struct ocelot_port *ocelot_port;
|
||||
u64 qos_class, rew_op;
|
||||
u8 *injection;
|
||||
|
||||
if (unlikely(skb_cow_head(skb, OCELOT_TAG_LEN) < 0)) {
|
||||
|
@ -149,19 +148,15 @@ static struct sk_buff *ocelot_xmit(struct sk_buff *skb,
|
|||
return NULL;
|
||||
}
|
||||
|
||||
ocelot_port = ocelot->ports[dp->index];
|
||||
|
||||
injection = skb_push(skb, OCELOT_TAG_LEN);
|
||||
|
||||
memset(injection, 0, OCELOT_TAG_LEN);
|
||||
|
||||
/* Set the source port as the CPU port module and not the NPI port */
|
||||
src = ocelot->num_phys_ports;
|
||||
dest = BIT(port);
|
||||
bypass = true;
|
||||
memcpy(injection, ocelot_port->xmit_template, OCELOT_TAG_LEN);
|
||||
/* Fix up the fields which are not statically determined
|
||||
* in the template
|
||||
*/
|
||||
qos_class = skb->priority;
|
||||
|
||||
packing(injection, &bypass, 127, 127, OCELOT_TAG_LEN, PACK, 0);
|
||||
packing(injection, &dest, 68, 56, OCELOT_TAG_LEN, PACK, 0);
|
||||
packing(injection, &src, 46, 43, OCELOT_TAG_LEN, PACK, 0);
|
||||
packing(injection, &qos_class, 19, 17, OCELOT_TAG_LEN, PACK, 0);
|
||||
|
||||
if (ocelot->ptp && (skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP)) {
|
||||
|
|
Loading…
Reference in New Issue