Commit 7981e67e authored by Vlastimil Babka's avatar Vlastimil Babka

mm/slab: Convert most struct page to struct slab by spatch

The majority of conversion from struct page to struct slab in SLAB
internals can be delegated to a coccinelle semantic patch. This includes
renaming of variables with 'page' in name to 'slab', and similar.

Big thanks to Julia Lawall and Luis Chamberlain for help with
coccinelle.

// Options: --include-headers --no-includes --smpl-spacing mm/slab.c
// Note: needs coccinelle 1.1.1 to avoid breaking whitespace, and ocaml for the
// embedded script

// build list of functions for applying the next rule
@initialize:ocaml@
@@

let ok_function p =
  not (List.mem (List.hd p).current_element ["kmem_getpages";"kmem_freepages"])

// convert the type in selected functions
@@
position p : script:ocaml() { ok_function p };
@@

- struct page@p
+ struct slab

@@
@@

-PageSlabPfmemalloc(page)
+slab_test_pfmemalloc(slab)

@@
@@

-ClearPageSlabPfmemalloc(page)
+slab_clear_pfmemalloc(slab)

@@
@@

obj_to_index(
 ...,
- page
+ slab_page(slab)
,...)

// for all functions, change any "struct slab *page" parameter to "struct slab
// *slab" in the signature, and generally all occurences of "page" to "slab" in
// the body - with some special cases.
@@
identifier fn;
expression E;
@@

 fn(...,
-   struct slab *page
+   struct slab *slab
    ,...)
 {
<...
(
- int page_node;
+ int slab_node;
|
- page_node
+ slab_node
|
- page_slab(page)
+ slab
|
- page_address(page)
+ slab_address(slab)
|
- page_size(page)
+ slab_size(slab)
|
- page_to_nid(page)
+ slab_nid(slab)
|
- virt_to_head_page(E)
+ virt_to_slab(E)
|
- page
+ slab
)
...>
 }

// rename a function parameter
@@
identifier fn;
expression E;
@@

 fn(...,
-   int page_node
+   int slab_node
    ,...)
 {
<...
- page_node
+ slab_node
...>
 }

// functions converted by previous rules that were temporarily called using
// slab_page(E) so we want to remove the wrapper now that they accept struct
// slab ptr directly
@@
identifier fn =~ "index_to_obj";
expression E;
@@

 fn(...,
- slab_page(E)
+ E
 ,...)

// functions that were returning struct page ptr and now will return struct
// slab ptr, including slab_page() wrapper removal
@@
identifier fn =~ "cache_grow_begin|get_valid_first_slab|get_first_slab";
expression E;
@@

 fn(...)
 {
<...
- slab_page(E)
+ E
...>
 }

// rename any former struct page * declarations
@@
@@

struct slab *
-page
+slab
;

// all functions (with exceptions) with a local "struct slab *page" variable
// that will be renamed to "struct slab *slab"
@@
identifier fn !~ "kmem_getpages|kmem_freepages";
expression E;
@@

 fn(...)
 {
<...
(
- page_slab(page)
+ slab
|
- page_to_nid(page)
+ slab_nid(slab)
|
- kasan_poison_slab(page)
+ kasan_poison_slab(slab_page(slab))
|
- page_address(page)
+ slab_address(slab)
|
- page_size(page)
+ slab_size(slab)
|
- page->pages
+ slab->slabs
|
- page = virt_to_head_page(E)
+ slab = virt_to_slab(E)
|
- virt_to_head_page(E)
+ virt_to_slab(E)
|
- page
+ slab
)
...>
 }
