net/mlx5: DR, Free unused buddy ICM memory
authorYevgeny Kliteynik <kliteyn@nvidia.com>
Mon, 14 Sep 2020 12:07:59 +0000 (15:07 +0300)
committerSaeed Mahameed <saeedm@nvidia.com>
Thu, 5 Nov 2020 20:09:30 +0000 (12:09 -0800)
Track buddy's used ICM memory, and free it if all
of the buddy's memory bacame unused.
Do this only for STEs.
MODIFY_ACTION buddies are much smaller, so in case there
is a large amount of modify_header actions, which result
in large amount of MODIFY_ACTION buddies, doing this
cleanup during sync will result in performance hit while
not freeing significant amount of memory.

Signed-off-by: Yevgeny Kliteynik <kliteyn@nvidia.com>
Reviewed-by: Alex Vesker <valex@nvidia.com>
Reviewed-by: Mark Bloch <mbloch@nvidia.com>
Signed-off-by: Saeed Mahameed <saeedm@nvidia.com>
drivers/net/ethernet/mellanox/mlx5/core/steering/dr_icm_pool.c
drivers/net/ethernet/mellanox/mlx5/core/steering/mlx5dr.h

index c49f8e8..66c2476 100644 (file)
@@ -175,10 +175,12 @@ get_chunk_icm_type(struct mlx5dr_icm_chunk *chunk)
        return chunk->buddy_mem->pool->icm_type;
 }
 
-static void dr_icm_chunk_destroy(struct mlx5dr_icm_chunk *chunk)
+static void dr_icm_chunk_destroy(struct mlx5dr_icm_chunk *chunk,
+                                struct mlx5dr_icm_buddy_mem *buddy)
 {
        enum mlx5dr_icm_type icm_type = get_chunk_icm_type(chunk);
 
+       buddy->used_memory -= chunk->byte_size;
        list_del(&chunk->chunk_list);
 
        if (icm_type == DR_ICM_TYPE_STE)
@@ -223,10 +225,10 @@ static void dr_icm_buddy_destroy(struct mlx5dr_icm_buddy_mem *buddy)
        struct mlx5dr_icm_chunk *chunk, *next;
 
        list_for_each_entry_safe(chunk, next, &buddy->hot_list, chunk_list)
-               dr_icm_chunk_destroy(chunk);
+               dr_icm_chunk_destroy(chunk, buddy);
 
        list_for_each_entry_safe(chunk, next, &buddy->used_list, chunk_list)
-               dr_icm_chunk_destroy(chunk);
+               dr_icm_chunk_destroy(chunk, buddy);
 
        dr_icm_pool_mr_destroy(buddy->icm_mr);
 
@@ -267,6 +269,7 @@ dr_icm_chunk_create(struct mlx5dr_icm_pool *pool,
                goto out_free_chunk;
        }
 
+       buddy_mem_pool->used_memory += chunk->byte_size;
        chunk->buddy_mem = buddy_mem_pool;
        INIT_LIST_HEAD(&chunk->chunk_list);
 
@@ -306,8 +309,11 @@ static int dr_icm_pool_sync_all_buddy_pools(struct mlx5dr_icm_pool *pool)
                        mlx5dr_buddy_free_mem(buddy, chunk->seg,
                                              ilog2(chunk->num_of_entries));
                        pool->hot_memory_size -= chunk->byte_size;
-                       dr_icm_chunk_destroy(chunk);
+                       dr_icm_chunk_destroy(chunk, buddy);
                }
+
+               if (!buddy->used_memory && pool->icm_type == DR_ICM_TYPE_STE)
+                       dr_icm_buddy_destroy(buddy);
        }
 
        return 0;
index a483d7d..4177786 100644 (file)
@@ -141,6 +141,7 @@ struct mlx5dr_icm_buddy_mem {
 
        /* This is the list of used chunks. HW may be accessing this memory */
        struct list_head        used_list;
+       u64                     used_memory;
 
        /* Hardware may be accessing this memory but at some future,
         * undetermined time, it might cease to do so.