This patch enables the watchdog to detect VF FLR when the link state
changes to down, and the default period is 2000us as defined by
IAVF_DEV_WATCHDOG_PERIOD.

In addition, the 'watchdog_period' devargs was added to adjust the watchdog
period(microseconds), or set to 0 to disable the watchdog.

Signed-off-by: Zhichao Zeng <zhichaox.z...@intel.com>

---
v3: add usage of new devarg in ice.rst
---
v2: enables watchdog when link status changes to down
---
 doc/guides/nics/ice.rst        | 13 ++++++
 drivers/net/iavf/iavf.h        |  5 ++-
 drivers/net/iavf/iavf_ethdev.c | 81 +++++++++++++++++++++++++---------
 drivers/net/iavf/iavf_vchnl.c  | 21 +++++++--
 4 files changed, 94 insertions(+), 26 deletions(-)

diff --git a/doc/guides/nics/ice.rst b/doc/guides/nics/ice.rst
index c351c6bd74..3cf038ede3 100644
--- a/doc/guides/nics/ice.rst
+++ b/doc/guides/nics/ice.rst
@@ -331,6 +331,19 @@ Additional Options
 
     -a 18:01.0,cap=dcf,acl=off
 
+- ``Control IAVF reset watchdog``
+
+  By default, the reset watchdog is enabled when link state changes to down.
+  The default period is 2000us, defined by ``IAVF_DEV_WATCHDOG_PERIOD``.
+  The user can set ``watchdog_period`` to adjust the watchdog period
+  (microseconds), or set it to 0 to disable the watchdog.
+
+    -a 18:01.0,watchdog_period=5000 (change period to 5000 microseconds)
+
+    or
+
+    -a 18:01.0,watchdog_period=0 (disable reset watchdog)
+
 .. _figure_ice_dcf:
 
 .. figure:: img/ice_dcf.*
diff --git a/drivers/net/iavf/iavf.h b/drivers/net/iavf/iavf.h
index aa18650ffa..98861e4242 100644
--- a/drivers/net/iavf/iavf.h
+++ b/drivers/net/iavf/iavf.h
@@ -31,7 +31,7 @@
 
 #define IAVF_NUM_MACADDR_MAX      64
 
-#define IAVF_DEV_WATCHDOG_PERIOD     0
+#define IAVF_DEV_WATCHDOG_PERIOD     2000 /* microseconds, set 0 to disable*/
 
 #define IAVF_DEFAULT_RX_PTHRESH      8
 #define IAVF_DEFAULT_RX_HTHRESH      8
@@ -304,6 +304,7 @@ struct iavf_devargs {
        uint8_t proto_xtr_dflt;
        uint8_t proto_xtr[IAVF_MAX_QUEUE_NUM];
        uint16_t quanta_size;
+       uint32_t watchdog_period;
 };
 
 struct iavf_security_ctx;
@@ -498,4 +499,6 @@ int iavf_flow_unsub(struct iavf_adapter *adapter,
                    struct iavf_fsub_conf *filter);
 int iavf_flow_sub_check(struct iavf_adapter *adapter,
                        struct iavf_fsub_conf *filter);
+void iavf_dev_watchdog_enable(struct iavf_adapter *adapter);
+void iavf_dev_watchdog_disable(struct iavf_adapter *adapter);
 #endif /* _IAVF_ETHDEV_H_ */
diff --git a/drivers/net/iavf/iavf_ethdev.c b/drivers/net/iavf/iavf_ethdev.c
index f6d68403ce..4cf4eb20cd 100644
--- a/drivers/net/iavf/iavf_ethdev.c
+++ b/drivers/net/iavf/iavf_ethdev.c
@@ -36,6 +36,7 @@
 /* devargs */
 #define IAVF_PROTO_XTR_ARG         "proto_xtr"
 #define IAVF_QUANTA_SIZE_ARG       "quanta_size"
+#define IAVF_RESET_WATCHDOG_ARG    "watchdog_period"
 
 uint64_t iavf_timestamp_dynflag;
 int iavf_timestamp_dynfield_offset = -1;
