net/octeontx: support set link up/down

Adding support for setting link up/down eth operation.
It is used to enable disable lmac.  Also implemented a
poll function for getting the link status at regular
intervals.

Signed-off-by: Harman Kalra <hkalra@marvell.com>
This commit is contained in:
Harman Kalra 2020-03-16 15:03:42 +05:30 committed by Ferruh Yigit
parent 56139e85ab
commit 8b42b07eef
6 changed files with 169 additions and 37 deletions

View File

@ -6,6 +6,7 @@
[Features]
Speed capabilities = Y
Link status = Y
Link status event = Y
Lock-free Tx queue = Y
Queue start/stop = P
MTU update = Y

View File

@ -207,6 +207,18 @@ octeontx_bgx_port_link_status(int port)
return link;
}
int
octeontx_bgx_port_set_link_state(int port, bool enable)
{
struct octeontx_mbox_hdr hdr;
hdr.coproc = OCTEONTX_BGX_COPROC;
hdr.msg = MBOX_BGX_PORT_SET_LINK_STATE;
hdr.vfid = port;
return octeontx_mbox_send(&hdr, &enable, sizeof(bool), NULL, 0);
}
int
octeontx_bgx_port_promisc_set(int port, int en)
{

View File

@ -5,6 +5,7 @@
#ifndef __OCTEONTX_BGX_H__
#define __OCTEONTX_BGX_H__
#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
@ -31,6 +32,7 @@
#define MBOX_BGX_PORT_ADD_MACADDR 15
#define MBOX_BGX_PORT_DEL_MACADDR 16
#define MBOX_BGX_PORT_GET_MACADDR_ENTRIES 17
#define MBOX_BGX_PORT_SET_LINK_STATE 20
/* BGX port configuration parameters: */
typedef struct octeontx_mbox_bgx_port_conf {
@ -132,6 +134,7 @@ int octeontx_bgx_port_mac_add(int port, uint8_t *mac_addr, int index);
int octeontx_bgx_port_mac_del(int port, uint32_t index);
int octeontx_bgx_port_mac_entries_get(int port);
int octeontx_bgx_port_mtu_set(int port, int mtu);
int octeontx_bgx_port_set_link_state(int port, bool en);
#endif /* __OCTEONTX_BGX_H__ */

View File

@ -11,6 +11,8 @@
#include <rte_alarm.h>
#include <rte_branch_prediction.h>
#include <rte_bus_vdev.h>
#include <rte_cycles.h>
#include <rte_debug.h>
#include <rte_devargs.h>
#include <rte_dev.h>
@ -18,7 +20,6 @@
#include <rte_malloc.h>
#include <rte_mbuf_pool_ops.h>
#include <rte_prefetch.h>
#include <rte_bus_vdev.h>
#include "octeontx_ethdev.h"
#include "octeontx_rxtx.h"
@ -153,11 +154,102 @@ octeontx_port_open(struct octeontx_nic *nic)
return res;
}
static void
octeontx_link_status_print(struct rte_eth_dev *eth_dev,
struct rte_eth_link *link)
{
if (link && link->link_status)
octeontx_log_info("Port %u: Link Up - speed %u Mbps - %s",
(eth_dev->data->port_id),
link->link_speed,
link->link_duplex == ETH_LINK_FULL_DUPLEX ?
"full-duplex" : "half-duplex");
else
octeontx_log_info("Port %d: Link Down",
(int)(eth_dev->data->port_id));
}
static void
octeontx_link_status_update(struct octeontx_nic *nic,
struct rte_eth_link *link)
{
memset(link, 0, sizeof(*link));
link->link_status = nic->link_up ? ETH_LINK_UP : ETH_LINK_DOWN;
switch (nic->speed) {
case OCTEONTX_LINK_SPEED_SGMII:
link->link_speed = ETH_SPEED_NUM_1G;
break;
case OCTEONTX_LINK_SPEED_XAUI:
link->link_speed = ETH_SPEED_NUM_10G;
break;
case OCTEONTX_LINK_SPEED_RXAUI:
case OCTEONTX_LINK_SPEED_10G_R:
link->link_speed = ETH_SPEED_NUM_10G;
break;
case OCTEONTX_LINK_SPEED_QSGMII:
link->link_speed = ETH_SPEED_NUM_5G;
break;
case OCTEONTX_LINK_SPEED_40G_R:
link->link_speed = ETH_SPEED_NUM_40G;
break;
case OCTEONTX_LINK_SPEED_RESERVE1:
case OCTEONTX_LINK_SPEED_RESERVE2:
default:
link->link_speed = ETH_SPEED_NUM_NONE;
octeontx_log_err("incorrect link speed %d", nic->speed);
break;
}
link->link_duplex = ETH_LINK_FULL_DUPLEX;
link->link_autoneg = ETH_LINK_AUTONEG;
}
static void
octeontx_link_status_poll(void *arg)
{
struct octeontx_nic *nic = arg;
struct rte_eth_link link;
struct rte_eth_dev *dev;
int res;
PMD_INIT_FUNC_TRACE();
dev = nic->dev;
res = octeontx_bgx_port_link_status(nic->port_id);
if (res < 0) {
octeontx_log_err("Failed to get port %d link status",
nic->port_id);
} else {
if (nic->link_up != (uint8_t)res) {
nic->link_up = (uint8_t)res;
octeontx_link_status_update(nic, &link);
octeontx_link_status_print(dev, &link);
rte_eth_linkstatus_set(dev, &link);
_rte_eth_dev_callback_process(dev,
RTE_ETH_EVENT_INTR_LSC,
NULL);
}
}
res = rte_eal_alarm_set(OCCTX_INTR_POLL_INTERVAL_MS * 1000,
octeontx_link_status_poll, nic);
if (res < 0)
octeontx_log_err("Failed to restart alarm for port %d, err: %d",
nic->port_id, res);
}
static void
octeontx_port_close(struct octeontx_nic *nic)
{
PMD_INIT_FUNC_TRACE();
rte_eal_alarm_cancel(octeontx_link_status_poll, nic);
octeontx_bgx_port_close(nic->port_id);
octeontx_log_dbg("port closed %d", nic->port_id);
}
@ -411,6 +503,8 @@ octeontx_dev_close(struct rte_eth_dev *dev)
rte_free(dev->data->mac_addrs);
dev->data->mac_addrs = NULL;
octeontx_port_close(nic);
dev->tx_pkt_burst = NULL;
dev->rx_pkt_burst = NULL;
}
@ -503,7 +597,7 @@ octeontx_dev_start(struct rte_eth_dev *dev)
{
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
struct octeontx_rxq *rxq;
int ret = 0, i;
int ret, i;
PMD_INIT_FUNC_TRACE();
/* Rechecking if any new offload set to update
@ -636,7 +730,10 @@ octeontx_port_link_status(struct octeontx_nic *nic)
return res;
}
nic->link_up = (uint8_t)res;
if (nic->link_up != (uint8_t)res || nic->print_flag == -1) {
nic->link_up = (uint8_t)res;
nic->print_flag = 1;
}
octeontx_log_dbg("port %d link status %d", nic->port_id, nic->link_up);
return res;
@ -661,39 +758,12 @@ octeontx_dev_link_update(struct rte_eth_dev *dev,
return res;
}
link.link_status = nic->link_up;
switch (nic->speed) {
case OCTEONTX_LINK_SPEED_SGMII:
link.link_speed = ETH_SPEED_NUM_1G;
break;
case OCTEONTX_LINK_SPEED_XAUI:
link.link_speed = ETH_SPEED_NUM_10G;
break;
case OCTEONTX_LINK_SPEED_RXAUI:
case OCTEONTX_LINK_SPEED_10G_R:
link.link_speed = ETH_SPEED_NUM_10G;
break;
case OCTEONTX_LINK_SPEED_QSGMII:
link.link_speed = ETH_SPEED_NUM_5G;
break;
case OCTEONTX_LINK_SPEED_40G_R:
link.link_speed = ETH_SPEED_NUM_40G;
break;
case OCTEONTX_LINK_SPEED_RESERVE1:
case OCTEONTX_LINK_SPEED_RESERVE2:
default:
link.link_speed = ETH_SPEED_NUM_NONE;
octeontx_log_err("incorrect link speed %d", nic->speed);
break;
octeontx_link_status_update(nic, &link);
if (nic->print_flag) {
octeontx_link_status_print(nic->dev, &link);
nic->print_flag = 0;
}
link.link_duplex = ETH_LINK_FULL_DUPLEX;
link.link_autoneg = ETH_LINK_AUTONEG;
return rte_eth_linkstatus_set(dev, &link);
}
@ -855,7 +925,7 @@ octeontx_vf_start_tx_queue(struct rte_eth_dev *dev, struct octeontx_nic *nic,
return res;
}
static int
int
octeontx_dev_tx_queue_start(struct rte_eth_dev *dev, uint16_t qidx)
{
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
@ -881,7 +951,7 @@ octeontx_vf_stop_tx_queue(struct rte_eth_dev *dev, struct octeontx_nic *nic,
return ret;
}
static int
int
octeontx_dev_tx_queue_stop(struct rte_eth_dev *dev, uint16_t qidx)
{
struct octeontx_nic *nic = octeontx_pmd_priv(dev);
@ -1201,6 +1271,8 @@ static const struct eth_dev_ops octeontx_dev_ops = {
.tx_queue_release = octeontx_dev_tx_queue_release,
.rx_queue_setup = octeontx_dev_rx_queue_setup,
.rx_queue_release = octeontx_dev_rx_queue_release,
.dev_set_link_up = octeontx_dev_set_link_up,
.dev_set_link_down = octeontx_dev_set_link_down,
.dev_supported_ptypes_get = octeontx_dev_supported_ptypes_get,
.mtu_set = octeontx_dev_mtu_set,
.pool_ops_supported = octeontx_pool_ops,
@ -1285,6 +1357,7 @@ octeontx_create(struct rte_vdev_device *dev, int port, uint8_t evdev,
nic->ev_queues = 1;
nic->ev_ports = 1;
nic->print_flag = -1;
data->dev_link.link_status = ETH_LINK_DOWN;
data->dev_started = 0;
@ -1321,6 +1394,13 @@ octeontx_create(struct rte_vdev_device *dev, int port, uint8_t evdev,
goto free_mac_addrs;
}
res = rte_eal_alarm_set(OCCTX_INTR_POLL_INTERVAL_MS * 1000,
octeontx_link_status_poll, nic);
if (res) {
octeontx_log_err("Failed to start link polling alarm");
goto err;
}
/* Update port_id mac to eth_dev */
memcpy(data->mac_addrs, nic->mac_addr, RTE_ETHER_ADDR_LEN);

View File

@ -30,7 +30,7 @@
#define OCTEONTX_MAX_LMAC_PER_BGX 4
#define OCCTX_RX_NB_SEG_MAX 6
#define OCCTX_INTR_POLL_INTERVAL_MS 1000
/* VLAN tag inserted by OCCTX_TX_VTAG_ACTION.
* In Tx space is always reserved for this in FRS.
*/
@ -121,6 +121,7 @@ struct octeontx_nic {
uint64_t tx_offloads;
uint16_t tx_offload_flags;
struct octeontx_vlan_info vlan_info;
int print_flag;
} __rte_cache_aligned;
struct octeontx_txq {
@ -143,10 +144,14 @@ void
octeontx_set_tx_function(struct rte_eth_dev *dev);
/* VLAN */
int octeontx_dev_tx_queue_start(struct rte_eth_dev *dev, uint16_t qidx);
int octeontx_dev_tx_queue_stop(struct rte_eth_dev *dev, uint16_t qidx);
int octeontx_dev_vlan_offload_init(struct rte_eth_dev *dev);
int octeontx_dev_vlan_offload_fini(struct rte_eth_dev *eth_dev);
int octeontx_dev_vlan_offload_set(struct rte_eth_dev *dev, int mask);
int octeontx_dev_vlan_filter_set(struct rte_eth_dev *dev,
uint16_t vlan_id, int on);
int octeontx_dev_set_link_up(struct rte_eth_dev *eth_dev);
int octeontx_dev_set_link_down(struct rte_eth_dev *eth_dev);
#endif /* __OCTEONTX_ETHDEV_H__ */

View File

@ -182,3 +182,34 @@ octeontx_dev_vlan_offload_fini(struct rte_eth_dev *dev)
return rc;
}
int
octeontx_dev_set_link_up(struct rte_eth_dev *eth_dev)
{
struct octeontx_nic *nic = octeontx_pmd_priv(eth_dev);
int rc, i;
rc = octeontx_bgx_port_set_link_state(nic->port_id, true);
if (rc)
goto done;
/* Start tx queues */
for (i = 0; i < eth_dev->data->nb_tx_queues; i++)
octeontx_dev_tx_queue_start(eth_dev, i);
done:
return rc;
}
int
octeontx_dev_set_link_down(struct rte_eth_dev *eth_dev)
{
struct octeontx_nic *nic = octeontx_pmd_priv(eth_dev);
int i;
/* Stop tx queues */
for (i = 0; i < eth_dev->data->nb_tx_queues; i++)
octeontx_dev_tx_queue_stop(eth_dev, i);
return octeontx_bgx_port_set_link_state(nic->port_id, false);
}