Skip to content

Commit

Permalink
drivers/net/ksz9477: Add simple port-based static VLAN configuration
Browse files Browse the repository at this point in the history
Add a static port-based VLAN configuration for KSZ9477 switch. This doesn't
use the VLAN tagging, but is a switch's internal mechanism to simply configure
if the packet forwarding is allowed from one port to another.

Signed-off-by: Jukka Laitinen <[email protected]>
  • Loading branch information
jlaitine committed Dec 19, 2023
1 parent bc2e551 commit 7bb149e
Show file tree
Hide file tree
Showing 4 changed files with 242 additions and 10 deletions.
64 changes: 54 additions & 10 deletions drivers/net/Kconfig
Original file line number Diff line number Diff line change
Expand Up @@ -392,6 +392,18 @@ menuconfig NET_W5500
References:
W5500 Datasheet, Version 1.0.9, 2013 WIZnet Co., Ltd.

if NET_W5500

config NET_W5500_NINTERFACES
int "Number of physical W5500 devices"
default 1
range 1 1
---help---
Specifies the number of physical WIZnet W5500
devices that will be supported.

endif # W5500

config NET_KSZ9477
bool "Management interface for ksz9477 ethernet switch"
default n
Expand All @@ -416,17 +428,49 @@ config NET_KSZ9477_SPI

endchoice

if NET_W5500

config NET_W5500_NINTERFACES
int "Number of physical W5500 devices"
default 1
range 1 1
config NET_KSZ9477_PORT_VLAN
bool "Use simple port-based VLAN configuration by default"
depends on NET_KSZ9477
default n
---help---
Specifies the number of physical WIZnet W5500
devices that will be supported.

endif # W5500
Set connections between switch ports by default at switch init.
For each port, set a bit mask indicating to which ports it is allowed
to forward packets. Bit 0 is for PHY1 port, bit 1 for PHY2 port etc.

config NET_KSZ9477_PORT_VLAN_PHY1
hex "Bitmask for PHY1 port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

config NET_KSZ9477_PORT_VLAN_PHY2
hex "Bitmask for PHY2 port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

config NET_KSZ9477_PORT_VLAN_PHY3
hex "Bitmask for PHY3 port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

config NET_KSZ9477_PORT_VLAN_PHY4
hex "Bitmask for PHY4 port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

config NET_KSZ9477_PORT_VLAN_PHY5
hex "Bitmask for PHY4 port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

config NET_KSZ9477_PORT_VLAN_RMII
hex "Bitmask for RMII port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

config NET_KSZ9477_PORT_VLAN_SGMII
hex "Bitmask for SGMII port connections"
depends on NET_KSZ9477_PORT_VLAN
default 0x1f

if ARCH_HAVE_PHY

Expand Down
124 changes: 124 additions & 0 deletions drivers/net/ksz9477.c
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,21 @@
* Private Data
****************************************************************************/

#ifdef CONFIG_NET_KSZ9477_PORT_VLAN

static uint8_t port_vlan_config[] =
{
CONFIG_NET_KSZ9477_PORT_VLAN_PHY1,
CONFIG_NET_KSZ9477_PORT_VLAN_PHY2,
CONFIG_NET_KSZ9477_PORT_VLAN_PHY3,
CONFIG_NET_KSZ9477_PORT_VLAN_PHY4,
CONFIG_NET_KSZ9477_PORT_VLAN_PHY5,
CONFIG_NET_KSZ9477_PORT_VLAN_RMII,
CONFIG_NET_KSZ9477_PORT_VLAN_SGMII
};

#endif

