2018-01-29 13:11:30 +00:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
* Copyright 2015 6WIND S.A.
|
2018-03-20 19:20:35 +00:00
|
|
|
* Copyright 2015 Mellanox Technologies, Ltd
|
2015-10-30 18:52:30 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <assert.h>
|
2018-01-30 15:34:58 +00:00
|
|
|
#include <dlfcn.h>
|
2015-10-30 18:52:30 +00:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
2016-06-24 13:17:50 +00:00
|
|
|
#include <errno.h>
|
2015-10-30 18:52:30 +00:00
|
|
|
#include <net/if.h>
|
2018-01-25 15:00:24 +00:00
|
|
|
#include <sys/mman.h>
|
2018-07-10 16:04:52 +00:00
|
|
|
#include <linux/netlink.h>
|
2018-04-05 15:07:19 +00:00
|
|
|
#include <linux/rtnetlink.h>
|
2015-10-30 18:52:30 +00:00
|
|
|
|
|
|
|
/* Verbs header. */
|
|
|
|
/* ISO C doesn't support unnamed structs/unions, disabling -pedantic. */
|
|
|
|
#ifdef PEDANTIC
|
2016-09-19 14:36:54 +00:00
|
|
|
#pragma GCC diagnostic ignored "-Wpedantic"
|
2015-10-30 18:52:30 +00:00
|
|
|
#endif
|
|
|
|
#include <infiniband/verbs.h>
|
|
|
|
#ifdef PEDANTIC
|
2016-09-19 14:36:54 +00:00
|
|
|
#pragma GCC diagnostic error "-Wpedantic"
|
2015-10-30 18:52:30 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <rte_malloc.h>
|
2018-01-22 00:16:22 +00:00
|
|
|
#include <rte_ethdev_driver.h>
|
2017-04-11 15:44:24 +00:00
|
|
|
#include <rte_ethdev_pci.h>
|
2015-10-30 18:52:30 +00:00
|
|
|
#include <rte_pci.h>
|
2017-10-26 10:06:08 +00:00
|
|
|
#include <rte_bus_pci.h>
|
2015-10-30 18:52:30 +00:00
|
|
|
#include <rte_common.h>
|
2018-01-30 15:34:58 +00:00
|
|
|
#include <rte_config.h>
|
2018-01-25 15:00:24 +00:00
|
|
|
#include <rte_eal_memconfig.h>
|
2016-06-24 13:17:50 +00:00
|
|
|
#include <rte_kvargs.h>
|
2018-05-24 14:36:49 +00:00
|
|
|
#include <rte_rwlock.h>
|
|
|
|
#include <rte_spinlock.h>
|
2018-07-10 16:04:48 +00:00
|
|
|
#include <rte_string_fns.h>
|
2015-10-30 18:52:30 +00:00
|
|
|
|
|
|
|
#include "mlx5.h"
|
|
|
|
#include "mlx5_utils.h"
|
2015-10-30 18:52:31 +00:00
|
|
|
#include "mlx5_rxtx.h"
|
2015-10-30 18:52:30 +00:00
|
|
|
#include "mlx5_autoconf.h"
|
2015-11-03 17:15:13 +00:00
|
|
|
#include "mlx5_defs.h"
|
2018-01-30 15:34:56 +00:00
|
|
|
#include "mlx5_glue.h"
|
net/mlx5: add new memory region support
This is the new design of Memory Region (MR) for mlx PMD, in order to:
- Accommodate the new memory hotplug model.
- Support non-contiguous Mempool.
There are multiple layers for MR search.
L0 is to look up the last-hit entry which is pointed by mr_ctrl->mru (Most
Recently Used). If L0 misses, L1 is to look up the address in a fixed-sized
array by linear search. L0/L1 is in an inline function -
mlx5_mr_lookup_cache().
If L1 misses, the bottom-half function is called to look up the address
from the bigger local cache of the queue. This is L2 - mlx5_mr_addr2mr_bh()
and it is not an inline function. Data structure for L2 is the Binary Tree.
If L2 misses, the search falls into the slowest path which takes locks in
order to access global device cache (priv->mr.cache) which is also a B-tree
and caches the original MR list (priv->mr.mr_list) of the device. Unless
the global cache is overflowed, it is all-inclusive of the MR list. This is
L3 - mlx5_mr_lookup_dev(). The size of the L3 cache table is limited and
can't be expanded on the fly due to deadlock. Refer to the comments in the
code for the details - mr_lookup_dev(). If L3 is overflowed, the list will
have to be searched directly bypassing the cache although it is slower.
If L3 misses, a new MR for the address should be created -
mlx5_mr_create(). When it creates a new MR, it tries to register adjacent
memsegs as much as possible which are virtually contiguous around the
address. This must take two locks - memory_hotplug_lock and
priv->mr.rwlock. Due to memory_hotplug_lock, there can't be any
allocation/free of memory inside.
In the free callback of the memory hotplug event, freed space is searched
from the MR list and corresponding bits are cleared from the bitmap of MRs.
This can fragment a MR and the MR will have multiple search entries in the
caches. Once there's a change by the event, the global cache must be
rebuilt and all the per-queue caches will be flushed as well. If memory is
frequently freed in run-time, that may cause jitter on dataplane processing
in the worst case by incurring MR cache flush and rebuild. But, it would be
the least probable scenario.
To guarantee the most optimal performance, it is highly recommended to use
an EAL option - '--socket-mem'. Then, the reserved memory will be pinned
and won't be freed dynamically. And it is also recommended to configure
per-lcore cache of Mempool. Even though there're many MRs for a device or
MRs are highly fragmented, the cache of Mempool will be much helpful to
reduce misses on per-queue caches anyway.
'--legacy-mem' is also supported.
Signed-off-by: Yongseok Koh <yskoh@mellanox.com>
2018-05-09 11:09:04 +00:00
|
|
|
#include "mlx5_mr.h"
|
2015-10-30 18:52:30 +00:00
|
|
|
|
2016-06-24 13:17:54 +00:00
|
|
|
/* Device parameter to enable RX completion queue compression. */
|
|
|
|
#define MLX5_RXQ_CQE_COMP_EN "rxq_cqe_comp_en"
|
|
|
|
|
2018-05-09 11:13:50 +00:00
|
|
|
/* Device parameter to enable Multi-Packet Rx queue. */
|
|
|
|
#define MLX5_RX_MPRQ_EN "mprq_en"
|
|
|
|
|
|
|
|
/* Device parameter to configure log 2 of the number of strides for MPRQ. */
|
|
|
|
#define MLX5_RX_MPRQ_LOG_STRIDE_NUM "mprq_log_stride_num"
|
|
|
|
|
|
|
|
/* Device parameter to limit the size of memcpy'd packet for MPRQ. */
|
|
|
|
#define MLX5_RX_MPRQ_MAX_MEMCPY_LEN "mprq_max_memcpy_len"
|
|
|
|
|
|
|
|
/* Device parameter to set the minimum number of Rx queues to enable MPRQ. */
|
|
|
|
#define MLX5_RXQS_MIN_MPRQ "rxqs_min_mprq"
|
|
|
|
|
2016-06-24 13:17:56 +00:00
|
|
|
/* Device parameter to configure inline send. */
|
|
|
|
#define MLX5_TXQ_INLINE "txq_inline"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Device parameter to configure the number of TX queues threshold for
|
|
|
|
* enabling inline send.
|
|
|
|
*/
|
|
|
|
#define MLX5_TXQS_MIN_INLINE "txqs_min_inline"
|
|
|
|
|
2016-06-24 13:17:57 +00:00
|
|
|
/* Device parameter to enable multi-packet send WQEs. */
|
|
|
|
#define MLX5_TXQ_MPW_EN "txq_mpw_en"
|
|
|
|
|
2017-03-15 23:55:44 +00:00
|
|
|
/* Device parameter to include 2 dsegs in the title WQEBB. */
|
|
|
|
#define MLX5_TXQ_MPW_HDR_DSEG_EN "txq_mpw_hdr_dseg_en"
|
|
|
|
|
|
|
|
/* Device parameter to limit the size of inlining packet. */
|
|
|
|
#define MLX5_TXQ_MAX_INLINE_LEN "txq_max_inline_len"
|
|
|
|
|
2017-08-02 15:32:56 +00:00
|
|
|
/* Device parameter to enable hardware Tx vector. */
|
|
|
|
#define MLX5_TX_VEC_EN "tx_vec_en"
|
|
|
|
|
|
|
|
/* Device parameter to enable hardware Rx vector. */
|
|
|
|
#define MLX5_RX_VEC_EN "rx_vec_en"
|
|
|
|
|
2018-04-23 12:33:02 +00:00
|
|
|
/* Allow L3 VXLAN flow creation. */
|
|
|
|
#define MLX5_L3_VXLAN_EN "l3_vxlan_en"
|
|
|
|
|
2018-04-05 15:07:21 +00:00
|
|
|
/* Activate Netlink support in VF mode. */
|
|
|
|
#define MLX5_VF_NL_EN "vf_nl_en"
|
|
|
|
|
2018-07-10 16:04:58 +00:00
|
|
|
/* Select port representors to instantiate. */
|
|
|
|
#define MLX5_REPRESENTOR "representor"
|
|
|
|
|
2017-09-26 15:38:24 +00:00
|
|
|
#ifndef HAVE_IBV_MLX5_MOD_MPW
|
|
|
|
#define MLX5DV_CONTEXT_FLAGS_MPW_ALLOWED (1 << 2)
|
|
|
|
#define MLX5DV_CONTEXT_FLAGS_ENHANCED_MPW (1 << 3)
|
|
|
|
#endif
|
|
|
|
|
2017-10-09 18:46:59 +00:00
|
|
|
#ifndef HAVE_IBV_MLX5_MOD_CQE_128B_COMP
|
|
|
|
#define MLX5DV_CONTEXT_FLAGS_CQE_128B_COMP (1 << 4)
|
|
|
|
#endif
|
|
|
|
|
net/mlx5: add new memory region support
This is the new design of Memory Region (MR) for mlx PMD, in order to:
- Accommodate the new memory hotplug model.
- Support non-contiguous Mempool.
There are multiple layers for MR search.
L0 is to look up the last-hit entry which is pointed by mr_ctrl->mru (Most
Recently Used). If L0 misses, L1 is to look up the address in a fixed-sized
array by linear search. L0/L1 is in an inline function -
mlx5_mr_lookup_cache().
If L1 misses, the bottom-half function is called to look up the address
from the bigger local cache of the queue. This is L2 - mlx5_mr_addr2mr_bh()
and it is not an inline function. Data structure for L2 is the Binary Tree.
If L2 misses, the search falls into the slowest path which takes locks in
order to access global device cache (priv->mr.cache) which is also a B-tree
and caches the original MR list (priv->mr.mr_list) of the device. Unless
the global cache is overflowed, it is all-inclusive of the MR list. This is
L3 - mlx5_mr_lookup_dev(). The size of the L3 cache table is limited and
can't be expanded on the fly due to deadlock. Refer to the comments in the
code for the details - mr_lookup_dev(). If L3 is overflowed, the list will
have to be searched directly bypassing the cache although it is slower.
If L3 misses, a new MR for the address should be created -
mlx5_mr_create(). When it creates a new MR, it tries to register adjacent
memsegs as much as possible which are virtually contiguous around the
address. This must take two locks - memory_hotplug_lock and
priv->mr.rwlock. Due to memory_hotplug_lock, there can't be any
allocation/free of memory inside.
In the free callback of the memory hotplug event, freed space is searched
from the MR list and corresponding bits are cleared from the bitmap of MRs.
This can fragment a MR and the MR will have multiple search entries in the
caches. Once there's a change by the event, the global cache must be
rebuilt and all the per-queue caches will be flushed as well. If memory is
frequently freed in run-time, that may cause jitter on dataplane processing
in the worst case by incurring MR cache flush and rebuild. But, it would be
the least probable scenario.
To guarantee the most optimal performance, it is highly recommended to use
an EAL option - '--socket-mem'. Then, the reserved memory will be pinned
and won't be freed dynamically. And it is also recommended to configure
per-lcore cache of Mempool. Even though there're many MRs for a device or
MRs are highly fragmented, the cache of Mempool will be much helpful to
reduce misses on per-queue caches anyway.
'--legacy-mem' is also supported.
Signed-off-by: Yongseok Koh <yskoh@mellanox.com>
2018-05-09 11:09:04 +00:00
|
|
|
static const char *MZ_MLX5_PMD_SHARED_DATA = "mlx5_pmd_shared_data";
|
|
|
|
|
|
|
|
/* Shared memory between primary and secondary processes. */
|
|
|
|
struct mlx5_shared_data *mlx5_shared_data;
|
|
|
|
|
|
|
|
/* Spinlock for mlx5_shared_data allocation. */
|
|
|
|
static rte_spinlock_t mlx5_shared_data_lock = RTE_SPINLOCK_INITIALIZER;
|
|
|
|
|
2018-03-13 09:23:56 +00:00
|
|
|
/** Driver-specific log messages type. */
|
|
|
|
int mlx5_logtype;
|
|
|
|
|
net/mlx5: add new memory region support
This is the new design of Memory Region (MR) for mlx PMD, in order to:
- Accommodate the new memory hotplug model.
- Support non-contiguous Mempool.
There are multiple layers for MR search.
L0 is to look up the last-hit entry which is pointed by mr_ctrl->mru (Most
Recently Used). If L0 misses, L1 is to look up the address in a fixed-sized
array by linear search. L0/L1 is in an inline function -
mlx5_mr_lookup_cache().
If L1 misses, the bottom-half function is called to look up the address
from the bigger local cache of the queue. This is L2 - mlx5_mr_addr2mr_bh()
and it is not an inline function. Data structure for L2 is the Binary Tree.
If L2 misses, the search falls into the slowest path which takes locks in
order to access global device cache (priv->mr.cache) which is also a B-tree
and caches the original MR list (priv->mr.mr_list) of the device. Unless
the global cache is overflowed, it is all-inclusive of the MR list. This is
L3 - mlx5_mr_lookup_dev(). The size of the L3 cache table is limited and
can't be expanded on the fly due to deadlock. Refer to the comments in the
code for the details - mr_lookup_dev(). If L3 is overflowed, the list will
have to be searched directly bypassing the cache although it is slower.
If L3 misses, a new MR for the address should be created -
mlx5_mr_create(). When it creates a new MR, it tries to register adjacent
memsegs as much as possible which are virtually contiguous around the
address. This must take two locks - memory_hotplug_lock and
priv->mr.rwlock. Due to memory_hotplug_lock, there can't be any
allocation/free of memory inside.
In the free callback of the memory hotplug event, freed space is searched
from the MR list and corresponding bits are cleared from the bitmap of MRs.
This can fragment a MR and the MR will have multiple search entries in the
caches. Once there's a change by the event, the global cache must be
rebuilt and all the per-queue caches will be flushed as well. If memory is
frequently freed in run-time, that may cause jitter on dataplane processing
in the worst case by incurring MR cache flush and rebuild. But, it would be
the least probable scenario.
To guarantee the most optimal performance, it is highly recommended to use
an EAL option - '--socket-mem'. Then, the reserved memory will be pinned
and won't be freed dynamically. And it is also recommended to configure
per-lcore cache of Mempool. Even though there're many MRs for a device or
MRs are highly fragmented, the cache of Mempool will be much helpful to
reduce misses on per-queue caches anyway.
'--legacy-mem' is also supported.
Signed-off-by: Yongseok Koh <yskoh@mellanox.com>
2018-05-09 11:09:04 +00:00
|
|
|
/**
|
|
|
|
* Prepare shared data between primary and secondary process.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mlx5_prepare_shared_data(void)
|
|
|
|
{
|
|
|
|
const struct rte_memzone *mz;
|
|
|
|
|
|
|
|
rte_spinlock_lock(&mlx5_shared_data_lock);
|
|
|
|
if (mlx5_shared_data == NULL) {
|
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
/* Allocate shared memory. */
|
|
|
|
mz = rte_memzone_reserve(MZ_MLX5_PMD_SHARED_DATA,
|
|
|
|
sizeof(*mlx5_shared_data),
|
|
|
|
SOCKET_ID_ANY, 0);
|
|
|
|
} else {
|
|
|
|
/* Lookup allocated shared memory. */
|
|
|
|
mz = rte_memzone_lookup(MZ_MLX5_PMD_SHARED_DATA);
|
|
|
|
}
|
|
|
|
if (mz == NULL)
|
|
|
|
rte_panic("Cannot allocate mlx5 shared data\n");
|
|
|
|
mlx5_shared_data = mz->addr;
|
|
|
|
/* Initialize shared data. */
|
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
LIST_INIT(&mlx5_shared_data->mem_event_cb_list);
|
|
|
|
rte_rwlock_init(&mlx5_shared_data->mem_event_rwlock);
|
|
|
|
}
|
2018-05-30 14:42:53 +00:00
|
|
|
rte_mem_event_callback_register("MLX5_MEM_EVENT_CB",
|
|
|
|
mlx5_mr_mem_event_cb, NULL);
|
net/mlx5: add new memory region support
This is the new design of Memory Region (MR) for mlx PMD, in order to:
- Accommodate the new memory hotplug model.
- Support non-contiguous Mempool.
There are multiple layers for MR search.
L0 is to look up the last-hit entry which is pointed by mr_ctrl->mru (Most
Recently Used). If L0 misses, L1 is to look up the address in a fixed-sized
array by linear search. L0/L1 is in an inline function -
mlx5_mr_lookup_cache().
If L1 misses, the bottom-half function is called to look up the address
from the bigger local cache of the queue. This is L2 - mlx5_mr_addr2mr_bh()
and it is not an inline function. Data structure for L2 is the Binary Tree.
If L2 misses, the search falls into the slowest path which takes locks in
order to access global device cache (priv->mr.cache) which is also a B-tree
and caches the original MR list (priv->mr.mr_list) of the device. Unless
the global cache is overflowed, it is all-inclusive of the MR list. This is
L3 - mlx5_mr_lookup_dev(). The size of the L3 cache table is limited and
can't be expanded on the fly due to deadlock. Refer to the comments in the
code for the details - mr_lookup_dev(). If L3 is overflowed, the list will
have to be searched directly bypassing the cache although it is slower.
If L3 misses, a new MR for the address should be created -
mlx5_mr_create(). When it creates a new MR, it tries to register adjacent
memsegs as much as possible which are virtually contiguous around the
address. This must take two locks - memory_hotplug_lock and
priv->mr.rwlock. Due to memory_hotplug_lock, there can't be any
allocation/free of memory inside.
In the free callback of the memory hotplug event, freed space is searched
from the MR list and corresponding bits are cleared from the bitmap of MRs.
This can fragment a MR and the MR will have multiple search entries in the
caches. Once there's a change by the event, the global cache must be
rebuilt and all the per-queue caches will be flushed as well. If memory is
frequently freed in run-time, that may cause jitter on dataplane processing
in the worst case by incurring MR cache flush and rebuild. But, it would be
the least probable scenario.
To guarantee the most optimal performance, it is highly recommended to use
an EAL option - '--socket-mem'. Then, the reserved memory will be pinned
and won't be freed dynamically. And it is also recommended to configure
per-lcore cache of Mempool. Even though there're many MRs for a device or
MRs are highly fragmented, the cache of Mempool will be much helpful to
reduce misses on per-queue caches anyway.
'--legacy-mem' is also supported.
Signed-off-by: Yongseok Koh <yskoh@mellanox.com>
2018-05-09 11:09:04 +00:00
|
|
|
}
|
|
|
|
rte_spinlock_unlock(&mlx5_shared_data_lock);
|
|
|
|
}
|
|
|
|
|
2016-03-17 15:38:57 +00:00
|
|
|
/**
|
|
|
|
* Retrieve integer value from environment variable.
|
|
|
|
*
|
|
|
|
* @param[in] name
|
|
|
|
* Environment variable name.
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* Integer value, 0 if the variable is not set.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
mlx5_getenv_int(const char *name)
|
|
|
|
{
|
|
|
|
const char *val = getenv(name);
|
|
|
|
|
|
|
|
if (val == NULL)
|
|
|
|
return 0;
|
|
|
|
return atoi(val);
|
|
|
|
}
|
|
|
|
|
2017-10-06 15:45:50 +00:00
|
|
|
/**
|
|
|
|
* Verbs callback to allocate a memory. This function should allocate the space
|
|
|
|
* according to the size provided residing inside a huge page.
|
|
|
|
* Please note that all allocation must respect the alignment from libmlx5
|
|
|
|
* (i.e. currently sysconf(_SC_PAGESIZE)).
|
|
|
|
*
|
|
|
|
* @param[in] size
|
|
|
|
* The size in bytes of the memory to allocate.
|
|
|
|
* @param[in] data
|
|
|
|
* A pointer to the callback data.
|
|
|
|
*
|
|
|
|
* @return
|
2018-03-05 12:21:06 +00:00
|
|
|
* Allocated buffer, NULL otherwise and rte_errno is set.
|
2017-10-06 15:45:50 +00:00
|
|
|
*/
|
|
|
|
static void *
|
|
|
|
mlx5_alloc_verbs_buf(size_t size, void *data)
|
|
|
|
{
|
|
|
|
struct priv *priv = data;
|
|
|
|
void *ret;
|
|
|
|
size_t alignment = sysconf(_SC_PAGESIZE);
|
2018-01-22 12:33:38 +00:00
|
|
|
unsigned int socket = SOCKET_ID_ANY;
|
2017-10-06 15:45:50 +00:00
|
|
|
|
2018-01-22 12:33:38 +00:00
|
|
|
if (priv->verbs_alloc_ctx.type == MLX5_VERBS_ALLOC_TYPE_TX_QUEUE) {
|
|
|
|
const struct mlx5_txq_ctrl *ctrl = priv->verbs_alloc_ctx.obj;
|
|
|
|
|
|
|
|
socket = ctrl->socket;
|
|
|
|
} else if (priv->verbs_alloc_ctx.type ==
|
|
|
|
MLX5_VERBS_ALLOC_TYPE_RX_QUEUE) {
|
|
|
|
const struct mlx5_rxq_ctrl *ctrl = priv->verbs_alloc_ctx.obj;
|
|
|
|
|
|
|
|
socket = ctrl->socket;
|
|
|
|
}
|
2017-10-06 15:45:50 +00:00
|
|
|
assert(data != NULL);
|
2018-01-22 12:33:38 +00:00
|
|
|
ret = rte_malloc_socket(__func__, size, alignment, socket);
|
2018-03-05 12:21:06 +00:00
|
|
|
if (!ret && size)
|
|
|
|
rte_errno = ENOMEM;
|
2017-10-06 15:45:50 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Verbs callback to free a memory.
|
|
|
|
*
|
|
|
|
* @param[in] ptr
|
|
|
|
* A pointer to the memory to free.
|
|
|
|
* @param[in] data
|
|
|
|
* A pointer to the callback data.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mlx5_free_verbs_buf(void *ptr, void *data __rte_unused)
|
|
|
|
{
|
|
|
|
assert(data != NULL);
|
|
|
|
rte_free(ptr);
|
|
|
|
}
|
|
|
|
|
2015-10-30 18:52:30 +00:00
|
|
|
/**
|
|
|
|
* DPDK callback to close the device.
|
|
|
|
*
|
|
|
|
* Destroy all queues and objects, free memory.
|
|
|
|
*
|
|
|
|
* @param dev
|
|
|
|
* Pointer to Ethernet device structure.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mlx5_dev_close(struct rte_eth_dev *dev)
|
|
|
|
{
|
2017-11-23 09:22:32 +00:00
|
|
|
struct priv *priv = dev->data->dev_private;
|
2015-10-30 18:52:31 +00:00
|
|
|
unsigned int i;
|
2017-10-09 14:44:42 +00:00
|
|
|
int ret;
|
2015-10-30 18:52:30 +00:00
|
|
|
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(DEBUG, "port %u closing device \"%s\"",
|
|
|
|
dev->data->port_id,
|
|
|
|
((priv->ctx != NULL) ? priv->ctx->device->name : ""));
|
2015-10-30 18:55:06 +00:00
|
|
|
/* In case mlx5_dev_stop() has not been called. */
|
2018-03-05 12:21:04 +00:00
|
|
|
mlx5_dev_interrupt_handler_uninstall(dev);
|
|
|
|
mlx5_traffic_disable(dev);
|
2015-10-30 18:52:31 +00:00
|
|
|
/* Prevent crashes when queues are still in use. */
|
|
|
|
dev->rx_pkt_burst = removed_rx_burst;
|
|
|
|
dev->tx_pkt_burst = removed_tx_burst;
|
|
|
|
if (priv->rxqs != NULL) {
|
|
|
|
/* XXX race condition if mlx5_rx_burst() is still running. */
|
|
|
|
usleep(1000);
|
2017-10-09 14:44:49 +00:00
|
|
|
for (i = 0; (i != priv->rxqs_n); ++i)
|
2018-03-05 12:21:04 +00:00
|
|
|
mlx5_rxq_release(dev, i);
|
2015-10-30 18:52:31 +00:00
|
|
|
priv->rxqs_n = 0;
|
|
|
|
priv->rxqs = NULL;
|
|
|
|
}
|
|
|
|
if (priv->txqs != NULL) {
|
|
|
|
/* XXX race condition if mlx5_tx_burst() is still running. */
|
|
|
|
usleep(1000);
|
2017-10-09 14:44:48 +00:00
|
|
|
for (i = 0; (i != priv->txqs_n); ++i)
|
2018-03-05 12:21:04 +00:00
|
|
|
mlx5_txq_release(dev, i);
|
2015-10-30 18:52:31 +00:00
|
|
|
priv->txqs_n = 0;
|
|
|
|
priv->txqs = NULL;
|
|
|
|
}
|
2018-05-09 11:13:50 +00:00
|
|
|
mlx5_mprq_free_mp(dev);
|
net/mlx5: add new memory region support
This is the new design of Memory Region (MR) for mlx PMD, in order to:
- Accommodate the new memory hotplug model.
- Support non-contiguous Mempool.
There are multiple layers for MR search.
L0 is to look up the last-hit entry which is pointed by mr_ctrl->mru (Most
Recently Used). If L0 misses, L1 is to look up the address in a fixed-sized
array by linear search. L0/L1 is in an inline function -
mlx5_mr_lookup_cache().
If L1 misses, the bottom-half function is called to look up the address
from the bigger local cache of the queue. This is L2 - mlx5_mr_addr2mr_bh()
and it is not an inline function. Data structure for L2 is the Binary Tree.
If L2 misses, the search falls into the slowest path which takes locks in
order to access global device cache (priv->mr.cache) which is also a B-tree
and caches the original MR list (priv->mr.mr_list) of the device. Unless
the global cache is overflowed, it is all-inclusive of the MR list. This is
L3 - mlx5_mr_lookup_dev(). The size of the L3 cache table is limited and
can't be expanded on the fly due to deadlock. Refer to the comments in the
code for the details - mr_lookup_dev(). If L3 is overflowed, the list will
have to be searched directly bypassing the cache although it is slower.
If L3 misses, a new MR for the address should be created -
mlx5_mr_create(). When it creates a new MR, it tries to register adjacent
memsegs as much as possible which are virtually contiguous around the
address. This must take two locks - memory_hotplug_lock and
priv->mr.rwlock. Due to memory_hotplug_lock, there can't be any
allocation/free of memory inside.
In the free callback of the memory hotplug event, freed space is searched
from the MR list and corresponding bits are cleared from the bitmap of MRs.
This can fragment a MR and the MR will have multiple search entries in the
caches. Once there's a change by the event, the global cache must be
rebuilt and all the per-queue caches will be flushed as well. If memory is
frequently freed in run-time, that may cause jitter on dataplane processing
in the worst case by incurring MR cache flush and rebuild. But, it would be
the least probable scenario.
To guarantee the most optimal performance, it is highly recommended to use
an EAL option - '--socket-mem'. Then, the reserved memory will be pinned
and won't be freed dynamically. And it is also recommended to configure
per-lcore cache of Mempool. Even though there're many MRs for a device or
MRs are highly fragmented, the cache of Mempool will be much helpful to
reduce misses on per-queue caches anyway.
'--legacy-mem' is also supported.
Signed-off-by: Yongseok Koh <yskoh@mellanox.com>
2018-05-09 11:09:04 +00:00
|
|
|
mlx5_mr_release(dev);
|
2015-10-30 18:52:30 +00:00
|
|
|
if (priv->pd != NULL) {
|
|
|
|
assert(priv->ctx != NULL);
|
2018-01-30 15:34:56 +00:00
|
|
|
claim_zero(mlx5_glue->dealloc_pd(priv->pd));
|
|
|
|
claim_zero(mlx5_glue->close_device(priv->ctx));
|
2015-10-30 18:52:30 +00:00
|
|
|
} else
|
|
|
|
assert(priv->ctx == NULL);
|
2017-10-09 14:44:56 +00:00
|
|
|
if (priv->rss_conf.rss_key != NULL)
|
|
|
|
rte_free(priv->rss_conf.rss_key);
|
2015-11-02 18:11:57 +00:00
|
|
|
if (priv->reta_idx != NULL)
|
|
|
|
rte_free(priv->reta_idx);
|
2018-02-06 12:54:22 +00:00
|
|
|
if (priv->primary_socket)
|
2018-03-05 12:21:04 +00:00
|
|
|
mlx5_socket_uninit(dev);
|
2018-04-05 15:07:19 +00:00
|
|
|
if (priv->config.vf)
|
|
|
|
mlx5_nl_mac_addr_flush(dev);
|
2018-07-10 16:04:52 +00:00
|
|
|
if (priv->nl_socket_route >= 0)
|
|
|
|
close(priv->nl_socket_route);
|
|
|
|
if (priv->nl_socket_rdma >= 0)
|
|
|
|
close(priv->nl_socket_rdma);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_hrxq_ibv_verify(dev);
|
2017-10-09 14:44:51 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some hash Rx queue still remain",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_ind_table_ibv_verify(dev);
|
2017-10-09 14:44:50 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some indirection table still remain",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_rxq_ibv_verify(dev);
|
2017-10-09 14:44:46 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some Verbs Rx queue still remain",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_rxq_verify(dev);
|
2017-10-09 14:44:49 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some Rx queues still remain",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_txq_ibv_verify(dev);
|
2017-10-09 14:44:47 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some Verbs Tx queue still remain",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_txq_verify(dev);
|
2017-10-09 14:44:48 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some Tx queues still remain",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:04 +00:00
|
|
|
ret = mlx5_flow_verify(dev);
|
2017-10-09 14:44:42 +00:00
|
|
|
if (ret)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "port %u some flows still remain",
|
|
|
|
dev->data->port_id);
|
2018-07-10 16:04:54 +00:00
|
|
|
if (priv->domain_id != RTE_ETH_DEV_SWITCH_DOMAIN_ID_INVALID) {
|
|
|
|
unsigned int c = 0;
|
|
|
|
unsigned int i = mlx5_dev_to_port_id(dev->device, NULL, 0);
|
|
|
|
uint16_t port_id[i];
|
|
|
|
|
|
|
|
i = RTE_MIN(mlx5_dev_to_port_id(dev->device, port_id, i), i);
|
|
|
|
while (i--) {
|
|
|
|
struct priv *opriv =
|
|
|
|
rte_eth_devices[port_id[i]].data->dev_private;
|
|
|
|
|
|
|
|
if (!opriv ||
|
|
|
|
opriv->domain_id != priv->domain_id ||
|
|
|
|
&rte_eth_devices[port_id[i]] == dev)
|
|
|
|
continue;
|
|
|
|
++c;
|
|
|
|
}
|
|
|
|
if (!c)
|
|
|
|
claim_zero(rte_eth_switch_domain_free(priv->domain_id));
|
|
|
|
}
|
2015-10-30 18:52:30 +00:00
|
|
|
memset(priv, 0, sizeof(*priv));
|
2018-07-10 16:04:54 +00:00
|
|
|
priv->domain_id = RTE_ETH_DEV_SWITCH_DOMAIN_ID_INVALID;
|
2015-10-30 18:52:30 +00:00
|
|
|
}
|
|
|
|
|
2017-10-09 14:45:06 +00:00
|
|
|
const struct eth_dev_ops mlx5_dev_ops = {
|
2015-10-30 18:52:33 +00:00
|
|
|
.dev_configure = mlx5_dev_configure,
|
|
|
|
.dev_start = mlx5_dev_start,
|
|
|
|
.dev_stop = mlx5_dev_stop,
|
2016-03-17 15:38:54 +00:00
|
|
|
.dev_set_link_down = mlx5_set_link_down,
|
|
|
|
.dev_set_link_up = mlx5_set_link_up,
|
2015-10-30 18:52:30 +00:00
|
|
|
.dev_close = mlx5_dev_close,
|
2015-10-30 18:52:37 +00:00
|
|
|
.promiscuous_enable = mlx5_promiscuous_enable,
|
|
|
|
.promiscuous_disable = mlx5_promiscuous_disable,
|
|
|
|
.allmulticast_enable = mlx5_allmulticast_enable,
|
|
|
|
.allmulticast_disable = mlx5_allmulticast_disable,
|
2015-10-30 18:52:38 +00:00
|
|
|
.link_update = mlx5_link_update,
|
2015-10-30 18:52:36 +00:00
|
|
|
.stats_get = mlx5_stats_get,
|
|
|
|
.stats_reset = mlx5_stats_reset,
|
2017-01-17 14:37:08 +00:00
|
|
|
.xstats_get = mlx5_xstats_get,
|
|
|
|
.xstats_reset = mlx5_xstats_reset,
|
|
|
|
.xstats_get_names = mlx5_xstats_get_names,
|
2015-10-30 18:52:33 +00:00
|
|
|
.dev_infos_get = mlx5_dev_infos_get,
|
2016-03-14 20:50:50 +00:00
|
|
|
.dev_supported_ptypes_get = mlx5_dev_supported_ptypes_get,
|
2015-10-30 18:52:40 +00:00
|
|
|
.vlan_filter_set = mlx5_vlan_filter_set,
|
2015-10-30 18:52:31 +00:00
|
|
|
.rx_queue_setup = mlx5_rx_queue_setup,
|
|
|
|
.tx_queue_setup = mlx5_tx_queue_setup,
|
|
|
|
.rx_queue_release = mlx5_rx_queue_release,
|
|
|
|
.tx_queue_release = mlx5_tx_queue_release,
|
2015-10-30 18:52:39 +00:00
|
|
|
.flow_ctrl_get = mlx5_dev_get_flow_ctrl,
|
|
|
|
.flow_ctrl_set = mlx5_dev_set_flow_ctrl,
|
2015-10-30 18:52:32 +00:00
|
|
|
.mac_addr_remove = mlx5_mac_addr_remove,
|
|
|
|
.mac_addr_add = mlx5_mac_addr_add,
|
2016-01-05 18:00:09 +00:00
|
|
|
.mac_addr_set = mlx5_mac_addr_set,
|
2018-04-23 11:09:28 +00:00
|
|
|
.set_mc_addr_list = mlx5_set_mc_addr_list,
|
2015-10-30 18:52:35 +00:00
|
|
|
.mtu_set = mlx5_dev_set_mtu,
|
2016-03-03 14:26:44 +00:00
|
|
|
.vlan_strip_queue_set = mlx5_vlan_strip_queue_set,
|
|
|
|
.vlan_offload_set = mlx5_vlan_offload_set,
|
2015-11-02 18:11:57 +00:00
|
|
|
.reta_update = mlx5_dev_rss_reta_update,
|
|
|
|
.reta_query = mlx5_dev_rss_reta_query,
|
2015-10-30 18:55:11 +00:00
|
|
|
.rss_hash_update = mlx5_rss_hash_update,
|
|
|
|
.rss_hash_conf_get = mlx5_rss_hash_conf_get,
|
2016-03-03 14:26:43 +00:00
|
|
|
.filter_ctrl = mlx5_dev_filter_ctrl,
|
2017-03-29 08:36:32 +00:00
|
|
|
.rx_descriptor_status = mlx5_rx_descriptor_status,
|
|
|
|
.tx_descriptor_status = mlx5_tx_descriptor_status,
|
2017-03-14 13:03:09 +00:00
|
|
|
.rx_queue_intr_enable = mlx5_rx_intr_enable,
|
|
|
|
.rx_queue_intr_disable = mlx5_rx_intr_disable,
|
2018-01-20 21:12:21 +00:00
|
|
|
.is_removed = mlx5_is_removed,
|
2015-10-30 18:52:30 +00:00
|
|
|
};
|
|
|
|
|
2017-10-06 15:45:51 +00:00
|
|
|
static const struct eth_dev_ops mlx5_dev_sec_ops = {
|
|
|
|
.stats_get = mlx5_stats_get,
|
|
|
|
.stats_reset = mlx5_stats_reset,
|
|
|
|
.xstats_get = mlx5_xstats_get,
|
|
|
|
.xstats_reset = mlx5_xstats_reset,
|
|
|
|
.xstats_get_names = mlx5_xstats_get_names,
|
|
|
|
.dev_infos_get = mlx5_dev_infos_get,
|
|
|
|
.rx_descriptor_status = mlx5_rx_descriptor_status,
|
|
|
|
.tx_descriptor_status = mlx5_tx_descriptor_status,
|
|
|
|
};
|
|
|
|
|
2017-10-09 14:45:06 +00:00
|
|
|
/* Available operators in flow isolated mode. */
|
|
|
|
const struct eth_dev_ops mlx5_dev_ops_isolate = {
|
|
|
|
.dev_configure = mlx5_dev_configure,
|
|
|
|
.dev_start = mlx5_dev_start,
|
|
|
|
.dev_stop = mlx5_dev_stop,
|
|
|
|
.dev_set_link_down = mlx5_set_link_down,
|
|
|
|
.dev_set_link_up = mlx5_set_link_up,
|
|
|
|
.dev_close = mlx5_dev_close,
|
|
|
|
.link_update = mlx5_link_update,
|
|
|
|
.stats_get = mlx5_stats_get,
|
|
|
|
.stats_reset = mlx5_stats_reset,
|
|
|
|
.xstats_get = mlx5_xstats_get,
|
|
|
|
.xstats_reset = mlx5_xstats_reset,
|
|
|
|
.xstats_get_names = mlx5_xstats_get_names,
|
|
|
|
.dev_infos_get = mlx5_dev_infos_get,
|
|
|
|
.dev_supported_ptypes_get = mlx5_dev_supported_ptypes_get,
|
|
|
|
.vlan_filter_set = mlx5_vlan_filter_set,
|
|
|
|
.rx_queue_setup = mlx5_rx_queue_setup,
|
|
|
|
.tx_queue_setup = mlx5_tx_queue_setup,
|
|
|
|
.rx_queue_release = mlx5_rx_queue_release,
|
|
|
|
.tx_queue_release = mlx5_tx_queue_release,
|
|
|
|
.flow_ctrl_get = mlx5_dev_get_flow_ctrl,
|
|
|
|
.flow_ctrl_set = mlx5_dev_set_flow_ctrl,
|
|
|
|
.mac_addr_remove = mlx5_mac_addr_remove,
|
|
|
|
.mac_addr_add = mlx5_mac_addr_add,
|
|
|
|
.mac_addr_set = mlx5_mac_addr_set,
|
2018-04-23 11:09:28 +00:00
|
|
|
.set_mc_addr_list = mlx5_set_mc_addr_list,
|
2017-10-09 14:45:06 +00:00
|
|
|
.mtu_set = mlx5_dev_set_mtu,
|
|
|
|
.vlan_strip_queue_set = mlx5_vlan_strip_queue_set,
|
|
|
|
.vlan_offload_set = mlx5_vlan_offload_set,
|
|
|
|
.filter_ctrl = mlx5_dev_filter_ctrl,
|
|
|
|
.rx_descriptor_status = mlx5_rx_descriptor_status,
|
|
|
|
.tx_descriptor_status = mlx5_tx_descriptor_status,
|
|
|
|
.rx_queue_intr_enable = mlx5_rx_intr_enable,
|
|
|
|
.rx_queue_intr_disable = mlx5_rx_intr_disable,
|
2018-01-20 21:12:21 +00:00
|
|
|
.is_removed = mlx5_is_removed,
|
2017-10-09 14:45:06 +00:00
|
|
|
};
|
|
|
|
|
2016-06-24 13:17:50 +00:00
|
|
|
/**
|
|
|
|
* Verify and store value for device argument.
|
|
|
|
*
|
|
|
|
* @param[in] key
|
|
|
|
* Key argument to verify.
|
|
|
|
* @param[in] val
|
|
|
|
* Value associated with key.
|
|
|
|
* @param opaque
|
|
|
|
* User data.
|
|
|
|
*
|
|
|
|
* @return
|
2018-03-05 12:21:06 +00:00
|
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
2016-06-24 13:17:50 +00:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mlx5_args_check(const char *key, const char *val, void *opaque)
|
|
|
|
{
|
2018-01-10 09:16:58 +00:00
|
|
|
struct mlx5_dev_config *config = opaque;
|
2016-06-24 13:17:54 +00:00
|
|
|
unsigned long tmp;
|
2016-06-24 13:17:50 +00:00
|
|
|
|
2018-07-10 16:04:58 +00:00
|
|
|
/* No-op, port representors are processed in mlx5_dev_spawn(). */
|
|
|
|
if (!strcmp(MLX5_REPRESENTOR, key))
|
|
|
|
return 0;
|
2016-06-24 13:17:54 +00:00
|
|
|
errno = 0;
|
|
|
|
tmp = strtoul(val, NULL, 0);
|
|
|
|
if (errno) {
|
2018-03-05 12:21:06 +00:00
|
|
|
rte_errno = errno;
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "%s: \"%s\" is not a valid integer", key, val);
|
2018-03-05 12:21:06 +00:00
|
|
|
return -rte_errno;
|
2016-06-24 13:17:54 +00:00
|
|
|
}
|
|
|
|
if (strcmp(MLX5_RXQ_CQE_COMP_EN, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->cqe_comp = !!tmp;
|
2018-05-09 11:13:50 +00:00
|
|
|
} else if (strcmp(MLX5_RX_MPRQ_EN, key) == 0) {
|
|
|
|
config->mprq.enabled = !!tmp;
|
|
|
|
} else if (strcmp(MLX5_RX_MPRQ_LOG_STRIDE_NUM, key) == 0) {
|
|
|
|
config->mprq.stride_num_n = tmp;
|
|
|
|
} else if (strcmp(MLX5_RX_MPRQ_MAX_MEMCPY_LEN, key) == 0) {
|
|
|
|
config->mprq.max_memcpy_len = tmp;
|
|
|
|
} else if (strcmp(MLX5_RXQS_MIN_MPRQ, key) == 0) {
|
|
|
|
config->mprq.min_rxqs_num = tmp;
|
2016-06-24 13:17:56 +00:00
|
|
|
} else if (strcmp(MLX5_TXQ_INLINE, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->txq_inline = tmp;
|
2016-06-24 13:17:56 +00:00
|
|
|
} else if (strcmp(MLX5_TXQS_MIN_INLINE, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->txqs_inline = tmp;
|
2016-06-24 13:17:57 +00:00
|
|
|
} else if (strcmp(MLX5_TXQ_MPW_EN, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->mps = !!tmp ? config->mps : 0;
|
2017-03-15 23:55:44 +00:00
|
|
|
} else if (strcmp(MLX5_TXQ_MPW_HDR_DSEG_EN, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->mpw_hdr_dseg = !!tmp;
|
2017-03-15 23:55:44 +00:00
|
|
|
} else if (strcmp(MLX5_TXQ_MAX_INLINE_LEN, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->inline_max_packet_sz = tmp;
|
2017-08-02 15:32:56 +00:00
|
|
|
} else if (strcmp(MLX5_TX_VEC_EN, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->tx_vec_en = !!tmp;
|
2017-08-02 15:32:56 +00:00
|
|
|
} else if (strcmp(MLX5_RX_VEC_EN, key) == 0) {
|
2018-01-10 09:16:58 +00:00
|
|
|
config->rx_vec_en = !!tmp;
|
2018-04-23 12:33:02 +00:00
|
|
|
} else if (strcmp(MLX5_L3_VXLAN_EN, key) == 0) {
|
|
|
|
config->l3_vxlan_en = !!tmp;
|
2018-04-05 15:07:21 +00:00
|
|
|
} else if (strcmp(MLX5_VF_NL_EN, key) == 0) {
|
|
|
|
config->vf_nl_en = !!tmp;
|
2016-06-24 13:17:54 +00:00
|
|
|
} else {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "%s: unknown parameter", key);
|
2018-03-05 12:21:06 +00:00
|
|
|
rte_errno = EINVAL;
|
|
|
|
return -rte_errno;
|
2016-06-24 13:17:54 +00:00
|
|
|
}
|
|
|
|
return 0;
|
2016-06-24 13:17:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse device parameters.
|
|
|
|
*
|
2018-01-10 09:16:58 +00:00
|
|
|
* @param config
|
|
|
|
* Pointer to device configuration structure.
|
2016-06-24 13:17:50 +00:00
|
|
|
* @param devargs
|
|
|
|
* Device arguments structure.
|
|
|
|
*
|
|
|
|
* @return
|
2018-03-05 12:21:06 +00:00
|
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
2016-06-24 13:17:50 +00:00
|
|
|
*/
|
|
|
|
static int
|
2018-01-10 09:16:58 +00:00
|
|
|
mlx5_args(struct mlx5_dev_config *config, struct rte_devargs *devargs)
|
2016-06-24 13:17:50 +00:00
|
|
|
{
|
|
|
|
const char **params = (const char *[]){
|
2016-06-24 13:17:54 +00:00
|
|
|
MLX5_RXQ_CQE_COMP_EN,
|
2018-05-09 11:13:50 +00:00
|
|
|
MLX5_RX_MPRQ_EN,
|
|
|
|
MLX5_RX_MPRQ_LOG_STRIDE_NUM,
|
|
|
|
MLX5_RX_MPRQ_MAX_MEMCPY_LEN,
|
|
|
|
MLX5_RXQS_MIN_MPRQ,
|
2016-06-24 13:17:56 +00:00
|
|
|
MLX5_TXQ_INLINE,
|
|
|
|
MLX5_TXQS_MIN_INLINE,
|
2016-06-24 13:17:57 +00:00
|
|
|
MLX5_TXQ_MPW_EN,
|
2017-03-15 23:55:44 +00:00
|
|
|
MLX5_TXQ_MPW_HDR_DSEG_EN,
|
|
|
|
MLX5_TXQ_MAX_INLINE_LEN,
|
2017-08-02 15:32:56 +00:00
|
|
|
MLX5_TX_VEC_EN,
|
|
|
|
MLX5_RX_VEC_EN,
|
2018-04-23 12:33:02 +00:00
|
|
|
MLX5_L3_VXLAN_EN,
|
2018-04-05 15:07:21 +00:00
|
|
|
MLX5_VF_NL_EN,
|
2018-07-10 16:04:58 +00:00
|
|
|
MLX5_REPRESENTOR,
|
2016-06-24 13:17:50 +00:00
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
struct rte_kvargs *kvlist;
|
|
|
|
int ret = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (devargs == NULL)
|
|
|
|
return 0;
|
|
|
|
/* Following UGLY cast is done to pass checkpatch. */
|
|
|
|
kvlist = rte_kvargs_parse(devargs->args, params);
|
|
|
|
if (kvlist == NULL)
|
|
|
|
return 0;
|
|
|
|
/* Process parameters. */
|
|
|
|
for (i = 0; (params[i] != NULL); ++i) {
|
|
|
|
if (rte_kvargs_count(kvlist, params[i])) {
|
|
|
|
ret = rte_kvargs_process(kvlist, params[i],
|
2018-01-10 09:16:58 +00:00
|
|
|
mlx5_args_check, config);
|
2018-03-05 12:21:06 +00:00
|
|
|
if (ret) {
|
|
|
|
rte_errno = EINVAL;
|
2017-01-22 08:24:47 +00:00
|
|
|
rte_kvargs_free(kvlist);
|
2018-03-05 12:21:06 +00:00
|
|
|
return -rte_errno;
|
2017-01-22 08:24:47 +00:00
|
|
|
}
|
2016-06-24 13:17:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
rte_kvargs_free(kvlist);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-04-11 15:44:24 +00:00
|
|
|
static struct rte_pci_driver mlx5_driver;
|
2015-10-30 18:52:30 +00:00
|
|
|
|
2018-01-25 15:00:24 +00:00
|
|
|
/*
|
|
|
|
* Reserved UAR address space for TXQ UAR(hw doorbell) mapping, process
|
|
|
|
* local resource used by both primary and secondary to avoid duplicate
|
|
|
|
* reservation.
|
|
|
|
* The space has to be available on both primary and secondary process,
|
|
|
|
* TXQ UAR maps to this area using fixed mmap w/o double check.
|
|
|
|
*/
|
|
|
|
static void *uar_base;
|
|
|
|
|
2018-04-11 12:30:03 +00:00
|
|
|
static int
|
mem: replace memseg with memseg lists
Before, we were aggregating multiple pages into one memseg, so the
number of memsegs was small. Now, each page gets its own memseg,
so the list of memsegs is huge. To accommodate the new memseg list
size and to keep the under-the-hood workings sane, the memseg list
is now not just a single list, but multiple lists. To be precise,
each hugepage size available on the system gets one or more memseg
lists, per socket.
In order to support dynamic memory allocation, we reserve all
memory in advance (unless we're in 32-bit legacy mode, in which
case we do not preallocate memory). As in, we do an anonymous
mmap() of the entire maximum size of memory per hugepage size, per
socket (which is limited to either RTE_MAX_MEMSEG_PER_TYPE pages or
RTE_MAX_MEM_MB_PER_TYPE megabytes worth of memory, whichever is the
smaller one), split over multiple lists (which are limited to
either RTE_MAX_MEMSEG_PER_LIST memsegs or RTE_MAX_MEM_MB_PER_LIST
megabytes per list, whichever is the smaller one). There is also
a global limit of CONFIG_RTE_MAX_MEM_MB megabytes, which is mainly
used for 32-bit targets to limit amounts of preallocated memory,
but can be used to place an upper limit on total amount of VA
memory that can be allocated by DPDK application.
So, for each hugepage size, we get (by default) up to 128G worth
of memory, per socket, split into chunks of up to 32G in size.
The address space is claimed at the start, in eal_common_memory.c.
The actual page allocation code is in eal_memalloc.c (Linux-only),
and largely consists of copied EAL memory init code.
Pages in the list are also indexed by address. That is, in order
to figure out where the page belongs, one can simply look at base
address for a memseg list. Similarly, figuring out IOVA address
of a memzone is a matter of finding the right memseg list, getting
offset and dividing by page size to get the appropriate memseg.
This commit also removes rte_eal_dump_physmem_layout() call,
according to deprecation notice [1], and removes that deprecation
notice as well.
On 32-bit targets due to limited VA space, DPDK will no longer
spread memory to different sockets like before. Instead, it will
(by default) allocate all of the memory on socket where master
lcore is. To override this behavior, --socket-mem must be used.
The rest of the changes are really ripple effects from the memseg
change - heap changes, compile fixes, and rewrites to support
fbarray-backed memseg lists. Due to earlier switch to _walk()
functions, most of the changes are simple fixes, however some
of the _walk() calls were switched to memseg list walk, where
it made sense to do so.
Additionally, we are also switching locks from flock() to fcntl().
Down the line, we will be introducing single-file segments option,
and we cannot use flock() locks to lock parts of the file. Therefore,
we will use fcntl() locks for legacy mem as well, in case someone is
unfortunate enough to accidentally start legacy mem primary process
alongside an already working non-legacy mem-based primary process.
[1] http://dpdk.org/dev/patchwork/patch/34002/
Signed-off-by: Anatoly Burakov <anatoly.burakov@intel.com>
Tested-by: Santosh Shukla <santosh.shukla@caviumnetworks.com>
Tested-by: Hemant Agrawal <hemant.agrawal@nxp.com>
Tested-by: Gowrishankar Muthukrishnan <gowrishankar.m@linux.vnet.ibm.com>
2018-04-11 12:30:24 +00:00
|
|
|
find_lower_va_bound(const struct rte_memseg_list *msl __rte_unused,
|
|
|
|
const struct rte_memseg *ms, void *arg)
|
2018-04-11 12:30:03 +00:00
|
|
|
{
|
|
|
|
void **addr = arg;
|
|
|
|
|
|
|
|
if (*addr == NULL)
|
|
|
|
*addr = ms->addr;
|
|
|
|
else
|
|
|
|
*addr = RTE_MIN(*addr, ms->addr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-25 15:00:24 +00:00
|
|
|
/**
|
|
|
|
* Reserve UAR address space for primary process.
|
|
|
|
*
|
2018-03-05 12:21:04 +00:00
|
|
|
* @param[in] dev
|
|
|
|
* Pointer to Ethernet device.
|
2018-01-25 15:00:24 +00:00
|
|
|
*
|
|
|
|
* @return
|
2018-03-05 12:21:06 +00:00
|
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
2018-01-25 15:00:24 +00:00
|
|
|
*/
|
|
|
|
static int
|
2018-03-05 12:21:04 +00:00
|
|
|
mlx5_uar_init_primary(struct rte_eth_dev *dev)
|
2018-01-25 15:00:24 +00:00
|
|
|
{
|
2018-03-05 12:21:04 +00:00
|
|
|
struct priv *priv = dev->data->dev_private;
|
2018-01-25 15:00:24 +00:00
|
|
|
void *addr = (void *)0;
|
|
|
|
|
|
|
|
if (uar_base) { /* UAR address space mapped. */
|
|
|
|
priv->uar_base = uar_base;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* find out lower bound of hugepage segments */
|
2018-04-11 12:30:03 +00:00
|
|
|
rte_memseg_walk(find_lower_va_bound, &addr);
|
|
|
|
|
2018-01-25 15:00:24 +00:00
|
|
|
/* keep distance to hugepages to minimize potential conflicts. */
|
|
|
|
addr = RTE_PTR_SUB(addr, MLX5_UAR_OFFSET + MLX5_UAR_SIZE);
|
|
|
|
/* anonymous mmap, no real memory consumption. */
|
|
|
|
addr = mmap(addr, MLX5_UAR_SIZE,
|
|
|
|
PROT_NONE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
|
|
|
|
if (addr == MAP_FAILED) {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(ERR,
|
|
|
|
"port %u failed to reserve UAR address space, please"
|
|
|
|
" adjust MLX5_UAR_SIZE or try --base-virtaddr",
|
|
|
|
dev->data->port_id);
|
2018-03-05 12:21:06 +00:00
|
|
|
rte_errno = ENOMEM;
|
|
|
|
return -rte_errno;
|
2018-01-25 15:00:24 +00:00
|
|
|
}
|
|
|
|
/* Accept either same addr or a new addr returned from mmap if target
|
|
|
|
* range occupied.
|
|
|
|
*/
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(INFO, "port %u reserved UAR address space: %p",
|
|
|
|
dev->data->port_id, addr);
|
2018-01-25 15:00:24 +00:00
|
|
|
priv->uar_base = addr; /* for primary and secondary UAR re-mmap. */
|
|
|
|
uar_base = addr; /* process local, don't reserve again. */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reserve UAR address space for secondary process, align with
|
|
|
|
* primary process.
|
|
|
|
*
|
2018-03-05 12:21:04 +00:00
|
|
|
* @param[in] dev
|
|
|
|
* Pointer to Ethernet device.
|
2018-01-25 15:00:24 +00:00
|
|
|
*
|
|
|
|
* @return
|
2018-03-05 12:21:06 +00:00
|
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
2018-01-25 15:00:24 +00:00
|
|
|
*/
|
|
|
|
static int
|
2018-03-05 12:21:04 +00:00
|
|
|
mlx5_uar_init_secondary(struct rte_eth_dev *dev)
|
2018-01-25 15:00:24 +00:00
|
|
|
{
|
2018-03-05 12:21:04 +00:00
|
|
|
struct priv *priv = dev->data->dev_private;
|
2018-01-25 15:00:24 +00:00
|
|
|
void *addr;
|
|
|
|
|
|
|
|
assert(priv->uar_base);
|
|
|
|
if (uar_base) { /* already reserved. */
|
|
|
|
assert(uar_base == priv->uar_base);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* anonymous mmap, no real memory consumption. */
|
|
|
|
addr = mmap(priv->uar_base, MLX5_UAR_SIZE,
|
|
|
|
PROT_NONE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
|
|
|
|
if (addr == MAP_FAILED) {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(ERR, "port %u UAR mmap failed: %p size: %llu",
|
|
|
|
dev->data->port_id, priv->uar_base, MLX5_UAR_SIZE);
|
2018-03-05 12:21:06 +00:00
|
|
|
rte_errno = ENXIO;
|
|
|
|
return -rte_errno;
|
2018-01-25 15:00:24 +00:00
|
|
|
}
|
|
|
|
if (priv->uar_base != addr) {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(ERR,
|
|
|
|
"port %u UAR address %p size %llu occupied, please"
|
|
|
|
" adjust MLX5_UAR_OFFSET or try EAL parameter"
|
|
|
|
" --base-virtaddr",
|
|
|
|
dev->data->port_id, priv->uar_base, MLX5_UAR_SIZE);
|
2018-03-05 12:21:06 +00:00
|
|
|
rte_errno = ENXIO;
|
|
|
|
return -rte_errno;
|
2018-01-25 15:00:24 +00:00
|
|
|
}
|
|
|
|
uar_base = addr; /* process local, don't reserve again */
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(INFO, "port %u reserved UAR address space: %p",
|
|
|
|
dev->data->port_id, addr);
|
2018-01-25 15:00:24 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-10-30 18:52:30 +00:00
|
|
|
/**
|
2018-07-10 16:04:48 +00:00
|
|
|
* Spawn an Ethernet device from Verbs information.
|
2015-10-30 18:52:30 +00:00
|
|
|
*
|
2018-07-10 16:04:48 +00:00
|
|
|
* @param dpdk_dev
|
|
|
|
* Backing DPDK device.
|
|
|
|
* @param ibv_dev
|
|
|
|
* Verbs device.
|
|
|
|
* @param vf
|
|
|
|
* If nonzero, enable VF-specific features.
|
2018-07-10 16:04:54 +00:00
|
|
|
* @param[in] switch_info
|
|
|
|
* Switch properties of Ethernet device.
|
2015-10-30 18:52:30 +00:00
|
|
|
*
|
|
|
|
* @return
|
2018-07-10 16:04:48 +00:00
|
|
|
* A valid Ethernet device object on success, NULL otherwise and rte_errno
|
2018-07-10 16:04:58 +00:00
|
|
|
* is set. The following error is defined:
|
|
|
|
*
|
|
|
|
* EBUSY: device is not supposed to be spawned.
|
2015-10-30 18:52:30 +00:00
|
|
|
*/
|
2018-07-10 16:04:48 +00:00
|
|
|
static struct rte_eth_dev *
|
|
|
|
mlx5_dev_spawn(struct rte_device *dpdk_dev,
|
|
|
|
struct ibv_device *ibv_dev,
|
2018-07-10 16:04:54 +00:00
|
|
|
int vf,
|
|
|
|
const struct mlx5_switch_info *switch_info)
|
2015-10-30 18:52:30 +00:00
|
|
|
{
|
2018-07-10 16:04:48 +00:00
|
|
|
struct ibv_context *ctx;
|
2018-07-10 16:04:44 +00:00
|
|
|
struct ibv_device_attr_ex attr;
|
2018-07-10 16:04:50 +00:00
|
|
|
struct ibv_port_attr port_attr;
|
2018-07-10 16:04:46 +00:00
|
|
|
struct ibv_pd *pd = NULL;
|
2018-07-10 16:04:42 +00:00
|
|
|
struct mlx5dv_context dv_attr = { .comp_mask = 0 };
|
2018-07-10 16:04:50 +00:00
|
|
|
struct mlx5_dev_config config = {
|
|
|
|
.vf = !!vf,
|
|
|
|
.tx_vec_en = 1,
|
|
|
|
.rx_vec_en = 1,
|
|
|
|
.mpw_hdr_dseg = 0,
|
|
|
|
.txq_inline = MLX5_ARG_UNSET,
|
|
|
|
.txqs_inline = MLX5_ARG_UNSET,
|
|
|
|
.inline_max_packet_sz = MLX5_ARG_UNSET,
|
|
|
|
.vf_nl_en = 1,
|
|
|
|
.mprq = {
|
|
|
|
.enabled = 0,
|
|
|
|
.stride_num_n = MLX5_MPRQ_STRIDE_NUM_N,
|
|
|
|
.max_memcpy_len = MLX5_MPRQ_MEMCPY_DEFAULT_LEN,
|
|
|
|
.min_rxqs_num = MLX5_MPRQ_MIN_RXQS,
|
|
|
|
},
|
|
|
|
};
|
2018-07-10 16:04:46 +00:00
|
|
|
struct rte_eth_dev *eth_dev = NULL;
|
|
|
|
struct priv *priv = NULL;
|
2015-10-30 18:52:30 +00:00
|
|
|
int err = 0;
|
2016-03-17 15:38:58 +00:00
|
|
|
unsigned int mps;
|
2017-10-09 18:46:59 +00:00
|
|
|
unsigned int cqe_comp;
|
2017-09-04 11:43:51 +00:00
|
|
|
unsigned int tunnel_en = 0;
|
2018-05-15 11:07:14 +00:00
|
|
|
unsigned int mpls_en = 0;
|
2018-04-08 12:41:20 +00:00
|
|
|
unsigned int swp = 0;
|
2018-05-09 11:13:50 +00:00
|
|
|
unsigned int mprq = 0;
|
|
|
|
unsigned int mprq_min_stride_size_n = 0;
|
|
|
|
unsigned int mprq_max_stride_size_n = 0;
|
|
|
|
unsigned int mprq_min_stride_num_n = 0;
|
|
|
|
unsigned int mprq_max_stride_num_n = 0;
|
2017-10-10 14:22:54 +00:00
|
|
|
#ifdef HAVE_IBV_DEVICE_COUNTERS_SET_SUPPORT
|
2018-06-12 11:38:11 +00:00
|
|
|
struct ibv_counter_set_description cs_desc = { .counter_type = 0 };
|
2017-10-10 14:22:54 +00:00
|
|
|
#endif
|
2018-07-10 16:04:50 +00:00
|
|
|
struct ether_addr mac;
|
|
|
|
char name[RTE_ETH_NAME_MAX_LEN];
|
2018-07-10 16:04:54 +00:00
|
|
|
int own_domain_id = 0;
|
|
|
|
unsigned int i;
|
2015-10-30 18:52:30 +00:00
|
|
|
|
2018-07-10 16:04:58 +00:00
|
|
|
/* Determine if this port representor is supposed to be spawned. */
|
|
|
|
if (switch_info->representor && dpdk_dev->devargs) {
|
|
|
|
struct rte_eth_devargs eth_da;
|
|
|
|
|
|
|
|
err = rte_eth_devargs_parse(dpdk_dev->devargs->args, ð_da);
|
|
|
|
if (err) {
|
|
|
|
rte_errno = -err;
|
|
|
|
DRV_LOG(ERR, "failed to process device arguments: %s",
|
|
|
|
strerror(rte_errno));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
for (i = 0; i < eth_da.nb_representor_ports; ++i)
|
|
|
|
if (eth_da.representor_ports[i] ==
|
|
|
|
(uint16_t)switch_info->port_name)
|
|
|
|
break;
|
|
|
|
if (i == eth_da.nb_representor_ports) {
|
|
|
|
rte_errno = EBUSY;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
net/mlx5: add new memory region support
This is the new design of Memory Region (MR) for mlx PMD, in order to:
- Accommodate the new memory hotplug model.
- Support non-contiguous Mempool.
There are multiple layers for MR search.
L0 is to look up the last-hit entry which is pointed by mr_ctrl->mru (Most
Recently Used). If L0 misses, L1 is to look up the address in a fixed-sized
array by linear search. L0/L1 is in an inline function -
mlx5_mr_lookup_cache().
If L1 misses, the bottom-half function is called to look up the address
from the bigger local cache of the queue. This is L2 - mlx5_mr_addr2mr_bh()
and it is not an inline function. Data structure for L2 is the Binary Tree.
If L2 misses, the search falls into the slowest path which takes locks in
order to access global device cache (priv->mr.cache) which is also a B-tree
and caches the original MR list (priv->mr.mr_list) of the device. Unless
the global cache is overflowed, it is all-inclusive of the MR list. This is
L3 - mlx5_mr_lookup_dev(). The size of the L3 cache table is limited and
can't be expanded on the fly due to deadlock. Refer to the comments in the
code for the details - mr_lookup_dev(). If L3 is overflowed, the list will
have to be searched directly bypassing the cache although it is slower.
If L3 misses, a new MR for the address should be created -
mlx5_mr_create(). When it creates a new MR, it tries to register adjacent
memsegs as much as possible which are virtually contiguous around the
address. This must take two locks - memory_hotplug_lock and
priv->mr.rwlock. Due to memory_hotplug_lock, there can't be any
allocation/free of memory inside.
In the free callback of the memory hotplug event, freed space is searched
from the MR list and corresponding bits are cleared from the bitmap of MRs.
This can fragment a MR and the MR will have multiple search entries in the
caches. Once there's a change by the event, the global cache must be
rebuilt and all the per-queue caches will be flushed as well. If memory is
frequently freed in run-time, that may cause jitter on dataplane processing
in the worst case by incurring MR cache flush and rebuild. But, it would be
the least probable scenario.
To guarantee the most optimal performance, it is highly recommended to use
an EAL option - '--socket-mem'. Then, the reserved memory will be pinned
and won't be freed dynamically. And it is also recommended to configure
per-lcore cache of Mempool. Even though there're many MRs for a device or
MRs are highly fragmented, the cache of Mempool will be much helpful to
reduce misses on per-queue caches anyway.
'--legacy-mem' is also supported.
Signed-off-by: Yongseok Koh <yskoh@mellanox.com>
2018-05-09 11:09:04 +00:00
|
|
|
/* Prepare shared data between primary and secondary process. */
|
|
|
|
mlx5_prepare_shared_data();
|
2018-07-10 16:04:48 +00:00
|
|
|
errno = 0;
|
|
|
|
ctx = mlx5_glue->open_device(ibv_dev);
|
|
|
|
if (!ctx) {
|
|
|
|
rte_errno = errno ? errno : ENODEV;
|
|
|
|
return NULL;
|
2015-10-30 18:52:30 +00:00
|
|
|
}
|
2018-04-08 12:41:20 +00:00
|
|
|
#ifdef HAVE_IBV_MLX5_MOD_SWP
|
2018-07-10 16:04:42 +00:00
|
|
|
dv_attr.comp_mask |= MLX5DV_CONTEXT_MASK_SWP;
|
2018-04-08 12:41:20 +00:00
|
|
|
#endif
|
2017-09-26 15:38:24 +00:00
|
|
|
/*
|
|
|
|
* Multi-packet send is supported by ConnectX-4 Lx PF as well
|
|
|
|
* as all ConnectX-5 devices.
|
|
|
|
*/
|
2018-02-25 07:28:37 +00:00
|
|
|
#ifdef HAVE_IBV_DEVICE_TUNNEL_SUPPORT
|
2018-07-10 16:04:42 +00:00
|
|
|
dv_attr.comp_mask |= MLX5DV_CONTEXT_MASK_TUNNEL_OFFLOADS;
|
2018-05-09 11:13:50 +00:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_IBV_DEVICE_STRIDING_RQ_SUPPORT
|
2018-07-10 16:04:42 +00:00
|
|
|
dv_attr.comp_mask |= MLX5DV_CONTEXT_MASK_STRIDING_RQ;
|
2018-02-25 07:28:37 +00:00
|
|
|
#endif
|
2018-07-10 16:04:44 +00:00
|
|
|
mlx5_glue->dv_query_device(ctx, &dv_attr);
|
2018-07-10 16:04:42 +00:00
|
|
|
if (dv_attr.flags & MLX5DV_CONTEXT_FLAGS_MPW_ALLOWED) {
|
|
|
|
if (dv_attr.flags & MLX5DV_CONTEXT_FLAGS_ENHANCED_MPW) {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(DEBUG, "enhanced MPW is supported");
|
2017-10-16 17:41:56 +00:00
|
|
|
mps = MLX5_MPW_ENHANCED;
|
|
|
|
} else {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(DEBUG, "MPW is supported");
|
2017-10-16 17:41:56 +00:00
|
|
|
mps = MLX5_MPW;
|
|
|
|
}
|
2017-09-26 15:38:24 +00:00
|
|
|
} else {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(DEBUG, "MPW isn't supported");
|
2017-09-26 15:38:24 +00:00
|
|
|
mps = MLX5_MPW_DISABLED;
|
|
|
|
}
|
2018-07-10 16:04:50 +00:00
|
|
|
config.mps = mps;
|
2018-04-08 12:41:20 +00:00
|
|
|
#ifdef HAVE_IBV_MLX5_MOD_SWP
|
2018-07-10 16:04:42 +00:00
|
|
|
if (dv_attr.comp_mask & MLX5DV_CONTEXT_MASK_SWP)
|
|
|
|
swp = dv_attr.sw_parsing_caps.sw_parsing_offloads;
|
2018-04-08 12:41:20 +00:00
|
|
|
DRV_LOG(DEBUG, "SWP support: %u", swp);
|
2018-05-09 11:13:50 +00:00
|
|
|
#endif
|
2018-07-10 16:04:50 +00:00
|
|
|
config.swp = !!swp;
|
2018-05-09 11:13:50 +00:00
|
|
|
#ifdef HAVE_IBV_DEVICE_STRIDING_RQ_SUPPORT
|
2018-07-10 16:04:42 +00:00
|
|
|
if (dv_attr.comp_mask & MLX5DV_CONTEXT_MASK_STRIDING_RQ) {
|
2018-05-09 11:13:50 +00:00
|
|
|
struct mlx5dv_striding_rq_caps mprq_caps =
|
2018-07-10 16:04:42 +00:00
|
|
|
dv_attr.striding_rq_caps;
|
2018-05-09 11:13:50 +00:00
|
|
|
|
|
|
|
DRV_LOG(DEBUG, "\tmin_single_stride_log_num_of_bytes: %d",
|
|
|
|
mprq_caps.min_single_stride_log_num_of_bytes);
|
|
|
|
DRV_LOG(DEBUG, "\tmax_single_stride_log_num_of_bytes: %d",
|
|
|
|
mprq_caps.max_single_stride_log_num_of_bytes);
|
|
|
|
DRV_LOG(DEBUG, "\tmin_single_wqe_log_num_of_strides: %d",
|
|
|
|
mprq_caps.min_single_wqe_log_num_of_strides);
|
|
|
|
DRV_LOG(DEBUG, "\tmax_single_wqe_log_num_of_strides: %d",
|
|
|
|
mprq_caps.max_single_wqe_log_num_of_strides);
|
|
|
|
DRV_LOG(DEBUG, "\tsupported_qpts: %d",
|
|
|
|
mprq_caps.supported_qpts);
|
|
|
|
DRV_LOG(DEBUG, "device supports Multi-Packet RQ");
|
|
|
|
mprq = 1;
|
|
|
|
mprq_min_stride_size_n =
|
|
|
|
mprq_caps.min_single_stride_log_num_of_bytes;
|
|
|
|
mprq_max_stride_size_n =
|
|
|
|
mprq_caps.max_single_stride_log_num_of_bytes;
|
|
|
|
mprq_min_stride_num_n =
|
|
|
|
mprq_caps.min_single_wqe_log_num_of_strides;
|
|
|
|
mprq_max_stride_num_n =
|
|
|
|
mprq_caps.max_single_wqe_log_num_of_strides;
|
2018-07-10 16:04:50 +00:00
|
|
|
config.mprq.stride_num_n = RTE_MAX(MLX5_MPRQ_STRIDE_NUM_N,
|
|
|
|
mprq_min_stride_num_n);
|
2018-05-09 11:13:50 +00:00
|
|
|
}
|
2018-04-08 12:41:20 +00:00
|
|
|
#endif
|
2017-10-09 18:46:59 +00:00
|
|
|
if (RTE_CACHE_LINE_SIZE == 128 &&
|
2018-07-10 16:04:42 +00:00
|
|
|
!(dv_attr.flags & MLX5DV_CONTEXT_FLAGS_CQE_128B_COMP))
|
2017-10-09 18:46:59 +00:00
|
|
|
cqe_comp = 0;
|
|
|
|
else
|
|
|
|
cqe_comp = 1;
|
2018-07-10 16:04:50 +00:00
|
|
|
config.cqe_comp = cqe_comp;
|
2018-02-25 07:28:37 +00:00
|
|
|
#ifdef HAVE_IBV_DEVICE_TUNNEL_SUPPORT
|
2018-07-10 16:04:42 +00:00
|
|
|
if (dv_attr.comp_mask & MLX5DV_CONTEXT_MASK_TUNNEL_OFFLOADS) {
|
|
|
|
tunnel_en = ((dv_attr.tunnel_offloads_caps &
|
2018-02-25 07:28:37 +00:00
|
|
|
MLX5DV_RAW_PACKET_CAP_TUNNELED_OFFLOAD_VXLAN) &&
|
2018-07-10 16:04:42 +00:00
|
|
|
(dv_attr.tunnel_offloads_caps &
|
2018-02-25 07:28:37 +00:00
|
|
|
MLX5DV_RAW_PACKET_CAP_TUNNELED_OFFLOAD_GRE));
|
|
|
|
}
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(DEBUG, "tunnel offloading is %ssupported",
|
|
|
|
tunnel_en ? "" : "not ");
|
2018-02-25 07:28:37 +00:00
|
|
|
#else
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING,
|
|
|
|
"tunnel offloading disabled due to old OFED/rdma-core version");
|
2018-05-15 11:07:14 +00:00
|
|
|
#endif
|
2018-07-10 16:04:50 +00:00
|
|
|
config.tunnel_en = tunnel_en;
|
2018-05-15 11:07:14 +00:00
|
|
|
#ifdef HAVE_IBV_DEVICE_MPLS_SUPPORT
|
2018-07-10 16:04:42 +00:00
|
|
|
mpls_en = ((dv_attr.tunnel_offloads_caps &
|
2018-05-15 11:07:14 +00:00
|
|
|
MLX5DV_RAW_PACKET_CAP_TUNNELED_OFFLOAD_CW_MPLS_OVER_GRE) &&
|
2018-07-10 16:04:42 +00:00
|
|
|
(dv_attr.tunnel_offloads_caps &
|
2018-05-15 11:07:14 +00:00
|
|
|
MLX5DV_RAW_PACKET_CAP_TUNNELED_OFFLOAD_CW_MPLS_OVER_UDP));
|
|
|
|
DRV_LOG(DEBUG, "MPLS over GRE/UDP tunnel offloading is %ssupported",
|
|
|
|
mpls_en ? "" : "not ");
|
|
|
|
#else
|
|
|
|
DRV_LOG(WARNING, "MPLS over GRE/UDP tunnel offloading disabled due to"
|
|
|
|
" old OFED/rdma-core version or firmware configuration");
|
2018-02-25 07:28:37 +00:00
|
|
|
#endif
|
2018-07-10 16:04:50 +00:00
|
|
|
config.mpls_en = mpls_en;
|
2018-07-10 16:04:44 +00:00
|
|
|
err = mlx5_glue->query_device_ex(ctx, NULL, &attr);
|
2018-05-03 07:59:37 +00:00
|
|
|
if (err) {
|
|
|
|
DEBUG("ibv_query_device_ex() failed");
|
2015-10-30 18:52:30 +00:00
|
|
|
goto error;
|
2018-03-05 12:21:06 +00:00
|
|
|
}
|
2018-07-10 16:04:54 +00:00
|
|
|
if (!switch_info->representor)
|
|
|
|
rte_strlcpy(name, dpdk_dev->name, sizeof(name));
|
|
|
|
else
|
|
|
|
snprintf(name, sizeof(name), "%s_representor_%u",
|
|
|
|
dpdk_dev->name, switch_info->port_name);
|
|
|
|
DRV_LOG(DEBUG, "naming Ethernet device \"%s\"", name);
|
2018-07-10 16:04:50 +00:00
|
|
|
if (rte_eal_process_type() == RTE_PROC_SECONDARY) {
|
|
|
|
eth_dev = rte_eth_dev_attach_secondary(name);
|
2018-03-05 12:21:04 +00:00
|
|
|
if (eth_dev == NULL) {
|
2018-07-10 16:04:50 +00:00
|
|
|
DRV_LOG(ERR, "can not attach rte ethdev");
|
|
|
|
rte_errno = ENOMEM;
|
|
|
|
err = rte_errno;
|
2018-07-10 16:04:46 +00:00
|
|
|
goto error;
|
2018-03-05 12:21:04 +00:00
|
|
|
}
|
2018-07-10 16:04:48 +00:00
|
|
|
eth_dev->device = dpdk_dev;
|
2018-07-10 16:04:50 +00:00
|
|
|
eth_dev->dev_ops = &mlx5_dev_sec_ops;
|
|
|
|
err = mlx5_uar_init_secondary(eth_dev);
|
2018-05-03 07:59:37 +00:00
|
|
|
if (err) {
|
|
|
|
err = rte_errno;
|
2018-07-10 16:04:46 +00:00
|
|
|
goto error;
|
2018-05-03 07:59:37 +00:00
|
|
|
}
|
2018-07-10 16:04:50 +00:00
|
|
|
/* Receive command fd from primary process */
|
|
|
|
err = mlx5_socket_connect(eth_dev);
|
|
|
|
if (err < 0) {
|
2018-05-03 07:59:37 +00:00
|
|
|
err = rte_errno;
|
2018-07-10 16:04:46 +00:00
|
|
|
goto error;
|
2018-05-03 07:59:37 +00:00
|
|
|
}
|
2018-07-10 16:04:50 +00:00
|
|
|
/* Remap UAR for Tx queues. */
|
|
|
|
err = mlx5_tx_uar_remap(eth_dev, err);
|
2018-04-23 12:33:00 +00:00
|
|
|
if (err) {
|
2018-05-03 07:59:37 +00:00
|
|
|
err = rte_errno;
|
2018-07-10 16:04:46 +00:00
|
|
|
goto error;
|
2018-04-23 12:33:00 +00:00
|
|
|
}
|
2018-05-26 13:27:35 +00:00
|
|
|
/*
|
2018-07-10 16:04:50 +00:00
|
|
|
* Ethdev pointer is still required as input since
|
|
|
|
* the primary device is not accessible from the
|
|
|
|
* secondary process.
|
2018-05-26 13:27:35 +00:00
|
|
|
*/
|
2018-07-10 16:04:50 +00:00
|
|
|
eth_dev->rx_pkt_burst = mlx5_select_rx_function(eth_dev);
|
|
|
|
eth_dev->tx_pkt_burst = mlx5_select_tx_function(eth_dev);
|
|
|
|
claim_zero(mlx5_glue->close_device(ctx));
|
2018-07-10 16:04:48 +00:00
|
|
|
return eth_dev;
|
2015-10-30 18:52:30 +00:00
|
|
|
}
|
2018-07-10 16:04:50 +00:00
|
|
|
/* Check port status. */
|
|
|
|
err = mlx5_glue->query_port(ctx, 1, &port_attr);
|
|
|
|
if (err) {
|
|
|
|
DRV_LOG(ERR, "port query failed: %s", strerror(err));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (port_attr.link_layer != IBV_LINK_LAYER_ETHERNET) {
|
|
|
|
DRV_LOG(ERR, "port is not configured in Ethernet mode");
|
|
|
|
err = EINVAL;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (port_attr.state != IBV_PORT_ACTIVE)
|
|
|
|
DRV_LOG(DEBUG, "port is not active: \"%s\" (%d)",
|
|
|
|
mlx5_glue->port_state_str(port_attr.state),
|
|
|
|
port_attr.state);
|
|
|
|
/* Allocate protection domain. */
|
|
|
|
pd = mlx5_glue->alloc_pd(ctx);
|
|
|
|
if (pd == NULL) {
|
|
|
|
DRV_LOG(ERR, "PD allocation failure");
|
|
|
|
err = ENOMEM;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
priv = rte_zmalloc("ethdev private structure",
|
|
|
|
sizeof(*priv),
|
|
|
|
RTE_CACHE_LINE_SIZE);
|
|
|
|
if (priv == NULL) {
|
|
|
|
DRV_LOG(ERR, "priv allocation failure");
|
|
|
|
err = ENOMEM;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
priv->ctx = ctx;
|
2018-07-10 16:04:54 +00:00
|
|
|
strncpy(priv->ibdev_name, priv->ctx->device->name,
|
|
|
|
sizeof(priv->ibdev_name));
|
2018-07-10 16:04:50 +00:00
|
|
|
strncpy(priv->ibdev_path, priv->ctx->device->ibdev_path,
|
|
|
|
sizeof(priv->ibdev_path));
|
|
|
|
priv->device_attr = attr;
|
|
|
|
priv->pd = pd;
|
|
|
|
priv->mtu = ETHER_MTU;
|
2018-07-10 16:04:52 +00:00
|
|
|
/* Some internal functions rely on Netlink sockets, open them now. */
|
|
|
|
priv->nl_socket_rdma = mlx5_nl_init(0, NETLINK_RDMA);
|
|
|
|
priv->nl_socket_route = mlx5_nl_init(RTMGRP_LINK, NETLINK_ROUTE);
|
|
|
|
priv->nl_sn = 0;
|
2018-07-10 16:04:54 +00:00
|
|
|
priv->representor = !!switch_info->representor;
|
|
|
|
priv->domain_id = RTE_ETH_DEV_SWITCH_DOMAIN_ID_INVALID;
|
|
|
|
priv->representor_id =
|
|
|
|
switch_info->representor ? switch_info->port_name : -1;
|
|
|
|
/*
|
|
|
|
* Look for sibling devices in order to reuse their switch domain
|
|
|
|
* if any, otherwise allocate one.
|
|
|
|
*/
|
|
|
|
i = mlx5_dev_to_port_id(dpdk_dev, NULL, 0);
|
|
|
|
if (i > 0) {
|
|
|
|
uint16_t port_id[i];
|
|
|
|
|
|
|
|
i = RTE_MIN(mlx5_dev_to_port_id(dpdk_dev, port_id, i), i);
|
|
|
|
while (i--) {
|
|
|
|
const struct priv *opriv =
|
|
|
|
rte_eth_devices[port_id[i]].data->dev_private;
|
|
|
|
|
|
|
|
if (!opriv ||
|
|
|
|
opriv->domain_id ==
|
|
|
|
RTE_ETH_DEV_SWITCH_DOMAIN_ID_INVALID)
|
|
|
|
continue;
|
|
|
|
priv->domain_id = opriv->domain_id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (priv->domain_id == RTE_ETH_DEV_SWITCH_DOMAIN_ID_INVALID) {
|
|
|
|
err = rte_eth_switch_domain_alloc(&priv->domain_id);
|
|
|
|
if (err) {
|
|
|
|
err = rte_errno;
|
|
|
|
DRV_LOG(ERR, "unable to allocate switch domain: %s",
|
|
|
|
strerror(rte_errno));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
own_domain_id = 1;
|
|
|
|
}
|
2018-07-10 16:04:50 +00:00
|
|
|
err = mlx5_args(&config, dpdk_dev->devargs);
|
|
|
|
if (err) {
|
|
|
|
err = rte_errno;
|
|
|
|
DRV_LOG(ERR, "failed to process device arguments: %s",
|
|
|
|
strerror(rte_errno));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
config.hw_csum = !!(attr.device_cap_flags_ex & IBV_DEVICE_RAW_IP_CSUM);
|
|
|
|
DRV_LOG(DEBUG, "checksum offloading is %ssupported",
|
|
|
|
(config.hw_csum ? "" : "not "));
|
|
|
|
#ifdef HAVE_IBV_DEVICE_COUNTERS_SET_SUPPORT
|
|
|
|
config.flow_counter_en = !!attr.max_counter_sets;
|
|
|
|
mlx5_glue->describe_counter_set(ctx, 0, &cs_desc);
|
|
|
|
DRV_LOG(DEBUG, "counter type = %d, num of cs = %ld, attributes = %d",
|
|
|
|
cs_desc.counter_type, cs_desc.num_of_cs,
|
|
|
|
cs_desc.attributes);
|
|
|
|
#endif
|
|
|
|
config.ind_table_max_size =
|
|
|
|
attr.rss_caps.max_rwq_indirection_table_size;
|
|
|
|
/*
|
|
|
|
* Remove this check once DPDK supports larger/variable
|
|
|
|
* indirection tables.
|
|
|
|
*/
|
|
|
|
if (config.ind_table_max_size > (unsigned int)ETH_RSS_RETA_SIZE_512)
|
|
|
|
config.ind_table_max_size = ETH_RSS_RETA_SIZE_512;
|
|
|
|
DRV_LOG(DEBUG, "maximum Rx indirection table size is %u",
|
|
|
|
config.ind_table_max_size);
|
|
|
|
config.hw_vlan_strip = !!(attr.raw_packet_caps &
|
|
|
|
IBV_RAW_PACKET_CAP_CVLAN_STRIPPING);
|
|
|
|
DRV_LOG(DEBUG, "VLAN stripping is %ssupported",
|
|
|
|
(config.hw_vlan_strip ? "" : "not "));
|
|
|
|
config.hw_fcs_strip = !!(attr.raw_packet_caps &
|
|
|
|
IBV_RAW_PACKET_CAP_SCATTER_FCS);
|
|
|
|
DRV_LOG(DEBUG, "FCS stripping configuration is %ssupported",
|
|
|
|
(config.hw_fcs_strip ? "" : "not "));
|
|
|
|
#ifdef HAVE_IBV_WQ_FLAG_RX_END_PADDING
|
|
|
|
config.hw_padding = !!attr.rx_pad_end_addr_align;
|
|
|
|
#endif
|
|
|
|
DRV_LOG(DEBUG, "hardware Rx end alignment padding is %ssupported",
|
|
|
|
(config.hw_padding ? "" : "not "));
|
|
|
|
config.tso = (attr.tso_caps.max_tso > 0 &&
|
|
|
|
(attr.tso_caps.supported_qpts &
|
|
|
|
(1 << IBV_QPT_RAW_PACKET)));
|
|
|
|
if (config.tso)
|
|
|
|
config.tso_max_payload_sz = attr.tso_caps.max_tso;
|
|
|
|
if (config.mps && !mps) {
|
|
|
|
DRV_LOG(ERR,
|
|
|
|
"multi-packet send not supported on this device"
|
|
|
|
" (" MLX5_TXQ_MPW_EN ")");
|
|
|
|
err = ENOTSUP;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
DRV_LOG(INFO, "%sMPS is %s",
|
|
|
|
config.mps == MLX5_MPW_ENHANCED ? "enhanced " : "",
|
|
|
|
config.mps != MLX5_MPW_DISABLED ? "enabled" : "disabled");
|
|
|
|
if (config.cqe_comp && !cqe_comp) {
|
|
|
|
DRV_LOG(WARNING, "Rx CQE compression isn't supported");
|
|
|
|
config.cqe_comp = 0;
|
|
|
|
}
|
|
|
|
if (config.mprq.enabled && mprq) {
|
|
|
|
if (config.mprq.stride_num_n > mprq_max_stride_num_n ||
|
|
|
|
config.mprq.stride_num_n < mprq_min_stride_num_n) {
|
|
|
|
config.mprq.stride_num_n =
|
|
|
|
RTE_MAX(MLX5_MPRQ_STRIDE_NUM_N,
|
|
|
|
mprq_min_stride_num_n);
|
|
|
|
DRV_LOG(WARNING,
|
|
|
|
"the number of strides"
|
|
|
|
" for Multi-Packet RQ is out of range,"
|
|
|
|
" setting default value (%u)",
|
|
|
|
1 << config.mprq.stride_num_n);
|
|
|
|
}
|
|
|
|
config.mprq.min_stride_size_n = mprq_min_stride_size_n;
|
|
|
|
config.mprq.max_stride_size_n = mprq_max_stride_size_n;
|
|
|
|
} else if (config.mprq.enabled && !mprq) {
|
|
|
|
DRV_LOG(WARNING, "Multi-Packet RQ isn't supported");
|
|
|
|
config.mprq.enabled = 0;
|
|
|
|
}
|
|
|
|
eth_dev = rte_eth_dev_allocate(name);
|
|
|
|
if (eth_dev == NULL) {
|
|
|
|
DRV_LOG(ERR, "can not allocate rte ethdev");
|
|
|
|
err = ENOMEM;
|
|
|
|
goto error;
|
|
|
|
}
|
2018-07-10 16:04:54 +00:00
|
|
|
if (priv->representor)
|
|
|
|
eth_dev->data->dev_flags |= RTE_ETH_DEV_REPRESENTOR;
|
2018-07-10 16:04:50 +00:00
|
|
|
eth_dev->data->dev_private = priv;
|
|
|
|
priv->dev_data = eth_dev->data;
|
|
|
|
eth_dev->data->mac_addrs = priv->mac;
|
|
|
|
eth_dev->device = dpdk_dev;
|
|
|
|
eth_dev->device->driver = &mlx5_driver.driver;
|
|
|
|
err = mlx5_uar_init_primary(eth_dev);
|
|
|
|
if (err) {
|
|
|
|
err = rte_errno;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
/* Configure the first MAC address by default. */
|
|
|
|
if (mlx5_get_mac(eth_dev, &mac.addr_bytes)) {
|
|
|
|
DRV_LOG(ERR,
|
|
|
|
"port %u cannot get MAC address, is mlx5_en"
|
|
|
|
" loaded? (errno: %s)",
|
|
|
|
eth_dev->data->port_id, strerror(rte_errno));
|
|
|
|
err = ENODEV;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
DRV_LOG(INFO,
|
|
|
|
"port %u MAC address is %02x:%02x:%02x:%02x:%02x:%02x",
|
|
|
|
eth_dev->data->port_id,
|
|
|
|
mac.addr_bytes[0], mac.addr_bytes[1],
|
|
|
|
mac.addr_bytes[2], mac.addr_bytes[3],
|
|
|
|
mac.addr_bytes[4], mac.addr_bytes[5]);
|
|
|
|
#ifndef NDEBUG
|
|
|
|
{
|
|
|
|
char ifname[IF_NAMESIZE];
|
|
|
|
|
|
|
|
if (mlx5_get_ifname(eth_dev, &ifname) == 0)
|
|
|
|
DRV_LOG(DEBUG, "port %u ifname is \"%s\"",
|
|
|
|
eth_dev->data->port_id, ifname);
|
|
|
|
else
|
|
|
|
DRV_LOG(DEBUG, "port %u ifname is unknown",
|
|
|
|
eth_dev->data->port_id);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* Get actual MTU if possible. */
|
|
|
|
err = mlx5_get_mtu(eth_dev, &priv->mtu);
|
|
|
|
if (err) {
|
|
|
|
err = rte_errno;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
DRV_LOG(DEBUG, "port %u MTU is %u", eth_dev->data->port_id,
|
|
|
|
priv->mtu);
|
|
|
|
/* Initialize burst functions to prevent crashes before link-up. */
|
|
|
|
eth_dev->rx_pkt_burst = removed_rx_burst;
|
|
|
|
eth_dev->tx_pkt_burst = removed_tx_burst;
|
|
|
|
eth_dev->dev_ops = &mlx5_dev_ops;
|
|
|
|
/* Register MAC address. */
|
|
|
|
claim_zero(mlx5_mac_addr_add(eth_dev, &mac, 0, 0));
|
2018-07-10 16:04:52 +00:00
|
|
|
if (vf && config.vf_nl_en)
|
2018-07-10 16:04:50 +00:00
|
|
|
mlx5_nl_mac_addr_sync(eth_dev);
|
|
|
|
TAILQ_INIT(&priv->flows);
|
|
|
|
TAILQ_INIT(&priv->ctrl_flows);
|
|
|
|
/* Hint libmlx5 to use PMD allocator for data plane resources */
|
|
|
|
struct mlx5dv_ctx_allocators alctr = {
|
|
|
|
.alloc = &mlx5_alloc_verbs_buf,
|
|
|
|
.free = &mlx5_free_verbs_buf,
|
|
|
|
.data = priv,
|
|
|
|
};
|
|
|
|
mlx5_glue->dv_set_context_attr(ctx, MLX5DV_CTX_ATTR_BUF_ALLOCATORS,
|
|
|
|
(void *)((uintptr_t)&alctr));
|
|
|
|
/* Bring Ethernet device up. */
|
|
|
|
DRV_LOG(DEBUG, "port %u forcing Ethernet interface up",
|
|
|
|
eth_dev->data->port_id);
|
|
|
|
mlx5_set_link_up(eth_dev);
|
|
|
|
/*
|
|
|
|
* Even though the interrupt handler is not installed yet,
|
|
|
|
* interrupts will still trigger on the asyn_fd from
|
|
|
|
* Verbs context returned by ibv_open_device().
|
|
|
|
*/
|
|
|
|
mlx5_link_update(eth_dev, 0);
|
|
|
|
/* Store device configuration on private structure. */
|
|
|
|
priv->config = config;
|
|
|
|
/* Supported Verbs flow priority number detection. */
|
2018-07-12 09:30:49 +00:00
|
|
|
err = mlx5_flow_discover_priorities(eth_dev);
|
|
|
|
if (err < 0)
|
|
|
|
goto error;
|
|
|
|
priv->config.flow_prio = err;
|
2018-07-10 16:04:50 +00:00
|
|
|
/*
|
|
|
|
* Once the device is added to the list of memory event
|
|
|
|
* callback, its global MR cache table cannot be expanded
|
|
|
|
* on the fly because of deadlock. If it overflows, lookup
|
|
|
|
* should be done by searching MR list linearly, which is slow.
|
|
|
|
*/
|
|
|
|
err = mlx5_mr_btree_init(&priv->mr.cache,
|
|
|
|
MLX5_MR_BTREE_CACHE_N * 2,
|
|
|
|
eth_dev->device->numa_node);
|
|
|
|
if (err) {
|
|
|
|
err = rte_errno;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
/* Add device to memory callback list. */
|
|
|
|
rte_rwlock_write_lock(&mlx5_shared_data->mem_event_rwlock);
|
|
|
|
LIST_INSERT_HEAD(&mlx5_shared_data->mem_event_cb_list,
|
|
|
|
priv, mem_event_cb);
|
|
|
|
rte_rwlock_write_unlock(&mlx5_shared_data->mem_event_rwlock);
|
|
|
|
return eth_dev;
|
2015-10-30 18:52:30 +00:00
|
|
|
error:
|
2018-07-10 16:04:52 +00:00
|
|
|
if (priv) {
|
|
|
|
if (priv->nl_socket_route >= 0)
|
|
|
|
close(priv->nl_socket_route);
|
|
|
|
if (priv->nl_socket_rdma >= 0)
|
|
|
|
close(priv->nl_socket_rdma);
|
2018-07-10 16:04:54 +00:00
|
|
|
if (own_domain_id)
|
|
|
|
claim_zero(rte_eth_switch_domain_free(priv->domain_id));
|
2018-07-10 16:04:46 +00:00
|
|
|
rte_free(priv);
|
2018-07-10 16:04:52 +00:00
|
|
|
}
|
2018-07-10 16:04:46 +00:00
|
|
|
if (pd)
|
|
|
|
claim_zero(mlx5_glue->dealloc_pd(pd));
|
|
|
|
if (eth_dev)
|
|
|
|
rte_eth_dev_release_port(eth_dev);
|
2018-07-10 16:04:44 +00:00
|
|
|
if (ctx)
|
|
|
|
claim_zero(mlx5_glue->close_device(ctx));
|
2018-07-10 16:04:48 +00:00
|
|
|
assert(err > 0);
|
|
|
|
rte_errno = err;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-07-10 16:04:56 +00:00
|
|
|
/** Data associated with devices to spawn. */
|
|
|
|
struct mlx5_dev_spawn_data {
|
|
|
|
unsigned int ifindex; /**< Network interface index. */
|
|
|
|
struct mlx5_switch_info info; /**< Switch information. */
|
|
|
|
struct ibv_device *ibv_dev; /**< Associated IB device. */
|
|
|
|
struct rte_eth_dev *eth_dev; /**< Associated Ethernet device. */
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Comparison callback to sort device data.
|
|
|
|
*
|
|
|
|
* This is meant to be used with qsort().
|
|
|
|
*
|
|
|
|
* @param a[in]
|
|
|
|
* Pointer to pointer to first data object.
|
|
|
|
* @param b[in]
|
|
|
|
* Pointer to pointer to second data object.
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* 0 if both objects are equal, less than 0 if the first argument is less
|
|
|
|
* than the second, greater than 0 otherwise.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mlx5_dev_spawn_data_cmp(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const struct mlx5_switch_info *si_a =
|
|
|
|
&((const struct mlx5_dev_spawn_data *)a)->info;
|
|
|
|
const struct mlx5_switch_info *si_b =
|
|
|
|
&((const struct mlx5_dev_spawn_data *)b)->info;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Master device first. */
|
|
|
|
ret = si_b->master - si_a->master;
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
/* Then representor devices. */
|
|
|
|
ret = si_b->representor - si_a->representor;
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
/* Unidentified devices come last in no specific order. */
|
|
|
|
if (!si_a->representor)
|
|
|
|
return 0;
|
|
|
|
/* Order representors by name. */
|
|
|
|
return si_a->port_name - si_b->port_name;
|
|
|
|
}
|
|
|
|
|
2018-07-10 16:04:48 +00:00
|
|
|
/**
|
|
|
|
* DPDK callback to register a PCI device.
|
|
|
|
*
|
2018-07-10 16:04:54 +00:00
|
|
|
* This function spawns Ethernet devices out of a given PCI device.
|
2018-07-10 16:04:48 +00:00
|
|
|
*
|
|
|
|
* @param[in] pci_drv
|
|
|
|
* PCI driver structure (mlx5_driver).
|
|
|
|
* @param[in] pci_dev
|
|
|
|
* PCI device information.
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* 0 on success, a negative errno value otherwise and rte_errno is set.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mlx5_pci_probe(struct rte_pci_driver *pci_drv __rte_unused,
|
|
|
|
struct rte_pci_device *pci_dev)
|
|
|
|
{
|
|
|
|
struct ibv_device **ibv_list;
|
2018-07-10 16:04:52 +00:00
|
|
|
unsigned int n = 0;
|
2018-07-10 16:04:48 +00:00
|
|
|
int vf;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
assert(pci_drv == &mlx5_driver);
|
|
|
|
errno = 0;
|
|
|
|
ibv_list = mlx5_glue->get_device_list(&ret);
|
|
|
|
if (!ibv_list) {
|
|
|
|
rte_errno = errno ? errno : ENOSYS;
|
|
|
|
DRV_LOG(ERR, "cannot list devices, is ib_uverbs loaded?");
|
2018-03-05 12:21:06 +00:00
|
|
|
return -rte_errno;
|
|
|
|
}
|
2018-07-10 16:04:52 +00:00
|
|
|
|
|
|
|
struct ibv_device *ibv_match[ret + 1];
|
|
|
|
|
2018-07-10 16:04:48 +00:00
|
|
|
while (ret-- > 0) {
|
|
|
|
struct rte_pci_addr pci_addr;
|
|
|
|
|
|
|
|
DRV_LOG(DEBUG, "checking device \"%s\"", ibv_list[ret]->name);
|
|
|
|
if (mlx5_ibv_device_to_pci_addr(ibv_list[ret], &pci_addr))
|
|
|
|
continue;
|
|
|
|
if (pci_dev->addr.domain != pci_addr.domain ||
|
|
|
|
pci_dev->addr.bus != pci_addr.bus ||
|
|
|
|
pci_dev->addr.devid != pci_addr.devid ||
|
|
|
|
pci_dev->addr.function != pci_addr.function)
|
|
|
|
continue;
|
2018-07-10 16:04:52 +00:00
|
|
|
DRV_LOG(INFO, "PCI information matches for device \"%s\"",
|
2018-07-10 16:04:48 +00:00
|
|
|
ibv_list[ret]->name);
|
2018-07-10 16:04:52 +00:00
|
|
|
ibv_match[n++] = ibv_list[ret];
|
|
|
|
}
|
|
|
|
ibv_match[n] = NULL;
|
|
|
|
|
2018-07-10 16:04:56 +00:00
|
|
|
struct mlx5_dev_spawn_data list[n];
|
2018-07-10 16:04:52 +00:00
|
|
|
int nl_route = n ? mlx5_nl_init(0, NETLINK_ROUTE) : -1;
|
|
|
|
int nl_rdma = n ? mlx5_nl_init(0, NETLINK_RDMA) : -1;
|
|
|
|
unsigned int i;
|
2018-07-10 16:04:54 +00:00
|
|
|
unsigned int u;
|
2018-07-10 16:04:52 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The existence of several matching entries (n > 1) means port
|
|
|
|
* representors have been instantiated. No existing Verbs call nor
|
|
|
|
* /sys entries can tell them apart, this can only be done through
|
|
|
|
* Netlink calls assuming kernel drivers are recent enough to
|
|
|
|
* support them.
|
|
|
|
*
|
|
|
|
* In the event of identification failure through Netlink, either:
|
|
|
|
*
|
|
|
|
* 1. No device matches (n == 0), complain and bail out.
|
|
|
|
* 2. A single IB device matches (n == 1) and is not a representor,
|
|
|
|
* assume no switch support.
|
|
|
|
* 3. Otherwise no safe assumptions can be made; complain louder and
|
|
|
|
* bail out.
|
|
|
|
*/
|
|
|
|
for (i = 0; i != n; ++i) {
|
2018-07-10 16:04:56 +00:00
|
|
|
list[i].ibv_dev = ibv_match[i];
|
|
|
|
list[i].eth_dev = NULL;
|
2018-07-10 16:04:52 +00:00
|
|
|
if (nl_rdma < 0)
|
2018-07-10 16:04:56 +00:00
|
|
|
list[i].ifindex = 0;
|
2018-07-10 16:04:52 +00:00
|
|
|
else
|
2018-07-10 16:04:56 +00:00
|
|
|
list[i].ifindex = mlx5_nl_ifindex
|
|
|
|
(nl_rdma, list[i].ibv_dev->name);
|
2018-07-10 16:04:52 +00:00
|
|
|
if (nl_route < 0 ||
|
2018-07-10 16:04:56 +00:00
|
|
|
!list[i].ifindex ||
|
|
|
|
mlx5_nl_switch_info(nl_route, list[i].ifindex,
|
|
|
|
&list[i].info)) {
|
|
|
|
list[i].ifindex = 0;
|
|
|
|
memset(&list[i].info, 0, sizeof(list[i].info));
|
2018-07-10 16:04:52 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nl_rdma >= 0)
|
|
|
|
close(nl_rdma);
|
|
|
|
if (nl_route >= 0)
|
|
|
|
close(nl_route);
|
2018-07-10 16:04:54 +00:00
|
|
|
/* Count unidentified devices. */
|
|
|
|
for (u = 0, i = 0; i != n; ++i)
|
2018-07-10 16:04:56 +00:00
|
|
|
if (!list[i].info.master && !list[i].info.representor)
|
2018-07-10 16:04:54 +00:00
|
|
|
++u;
|
|
|
|
if (u) {
|
|
|
|
if (n == 1 && u == 1) {
|
2018-07-10 16:04:52 +00:00
|
|
|
/* Case #2. */
|
|
|
|
DRV_LOG(INFO, "no switch support detected");
|
|
|
|
} else {
|
|
|
|
/* Case #3. */
|
|
|
|
DRV_LOG(ERR,
|
|
|
|
"unable to tell which of the matching devices"
|
|
|
|
" is the master (lack of kernel support?)");
|
|
|
|
n = 0;
|
|
|
|
}
|
|
|
|
}
|
2018-07-10 16:04:56 +00:00
|
|
|
/*
|
|
|
|
* Sort list to probe devices in natural order for users convenience
|
|
|
|
* (i.e. master first, then representors from lowest to highest ID).
|
|
|
|
*/
|
|
|
|
if (n)
|
|
|
|
qsort(list, n, sizeof(*list), mlx5_dev_spawn_data_cmp);
|
2018-07-10 16:04:48 +00:00
|
|
|
switch (pci_dev->id.device_id) {
|
|
|
|
case PCI_DEVICE_ID_MELLANOX_CONNECTX4VF:
|
|
|
|
case PCI_DEVICE_ID_MELLANOX_CONNECTX4LXVF:
|
|
|
|
case PCI_DEVICE_ID_MELLANOX_CONNECTX5VF:
|
|
|
|
case PCI_DEVICE_ID_MELLANOX_CONNECTX5EXVF:
|
|
|
|
vf = 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
vf = 0;
|
|
|
|
}
|
2018-07-10 16:04:54 +00:00
|
|
|
for (i = 0; i != n; ++i) {
|
|
|
|
uint32_t restore;
|
|
|
|
|
2018-07-10 16:04:56 +00:00
|
|
|
list[i].eth_dev = mlx5_dev_spawn
|
|
|
|
(&pci_dev->device, list[i].ibv_dev, vf, &list[i].info);
|
2018-07-10 16:04:58 +00:00
|
|
|
if (!list[i].eth_dev) {
|
|
|
|
if (rte_errno != EBUSY)
|
|
|
|
break;
|
|
|
|
/* Device is disabled, ignore it. */
|
|
|
|
continue;
|
|
|
|
}
|
2018-07-10 16:04:56 +00:00
|
|
|
restore = list[i].eth_dev->data->dev_flags;
|
|
|
|
rte_eth_copy_pci_info(list[i].eth_dev, pci_dev);
|
2018-07-10 16:04:54 +00:00
|
|
|
/* Restore non-PCI flags cleared by the above call. */
|
2018-07-10 16:04:56 +00:00
|
|
|
list[i].eth_dev->data->dev_flags |= restore;
|
|
|
|
rte_eth_dev_probing_finish(list[i].eth_dev);
|
2018-07-10 16:04:54 +00:00
|
|
|
}
|
2018-07-10 16:04:48 +00:00
|
|
|
mlx5_glue->free_device_list(ibv_list);
|
2018-07-10 16:04:52 +00:00
|
|
|
if (!n) {
|
2018-07-10 16:04:48 +00:00
|
|
|
DRV_LOG(WARNING,
|
|
|
|
"no Verbs device matches PCI device " PCI_PRI_FMT ","
|
|
|
|
" are kernel drivers loaded?",
|
|
|
|
pci_dev->addr.domain, pci_dev->addr.bus,
|
|
|
|
pci_dev->addr.devid, pci_dev->addr.function);
|
|
|
|
rte_errno = ENOENT;
|
|
|
|
ret = -rte_errno;
|
2018-07-10 16:04:54 +00:00
|
|
|
} else if (i != n) {
|
2018-07-10 16:04:48 +00:00
|
|
|
DRV_LOG(ERR,
|
|
|
|
"probe of PCI device " PCI_PRI_FMT " aborted after"
|
|
|
|
" encountering an error: %s",
|
|
|
|
pci_dev->addr.domain, pci_dev->addr.bus,
|
|
|
|
pci_dev->addr.devid, pci_dev->addr.function,
|
|
|
|
strerror(rte_errno));
|
|
|
|
ret = -rte_errno;
|
2018-07-10 16:04:54 +00:00
|
|
|
/* Roll back. */
|
|
|
|
while (i--) {
|
2018-07-10 16:04:58 +00:00
|
|
|
if (!list[i].eth_dev)
|
|
|
|
continue;
|
2018-07-10 16:04:56 +00:00
|
|
|
mlx5_dev_close(list[i].eth_dev);
|
2018-07-10 16:04:54 +00:00
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY)
|
2018-07-10 16:04:56 +00:00
|
|
|
rte_free(list[i].eth_dev->data->dev_private);
|
|
|
|
claim_zero(rte_eth_dev_release_port(list[i].eth_dev));
|
2018-07-10 16:04:54 +00:00
|
|
|
}
|
|
|
|
/* Restore original error. */
|
|
|
|
rte_errno = -ret;
|
2018-07-10 16:04:48 +00:00
|
|
|
} else {
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
return ret;
|
2015-10-30 18:52:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct rte_pci_id mlx5_pci_id_map[] = {
|
|
|
|
{
|
2016-06-24 13:17:40 +00:00
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX4)
|
2015-10-30 18:52:30 +00:00
|
|
|
},
|
|
|
|
{
|
2016-06-24 13:17:40 +00:00
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX4VF)
|
2015-10-30 18:52:30 +00:00
|
|
|
},
|
|
|
|
{
|
2016-06-24 13:17:40 +00:00
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX4LX)
|
2015-10-30 18:52:30 +00:00
|
|
|
},
|
|
|
|
{
|
2016-06-24 13:17:40 +00:00
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX4LXVF)
|
2015-10-30 18:52:30 +00:00
|
|
|
},
|
2017-01-06 00:49:31 +00:00
|
|
|
{
|
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX5)
|
|
|
|
},
|
|
|
|
{
|
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX5VF)
|
|
|
|
},
|
|
|
|
{
|
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX5EX)
|
|
|
|
},
|
|
|
|
{
|
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX5EXVF)
|
|
|
|
},
|
2018-05-15 06:12:50 +00:00
|
|
|
{
|
|
|
|
RTE_PCI_DEVICE(PCI_VENDOR_ID_MELLANOX,
|
|
|
|
PCI_DEVICE_ID_MELLANOX_CONNECTX5BF)
|
|
|
|
},
|
2015-10-30 18:52:30 +00:00
|
|
|
{
|
|
|
|
.vendor_id = 0
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-04-11 15:44:24 +00:00
|
|
|
static struct rte_pci_driver mlx5_driver = {
|
|
|
|
.driver = {
|
|
|
|
.name = MLX5_DRIVER_NAME
|
2015-10-30 18:52:30 +00:00
|
|
|
},
|
2017-04-11 15:44:24 +00:00
|
|
|
.id_table = mlx5_pci_id_map,
|
|
|
|
.probe = mlx5_pci_probe,
|
2017-09-08 10:47:45 +00:00
|
|
|
.drv_flags = RTE_PCI_DRV_INTR_LSC | RTE_PCI_DRV_INTR_RMV,
|
2015-10-30 18:52:30 +00:00
|
|
|
};
|
|
|
|
|
2018-01-30 15:34:58 +00:00
|
|
|
#ifdef RTE_LIBRTE_MLX5_DLOPEN_DEPS
|
|
|
|
|
2018-03-02 14:15:17 +00:00
|
|
|
/**
|
|
|
|
* Suffix RTE_EAL_PMD_PATH with "-glue".
|
|
|
|
*
|
|
|
|
* This function performs a sanity check on RTE_EAL_PMD_PATH before
|
|
|
|
* suffixing its last component.
|
|
|
|
*
|
|
|
|
* @param buf[out]
|
|
|
|
* Output buffer, should be large enough otherwise NULL is returned.
|
|
|
|
* @param size
|
|
|
|
* Size of @p out.
|
|
|
|
*
|
|
|
|
* @return
|
|
|
|
* Pointer to @p buf or @p NULL in case suffix cannot be appended.
|
|
|
|
*/
|
|
|
|
static char *
|
|
|
|
mlx5_glue_path(char *buf, size_t size)
|
|
|
|
{
|
|
|
|
static const char *const bad[] = { "/", ".", "..", NULL };
|
|
|
|
const char *path = RTE_EAL_PMD_PATH;
|
|
|
|
size_t len = strlen(path);
|
|
|
|
size_t off;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
while (len && path[len - 1] == '/')
|
|
|
|
--len;
|
|
|
|
for (off = len; off && path[off - 1] != '/'; --off)
|
|
|
|
;
|
|
|
|
for (i = 0; bad[i]; ++i)
|
|
|
|
if (!strncmp(path + off, bad[i], (int)(len - off)))
|
|
|
|
goto error;
|
|
|
|
i = snprintf(buf, size, "%.*s-glue", (int)len, path);
|
|
|
|
if (i == -1 || (size_t)i >= size)
|
|
|
|
goto error;
|
|
|
|
return buf;
|
|
|
|
error:
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(ERR,
|
|
|
|
"unable to append \"-glue\" to last component of"
|
|
|
|
" RTE_EAL_PMD_PATH (\"" RTE_EAL_PMD_PATH "\"),"
|
|
|
|
" please re-configure DPDK");
|
2018-03-02 14:15:17 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-01-30 15:34:58 +00:00
|
|
|
/**
|
|
|
|
* Initialization routine for run-time dependency on rdma-core.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mlx5_glue_init(void)
|
|
|
|
{
|
2018-03-02 14:15:17 +00:00
|
|
|
char glue_path[sizeof(RTE_EAL_PMD_PATH) - 1 + sizeof("-glue")];
|
2018-02-02 16:46:18 +00:00
|
|
|
const char *path[] = {
|
|
|
|
/*
|
|
|
|
* A basic security check is necessary before trusting
|
|
|
|
* MLX5_GLUE_PATH, which may override RTE_EAL_PMD_PATH.
|
|
|
|
*/
|
|
|
|
(geteuid() == getuid() && getegid() == getgid() ?
|
|
|
|
getenv("MLX5_GLUE_PATH") : NULL),
|
2018-03-02 14:15:17 +00:00
|
|
|
/*
|
|
|
|
* When RTE_EAL_PMD_PATH is set, use its glue-suffixed
|
|
|
|
* variant, otherwise let dlopen() look up libraries on its
|
|
|
|
* own.
|
|
|
|
*/
|
|
|
|
(*RTE_EAL_PMD_PATH ?
|
|
|
|
mlx5_glue_path(glue_path, sizeof(glue_path)) : ""),
|
2018-02-02 16:46:18 +00:00
|
|
|
};
|
|
|
|
unsigned int i = 0;
|
2018-01-30 15:34:58 +00:00
|
|
|
void *handle = NULL;
|
|
|
|
void **sym;
|
|
|
|
const char *dlmsg;
|
|
|
|
|
2018-02-02 16:46:18 +00:00
|
|
|
while (!handle && i != RTE_DIM(path)) {
|
|
|
|
const char *end;
|
|
|
|
size_t len;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!path[i]) {
|
|
|
|
++i;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
end = strpbrk(path[i], ":;");
|
|
|
|
if (!end)
|
|
|
|
end = path[i] + strlen(path[i]);
|
|
|
|
len = end - path[i];
|
|
|
|
ret = 0;
|
|
|
|
do {
|
|
|
|
char name[ret + 1];
|
|
|
|
|
|
|
|
ret = snprintf(name, sizeof(name), "%.*s%s" MLX5_GLUE,
|
|
|
|
(int)len, path[i],
|
|
|
|
(!len || *(end - 1) == '/') ? "" : "/");
|
|
|
|
if (ret == -1)
|
|
|
|
break;
|
|
|
|
if (sizeof(name) != (size_t)ret + 1)
|
|
|
|
continue;
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(DEBUG, "looking for rdma-core glue as \"%s\"",
|
|
|
|
name);
|
2018-02-02 16:46:18 +00:00
|
|
|
handle = dlopen(name, RTLD_LAZY);
|
|
|
|
break;
|
|
|
|
} while (1);
|
|
|
|
path[i] = end + 1;
|
|
|
|
if (!*end)
|
|
|
|
++i;
|
|
|
|
}
|
2018-01-30 15:34:58 +00:00
|
|
|
if (!handle) {
|
|
|
|
rte_errno = EINVAL;
|
|
|
|
dlmsg = dlerror();
|
|
|
|
if (dlmsg)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING, "cannot load glue library: %s", dlmsg);
|
2018-01-30 15:34:58 +00:00
|
|
|
goto glue_error;
|
|
|
|
}
|
|
|
|
sym = dlsym(handle, "mlx5_glue");
|
|
|
|
if (!sym || !*sym) {
|
|
|
|
rte_errno = EINVAL;
|
|
|
|
dlmsg = dlerror();
|
|
|
|
if (dlmsg)
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(ERR, "cannot resolve glue symbol: %s", dlmsg);
|
2018-01-30 15:34:58 +00:00
|
|
|
goto glue_error;
|
|
|
|
}
|
|
|
|
mlx5_glue = *sym;
|
|
|
|
return 0;
|
|
|
|
glue_error:
|
|
|
|
if (handle)
|
|
|
|
dlclose(handle);
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(WARNING,
|
|
|
|
"cannot initialize PMD due to missing run-time dependency on"
|
|
|
|
" rdma-core libraries (libibverbs, libmlx5)");
|
2018-01-30 15:34:58 +00:00
|
|
|
return -rte_errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2015-10-30 18:52:30 +00:00
|
|
|
/**
|
|
|
|
* Driver initialization routine.
|
|
|
|
*/
|
2018-06-18 12:32:21 +00:00
|
|
|
RTE_INIT(rte_mlx5_pmd_init)
|
2015-10-30 18:52:30 +00:00
|
|
|
{
|
2018-06-13 18:46:26 +00:00
|
|
|
/* Initialize driver log type. */
|
|
|
|
mlx5_logtype = rte_log_register("pmd.net.mlx5");
|
|
|
|
if (mlx5_logtype >= 0)
|
|
|
|
rte_log_set_level(mlx5_logtype, RTE_LOG_NOTICE);
|
|
|
|
|
2018-04-08 12:41:20 +00:00
|
|
|
/* Build the static tables for Verbs conversion. */
|
2017-07-26 19:29:33 +00:00
|
|
|
mlx5_set_ptype_table();
|
2018-04-08 12:41:20 +00:00
|
|
|
mlx5_set_cksum_table();
|
|
|
|
mlx5_set_swp_types_table();
|
2015-10-30 18:52:30 +00:00
|
|
|
/*
|
|
|
|
* RDMAV_HUGEPAGES_SAFE tells ibv_fork_init() we intend to use
|
|
|
|
* huge pages. Calling ibv_fork_init() during init allows
|
|
|
|
* applications to use fork() safely for purposes other than
|
|
|
|
* using this PMD, which is not supported in forked processes.
|
|
|
|
*/
|
|
|
|
setenv("RDMAV_HUGEPAGES_SAFE", "1", 1);
|
2017-10-09 18:46:58 +00:00
|
|
|
/* Match the size of Rx completion entry to the size of a cacheline. */
|
|
|
|
if (RTE_CACHE_LINE_SIZE == 128)
|
|
|
|
setenv("MLX5_CQE_SIZE", "128", 0);
|
2018-07-02 05:13:18 +00:00
|
|
|
/*
|
|
|
|
* MLX5_DEVICE_FATAL_CLEANUP tells ibv_destroy functions to
|
|
|
|
* cleanup all the Verbs resources even when the device was removed.
|
|
|
|
*/
|
|
|
|
setenv("MLX5_DEVICE_FATAL_CLEANUP", "1", 1);
|
2018-01-30 15:34:58 +00:00
|
|
|
#ifdef RTE_LIBRTE_MLX5_DLOPEN_DEPS
|
|
|
|
if (mlx5_glue_init())
|
|
|
|
return;
|
|
|
|
assert(mlx5_glue);
|
2018-02-02 16:46:12 +00:00
|
|
|
#endif
|
|
|
|
#ifndef NDEBUG
|
|
|
|
/* Glue structure must not contain any NULL pointers. */
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i != sizeof(*mlx5_glue) / sizeof(void *); ++i)
|
|
|
|
assert(((const void *const *)mlx5_glue)[i]);
|
|
|
|
}
|
2018-01-30 15:34:58 +00:00
|
|
|
#endif
|
2018-02-02 16:46:16 +00:00
|
|
|
if (strcmp(mlx5_glue->version, MLX5_GLUE_VERSION)) {
|
2018-03-13 09:23:56 +00:00
|
|
|
DRV_LOG(ERR,
|
|
|
|
"rdma-core glue \"%s\" mismatch: \"%s\" is required",
|
|
|
|
mlx5_glue->version, MLX5_GLUE_VERSION);
|
2018-02-02 16:46:16 +00:00
|
|
|
return;
|
|
|
|
}
|
2018-01-30 15:34:56 +00:00
|
|
|
mlx5_glue->fork_init();
|
2017-05-04 14:48:59 +00:00
|
|
|
rte_pci_register(&mlx5_driver);
|
2015-10-30 18:52:30 +00:00
|
|
|
}
|
|
|
|
|
2016-10-10 05:43:15 +00:00
|
|
|
RTE_PMD_EXPORT_NAME(net_mlx5, __COUNTER__);
|
|
|
|
RTE_PMD_REGISTER_PCI_TABLE(net_mlx5, mlx5_pci_id_map);
|
2016-12-15 13:46:39 +00:00
|
|
|
RTE_PMD_REGISTER_KMOD_DEP(net_mlx5, "* ib_uverbs & mlx5_core & mlx5_ib");
|