@@ -43,6 +44,7 @@ int iavf_timestamp_dynfield_offset = -1;
 static const char * const iavf_valid_args[] = {
        IAVF_PROTO_XTR_ARG,
        IAVF_QUANTA_SIZE_ARG,
+       IAVF_RESET_WATCHDOG_ARG,
        NULL
 };
 
@@ -301,40 +303,46 @@ iavf_dev_watchdog(void *cb_arg)
 
                        /* enter reset state with VFLR event */
                        adapter->vf.vf_reset = true;
+                       adapter->vf.link_up = false;
 
                        rte_eth_dev_callback_process(adapter->vf.eth_dev,
                                RTE_ETH_EVENT_INTR_RESET, NULL);
                }
        }
 
-       /* re-alarm watchdog */
-       rc = rte_eal_alarm_set(IAVF_DEV_WATCHDOG_PERIOD,
-                       &iavf_dev_watchdog, cb_arg);
+       if (adapter->devargs.watchdog_period) {
+               /* re-alarm watchdog */
+               rc = rte_eal_alarm_set(adapter->devargs.watchdog_period,
+                                       &iavf_dev_watchdog, cb_arg);
 
-       if (rc)
-               PMD_DRV_LOG(ERR, "Failed \"%s\" to reset device watchdog alarm",
-                       adapter->vf.eth_dev->data->name);
+               if (rc)
+                       PMD_DRV_LOG(ERR, "Failed \"%s\" to reset device 
watchdog alarm",
+                               adapter->vf.eth_dev->data->name);
+       }
 }
 
-static void
-iavf_dev_watchdog_enable(struct iavf_adapter *adapter __rte_unused)
-{
-#if (IAVF_DEV_WATCHDOG_PERIOD > 0)
-       PMD_DRV_LOG(INFO, "Enabling device watchdog");
-       adapter->vf.watchdog_enabled = true;
-       if (rte_eal_alarm_set(IAVF_DEV_WATCHDOG_PERIOD,
-                       &iavf_dev_watchdog, (void *)adapter))
-               PMD_DRV_LOG(ERR, "Failed to enabled device watchdog");
-#endif
+void
+iavf_dev_watchdog_enable(struct iavf_adapter *adapter)
+{
+       if (adapter->devargs.watchdog_period && !adapter->vf.watchdog_enabled) {
+               PMD_DRV_LOG(INFO, "Enabling device watchdog, period is %dμs",
+                                       adapter->devargs.watchdog_period);
+               adapter->vf.watchdog_enabled = true;
+               if (rte_eal_alarm_set(adapter->devargs.watchdog_period,
+                                       &iavf_dev_watchdog, (void *)adapter))
+                       PMD_DRV_LOG(ERR, "Failed to enabled device watchdog");
+       } else {
+               PMD_DRV_LOG(INFO, "Device watchdog is disabled");
+       }
 }
 
-static void
-iavf_dev_watchdog_disable(struct iavf_adapter *adapter __rte_unused)
+void
+iavf_dev_watchdog_disable(struct iavf_adapter *adapter)
 {
-#if (IAVF_DEV_WATCHDOG_PERIOD > 0)
-       PMD_DRV_LOG(INFO, "Disabling device watchdog");
-       adapter->vf.watchdog_enabled = false;
-#endif
+       if (adapter->devargs.watchdog_period && adapter->vf.watchdog_enabled) {
+               PMD_DRV_LOG(INFO, "Disabling device watchdog");
+               adapter->vf.watchdog_enabled = false;
+       }
 }
 
 static int
@@ -2201,6 +2209,25 @@ parse_u16(__rte_unused const char *key, const char 
*value, void *args)
        return 0;
 }
 