/****************************************************************************
* Private Functions
****************************************************************************/
Expand Down Expand Up @@ -197,6 +212,95 @@ static int ksz9477_sgmii_write_indirect(uint32_t address, uint16_t *value,
* Public Functions
****************************************************************************/

/****************************************************************************
* Name: ksz9477_enable_port_vlan
*
* Description:
* Enables static port-based VLAN, which can be configured in the switch
* queue management's port control registers
*
* Input Parameters:
* None
*
* Returned Value:
* OK or negative error number
*
****************************************************************************/

int ksz9477_enable_port_vlan(void)
{
uint32_t reg;
int ret = ksz9477_reg_read32(KSZ9477_Q_MGMT_CONTROL0, &reg);
if (ret)
{
reg |= KSZ9477_Q_MGMT_PORT_VLAN_ENABLE;
ret = ksz9477_reg_write32(KSZ9477_Q_MGMT_CONTROL0, reg);
}

return ret;
}

/****************************************************************************
* Name: ksz9477_disable_port_vlan
*
* Description:
* Disables the static port-based VLAN
*
* Input Parameters:
* None
*
* Returned Value:
* OK or negative error number
*
****************************************************************************/

int ksz9477_disable_port_vlan(void)
{
uint32_t reg;
int ret = ksz9477_reg_read32(KSZ9477_Q_MGMT_CONTROL0, &reg);
if (ret)
{
reg &= ~KSZ9477_Q_MGMT_PORT_VLAN_ENABLE;
ret = ksz9477_reg_write32(KSZ9477_Q_MGMT_CONTROL0, reg);
}

return ret;
}

/****************************************************************************
* Name: ksz9477_configure_port_vlan
*
* Description:
* Configures the static port-based VLAN for a single port
* The change will become effective next time when the switch is
* initialized.
*
* Input Parameters:
* port: The port being configured (1-7)
* disable: Bitmask of ports where frames may not be forwarded to.
* Bit 0 is for port 1, bit 1 for port 2 etc.
* enable: Bitmask of ports where frames may be forwarded to.
* Bit 0 is for port 1, bit 1 for port 2 etc.
*
* Returned Value:
* OK or negative error number
*
****************************************************************************/

int ksz9477_configure_port_vlan(ksz9477_port_t port, uint8_t disable,
uint8_t enable)
{
if (port < KSZ9477_PORT_PHY1 || port > KSZ9477_PORT_SGMII)
{
return -EINVAL;
}

port_vlan_config[port - KSZ9477_PORT_PHY1] &= ~disable;
port_vlan_config[port - KSZ9477_PORT_PHY1] |= enable;

return OK;
}

/****************************************************************************
* Name: ksz9477_init
*
Expand All @@ -215,6 +319,7 @@ static int ksz9477_sgmii_write_indirect(uint32_t address, uint16_t *value,
int ksz9477_init(ksz9477_port_t master_port)
{
int ret;
int i;
uint16_t regval16;
uint32_t regval32;

Expand Down Expand Up @@ -256,5 +361,24 @@ int ksz9477_init(ksz9477_port_t master_port)
&regval16, 1);
}

/* Configure the static port-based VLANs */

#ifdef CONFIG_NET_KSZ9477_PORT_VLAN

/* Restrict traffic according to Q_MGMT_PORT_CONTROL1 registers */

ret = ksz9477_enable_port_vlan();

/* Configure traffic control for each port */

for (i = 0; ret == OK && i < 7; i++)
{
ret = ksz9477_reg_write32(
KSZ9477_Q_MGMT_PORT_CONTROL1(KSZ9477_PORT_PHY1 + i),
port_vlan_config[i]);
}

#endif

return ret;
}
8 changes: 8 additions & 0 deletions drivers/net/ksz9477_reg.h
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,14 @@
#define KSZ9477_PORT_ADDRESS(p) KSZ9477_PORT_REG(p, 0x200)
#define KSZ9477_PORT_DATA(p) KSZ9477_PORT_REG(p, 0x204)

/* Switch queue management registers */

#define KSZ9477_Q_MGMT_CONTROL0 0x0390
#define KSZ9477_Q_MGMT_PORT_VLAN_ENABLE (1 << 1)

#define KSZ9477_Q_MGMT_PORT_CONTROL0(p) KSZ9477_PORT_REG(p, 0xA00)
#define KSZ9477_Q_MGMT_PORT_CONTROL1(p) KSZ9477_PORT_REG(p, 0xA04)

#define KSZ9477_SGMII_PORT_ADDRESS KSZ9477_PORT_ADDRESS(7)
#define KSZ9477_SGMII_PORT_DATA KSZ9477_PORT_DATA(7)

Expand Down
56 changes: 56 additions & 0 deletions include/nuttx/net/ksz9477.h
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,62 @@ int ksz9477_i2c_init(struct i2c_master_s *i2c_bus,
# error Only I2c interface currently supported
#endif

/****************************************************************************
* Name: ksz9477_enable_port_vlan
*
* Description:
* Enables static port-based VLAN, which can be configured in the switch
* queue management's port control registers
*
* Input Parameters:
* None
*
* Returned Value:
* OK or negative error number
*
****************************************************************************/

int ksz9477_enable_port_vlan(void);

/****************************************************************************
* Name: ksz9477_disable_port_vlan
*
* Description:
* Disables the static port-based VLAN
*
* Input Parameters:
* None
*
* Returned Value:
* OK or negative error number
*
****************************************************************************/

int ksz9477_disable_port_vlan(void);

/****************************************************************************
* Name: ksz9477_configure_port_vlan
*
* Description:
* Configures the static port-based VLAN for a single port
* The change will become effective next time when the switch is
* initialized.
*
* Input Parameters:
* port: The port being configured (1-7)
* disable: Bitmask of ports where frames may not be forwarded to.
* Bit 0 is for port 1, bit 1 for port 2 etc.
* enable: Bitmask of ports where frames may be forwarded to.
* Bit 0 is for port 1, bit 1 for port 2 etc.
*
* Returned Value:
* OK or negative error number
*
****************************************************************************/

int ksz9477_configure_port_vlan(ksz9477_port_t port, uint8_t disable,
uint8_t enable);

#if defined(__cplusplus)
}
#endif
Expand Down

0 comments on commit 7bb149e

Please sign in to comment.