Browse Source

cnss_prealloc: use slab_cache from slab struct to identify mempool id

CNSS prealloc maintains various prealloc pools of 8Kb, 16Kb, 32Kb
and so on, and allocates buffer from the pool for wlan driver. When
wlan driver requests to free the memory buffer then CNSS prealloc
needs to first find out mempool id from which this buffer was allocated.
Use slab_cache from slab struct to identify memory pool id. Until
kernel 5.16, slab_cache was part of page struct but in kernel 5.17,
slab_cache is moved out to mm/slab.h file.

Also change WCNSS_PRE_ALLOC_GET_THRESHOLD to 8Kb to match with
cnss_pool_alloc_threshold.

Change-Id: I4e34d8f0b855c210cc9af30c1f4a0d6c7e43ab00
CRs-Fixed: 3414037
Rajesh Chauhan 2 years ago
parent
commit
ff372a6f1d
2 changed files with 55 additions and 48 deletions
  1. 50 46
      cnss_prealloc/cnss_prealloc.c
  2. 5 2
      inc/cnss_prealloc.h

+ 50 - 46
cnss_prealloc/cnss_prealloc.c

@@ -1,7 +1,7 @@
 // SPDX-License-Identifier: GPL-2.0-only
 /*
  * Copyright (c) 2012,2014-2017,2019-2021 The Linux Foundation. All rights reserved.
- * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
+ * Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved.
  */
 
 #include <linux/module.h>
@@ -16,6 +16,17 @@
 #else
 #include <net/cnss_prealloc.h>
 #endif
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 17, 0))
+/* Ideally header should be from standard include path. So this is not an
+ * ideal way of header inclusion but use of slab struct to derive cache
+ * from a mem ptr helps in avoiding additional tracking and/or adding headroom
+ * of 8 bytes for cache in the beginning of buffer and wasting extra memory,
+ * particulary in the case when size of memory requested falls around the edge
+ * of a page boundary. We also have precedence of minidump_memory.c which
+ * includes mm/slab.h using this style.
+ */
+#include "../mm/slab.h"
+#endif
 
 MODULE_LICENSE("GPL v2");
 MODULE_DESCRIPTION("CNSS prealloc driver");
@@ -146,64 +157,58 @@ static void cnss_pool_deinit(void)
 	}
 }
 
-#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 17, 0))
-/* In kernel 5.17, slab_cache is removed from page struct, so
- * store cache in the beginning of memory buffer.
+/**
+ * cnss_pool_get_index() - Get the index of memory pool
+ * @mem: Allocated memory
+ *
+ * Returns the index of the memory pool which fits the reqested memory. The
+ * complexity of this check is O(num of memory pools). Returns a negative
+ * value with error code in case of failure.
+ *
  */
-static inline void cnss_pool_put_cache_in_mem(void *mem, struct kmem_cache *cache)
-{
-	/* put cache at the beginnging of mem */
-	(*(struct kmem_cache **)mem) = cache;
-}
-
-static inline struct kmem_cache *cnss_pool_get_cache_from_mem(void *mem)
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 17, 0))
+static int cnss_pool_get_index(void *mem)
 {
+	struct slab *slab;
 	struct kmem_cache *cache;
+	int i;
+
+	if (!virt_addr_valid(mem))
+		return -EINVAL;
+
+	/* mem -> slab -> cache */
+	slab = virt_to_slab(mem);
+	if (!slab)
+		return -ENOENT;
 
-	/* read cache from the beginnging of mem */
-	cache = (struct kmem_cache *)(*(struct kmem_cache **)mem);
+	cache = slab->slab_cache;
+	if (!cache)
+		return -ENOENT;
 
-	return cache;
+	/* Check if memory belongs to a pool */
+	for (i = 0; i < ARRAY_SIZE(cnss_pools); i++) {
+		if (cnss_pools[i].cache == cache)
+			return i;
+	}
+
+	return -ENOENT;
 }
 #else /* (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 17, 0)) */
-/* for older kernel < 5.17, we use page->slab_cache. In such case
- * we do not reserve headroom in memory buffer to store cache.
- */
-static inline void cnss_pool_put_cache_in_mem(void *mem, struct kmem_cache *cache)
-{
-}
-
-static inline struct kmem_cache *cnss_pool_get_cache_from_mem(void *mem)
+static int cnss_pool_get_index(void *mem)
 {
 	struct page *page;
+	struct kmem_cache *cache;
+	int i;
 
 	if (!virt_addr_valid(mem))
-		return NULL;
+		return -EINVAL;
 
 	/* mem -> page -> cache */
 	page = virt_to_head_page(mem);
 	if (!page)
-		return NULL;
-
-	return page->slab_cache;
-}
-#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 17, 0)) */
-
-/**
- * cnss_pool_get_index() - Get the index of memory pool
- * @mem: Allocated memory
- *
- * Returns the index of the memory pool which fits the reqested memory. The
- * complexity of this check is O(num of memory pools). Returns a negative
- * value with error code in case of failure.
- *
- */
-static int cnss_pool_get_index(void *mem)
-{
-	struct kmem_cache *cache;
-	int i;
+		return -ENOENT;
 
-	cache = cnss_pool_get_cache_from_mem(mem);
+	cache = page->slab_cache;
 	if (!cache)
 		return -ENOENT;
 
@@ -215,6 +220,7 @@ static int cnss_pool_get_index(void *mem)
 
 	return -ENOENT;
 }
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 17, 0)) */
 
 /**
  * wcnss_prealloc_get() - Get preallocated memory from a pool
@@ -242,10 +248,8 @@ void *wcnss_prealloc_get(size_t size)
 		for (i = 0; i < ARRAY_SIZE(cnss_pools); i++) {
 			if (cnss_pools[i].size >= size && cnss_pools[i].mp) {
 				mem = mempool_alloc(cnss_pools[i].mp, gfp_mask);
-				if (mem) {
-					cnss_pool_put_cache_in_mem(mem, cnss_pools[i].cache);
+				if (mem)
 					break;
-				}
 			}
 		}
 	}

+ 5 - 2
inc/cnss_prealloc.h

@@ -1,12 +1,15 @@
 /* SPDX-License-Identifier: GPL-2.0-only */
-/* Copyright (c) 2015-2016,2019 The Linux Foundation. All rights reserved. */
+/*
+ * Copyright (c) 2015-2016,2019 The Linux Foundation. All rights reserved.
+ * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
+ */
 
 #ifndef _NET_CNSS_PREALLOC_H_
 #define _NET_CNSS_PREALLOC_H_
 
 #include <linux/types.h>
 
-#define WCNSS_PRE_ALLOC_GET_THRESHOLD (4*1024)
+#define WCNSS_PRE_ALLOC_GET_THRESHOLD (8*1024)
 
 extern void *wcnss_prealloc_get(size_t size);
 extern int wcnss_prealloc_put(void *ptr);