[dpdk-dev] [PATCH v2] i40e: fix the write back issue in FVL VF
Liang, Cunming
cunming.liang at intel.com
Fri Oct 30 06:30:29 CET 2015
Hi,
> -----Original Message-----
> From: dev [mailto:dev-bounces at dpdk.org] On Behalf Of Jingjing Wu
> Sent: Wednesday, October 21, 2015 10:18 AM
> To: dev at dpdk.org
> Cc: Pei, Yulong
> Subject: [dpdk-dev] [PATCH v2] i40e: fix the write back issue in FVL VF
>
> If DPDK is used on VF while the host is using Linux Kernel driver
> as PF driver on FVL NIC, then VF Rx is reported only in batches of
> 4 packets. It is due to the kernel driver assumes VF driver is working
> in interrupt mode, but DPDK VF is working in Polling mode.
> This patch fixes this issue by using the V1.1 virtual channel with
> Linux i40e PF driver.
>
> Signed-off-by: Jingjing Wu <jingjing.wu at intel.com>
> ---
> v2:
> fix coding style issue.
>
> drivers/net/i40e/i40e_ethdev.h | 5 +++
> drivers/net/i40e/i40e_ethdev_vf.c | 65 +++++++++++++++++++++++++++++---
> -------
> 2 files changed, 53 insertions(+), 17 deletions(-)
>
> diff --git a/drivers/net/i40e/i40e_ethdev.h b/drivers/net/i40e/i40e_ethdev.h
> index 6185657..d42487d 100644
> --- a/drivers/net/i40e/i40e_ethdev.h
> +++ b/drivers/net/i40e/i40e_ethdev.h
> @@ -91,6 +91,11 @@
> #define I40E_48_BIT_WIDTH (CHAR_BIT * 6)
> #define I40E_48_BIT_MASK RTE_LEN2MASK(I40E_48_BIT_WIDTH, uint64_t)
>
> +/* Linux PF host with virtchnl version 1.1 */
> +#define PF_IS_V11(vf) \
> + (((vf)->version_major == I40E_VIRTCHNL_VERSION_MAJOR) && \
> + ((vf)->version_minor == 1))
> +
> /* index flex payload per layer */
> enum i40e_flxpld_layer_idx {
> I40E_FLXPLD_L2_IDX = 0,
> diff --git a/drivers/net/i40e/i40e_ethdev_vf.c
> b/drivers/net/i40e/i40e_ethdev_vf.c
> index b694400..1324281 100644
> --- a/drivers/net/i40e/i40e_ethdev_vf.c
> +++ b/drivers/net/i40e/i40e_ethdev_vf.c
> @@ -67,12 +67,15 @@
> #include "i40e_rxtx.h"
> #include "i40e_ethdev.h"
> #include "i40e_pf.h"
> -#define I40EVF_VSI_DEFAULT_MSIX_INTR 1
> +#define I40EVF_VSI_DEFAULT_MSIX_INTR 1
> +#define I40EVF_VSI_DEFAULT_MSIX_INTR_LNX 0
>
> /* busy wait delay in msec */
> #define I40EVF_BUSY_WAIT_DELAY 10
> #define I40EVF_BUSY_WAIT_COUNT 50
> #define MAX_RESET_WAIT_CNT 20
> +/*ITR index for NOITR*/
> +#define I40E_QINT_RQCTL_MSIX_INDX_NOITR 3
>
> struct i40evf_arq_msg_info {
> enum i40e_virtchnl_ops ops;
> @@ -412,7 +415,7 @@ i40evf_check_api_version(struct rte_eth_dev *dev)
> if (vf->version_major == I40E_DPDK_VERSION_MAJOR)
> PMD_DRV_LOG(INFO, "Peer is DPDK PF host");
> else if ((vf->version_major == I40E_VIRTCHNL_VERSION_MAJOR) &&
> - (vf->version_minor == I40E_VIRTCHNL_VERSION_MINOR))
> + (vf->version_minor <= I40E_VIRTCHNL_VERSION_MINOR))
> PMD_DRV_LOG(INFO, "Peer is Linux PF host");
> else {
> PMD_INIT_LOG(ERR, "PF/VF API version mismatch:(%u.%u)-
> (%u.%u)",
> @@ -432,14 +435,23 @@ i40evf_get_vf_resource(struct rte_eth_dev *dev)
> struct i40e_vf *vf = I40EVF_DEV_PRIVATE_TO_VF(dev->data-
> >dev_private);
> int err;
> struct vf_cmd_info args;
> - uint32_t len;
> + uint32_t caps, len;
>
> args.ops = I40E_VIRTCHNL_OP_GET_VF_RESOURCES;
> - args.in_args = NULL;
> - args.in_args_size = 0;
> args.out_buffer = cmd_result_buffer;
> args.out_size = I40E_AQ_BUF_SZ;
> -
> + if (PF_IS_V11(vf)) {
> + caps = I40E_VIRTCHNL_VF_OFFLOAD_L2 |
> + I40E_VIRTCHNL_VF_OFFLOAD_RSS_AQ |
> + I40E_VIRTCHNL_VF_OFFLOAD_RSS_REG |
> + I40E_VIRTCHNL_VF_OFFLOAD_VLAN |
> + I40E_VIRTCHNL_VF_OFFLOAD_RX_POLLING;
> + args.in_args = (uint8_t *)∩︀
> + args.in_args_size = sizeof(caps);
> + } else {
> + args.in_args = NULL;
> + args.in_args_size = 0;
> + }
> err = i40evf_execute_vf_cmd(dev, &args);
>
> if (err) {
> @@ -703,11 +715,14 @@ i40evf_config_irq_map(struct rte_eth_dev *dev)
> int i, err;
> map_info = (struct i40e_virtchnl_irq_map_info *)cmd_buffer;
> map_info->num_vectors = 1;
> - map_info->vecmap[0].rxitr_idx = RTE_LIBRTE_I40E_ITR_INTERVAL / 2;
> - map_info->vecmap[0].txitr_idx = RTE_LIBRTE_I40E_ITR_INTERVAL / 2;
> + map_info->vecmap[0].rxitr_idx = I40E_QINT_RQCTL_MSIX_INDX_NOITR;
> map_info->vecmap[0].vsi_id = vf->vsi_res->vsi_id;
> /* Alway use default dynamic MSIX interrupt */
> - map_info->vecmap[0].vector_id = I40EVF_VSI_DEFAULT_MSIX_INTR;
> + if (vf->version_major == I40E_DPDK_VERSION_MAJOR)
> + map_info->vecmap[0].vector_id =
> I40EVF_VSI_DEFAULT_MSIX_INTR;
> + else
> + map_info->vecmap[0].vector_id =
> I40EVF_VSI_DEFAULT_MSIX_INTR_LNX;
> +
> /* Don't map any tx queue */
> map_info->vecmap[0].txq_map = 0;
> map_info->vecmap[0].rxq_map = 0;
> @@ -1546,18 +1561,36 @@ i40evf_tx_init(struct rte_eth_dev *dev)
> }
>
> static inline void
> -i40evf_enable_queues_intr(struct i40e_hw *hw)
> +i40evf_enable_queues_intr(struct rte_eth_dev *dev)
> {
> - I40E_WRITE_REG(hw,
> I40E_VFINT_DYN_CTLN1(I40EVF_VSI_DEFAULT_MSIX_INTR - 1),
> + struct i40e_vf *vf = I40EVF_DEV_PRIVATE_TO_VF(dev->data-
> >dev_private);
> + struct i40e_hw *hw = I40E_DEV_PRIVATE_TO_HW(dev->data-
> >dev_private);
> +
> + if (vf->version_major == I40E_DPDK_VERSION_MAJOR)
> + /* To support DPDK PF host */
> + I40E_WRITE_REG(hw,
> +
> I40E_VFINT_DYN_CTLN1(I40EVF_VSI_DEFAULT_MSIX_INTR - 1),
> I40E_VFINT_DYN_CTLN1_INTENA_MASK |
> I40E_VFINT_DYN_CTLN_CLEARPBA_MASK);
> + else
> + /* To support Linux PF host */
> + I40E_WRITE_REG(hw, I40E_VFINT_DYN_CTL01,
> + I40E_VFINT_DYN_CTL01_INTENA_MASK |
> + I40E_VFINT_DYN_CTL01_CLEARPBA_MASK);
> }
>
> static inline void
> -i40evf_disable_queues_intr(struct i40e_hw *hw)
> +i40evf_disable_queues_intr(struct rte_eth_dev *dev)
> {
> - I40E_WRITE_REG(hw,
> I40E_VFINT_DYN_CTLN1(I40EVF_VSI_DEFAULT_MSIX_INTR - 1),
> + struct i40e_vf *vf = I40EVF_DEV_PRIVATE_TO_VF(dev->data-
> >dev_private);
> + struct i40e_hw *hw = I40E_DEV_PRIVATE_TO_HW(dev->data-
> >dev_private);
> +
> + if (vf->version_major == I40E_DPDK_VERSION_MAJOR)
> + I40E_WRITE_REG(hw,
> +
> I40E_VFINT_DYN_CTLN1(I40EVF_VSI_DEFAULT_MSIX_INTR - 1),
> 0);
> + else
> + I40E_WRITE_REG(hw, I40E_VFINT_DYN_CTL01, 0);
> }
>
> static int
> @@ -1604,7 +1637,7 @@ i40evf_dev_start(struct rte_eth_dev *dev)
> goto err_mac;
> }
>
> - i40evf_enable_queues_intr(hw);
> + i40evf_enable_queues_intr(dev);
> return 0;
>
> err_mac:
> @@ -1616,11 +1649,9 @@ err_queue:
> static void
> i40evf_dev_stop(struct rte_eth_dev *dev)
> {
> - struct i40e_hw *hw = I40E_DEV_PRIVATE_TO_HW(dev->data-
> >dev_private);
> -
> PMD_INIT_FUNC_TRACE();
>
> - i40evf_disable_queues_intr(hw);
> + i40evf_disable_queues_intr(dev);
> i40evf_stop_queues(dev);
> i40e_dev_clear_queues(dev);
> }
> --
> 2.4.0
Acked-by: Cunming Liang <cunming.liang at intel.com>
More information about the dev
mailing list