+static int
+iavf_parse_watchdog_period(__rte_unused const char *key, const char *value, 
void *args)
+{
+       int *num = (int *)args;
+       int tmp;
+
+       errno = 0;
+       tmp = atoi(value);
+       if (tmp < 0) {
+               PMD_DRV_LOG(WARNING, "%s: \"%s\" is not greater than or equal 
to zero",
+                               key, value);
+               return -1;
+       }
+
+       *num = tmp;
+
+       return 0;
+}
+
 static int iavf_parse_devargs(struct rte_eth_dev *dev)
 {
        struct iavf_adapter *ad =
@@ -2208,6 +2235,7 @@ static int iavf_parse_devargs(struct rte_eth_dev *dev)
        struct rte_devargs *devargs = dev->device->devargs;
        struct rte_kvargs *kvlist;
        int ret;
+       int watchdog_period = -1;
 
        if (!devargs)
                return 0;
@@ -2232,6 +2260,15 @@ static int iavf_parse_devargs(struct rte_eth_dev *dev)
        if (ret)
                goto bail;
 
+       ret = rte_kvargs_process(kvlist, IAVF_RESET_WATCHDOG_ARG,
+                                &iavf_parse_watchdog_period, &watchdog_period);
+       if (ret)
+               goto bail;
+       if (watchdog_period == -1)
+               ad->devargs.watchdog_period = IAVF_DEV_WATCHDOG_PERIOD;
+       else
+               ad->devargs.watchdog_period = watchdog_period;
+
        if (ad->devargs.quanta_size != 0 &&
            (ad->devargs.quanta_size < 256 || ad->devargs.quanta_size > 4096 ||
             ad->devargs.quanta_size & 0x40)) {
diff --git a/drivers/net/iavf/iavf_vchnl.c b/drivers/net/iavf/iavf_vchnl.c
index 9adaadb173..f2d7331f55 100644
--- a/drivers/net/iavf/iavf_vchnl.c
+++ b/drivers/net/iavf/iavf_vchnl.c
@@ -256,6 +256,12 @@ iavf_read_msg_from_pf(struct iavf_adapter *adapter, 
uint16_t buf_len,
                                vf->link_speed = iavf_convert_link_speed(speed);
                        }
                        iavf_dev_link_update(vf->eth_dev, 0);
+                       if (vf->link_up && !vf->vf_reset) {
+                               iavf_dev_watchdog_disable(adapter);
+                       } else {
+                               if (!vf->link_up)
+                                       iavf_dev_watchdog_enable(adapter);
+                       }
                        PMD_DRV_LOG(INFO, "Link status update:%s",
                                        vf->link_up ? "up" : "down");
                        break;
@@ -433,9 +439,12 @@ iavf_handle_pf_event_msg(struct rte_eth_dev *dev, uint8_t 
*msg,
        switch (pf_msg->event) {
        case VIRTCHNL_EVENT_RESET_IMPENDING:
                PMD_DRV_LOG(DEBUG, "VIRTCHNL_EVENT_RESET_IMPENDING event");
-               vf->vf_reset = true;
-               iavf_dev_event_post(dev, RTE_ETH_EVENT_INTR_RESET,
-                                             NULL, 0);
+               vf->link_up = false;
+               if (!vf->vf_reset) {
+                       vf->vf_reset = true;
+                       iavf_dev_event_post(dev, RTE_ETH_EVENT_INTR_RESET,
+                               NULL, 0);
+               }
                break;
        case VIRTCHNL_EVENT_LINK_CHANGE:
                PMD_DRV_LOG(DEBUG, "VIRTCHNL_EVENT_LINK_CHANGE event");
@@ -449,6 +458,12 @@ iavf_handle_pf_event_msg(struct rte_eth_dev *dev, uint8_t 
*msg,
                        vf->link_speed = iavf_convert_link_speed(speed);
                }
                iavf_dev_link_update(dev, 0);
+               if (vf->link_up && !vf->vf_reset) {
+                       iavf_dev_watchdog_disable(adapter);
+               } else {
+                       if (!vf->link_up)
+                               iavf_dev_watchdog_enable(adapter);
+               }
                iavf_dev_event_post(dev, RTE_ETH_EVENT_INTR_LSC, NULL, 0);
                break;
        case VIRTCHNL_EVENT_PF_DRIVER_CLOSE:
-- 
2.34.1

Reply via email to