net/mlx5: Maintain separate page trees for ECPF and PF functions
Pages for the host PF and ECPF were stored in the same tree, so the ECPF pages were being freed along with the host PF's when the host driver unloaded. Combine the function ID and ECPF flag to use as an index into the x-array containing the trees to get a different tree for the host PF and ECPF. Fixes: c6168161f693 ("net/mlx5: Add support for release all pages event") Signed-off-by: Daniel Jurgens <danielj@nvidia.com> Signed-off-by: Saeed Mahameed <saeedm@nvidia.com>
This commit is contained in:
parent
45c9a30835
commit
0aa128475d
@ -58,7 +58,7 @@ struct fw_page {
|
|||||||
struct rb_node rb_node;
|
struct rb_node rb_node;
|
||||||
u64 addr;
|
u64 addr;
|
||||||
struct page *page;
|
struct page *page;
|
||||||
u16 func_id;
|
u32 function;
|
||||||
unsigned long bitmask;
|
unsigned long bitmask;
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
unsigned free_count;
|
unsigned free_count;
|
||||||
@ -74,12 +74,17 @@ enum {
|
|||||||
MLX5_NUM_4K_IN_PAGE = PAGE_SIZE / MLX5_ADAPTER_PAGE_SIZE,
|
MLX5_NUM_4K_IN_PAGE = PAGE_SIZE / MLX5_ADAPTER_PAGE_SIZE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct rb_root *page_root_per_func_id(struct mlx5_core_dev *dev, u16 func_id)
|
static u32 get_function(u16 func_id, bool ec_function)
|
||||||
|
{
|
||||||
|
return func_id & (ec_function << 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct rb_root *page_root_per_function(struct mlx5_core_dev *dev, u32 function)
|
||||||
{
|
{
|
||||||
struct rb_root *root;
|
struct rb_root *root;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
root = xa_load(&dev->priv.page_root_xa, func_id);
|
root = xa_load(&dev->priv.page_root_xa, function);
|
||||||
if (root)
|
if (root)
|
||||||
return root;
|
return root;
|
||||||
|
|
||||||
@ -87,7 +92,7 @@ static struct rb_root *page_root_per_func_id(struct mlx5_core_dev *dev, u16 func
|
|||||||
if (!root)
|
if (!root)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
err = xa_insert(&dev->priv.page_root_xa, func_id, root, GFP_KERNEL);
|
err = xa_insert(&dev->priv.page_root_xa, function, root, GFP_KERNEL);
|
||||||
if (err) {
|
if (err) {
|
||||||
kfree(root);
|
kfree(root);
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
@ -98,7 +103,7 @@ static struct rb_root *page_root_per_func_id(struct mlx5_core_dev *dev, u16 func
|
|||||||
return root;
|
return root;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int insert_page(struct mlx5_core_dev *dev, u64 addr, struct page *page, u16 func_id)
|
static int insert_page(struct mlx5_core_dev *dev, u64 addr, struct page *page, u32 function)
|
||||||
{
|
{
|
||||||
struct rb_node *parent = NULL;
|
struct rb_node *parent = NULL;
|
||||||
struct rb_root *root;
|
struct rb_root *root;
|
||||||
@ -107,7 +112,7 @@ static int insert_page(struct mlx5_core_dev *dev, u64 addr, struct page *page, u
|
|||||||
struct fw_page *tfp;
|
struct fw_page *tfp;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
root = page_root_per_func_id(dev, func_id);
|
root = page_root_per_function(dev, function);
|
||||||
if (IS_ERR(root))
|
if (IS_ERR(root))
|
||||||
return PTR_ERR(root);
|
return PTR_ERR(root);
|
||||||
|
|
||||||
@ -130,7 +135,7 @@ static int insert_page(struct mlx5_core_dev *dev, u64 addr, struct page *page, u
|
|||||||
|
|
||||||
nfp->addr = addr;
|
nfp->addr = addr;
|
||||||
nfp->page = page;
|
nfp->page = page;
|
||||||
nfp->func_id = func_id;
|
nfp->function = function;
|
||||||
nfp->free_count = MLX5_NUM_4K_IN_PAGE;
|
nfp->free_count = MLX5_NUM_4K_IN_PAGE;
|
||||||
for (i = 0; i < MLX5_NUM_4K_IN_PAGE; i++)
|
for (i = 0; i < MLX5_NUM_4K_IN_PAGE; i++)
|
||||||
set_bit(i, &nfp->bitmask);
|
set_bit(i, &nfp->bitmask);
|
||||||
@ -143,14 +148,14 @@ static int insert_page(struct mlx5_core_dev *dev, u64 addr, struct page *page, u
|
|||||||
}
|
}
|
||||||
|
|
||||||
static struct fw_page *find_fw_page(struct mlx5_core_dev *dev, u64 addr,
|
static struct fw_page *find_fw_page(struct mlx5_core_dev *dev, u64 addr,
|
||||||
u32 func_id)
|
u32 function)
|
||||||
{
|
{
|
||||||
struct fw_page *result = NULL;
|
struct fw_page *result = NULL;
|
||||||
struct rb_root *root;
|
struct rb_root *root;
|
||||||
struct rb_node *tmp;
|
struct rb_node *tmp;
|
||||||
struct fw_page *tfp;
|
struct fw_page *tfp;
|
||||||
|
|
||||||
root = xa_load(&dev->priv.page_root_xa, func_id);
|
root = xa_load(&dev->priv.page_root_xa, function);
|
||||||
if (WARN_ON_ONCE(!root))
|
if (WARN_ON_ONCE(!root))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
@ -194,14 +199,14 @@ static int mlx5_cmd_query_pages(struct mlx5_core_dev *dev, u16 *func_id,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int alloc_4k(struct mlx5_core_dev *dev, u64 *addr, u16 func_id)
|
static int alloc_4k(struct mlx5_core_dev *dev, u64 *addr, u32 function)
|
||||||
{
|
{
|
||||||
struct fw_page *fp = NULL;
|
struct fw_page *fp = NULL;
|
||||||
struct fw_page *iter;
|
struct fw_page *iter;
|
||||||
unsigned n;
|
unsigned n;
|
||||||
|
|
||||||
list_for_each_entry(iter, &dev->priv.free_list, list) {
|
list_for_each_entry(iter, &dev->priv.free_list, list) {
|
||||||
if (iter->func_id != func_id)
|
if (iter->function != function)
|
||||||
continue;
|
continue;
|
||||||
fp = iter;
|
fp = iter;
|
||||||
}
|
}
|
||||||
@ -231,7 +236,7 @@ static void free_fwp(struct mlx5_core_dev *dev, struct fw_page *fwp,
|
|||||||
{
|
{
|
||||||
struct rb_root *root;
|
struct rb_root *root;
|
||||||
|
|
||||||
root = xa_load(&dev->priv.page_root_xa, fwp->func_id);
|
root = xa_load(&dev->priv.page_root_xa, fwp->function);
|
||||||
if (WARN_ON_ONCE(!root))
|
if (WARN_ON_ONCE(!root))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -244,12 +249,12 @@ static void free_fwp(struct mlx5_core_dev *dev, struct fw_page *fwp,
|
|||||||
kfree(fwp);
|
kfree(fwp);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void free_4k(struct mlx5_core_dev *dev, u64 addr, u32 func_id)
|
static void free_4k(struct mlx5_core_dev *dev, u64 addr, u32 function)
|
||||||
{
|
{
|
||||||
struct fw_page *fwp;
|
struct fw_page *fwp;
|
||||||
int n;
|
int n;
|
||||||
|
|
||||||
fwp = find_fw_page(dev, addr & MLX5_U64_4K_PAGE_MASK, func_id);
|
fwp = find_fw_page(dev, addr & MLX5_U64_4K_PAGE_MASK, function);
|
||||||
if (!fwp) {
|
if (!fwp) {
|
||||||
mlx5_core_warn_rl(dev, "page not found\n");
|
mlx5_core_warn_rl(dev, "page not found\n");
|
||||||
return;
|
return;
|
||||||
@ -263,7 +268,7 @@ static void free_4k(struct mlx5_core_dev *dev, u64 addr, u32 func_id)
|
|||||||
list_add(&fwp->list, &dev->priv.free_list);
|
list_add(&fwp->list, &dev->priv.free_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int alloc_system_page(struct mlx5_core_dev *dev, u16 func_id)
|
static int alloc_system_page(struct mlx5_core_dev *dev, u32 function)
|
||||||
{
|
{
|
||||||
struct device *device = mlx5_core_dma_dev(dev);
|
struct device *device = mlx5_core_dma_dev(dev);
|
||||||
int nid = dev_to_node(device);
|
int nid = dev_to_node(device);
|
||||||
@ -291,7 +296,7 @@ map:
|
|||||||
goto map;
|
goto map;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = insert_page(dev, addr, page, func_id);
|
err = insert_page(dev, addr, page, function);
|
||||||
if (err) {
|
if (err) {
|
||||||
mlx5_core_err(dev, "failed to track allocated page\n");
|
mlx5_core_err(dev, "failed to track allocated page\n");
|
||||||
dma_unmap_page(device, addr, PAGE_SIZE, DMA_BIDIRECTIONAL);
|
dma_unmap_page(device, addr, PAGE_SIZE, DMA_BIDIRECTIONAL);
|
||||||
@ -328,6 +333,7 @@ static void page_notify_fail(struct mlx5_core_dev *dev, u16 func_id,
|
|||||||
static int give_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
|
static int give_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
|
||||||
int notify_fail, bool ec_function)
|
int notify_fail, bool ec_function)
|
||||||
{
|
{
|
||||||
|
u32 function = get_function(func_id, ec_function);
|
||||||
u32 out[MLX5_ST_SZ_DW(manage_pages_out)] = {0};
|
u32 out[MLX5_ST_SZ_DW(manage_pages_out)] = {0};
|
||||||
int inlen = MLX5_ST_SZ_BYTES(manage_pages_in);
|
int inlen = MLX5_ST_SZ_BYTES(manage_pages_in);
|
||||||
u64 addr;
|
u64 addr;
|
||||||
@ -345,10 +351,10 @@ static int give_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
|
|||||||
|
|
||||||
for (i = 0; i < npages; i++) {
|
for (i = 0; i < npages; i++) {
|
||||||
retry:
|
retry:
|
||||||
err = alloc_4k(dev, &addr, func_id);
|
err = alloc_4k(dev, &addr, function);
|
||||||
if (err) {
|
if (err) {
|
||||||
if (err == -ENOMEM)
|
if (err == -ENOMEM)
|
||||||
err = alloc_system_page(dev, func_id);
|
err = alloc_system_page(dev, function);
|
||||||
if (err)
|
if (err)
|
||||||
goto out_4k;
|
goto out_4k;
|
||||||
|
|
||||||
@ -384,7 +390,7 @@ retry:
|
|||||||
|
|
||||||
out_4k:
|
out_4k:
|
||||||
for (i--; i >= 0; i--)
|
for (i--; i >= 0; i--)
|
||||||
free_4k(dev, MLX5_GET64(manage_pages_in, in, pas[i]), func_id);
|
free_4k(dev, MLX5_GET64(manage_pages_in, in, pas[i]), function);
|
||||||
out_free:
|
out_free:
|
||||||
kvfree(in);
|
kvfree(in);
|
||||||
if (notify_fail)
|
if (notify_fail)
|
||||||
@ -392,14 +398,15 @@ out_free:
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void release_all_pages(struct mlx5_core_dev *dev, u32 func_id,
|
static void release_all_pages(struct mlx5_core_dev *dev, u16 func_id,
|
||||||
bool ec_function)
|
bool ec_function)
|
||||||
{
|
{
|
||||||
|
u32 function = get_function(func_id, ec_function);
|
||||||
struct rb_root *root;
|
struct rb_root *root;
|
||||||
struct rb_node *p;
|
struct rb_node *p;
|
||||||
int npages = 0;
|
int npages = 0;
|
||||||
|
|
||||||
root = xa_load(&dev->priv.page_root_xa, func_id);
|
root = xa_load(&dev->priv.page_root_xa, function);
|
||||||
if (WARN_ON_ONCE(!root))
|
if (WARN_ON_ONCE(!root))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -446,6 +453,7 @@ static int reclaim_pages_cmd(struct mlx5_core_dev *dev,
|
|||||||
struct rb_root *root;
|
struct rb_root *root;
|
||||||
struct fw_page *fwp;
|
struct fw_page *fwp;
|
||||||
struct rb_node *p;
|
struct rb_node *p;
|
||||||
|
bool ec_function;
|
||||||
u32 func_id;
|
u32 func_id;
|
||||||
u32 npages;
|
u32 npages;
|
||||||
u32 i = 0;
|
u32 i = 0;
|
||||||
@ -456,8 +464,9 @@ static int reclaim_pages_cmd(struct mlx5_core_dev *dev,
|
|||||||
/* No hard feelings, we want our pages back! */
|
/* No hard feelings, we want our pages back! */
|
||||||
npages = MLX5_GET(manage_pages_in, in, input_num_entries);
|
npages = MLX5_GET(manage_pages_in, in, input_num_entries);
|
||||||
func_id = MLX5_GET(manage_pages_in, in, function_id);
|
func_id = MLX5_GET(manage_pages_in, in, function_id);
|
||||||
|
ec_function = MLX5_GET(manage_pages_in, in, embedded_cpu_function);
|
||||||
|
|
||||||
root = xa_load(&dev->priv.page_root_xa, func_id);
|
root = xa_load(&dev->priv.page_root_xa, get_function(func_id, ec_function));
|
||||||
if (WARN_ON_ONCE(!root))
|
if (WARN_ON_ONCE(!root))
|
||||||
return -EEXIST;
|
return -EEXIST;
|
||||||
|
|
||||||
@ -473,9 +482,10 @@ static int reclaim_pages_cmd(struct mlx5_core_dev *dev,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int reclaim_pages(struct mlx5_core_dev *dev, u32 func_id, int npages,
|
static int reclaim_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
|
||||||
int *nclaimed, bool ec_function)
|
int *nclaimed, bool ec_function)
|
||||||
{
|
{
|
||||||
|
u32 function = get_function(func_id, ec_function);
|
||||||
int outlen = MLX5_ST_SZ_BYTES(manage_pages_out);
|
int outlen = MLX5_ST_SZ_BYTES(manage_pages_out);
|
||||||
u32 in[MLX5_ST_SZ_DW(manage_pages_in)] = {};
|
u32 in[MLX5_ST_SZ_DW(manage_pages_in)] = {};
|
||||||
int num_claimed;
|
int num_claimed;
|
||||||
@ -514,7 +524,7 @@ static int reclaim_pages(struct mlx5_core_dev *dev, u32 func_id, int npages,
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < num_claimed; i++)
|
for (i = 0; i < num_claimed; i++)
|
||||||
free_4k(dev, MLX5_GET64(manage_pages_out, out, pas[i]), func_id);
|
free_4k(dev, MLX5_GET64(manage_pages_out, out, pas[i]), function);
|
||||||
|
|
||||||
if (nclaimed)
|
if (nclaimed)
|
||||||
*nclaimed = num_claimed;
|
*nclaimed = num_claimed;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user