Signed-off-by: default avatarVlastimil Babka <vbabka@suse.cz>
Reviewed-by: default avatarRoman Gushchin <guro@fb.com>
Tested-by: default avatarHyeonggon Yoo <42.hyeyoo@gmail.com>
Cc: Julia Lawall <julia.lawall@inria.fr>
Cc: Luis Chamberlain <mcgrof@kernel.org>
parent 42c0faac
...@@ -218,7 +218,7 @@ static void cache_reap(struct work_struct *unused); ...@@ -218,7 +218,7 @@ static void cache_reap(struct work_struct *unused);
static inline void fixup_objfreelist_debug(struct kmem_cache *cachep, static inline void fixup_objfreelist_debug(struct kmem_cache *cachep,
void **list); void **list);
static inline void fixup_slab_list(struct kmem_cache *cachep, static inline void fixup_slab_list(struct kmem_cache *cachep,
struct kmem_cache_node *n, struct page *page, struct kmem_cache_node *n, struct slab *slab,
void **list); void **list);
static int slab_early_init = 1; static int slab_early_init = 1;
...@@ -373,9 +373,9 @@ static int slab_max_order = SLAB_MAX_ORDER_LO; ...@@ -373,9 +373,9 @@ static int slab_max_order = SLAB_MAX_ORDER_LO;
static bool slab_max_order_set __initdata; static bool slab_max_order_set __initdata;
static inline void *index_to_obj(struct kmem_cache *cache, static inline void *index_to_obj(struct kmem_cache *cache,
const struct page *page, unsigned int idx) const struct slab *slab, unsigned int idx)
{ {
return page->s_mem + cache->size * idx; return slab->s_mem + cache->size * idx;
} }
#define BOOT_CPUCACHE_ENTRIES 1 #define BOOT_CPUCACHE_ENTRIES 1
...@@ -550,17 +550,17 @@ static struct array_cache *alloc_arraycache(int node, int entries, ...@@ -550,17 +550,17 @@ static struct array_cache *alloc_arraycache(int node, int entries,
} }
static noinline void cache_free_pfmemalloc(struct kmem_cache *cachep, static noinline void cache_free_pfmemalloc(struct kmem_cache *cachep,
struct page *page, void *objp) struct slab *slab, void *objp)
{ {
struct kmem_cache_node *n; struct kmem_cache_node *n;
int page_node; int slab_node;
LIST_HEAD(list); LIST_HEAD(list);
page_node = page_to_nid(page); slab_node = slab_nid(slab);
n = get_node(cachep, page_node); n = get_node(cachep, slab_node);
spin_lock(&n->list_lock); spin_lock(&n->list_lock);
free_block(cachep, &objp, 1, page_node, &list); free_block(cachep, &objp, 1, slab_node, &list);
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
slabs_destroy(cachep, &list); slabs_destroy(cachep, &list);
...@@ -761,7 +761,7 @@ static void drain_alien_cache(struct kmem_cache *cachep, ...@@ -761,7 +761,7 @@ static void drain_alien_cache(struct kmem_cache *cachep,
} }
static int __cache_free_alien(struct kmem_cache *cachep, void *objp, static int __cache_free_alien(struct kmem_cache *cachep, void *objp,
int node, int page_node) int node, int slab_node)
{ {
struct kmem_cache_node *n; struct kmem_cache_node *n;
struct alien_cache *alien = NULL; struct alien_cache *alien = NULL;
...@@ -770,21 +770,21 @@ static int __cache_free_alien(struct kmem_cache *cachep, void *objp, ...@@ -770,21 +770,21 @@ static int __cache_free_alien(struct kmem_cache *cachep, void *objp,
n = get_node(cachep, node); n = get_node(cachep, node);
STATS_INC_NODEFREES(cachep); STATS_INC_NODEFREES(cachep);
if (n->alien && n->alien[page_node]) { if (n->alien && n->alien[slab_node]) {
alien = n->alien[page_node]; alien = n->alien[slab_node];
ac = &alien->ac; ac = &alien->ac;
spin_lock(&alien->lock); spin_lock(&alien->lock);
if (unlikely(ac->avail == ac->limit)) { if (unlikely(ac->avail == ac->limit)) {
STATS_INC_ACOVERFLOW(cachep); STATS_INC_ACOVERFLOW(cachep);
__drain_alien_cache(cachep, ac, page_node, &list); __drain_alien_cache(cachep, ac, slab_node, &list);
} }
__free_one(ac, objp); __free_one(ac, objp);
spin_unlock(&alien->lock); spin_unlock(&alien->lock);
slabs_destroy(cachep, &list); slabs_destroy(cachep, &list);
} else { } else {
n = get_node(cachep, page_node); n = get_node(cachep, slab_node);
spin_lock(&n->list_lock); spin_lock(&n->list_lock);
free_block(cachep, &objp, 1, page_node, &list); free_block(cachep, &objp, 1, slab_node, &list);
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
slabs_destroy(cachep, &list); slabs_destroy(cachep, &list);
} }
...@@ -1556,18 +1556,18 @@ static void check_poison_obj(struct kmem_cache *cachep, void *objp) ...@@ -1556,18 +1556,18 @@ static void check_poison_obj(struct kmem_cache *cachep, void *objp)
/* Print some data about the neighboring objects, if they /* Print some data about the neighboring objects, if they
* exist: * exist:
*/ */
struct page *page = virt_to_head_page(objp); struct slab *slab = virt_to_slab(objp);
unsigned int objnr; unsigned int objnr;
objnr = obj_to_index(cachep, page, objp); objnr = obj_to_index(cachep, slab_page(slab), objp);
if (objnr) { if (objnr) {
objp = index_to_obj(cachep, page, objnr - 1); objp = index_to_obj(cachep, slab, objnr - 1);
realobj = (char *)objp + obj_offset(cachep); realobj = (char *)objp + obj_offset(cachep);
pr_err("Prev obj: start=%px, len=%d\n", realobj, size); pr_err("Prev obj: start=%px, len=%d\n", realobj, size);
print_objinfo(cachep, objp, 2); print_objinfo(cachep, objp, 2);
} }
if (objnr + 1 < cachep->num) { if (objnr + 1 < cachep->num) {
objp = index_to_obj(cachep, page, objnr + 1); objp = index_to_obj(cachep, slab, objnr + 1);
realobj = (char *)objp + obj_offset(cachep); realobj = (char *)objp + obj_offset(cachep);
pr_err("Next obj: start=%px, len=%d\n", realobj, size); pr_err("Next obj: start=%px, len=%d\n", realobj, size);
print_objinfo(cachep, objp, 2); print_objinfo(cachep, objp, 2);
...@@ -1578,17 +1578,17 @@ static void check_poison_obj(struct kmem_cache *cachep, void *objp) ...@@ -1578,17 +1578,17 @@ static void check_poison_obj(struct kmem_cache *cachep, void *objp)
#if DEBUG #if DEBUG
static void slab_destroy_debugcheck(struct kmem_cache *cachep, static void slab_destroy_debugcheck(struct kmem_cache *cachep,
struct page *page) struct slab *slab)
{ {
int i; int i;
if (OBJFREELIST_SLAB(cachep) && cachep->flags & SLAB_POISON) { if (OBJFREELIST_SLAB(cachep) && cachep->flags & SLAB_POISON) {
poison_obj(cachep, page->freelist - obj_offset(cachep), poison_obj(cachep, slab->freelist - obj_offset(cachep),
POISON_FREE); POISON_FREE);
} }
for (i = 0; i < cachep->num; i++) { for (i = 0; i < cachep->num; i++) {
void *objp = index_to_obj(cachep, page, i); void *objp = index_to_obj(cachep, slab, i);
if (cachep->flags & SLAB_POISON) { if (cachep->flags & SLAB_POISON) {
check_poison_obj(cachep, objp); check_poison_obj(cachep, objp);
...@@ -1604,7 +1604,7 @@ static void slab_destroy_debugcheck(struct kmem_cache *cachep, ...@@ -1604,7 +1604,7 @@ static void slab_destroy_debugcheck(struct kmem_cache *cachep,
} }
#else #else
static void slab_destroy_debugcheck(struct kmem_cache *cachep, static void slab_destroy_debugcheck(struct kmem_cache *cachep,
struct page *page) struct slab *slab)
{ {
} }
#endif #endif
...@@ -1618,16 +1618,16 @@ static void slab_destroy_debugcheck(struct kmem_cache *cachep, ...@@ -1618,16 +1618,16 @@ static void slab_destroy_debugcheck(struct kmem_cache *cachep,
* Before calling the slab page must have been unlinked from the cache. The * Before calling the slab page must have been unlinked from the cache. The
* kmem_cache_node ->list_lock is not held/needed. * kmem_cache_node ->list_lock is not held/needed.
*/ */
static void slab_destroy(struct kmem_cache *cachep, struct page *page) static void slab_destroy(struct kmem_cache *cachep, struct slab *slab)
{ {
void *freelist; void *freelist;
freelist = page->freelist; freelist = slab->freelist;
slab_destroy_debugcheck(cachep, page); slab_destroy_debugcheck(cachep, slab);
if (unlikely(cachep->flags & SLAB_TYPESAFE_BY_RCU)) if (unlikely(cachep->flags & SLAB_TYPESAFE_BY_RCU))
call_rcu(&page->rcu_head, kmem_rcu_free); call_rcu(&slab->rcu_head, kmem_rcu_free);
else else
kmem_freepages(cachep, page_slab(page)); kmem_freepages(cachep, slab);
/* /*
* From now on, we don't use freelist * From now on, we don't use freelist
...@@ -1643,11 +1643,11 @@ static void slab_destroy(struct kmem_cache *cachep, struct page *page) ...@@ -1643,11 +1643,11 @@ static void slab_destroy(struct kmem_cache *cachep, struct page *page)
*/ */
static void slabs_destroy(struct kmem_cache *cachep, struct list_head *list) static void slabs_destroy(struct kmem_cache *cachep, struct list_head *list)
{ {
struct page *page, *n; struct slab *slab, *n;
list_for_each_entry_safe(page, n, list, slab_list) { list_for_each_entry_safe(slab, n, list, slab_list) {
list_del(&page->slab_list); list_del(&slab->slab_list);
slab_destroy(cachep, page); slab_destroy(cachep, slab);
} }
} }
...@@ -2197,7 +2197,7 @@ static int drain_freelist(struct kmem_cache *cache, ...@@ -2197,7 +2197,7 @@ static int drain_freelist(struct kmem_cache *cache,
{ {
struct list_head *p; struct list_head *p;
int nr_freed; int nr_freed;
struct page *page; struct slab *slab;
nr_freed = 0; nr_freed = 0;
while (nr_freed < tofree && !list_empty(&n->slabs_free)) { while (nr_freed < tofree && !list_empty(&n->slabs_free)) {
...@@ -2209,8 +2209,8 @@ static int drain_freelist(struct kmem_cache *cache, ...@@ -2209,8 +2209,8 @@ static int drain_freelist(struct kmem_cache *cache,
goto out; goto out;
} }
page = list_entry(p, struct page, slab_list); slab = list_entry(p, struct slab, slab_list);
list_del(&page->slab_list); list_del(&slab->slab_list);
n->free_slabs--; n->free_slabs--;
n->total_slabs--; n->total_slabs--;
/* /*
...@@ -2219,7 +2219,7 @@ static int drain_freelist(struct kmem_cache *cache, ...@@ -2219,7 +2219,7 @@ static int drain_freelist(struct kmem_cache *cache,
*/ */
n->free_objects -= cache->num; n->free_objects -= cache->num;
spin_unlock_irq(&n->list_lock); spin_unlock_irq(&n->list_lock);
slab_destroy(cache, page); slab_destroy(cache, slab);
nr_freed++; nr_freed++;
} }
out: out:
...@@ -2294,14 +2294,14 @@ void __kmem_cache_release(struct kmem_cache *cachep) ...@@ -2294,14 +2294,14 @@ void __kmem_cache_release(struct kmem_cache *cachep)
* which are all initialized during kmem_cache_init(). * which are all initialized during kmem_cache_init().
*/ */
static void *alloc_slabmgmt(struct kmem_cache *cachep, static void *alloc_slabmgmt(struct kmem_cache *cachep,
struct page *page, int colour_off, struct slab *slab, int colour_off,
gfp_t local_flags, int nodeid) gfp_t local_flags, int nodeid)
{ {
void *freelist; void *freelist;
void *addr = page_address(page); void *addr = slab_address(slab);
page->s_mem = addr + colour_off; slab->s_mem = addr + colour_off;
page->active = 0; slab->active = 0;
if (OBJFREELIST_SLAB(cachep)) if (OBJFREELIST_SLAB(cachep))
freelist = NULL; freelist = NULL;
...@@ -2318,24 +2318,24 @@ static void *alloc_slabmgmt(struct kmem_cache *cachep, ...@@ -2318,24 +2318,24 @@ static void *alloc_slabmgmt(struct kmem_cache *cachep,
return freelist; return freelist;
} }
static inline freelist_idx_t get_free_obj(struct page *page, unsigned int idx) static inline freelist_idx_t get_free_obj(struct slab *slab, unsigned int idx)
{ {
return ((freelist_idx_t *)page->freelist)[idx]; return ((freelist_idx_t *) slab->freelist)[idx];
} }
static inline void set_free_obj(struct page *page, static inline void set_free_obj(struct slab *slab,
unsigned int idx, freelist_idx_t val) unsigned int idx, freelist_idx_t val)
{ {
((freelist_idx_t *)(page->freelist))[idx] = val; ((freelist_idx_t *)(slab->freelist))[idx] = val;
} }
static void cache_init_objs_debug(struct kmem_cache *cachep, struct page *page) static void cache_init_objs_debug(struct kmem_cache *cachep, struct slab *slab)
{ {
#if DEBUG #if DEBUG
int i; int i;
for (i = 0; i < cachep->num; i++) { for (i = 0; i < cachep->num; i++) {
void *objp = index_to_obj(cachep, page, i); void *objp = index_to_obj(cachep, slab, i);
if (cachep->flags & SLAB_STORE_USER) if (cachep->flags & SLAB_STORE_USER)
*dbg_userword(cachep, objp) = NULL; *dbg_userword(cachep, objp) = NULL;
...@@ -2419,17 +2419,17 @@ static freelist_idx_t next_random_slot(union freelist_init_state *state) ...@@ -2419,17 +2419,17 @@ static freelist_idx_t next_random_slot(union freelist_init_state *state)
} }
/* Swap two freelist entries */ /* Swap two freelist entries */
static void swap_free_obj(struct page *page, unsigned int a, unsigned int b) static void swap_free_obj(struct slab *slab, unsigned int a, unsigned int b)
{ {
swap(((freelist_idx_t *)page->freelist)[a], swap(((freelist_idx_t *) slab->freelist)[a],
((freelist_idx_t *)page->freelist)[b]); ((freelist_idx_t *) slab->freelist)[b]);
} }
/* /*
* Shuffle the freelist initialization state based on pre-computed lists. * Shuffle the freelist initialization state based on pre-computed lists.
* return true if the list was successfully shuffled, false otherwise. * return true if the list was successfully shuffled, false otherwise.
*/ */
static bool shuffle_freelist(struct kmem_cache *cachep, struct page *page) static bool shuffle_freelist(struct kmem_cache *cachep, struct slab *slab)
{ {
unsigned int objfreelist = 0, i, rand, count = cachep->num; unsigned int objfreelist = 0, i, rand, count = cachep->num;
union freelist_init_state state; union freelist_init_state state;
...@@ -2446,7 +2446,7 @@ static bool shuffle_freelist(struct kmem_cache *cachep, struct page *page) ...@@ -2446,7 +2446,7 @@ static bool shuffle_freelist(struct kmem_cache *cachep, struct page *page)
objfreelist = count - 1; objfreelist = count - 1;
else else
objfreelist = next_random_slot(&state); objfreelist = next_random_slot(&state);
page->freelist = index_to_obj(cachep, page, objfreelist) + slab->freelist = index_to_obj(cachep, slab, objfreelist) +
obj_offset(cachep); obj_offset(cachep);
count--; count--;
} }
...@@ -2457,51 +2457,51 @@ static bool shuffle_freelist(struct kmem_cache *cachep, struct page *page) ...@@ -2457,51 +2457,51 @@ static bool shuffle_freelist(struct kmem_cache *cachep, struct page *page)
*/ */
if (!precomputed) { if (!precomputed) {
for (i = 0; i < count; i++) for (i = 0; i < count; i++)
set_free_obj(page, i, i); set_free_obj(slab, i, i);
/* Fisher-Yates shuffle */ /* Fisher-Yates shuffle */
for (i = count - 1; i > 0; i--) { for (i = count - 1; i > 0; i--) {
rand = prandom_u32_state(&state.rnd_state); rand = prandom_u32_state(&state.rnd_state);
rand %= (i + 1); rand %= (i + 1);
swap_free_obj(page, i, rand); swap_free_obj(slab, i, rand);
} }
} else { } else {
for (i = 0; i < count; i++) for (i = 0; i < count; i++)
set_free_obj(page, i, next_random_slot(&state)); set_free_obj(slab, i, next_random_slot(&state));
} }
if (OBJFREELIST_SLAB(cachep)) if (OBJFREELIST_SLAB(cachep))
set_free_obj(page, cachep->num - 1, objfreelist); set_free_obj(slab, cachep->num - 1, objfreelist);
return true; return true;
} }
#else #else
static inline bool shuffle_freelist(struct kmem_cache *cachep, static inline bool shuffle_freelist(struct kmem_cache *cachep,
struct page *page) struct slab *slab)
{ {
return false; return false;
} }
#endif /* CONFIG_SLAB_FREELIST_RANDOM */ #endif /* CONFIG_SLAB_FREELIST_RANDOM */
static void cache_init_objs(struct kmem_cache *cachep, static void cache_init_objs(struct kmem_cache *cachep,
struct page *page) struct slab *slab)
{ {
int i; int i;
void *objp; void *objp;
bool shuffled; bool shuffled;
cache_init_objs_debug(cachep, page); cache_init_objs_debug(cachep, slab);
/* Try to randomize the freelist if enabled */ /* Try to randomize the freelist if enabled */
shuffled = shuffle_freelist(cachep, page); shuffled = shuffle_freelist(cachep, slab);
if (!shuffled && OBJFREELIST_SLAB(cachep)) { if (!shuffled && OBJFREELIST_SLAB(cachep)) {
page->freelist = index_to_obj(cachep, page, cachep->num - 1) + slab->freelist = index_to_obj(cachep, slab, cachep->num - 1) +
obj_offset(cachep); obj_offset(cachep);
} }
for (i = 0; i < cachep->num; i++) { for (i = 0; i < cachep->num; i++) {
objp = index_to_obj(cachep, page, i); objp = index_to_obj(cachep, slab, i);
objp = kasan_init_slab_obj(cachep, objp); objp = kasan_init_slab_obj(cachep, objp);
/* constructor could break poison info */ /* constructor could break poison info */
...@@ -2512,48 +2512,48 @@ static void cache_init_objs(struct kmem_cache *cachep, ...@@ -2512,48 +2512,48 @@ static void cache_init_objs(struct kmem_cache *cachep,
} }
if (!shuffled) if (!shuffled)
set_free_obj(page, i, i); set_free_obj(slab, i, i);
} }
} }
static void *slab_get_obj(struct kmem_cache *cachep, struct page *page) static void *slab_get_obj(struct kmem_cache *cachep, struct slab *slab)
{ {
void *objp; void *objp;
objp = index_to_obj(cachep, page, get_free_obj(page, page->active)); objp = index_to_obj(cachep, slab, get_free_obj(slab, slab->active));
page->active++; slab->active++;
return objp; return objp;
} }
static void slab_put_obj(struct kmem_cache *cachep, static void slab_put_obj(struct kmem_cache *cachep,
struct page *page, void *objp) struct slab *slab, void *objp)
{ {
unsigned int objnr = obj_to_index(cachep, page, objp); unsigned int objnr = obj_to_index(cachep, slab_page(slab), objp);
#if DEBUG #if DEBUG
unsigned int i; unsigned int i;
/* Verify double free bug */ /* Verify double free bug */
for (i = page->active; i < cachep->num; i++) { for (i = slab->active; i < cachep->num; i++) {
if (get_free_obj(page, i) == objnr) { if (get_free_obj(slab, i) == objnr) {
pr_err("slab: double free detected in cache '%s', objp %px\n", pr_err("slab: double free detected in cache '%s', objp %px\n",
cachep->name, objp); cachep->name, objp);
BUG(); BUG();
} }
} }
#endif #endif
page->active--; slab->active--;
if (!page->freelist) if (!slab->freelist)
page->freelist = objp + obj_offset(cachep); slab->freelist = objp + obj_offset(cachep);
set_free_obj(page, page->active, objnr); set_free_obj(slab, slab->active, objnr);
} }
/* /*
* Grow (by 1) the number of slabs within a cache. This is called by * Grow (by 1) the number of slabs within a cache. This is called by
* kmem_cache_alloc() when there are no active objs left in a cache. * kmem_cache_alloc() when there are no active objs left in a cache.
*/ */
static struct page *cache_grow_begin(struct kmem_cache *cachep, static struct slab *cache_grow_begin(struct kmem_cache *cachep,
gfp_t flags, int nodeid) gfp_t flags, int nodeid)
{ {
void *freelist; void *freelist;
...@@ -2561,7 +2561,7 @@ static struct page *cache_grow_begin(struct kmem_cache *cachep, ...@@ -2561,7 +2561,7 @@ static struct page *cache_grow_begin(struct kmem_cache *cachep,
gfp_t local_flags; gfp_t local_flags;
int page_node; int page_node;
struct kmem_cache_node *n; struct kmem_cache_node *n;
struct page *page; struct slab *slab;
/* /*
* Be lazy and only check for valid flags here, keeping it out of the * Be lazy and only check for valid flags here, keeping it out of the
...@@ -2581,11 +2581,11 @@ static struct page *cache_grow_begin(struct kmem_cache *cachep, ...@@ -2581,11 +2581,11 @@ static struct page *cache_grow_begin(struct kmem_cache *cachep,
* Get mem for the objs. Attempt to allocate a physical page from * Get mem for the objs. Attempt to allocate a physical page from
* 'nodeid'. * 'nodeid'.
*/ */
page = slab_page(kmem_getpages(cachep, local_flags, nodeid)); slab = kmem_getpages(cachep, local_flags, nodeid);
if (!page) if (!slab)
goto failed; goto failed;
page_node = page_to_nid(page); page_node = slab_nid(slab);
n = get_node(cachep, page_node); n = get_node(cachep, page_node);
/* Get colour for the slab, and cal the next value. */ /* Get colour for the slab, and cal the next value. */
...@@ -2604,55 +2604,55 @@ static struct page *cache_grow_begin(struct kmem_cache *cachep, ...@@ -2604,55 +2604,55 @@ static struct page *cache_grow_begin(struct kmem_cache *cachep,
* page_address() in the latter returns a non-tagged pointer, * page_address() in the latter returns a non-tagged pointer,
* as it should be for slab pages. * as it should be for slab pages.
*/ */
kasan_poison_slab(page); kasan_poison_slab(slab_page(slab));
/* Get slab management. */ /* Get slab management. */
freelist = alloc_slabmgmt(cachep, page, offset, freelist = alloc_slabmgmt(cachep, slab, offset,
local_flags & ~GFP_CONSTRAINT_MASK, page_node); local_flags & ~GFP_CONSTRAINT_MASK, page_node);
if (OFF_SLAB(cachep) && !freelist) if (OFF_SLAB(cachep) && !freelist)
goto opps1; goto opps1;
page->slab_cache = cachep; slab->slab_cache = cachep;
page->freelist = freelist; slab->freelist = freelist;
cache_init_objs(cachep, page); cache_init_objs(cachep, slab);
if (gfpflags_allow_blocking(local_flags)) if (gfpflags_allow_blocking(local_flags))
local_irq_disable(); local_irq_disable();
return page; return slab;
opps1: opps1:
kmem_freepages(cachep, page_slab(page)); kmem_freepages(cachep, slab);
failed: failed:
if (gfpflags_allow_blocking(local_flags)) if (gfpflags_allow_blocking(local_flags))
local_irq_disable(); local_irq_disable();
return NULL; return NULL;
} }
static void cache_grow_end(struct kmem_cache *cachep, struct page *page) static void cache_grow_end(struct kmem_cache *cachep, struct slab *slab)
{ {
struct kmem_cache_node *n; struct kmem_cache_node *n;
void *list = NULL; void *list = NULL;
check_irq_off(); check_irq_off();
if (!page) if (!slab)
return; return;
INIT_LIST_HEAD(&page->slab_list); INIT_LIST_HEAD(&slab->slab_list);
n = get_node(cachep, page_to_nid(page)); n = get_node(cachep, slab_nid(slab));
spin_lock(&n->list_lock); spin_lock(&n->list_lock);
n->total_slabs++; n->total_slabs++;
if (!page->active) { if (!slab->active) {
list_add_tail(&page->slab_list, &n->slabs_free); list_add_tail(&slab->slab_list, &n->slabs_free);
n->free_slabs++; n->free_slabs++;
} else } else
fixup_slab_list(cachep, n, page, &list); fixup_slab_list(cachep, n, slab, &list);
STATS_INC_GROWN(cachep); STATS_INC_GROWN(cachep);
n->free_objects += cachep->num - page->active; n->free_objects += cachep->num - slab->active;
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
fixup_objfreelist_debug(cachep, &list); fixup_objfreelist_debug(cachep, &list);
...@@ -2700,13 +2700,13 @@ static void *cache_free_debugcheck(struct kmem_cache *cachep, void *objp, ...@@ -2700,13 +2700,13 @@ static void *cache_free_debugcheck(struct kmem_cache *cachep, void *objp,
unsigned long caller) unsigned long caller)
{ {
unsigned int objnr; unsigned int objnr;
struct page *page; struct slab *slab;
BUG_ON(virt_to_cache(objp) != cachep); BUG_ON(virt_to_cache(objp) != cachep);
objp -= obj_offset(cachep); objp -= obj_offset(cachep);
kfree_debugcheck(objp); kfree_debugcheck(objp);
page = virt_to_head_page(objp); slab = virt_to_slab(objp);
if (cachep->flags & SLAB_RED_ZONE) { if (cachep->flags & SLAB_RED_ZONE) {
verify_redzone_free(cachep, objp); verify_redzone_free(cachep, objp);
...@@ -2716,10 +2716,10 @@ static void *cache_free_debugcheck(struct kmem_cache *cachep, void *objp, ...@@ -2716,10 +2716,10 @@ static void *cache_free_debugcheck(struct kmem_cache *cachep, void *objp,
if (cachep->flags & SLAB_STORE_USER) if (cachep->flags & SLAB_STORE_USER)
*dbg_userword(cachep, objp) = (void *)caller; *dbg_userword(cachep, objp) = (void *)caller;
objnr = obj_to_index(cachep, page, objp); objnr = obj_to_index(cachep, slab_page(slab), objp);
BUG_ON(objnr >= cachep->num); BUG_ON(objnr >= cachep->num);
BUG_ON(objp != index_to_obj(cachep, page, objnr)); BUG_ON(objp != index_to_obj(cachep, slab, objnr));
if (cachep->flags & SLAB_POISON) { if (cachep->flags & SLAB_POISON) {
poison_obj(cachep, objp, POISON_FREE); poison_obj(cachep, objp, POISON_FREE);
...@@ -2749,97 +2749,97 @@ static inline void fixup_objfreelist_debug(struct kmem_cache *cachep, ...@@ -2749,97 +2749,97 @@ static inline void fixup_objfreelist_debug(struct kmem_cache *cachep,
} }
static inline void fixup_slab_list(struct kmem_cache *cachep, static inline void fixup_slab_list(struct kmem_cache *cachep,
struct kmem_cache_node *n, struct page *page, struct kmem_cache_node *n, struct slab *slab,
void **list) void **list)
{ {
/* move slabp to correct slabp list: */ /* move slabp to correct slabp list: */
list_del(&page->slab_list); list_del(&slab->slab_list);
if (page->active == cachep->num) { if (slab->active == cachep->num) {
list_add(&page->slab_list, &n->slabs_full); list_add(&slab->slab_list, &n->slabs_full);
if (OBJFREELIST_SLAB(cachep)) { if (OBJFREELIST_SLAB(cachep)) {
#if DEBUG #if DEBUG
/* Poisoning will be done without holding the lock */ /* Poisoning will be done without holding the lock */
if (cachep->flags & SLAB_POISON) { if (cachep->flags & SLAB_POISON) {
void **objp = page->freelist; void **objp = slab->freelist;
*objp = *list; *objp = *list;
*list = objp; *list = objp;
} }
#endif #endif
page->freelist = NULL; slab->freelist = NULL;
} }
} else } else
list_add(&page->slab_list, &n->slabs_partial); list_add(&slab->slab_list, &n->slabs_partial);
} }
/* Try to find non-pfmemalloc slab if needed */ /* Try to find non-pfmemalloc slab if needed */
static noinline struct page *get_valid_first_slab(struct kmem_cache_node *n, static noinline struct slab *get_valid_first_slab(struct kmem_cache_node *n,
struct page *page, bool pfmemalloc) struct slab *slab, bool pfmemalloc)
{ {
if (!page) if (!slab)
return NULL; return NULL;
if (pfmemalloc) if (pfmemalloc)
return page; return slab;
if (!PageSlabPfmemalloc(page)) if (!slab_test_pfmemalloc(slab))
return page; return slab;
/* No need to keep pfmemalloc slab if we have enough free objects */ /* No need to keep pfmemalloc slab if we have enough free objects */
if (n->free_objects > n->free_limit) { if (n->free_objects > n->free_limit) {
ClearPageSlabPfmemalloc(page); slab_clear_pfmemalloc(slab);
return page; return slab;
} }
/* Move pfmemalloc slab to the end of list to speed up next search */ /* Move pfmemalloc slab to the end of list to speed up next search */
list_del(&page->slab_list); list_del(&slab->slab_list);
if (!page->active) { if (!slab->active) {
list_add_tail(&page->slab_list, &n->slabs_free); list_add_tail(&slab->slab_list, &n->slabs_free);
n->free_slabs++; n->free_slabs++;
} else } else
list_add_tail(&page->slab_list, &n->slabs_partial); list_add_tail(&slab->slab_list, &n->slabs_partial);
list_for_each_entry(page, &n->slabs_partial, slab_list) { list_for_each_entry(slab, &n->slabs_partial, slab_list) {
if (!PageSlabPfmemalloc(page)) if (!slab_test_pfmemalloc(slab))
return page; return slab;
} }
n->free_touched = 1; n->free_touched = 1;
list_for_each_entry(page, &n->slabs_free, slab_list) { list_for_each_entry(slab, &n->slabs_free, slab_list) {
if (!PageSlabPfmemalloc(page)) { if (!slab_test_pfmemalloc(slab)) {
n->free_slabs--; n->free_slabs--;
return page; return slab;
} }
} }
return NULL; return NULL;
} }
static struct page *get_first_slab(struct kmem_cache_node *n, bool pfmemalloc) static struct slab *get_first_slab(struct kmem_cache_node *n, bool pfmemalloc)
{ {
struct page *page; struct slab *slab;
assert_spin_locked(&n->list_lock); assert_spin_locked(&n->list_lock);
page = list_first_entry_or_null(&n->slabs_partial, struct page, slab = list_first_entry_or_null(&n->slabs_partial, struct slab,
slab_list); slab_list);
if (!page) { if (!slab) {
n->free_touched = 1; n->free_touched = 1;
page = list_first_entry_or_null(&n->slabs_free, struct page, slab = list_first_entry_or_null(&n->slabs_free, struct slab,
slab_list); slab_list);
if (page) if (slab)
n->free_slabs--; n->free_slabs--;
} }
if (sk_memalloc_socks()) if (sk_memalloc_socks())
page = get_valid_first_slab(n, page, pfmemalloc); slab = get_valid_first_slab(n, slab, pfmemalloc);
return page; return slab;
} }
static noinline void *cache_alloc_pfmemalloc(struct kmem_cache *cachep, static noinline void *cache_alloc_pfmemalloc(struct kmem_cache *cachep,
struct kmem_cache_node *n, gfp_t flags) struct kmem_cache_node *n, gfp_t flags)
{ {
struct page *page; struct slab *slab;
void *obj; void *obj;
void *list = NULL; void *list = NULL;
...@@ -2847,16 +2847,16 @@ static noinline void *cache_alloc_pfmemalloc(struct kmem_cache *cachep, ...@@ -2847,16 +2847,16 @@ static noinline void *cache_alloc_pfmemalloc(struct kmem_cache *cachep,
return NULL; return NULL;
spin_lock(&n->list_lock); spin_lock(&n->list_lock);
page = get_first_slab(n, true); slab = get_first_slab(n, true);
if (!page) { if (!slab) {
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
return NULL; return NULL;
} }
obj = slab_get_obj(cachep, page); obj = slab_get_obj(cachep, slab);
n->free_objects--; n->free_objects--;
fixup_slab_list(cachep, n, page, &list); fixup_slab_list(cachep, n, slab, &list);
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
fixup_objfreelist_debug(cachep, &list); fixup_objfreelist_debug(cachep, &list);
...@@ -2869,20 +2869,20 @@ static noinline void *cache_alloc_pfmemalloc(struct kmem_cache *cachep, ...@@ -2869,20 +2869,20 @@ static noinline void *cache_alloc_pfmemalloc(struct kmem_cache *cachep,
* or cache_grow_end() for new slab * or cache_grow_end() for new slab
*/ */
static __always_inline int alloc_block(struct kmem_cache *cachep, static __always_inline int alloc_block(struct kmem_cache *cachep,
struct array_cache *ac, struct page *page, int batchcount) struct array_cache *ac, struct slab *slab, int batchcount)
{ {
/* /*
* There must be at least one object available for * There must be at least one object available for
* allocation. * allocation.
*/ */
BUG_ON(page->active >= cachep->num); BUG_ON(slab->active >= cachep->num);
while (page->active < cachep->num && batchcount--) { while (slab->active < cachep->num && batchcount--) {
STATS_INC_ALLOCED(cachep); STATS_INC_ALLOCED(cachep);
STATS_INC_ACTIVE(cachep); STATS_INC_ACTIVE(cachep);
STATS_SET_HIGH(cachep); STATS_SET_HIGH(cachep);
ac->entry[ac->avail++] = slab_get_obj(cachep, page); ac->entry[ac->avail++] = slab_get_obj(cachep, slab);
} }
return batchcount; return batchcount;
...@@ -2895,7 +2895,7 @@ static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags) ...@@ -2895,7 +2895,7 @@ static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags)
struct array_cache *ac, *shared; struct array_cache *ac, *shared;
int node; int node;
void *list = NULL; void *list = NULL;
struct page *page; struct slab *slab;
check_irq_off(); check_irq_off();
node = numa_mem_id(); node = numa_mem_id();
...@@ -2928,14 +2928,14 @@ static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags) ...@@ -2928,14 +2928,14 @@ static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags)
while (batchcount > 0) { while (batchcount > 0) {
/* Get slab alloc is to come from. */ /* Get slab alloc is to come from. */
page = get_first_slab(n, false); slab = get_first_slab(n, false);
if (!page) if (!slab)
goto must_grow; goto must_grow;
check_spinlock_acquired(cachep); check_spinlock_acquired(cachep);
batchcount = alloc_block(cachep, ac, page, batchcount); batchcount = alloc_block(cachep, ac, slab, batchcount);
fixup_slab_list(cachep, n, page, &list); fixup_slab_list(cachep, n, slab, &list);
} }
must_grow: must_grow:
...@@ -2954,16 +2954,16 @@ static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags) ...@@ -2954,16 +2954,16 @@ static void *cache_alloc_refill(struct kmem_cache *cachep, gfp_t flags)
return obj; return obj;
} }
page = cache_grow_begin(cachep, gfp_exact_node(flags), node); slab = cache_grow_begin(cachep, gfp_exact_node(flags), node);
/* /*
* cache_grow_begin() can reenable interrupts, * cache_grow_begin() can reenable interrupts,
* then ac could change. * then ac could change.
*/ */
ac = cpu_cache_get(cachep); ac = cpu_cache_get(cachep);
if (!ac->avail && page) if (!ac->avail && slab)
alloc_block(cachep, ac, page, batchcount); alloc_block(cachep, ac, slab, batchcount);
cache_grow_end(cachep, page); cache_grow_end(cachep, slab);
if (!ac->avail) if (!ac->avail)
return NULL; return NULL;
...@@ -3093,7 +3093,7 @@ static void *fallback_alloc(struct kmem_cache *cache, gfp_t flags) ...@@ -3093,7 +3093,7 @@ static void *fallback_alloc(struct kmem_cache *cache, gfp_t flags)
struct zone *zone; struct zone *zone;
enum zone_type highest_zoneidx = gfp_zone(flags); enum zone_type highest_zoneidx = gfp_zone(flags);
void *obj = NULL; void *obj = NULL;
struct page *page; struct slab *slab;
int nid; int nid;
unsigned int cpuset_mems_cookie; unsigned int cpuset_mems_cookie;
...@@ -3129,10 +3129,10 @@ static void *fallback_alloc(struct kmem_cache *cache, gfp_t flags) ...@@ -3129,10 +3129,10 @@ static void *fallback_alloc(struct kmem_cache *cache, gfp_t flags)
* We may trigger various forms of reclaim on the allowed * We may trigger various forms of reclaim on the allowed
* set and go into memory reserves if necessary. * set and go into memory reserves if necessary.
*/ */
page = cache_grow_begin(cache, flags, numa_mem_id()); slab = cache_grow_begin(cache, flags, numa_mem_id());
cache_grow_end(cache, page); cache_grow_end(cache, slab);
if (page) { if (slab) {
nid = page_to_nid(page); nid = slab_nid(slab);
obj = ____cache_alloc_node(cache, obj = ____cache_alloc_node(cache,
gfp_exact_node(flags), nid); gfp_exact_node(flags), nid);
...@@ -3156,7 +3156,7 @@ static void *fallback_alloc(struct kmem_cache *cache, gfp_t flags) ...@@ -3156,7 +3156,7 @@ static void *fallback_alloc(struct kmem_cache *cache, gfp_t flags)
static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags, static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags,
int nodeid) int nodeid)
{ {
struct page *page; struct slab *slab;
struct kmem_cache_node *n; struct kmem_cache_node *n;
void *obj = NULL; void *obj = NULL;
void *list = NULL; void *list = NULL;
...@@ -3167,8 +3167,8 @@ static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags, ...@@ -3167,8 +3167,8 @@ static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags,
check_irq_off(); check_irq_off();
spin_lock(&n->list_lock); spin_lock(&n->list_lock);
page = get_first_slab(n, false); slab = get_first_slab(n, false);
if (!page) if (!slab)
goto must_grow; goto must_grow;
check_spinlock_acquired_node(cachep, nodeid); check_spinlock_acquired_node(cachep, nodeid);
...@@ -3177,12 +3177,12 @@ static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags, ...@@ -3177,12 +3177,12 @@ static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags,
STATS_INC_ACTIVE(cachep); STATS_INC_ACTIVE(cachep);
STATS_SET_HIGH(cachep); STATS_SET_HIGH(cachep);
BUG_ON(page->active == cachep->num); BUG_ON(slab->active == cachep->num);
obj = slab_get_obj(cachep, page); obj = slab_get_obj(cachep, slab);
n->free_objects--; n->free_objects--;
fixup_slab_list(cachep, n, page, &list); fixup_slab_list(cachep, n, slab, &list);
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
fixup_objfreelist_debug(cachep, &list); fixup_objfreelist_debug(cachep, &list);
...@@ -3190,12 +3190,12 @@ static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags, ...@@ -3190,12 +3190,12 @@ static void *____cache_alloc_node(struct kmem_cache *cachep, gfp_t flags,
must_grow: must_grow:
spin_unlock(&n->list_lock); spin_unlock(&n->list_lock);
page = cache_grow_begin(cachep, gfp_exact_node(flags), nodeid); slab = cache_grow_begin(cachep, gfp_exact_node(flags), nodeid);
if (page) { if (slab) {
/* This slab isn't counted yet so don't update free_objects */ /* This slab isn't counted yet so don't update free_objects */
obj = slab_get_obj(cachep, page); obj = slab_get_obj(cachep, slab);
} }
cache_grow_end(cachep, page); cache_grow_end(cachep, slab);
return obj ? obj : fallback_alloc(cachep, flags); return obj ? obj : fallback_alloc(cachep, flags);
} }
...@@ -3325,40 +3325,40 @@ static void free_block(struct kmem_cache *cachep, void **objpp, ...@@ -3325,40 +3325,40 @@ static void free_block(struct kmem_cache *cachep, void **objpp,
{ {
int i; int i;
struct kmem_cache_node *n = get_node(cachep, node); struct kmem_cache_node *n = get_node(cachep, node);
struct page *page; struct slab *slab;
n->free_objects += nr_objects; n->free_objects += nr_objects;
for (i = 0; i < nr_objects; i++) { for (i = 0; i < nr_objects; i++) {
void *objp; void *objp;
struct page *page; struct slab *slab;
objp = objpp[i]; objp = objpp[i];
page = virt_to_head_page(objp); slab = virt_to_slab(objp);
list_del(&page->slab_list); list_del(&slab->slab_list);
check_spinlock_acquired_node(cachep, node); check_spinlock_acquired_node(cachep, node);
slab_put_obj(cachep, page, objp); slab_put_obj(cachep, slab, objp);
STATS_DEC_ACTIVE(cachep); STATS_DEC_ACTIVE(cachep);
/* fixup slab chains */ /* fixup slab chains */
if (page->active == 0) { if (slab->active == 0) {
list_add(&page->slab_list, &n->slabs_free); list_add(&slab->slab_list, &n->slabs_free);
n->free_slabs++; n->free_slabs++;
} else { } else {
/* Unconditionally move a slab to the end of the /* Unconditionally move a slab to the end of the
* partial list on free - maximum time for the * partial list on free - maximum time for the
* other objects to be freed, too. * other objects to be freed, too.
*/ */
list_add_tail(&page->slab_list, &n->slabs_partial); list_add_tail(&slab->slab_list, &n->slabs_partial);
} }
} }
while (n->free_objects > n->free_limit && !list_empty(&n->slabs_free)) { while (n->free_objects > n->free_limit && !list_empty(&n->slabs_free)) {
n->free_objects -= cachep->num; n->free_objects -= cachep->num;
page = list_last_entry(&n->slabs_free, struct page, slab_list); slab = list_last_entry(&n->slabs_free, struct slab, slab_list);
list_move(&page->slab_list, list); list_move(&slab->slab_list, list);
n->free_slabs--; n->free_slabs--;
n->total_slabs--; n->total_slabs--;
} }
...@@ -3394,10 +3394,10 @@ static void cache_flusharray(struct kmem_cache *cachep, struct array_cache *ac) ...@@ -3394,10 +3394,10 @@ static void cache_flusharray(struct kmem_cache *cachep, struct array_cache *ac)
#if STATS #if STATS
{ {
int i = 0; int i = 0;
struct page *page; struct slab *slab;
list_for_each_entry(page, &n->slabs_free, slab_list) { list_for_each_entry(slab, &n->slabs_free, slab_list) {
BUG_ON(page->active); BUG_ON(slab->active);
i++; i++;
} }
...@@ -3473,10 +3473,10 @@ void ___cache_free(struct kmem_cache *cachep, void *objp, ...@@ -3473,10 +3473,10 @@ void ___cache_free(struct kmem_cache *cachep, void *objp,
} }
if (sk_memalloc_socks()) { if (sk_memalloc_socks()) {
struct page *page = virt_to_head_page(objp); struct slab *slab = virt_to_slab(objp);
if (unlikely(PageSlabPfmemalloc(page))) { if (unlikely(slab_test_pfmemalloc(slab))) {
cache_free_pfmemalloc(cachep, page, objp); cache_free_pfmemalloc(cachep, slab, objp);
return; return;
} }
} }
...@@ -3663,7 +3663,7 @@ void kmem_obj_info(struct kmem_obj_info *kpp, void *object, struct slab *slab) ...@@ -3663,7 +3663,7 @@ void kmem_obj_info(struct kmem_obj_info *kpp, void *object, struct slab *slab)
kpp->kp_data_offset = obj_offset(cachep); kpp->kp_data_offset = obj_offset(cachep);
slab = virt_to_slab(objp); slab = virt_to_slab(objp);
objnr = obj_to_index(cachep, slab_page(slab), objp); objnr = obj_to_index(cachep, slab_page(slab), objp);
objp = index_to_obj(cachep, slab_page(slab), objnr); objp = index_to_obj(cachep, slab, objnr);
kpp->kp_objp = objp; kpp->kp_objp = objp;
if (DEBUG && cachep->flags & SLAB_STORE_USER) if (DEBUG && cachep->flags & SLAB_STORE_USER)
kpp->kp_ret = *dbg_userword(cachep, objp); kpp->kp_ret = *dbg_userword(cachep, objp);
...@@ -4187,7 +4187,7 @@ void __check_heap_object(const void *ptr, unsigned long n, ...@@ -4187,7 +4187,7 @@ void __check_heap_object(const void *ptr, unsigned long n,
if (is_kfence_address(ptr)) if (is_kfence_address(ptr))
offset = ptr - kfence_object_start(ptr); offset = ptr - kfence_object_start(ptr);
else else
offset = ptr - index_to_obj(cachep, slab_page(slab), objnr) - obj_offset(cachep); offset = ptr - index_to_obj(cachep, slab, objnr) - obj_offset(cachep);
/* Allow address range falling entirely within usercopy region. */ /* Allow address range falling entirely within usercopy region. */
if (offset >= cachep->useroffset && if (offset >= cachep->useroffset &&
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment