mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-07 13:43:51 +00:00
mm: Convert struct page to struct slab in functions used by other subsystems
KASAN, KFENCE and memcg interact with SLAB or SLUB internals through functions nearest_obj(), obj_to_index() and objs_per_slab() that use struct page as parameter. This patch converts it to struct slab including all callers, through a coccinelle semantic patch. // Options: --include-headers --no-includes --smpl-spacing include/linux/slab_def.h include/linux/slub_def.h mm/slab.h mm/kasan/*.c mm/kfence/kfence_test.c mm/memcontrol.c mm/slab.c mm/slub.c // Note: needs coccinelle 1.1.1 to avoid breaking whitespace @@ @@ -objs_per_slab_page( +objs_per_slab( ... ) { ... } @@ @@ -objs_per_slab_page( +objs_per_slab( ... ) @@ identifier fn =~ "obj_to_index|objs_per_slab"; @@ fn(..., - const struct page *page + const struct slab *slab ,...) { <... ( - page_address(page) + slab_address(slab) | - page + slab ) ...> } @@ identifier fn =~ "nearest_obj"; @@ fn(..., - struct page *page + const struct slab *slab ,...) { <... ( - page_address(page) + slab_address(slab) | - page + slab ) ...> } @@ identifier fn =~ "nearest_obj|obj_to_index|objs_per_slab"; expression E; @@ fn(..., ( - slab_page(E) + E | - virt_to_page(E) + virt_to_slab(E) | - virt_to_head_page(E) + virt_to_slab(E) | - page + page_slab(page) ) ,...) Signed-off-by: Vlastimil Babka <vbabka@suse.cz> Reviewed-by: Andrey Konovalov <andreyknvl@gmail.com> Reviewed-by: Roman Gushchin <guro@fb.com> Acked-by: Johannes Weiner <hannes@cmpxchg.org> Cc: Julia Lawall <julia.lawall@inria.fr> Cc: Luis Chamberlain <mcgrof@kernel.org> Cc: Andrey Ryabinin <ryabinin.a.a@gmail.com> Cc: Alexander Potapenko <glider@google.com> Cc: Andrey Konovalov <andreyknvl@gmail.com> Cc: Dmitry Vyukov <dvyukov@google.com> Cc: Marco Elver <elver@google.com> Cc: Johannes Weiner <hannes@cmpxchg.org> Cc: Michal Hocko <mhocko@kernel.org> Cc: Vladimir Davydov <vdavydov.dev@gmail.com> Cc: <kasan-dev@googlegroups.com> Cc: <cgroups@vger.kernel.org>
This commit is contained in:
parent
dd35f71a1d
commit
40f3bf0cb0
@ -87,11 +87,11 @@ struct kmem_cache {
|
|||||||
struct kmem_cache_node *node[MAX_NUMNODES];
|
struct kmem_cache_node *node[MAX_NUMNODES];
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline void *nearest_obj(struct kmem_cache *cache, struct page *page,
|
static inline void *nearest_obj(struct kmem_cache *cache, const struct slab *slab,
|
||||||
void *x)
|
void *x)
|
||||||
{
|
{
|
||||||
void *object = x - (x - page->s_mem) % cache->size;
|
void *object = x - (x - slab->s_mem) % cache->size;
|
||||||
void *last_object = page->s_mem + (cache->num - 1) * cache->size;
|
void *last_object = slab->s_mem + (cache->num - 1) * cache->size;
|
||||||
|
|
||||||
if (unlikely(object > last_object))
|
if (unlikely(object > last_object))
|
||||||
return last_object;
|
return last_object;
|
||||||
@ -106,16 +106,16 @@ static inline void *nearest_obj(struct kmem_cache *cache, struct page *page,
|
|||||||
* reciprocal_divide(offset, cache->reciprocal_buffer_size)
|
* reciprocal_divide(offset, cache->reciprocal_buffer_size)
|
||||||
*/
|
*/
|
||||||
static inline unsigned int obj_to_index(const struct kmem_cache *cache,
|
static inline unsigned int obj_to_index(const struct kmem_cache *cache,
|
||||||
const struct page *page, void *obj)
|
const struct slab *slab, void *obj)
|
||||||
{
|
{
|
||||||
u32 offset = (obj - page->s_mem);
|
u32 offset = (obj - slab->s_mem);
|
||||||
return reciprocal_divide(offset, cache->reciprocal_buffer_size);
|
return reciprocal_divide(offset, cache->reciprocal_buffer_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int objs_per_slab_page(const struct kmem_cache *cache,
|
static inline int objs_per_slab(const struct kmem_cache *cache,
|
||||||
const struct page *page)
|
const struct slab *slab)
|
||||||
{
|
{
|
||||||
if (is_kfence_address(page_address(page)))
|
if (is_kfence_address(slab_address(slab)))
|
||||||
return 1;
|
return 1;
|
||||||
return cache->num;
|
return cache->num;
|
||||||
}
|
}
|
||||||
|
@ -158,11 +158,11 @@ static inline void sysfs_slab_release(struct kmem_cache *s)
|
|||||||
|
|
||||||
void *fixup_red_left(struct kmem_cache *s, void *p);
|
void *fixup_red_left(struct kmem_cache *s, void *p);
|
||||||
|
|
||||||
static inline void *nearest_obj(struct kmem_cache *cache, struct page *page,
|
static inline void *nearest_obj(struct kmem_cache *cache, const struct slab *slab,
|
||||||
void *x) {
|
void *x) {
|
||||||
void *object = x - (x - page_address(page)) % cache->size;
|
void *object = x - (x - slab_address(slab)) % cache->size;
|
||||||
void *last_object = page_address(page) +
|
void *last_object = slab_address(slab) +
|
||||||
(page->objects - 1) * cache->size;
|
(slab->objects - 1) * cache->size;
|
||||||
void *result = (unlikely(object > last_object)) ? last_object : object;
|
void *result = (unlikely(object > last_object)) ? last_object : object;
|
||||||
|
|
||||||
result = fixup_red_left(cache, result);
|
result = fixup_red_left(cache, result);
|
||||||
@ -178,16 +178,16 @@ static inline unsigned int __obj_to_index(const struct kmem_cache *cache,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline unsigned int obj_to_index(const struct kmem_cache *cache,
|
static inline unsigned int obj_to_index(const struct kmem_cache *cache,
|
||||||
const struct page *page, void *obj)
|
const struct slab *slab, void *obj)
|
||||||
{
|
{
|
||||||
if (is_kfence_address(obj))
|
if (is_kfence_address(obj))
|
||||||
return 0;
|
return 0;
|
||||||
return __obj_to_index(cache, page_address(page), obj);
|
return __obj_to_index(cache, slab_address(slab), obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int objs_per_slab_page(const struct kmem_cache *cache,
|
static inline int objs_per_slab(const struct kmem_cache *cache,
|
||||||
const struct page *page)
|
const struct slab *slab)
|
||||||
{
|
{
|
||||||
return page->objects;
|
return slab->objects;
|
||||||
}
|
}
|
||||||
#endif /* _LINUX_SLUB_DEF_H */
|
#endif /* _LINUX_SLUB_DEF_H */
|
||||||
|
@ -298,7 +298,7 @@ static inline u8 assign_tag(struct kmem_cache *cache,
|
|||||||
/* For caches that either have a constructor or SLAB_TYPESAFE_BY_RCU: */
|
/* For caches that either have a constructor or SLAB_TYPESAFE_BY_RCU: */
|
||||||
#ifdef CONFIG_SLAB
|
#ifdef CONFIG_SLAB
|
||||||
/* For SLAB assign tags based on the object index in the freelist. */
|
/* For SLAB assign tags based on the object index in the freelist. */
|
||||||
return (u8)obj_to_index(cache, virt_to_head_page(object), (void *)object);
|
return (u8)obj_to_index(cache, virt_to_slab(object), (void *)object);
|
||||||
#else
|
#else
|
||||||
/*
|
/*
|
||||||
* For SLUB assign a random tag during slab creation, otherwise reuse
|
* For SLUB assign a random tag during slab creation, otherwise reuse
|
||||||
@ -341,7 +341,7 @@ static inline bool ____kasan_slab_free(struct kmem_cache *cache, void *object,
|
|||||||
if (is_kfence_address(object))
|
if (is_kfence_address(object))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (unlikely(nearest_obj(cache, virt_to_head_page(object), object) !=
|
if (unlikely(nearest_obj(cache, virt_to_slab(object), object) !=
|
||||||
object)) {
|
object)) {
|
||||||
kasan_report_invalid_free(tagged_object, ip);
|
kasan_report_invalid_free(tagged_object, ip);
|
||||||
return true;
|
return true;
|
||||||
|
@ -339,7 +339,7 @@ static void __kasan_record_aux_stack(void *addr, bool can_alloc)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
cache = page->slab_cache;
|
cache = page->slab_cache;
|
||||||
object = nearest_obj(cache, page, addr);
|
object = nearest_obj(cache, page_slab(page), addr);
|
||||||
alloc_meta = kasan_get_alloc_meta(cache, object);
|
alloc_meta = kasan_get_alloc_meta(cache, object);
|
||||||
if (!alloc_meta)
|
if (!alloc_meta)
|
||||||
return;
|
return;
|
||||||
|
@ -249,7 +249,7 @@ static void print_address_description(void *addr, u8 tag)
|
|||||||
|
|
||||||
if (page && PageSlab(page)) {
|
if (page && PageSlab(page)) {
|
||||||
struct kmem_cache *cache = page->slab_cache;
|
struct kmem_cache *cache = page->slab_cache;
|
||||||
void *object = nearest_obj(cache, page, addr);
|
void *object = nearest_obj(cache, page_slab(page), addr);
|
||||||
|
|
||||||
describe_object(cache, object, addr, tag);
|
describe_object(cache, object, addr, tag);
|
||||||
}
|
}
|
||||||
|
@ -23,7 +23,7 @@ const char *kasan_get_bug_type(struct kasan_access_info *info)
|
|||||||
page = kasan_addr_to_page(addr);
|
page = kasan_addr_to_page(addr);
|
||||||
if (page && PageSlab(page)) {
|
if (page && PageSlab(page)) {
|
||||||
cache = page->slab_cache;
|
cache = page->slab_cache;
|
||||||
object = nearest_obj(cache, page, (void *)addr);
|
object = nearest_obj(cache, page_slab(page), (void *)addr);
|
||||||
alloc_meta = kasan_get_alloc_meta(cache, object);
|
alloc_meta = kasan_get_alloc_meta(cache, object);
|
||||||
|
|
||||||
if (alloc_meta) {
|
if (alloc_meta) {
|
||||||
|
@ -291,8 +291,8 @@ static void *test_alloc(struct kunit *test, size_t size, gfp_t gfp, enum allocat
|
|||||||
* even for KFENCE objects; these are required so that
|
* even for KFENCE objects; these are required so that
|
||||||
* memcg accounting works correctly.
|
* memcg accounting works correctly.
|
||||||
*/
|
*/
|
||||||
KUNIT_EXPECT_EQ(test, obj_to_index(s, page, alloc), 0U);
|
KUNIT_EXPECT_EQ(test, obj_to_index(s, page_slab(page), alloc), 0U);
|
||||||
KUNIT_EXPECT_EQ(test, objs_per_slab_page(s, page), 1);
|
KUNIT_EXPECT_EQ(test, objs_per_slab(s, page_slab(page)), 1);
|
||||||
|
|
||||||
if (policy == ALLOCATE_ANY)
|
if (policy == ALLOCATE_ANY)
|
||||||
return alloc;
|
return alloc;
|
||||||
|
@ -2819,7 +2819,7 @@ static inline void mod_objcg_mlstate(struct obj_cgroup *objcg,
|
|||||||
int memcg_alloc_page_obj_cgroups(struct page *page, struct kmem_cache *s,
|
int memcg_alloc_page_obj_cgroups(struct page *page, struct kmem_cache *s,
|
||||||
gfp_t gfp, bool new_page)
|
gfp_t gfp, bool new_page)
|
||||||
{
|
{
|
||||||
unsigned int objects = objs_per_slab_page(s, page);
|
unsigned int objects = objs_per_slab(s, page_slab(page));
|
||||||
unsigned long memcg_data;
|
unsigned long memcg_data;
|
||||||
void *vec;
|
void *vec;
|
||||||
|
|
||||||
@ -2881,7 +2881,7 @@ struct mem_cgroup *mem_cgroup_from_obj(void *p)
|
|||||||
struct obj_cgroup *objcg;
|
struct obj_cgroup *objcg;
|
||||||
unsigned int off;
|
unsigned int off;
|
||||||
|
|
||||||
off = obj_to_index(page->slab_cache, page, p);
|
off = obj_to_index(page->slab_cache, page_slab(page), p);
|
||||||
objcg = page_objcgs(page)[off];
|
objcg = page_objcgs(page)[off];
|
||||||
if (objcg)
|
if (objcg)
|
||||||
return obj_cgroup_memcg(objcg);
|
return obj_cgroup_memcg(objcg);
|
||||||
|
10
mm/slab.c
10
mm/slab.c
@ -1559,7 +1559,7 @@ static void check_poison_obj(struct kmem_cache *cachep, void *objp)
|
|||||||
struct slab *slab = virt_to_slab(objp);
|
struct slab *slab = virt_to_slab(objp);
|
||||||
unsigned int objnr;
|
unsigned int objnr;
|
||||||
|
|
||||||
objnr = obj_to_index(cachep, slab_page(slab), objp);
|
objnr = obj_to_index(cachep, slab, objp);
|
||||||
if (objnr) {
|
if (objnr) {
|
||||||
objp = index_to_obj(cachep, slab, objnr - 1);
|
objp = index_to_obj(cachep, slab, objnr - 1);
|
||||||
realobj = (char *)objp + obj_offset(cachep);
|
realobj = (char *)objp + obj_offset(cachep);
|
||||||
@ -2529,7 +2529,7 @@ static void *slab_get_obj(struct kmem_cache *cachep, struct slab *slab)
|
|||||||
static void slab_put_obj(struct kmem_cache *cachep,
|
static void slab_put_obj(struct kmem_cache *cachep,
|
||||||
struct slab *slab, void *objp)
|
struct slab *slab, void *objp)
|
||||||
{
|
{
|
||||||
unsigned int objnr = obj_to_index(cachep, slab_page(slab), objp);
|
unsigned int objnr = obj_to_index(cachep, slab, objp);
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
@ -2716,7 +2716,7 @@ 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, slab_page(slab), objp);
|
objnr = obj_to_index(cachep, slab, objp);
|
||||||
|
|
||||||
BUG_ON(objnr >= cachep->num);
|
BUG_ON(objnr >= cachep->num);
|
||||||
BUG_ON(objp != index_to_obj(cachep, slab, objnr));
|
BUG_ON(objp != index_to_obj(cachep, slab, objnr));
|
||||||
@ -3662,7 +3662,7 @@ void kmem_obj_info(struct kmem_obj_info *kpp, void *object, struct slab *slab)
|
|||||||
objp = object - obj_offset(cachep);
|
objp = object - obj_offset(cachep);
|
||||||
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, objp);
|
||||||
objp = index_to_obj(cachep, 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)
|
||||||
@ -4180,7 +4180,7 @@ void __check_heap_object(const void *ptr, unsigned long n,
|
|||||||
|
|
||||||
/* Find and validate object. */
|
/* Find and validate object. */
|
||||||
cachep = slab->slab_cache;
|
cachep = slab->slab_cache;
|
||||||
objnr = obj_to_index(cachep, slab_page(slab), (void *)ptr);
|
objnr = obj_to_index(cachep, slab, (void *)ptr);
|
||||||
BUG_ON(objnr >= cachep->num);
|
BUG_ON(objnr >= cachep->num);
|
||||||
|
|
||||||
/* Find offset within object. */
|
/* Find offset within object. */
|
||||||
|
@ -483,7 +483,7 @@ static inline void memcg_slab_post_alloc_hook(struct kmem_cache *s,
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
off = obj_to_index(s, page, p[i]);
|
off = obj_to_index(s, page_slab(page), p[i]);
|
||||||
obj_cgroup_get(objcg);
|
obj_cgroup_get(objcg);
|
||||||
page_objcgs(page)[off] = objcg;
|
page_objcgs(page)[off] = objcg;
|
||||||
mod_objcg_state(objcg, page_pgdat(page),
|
mod_objcg_state(objcg, page_pgdat(page),
|
||||||
@ -522,7 +522,7 @@ static inline void memcg_slab_free_hook(struct kmem_cache *s_orig,
|
|||||||
else
|
else
|
||||||
s = s_orig;
|
s = s_orig;
|
||||||
|
|
||||||
off = obj_to_index(s, page, p[i]);
|
off = obj_to_index(s, page_slab(page), p[i]);
|
||||||
objcg = objcgs[off];
|
objcg = objcgs[off];
|
||||||
if (!objcg)
|
if (!objcg)
|
||||||
continue;
|
continue;
|
||||||
|
@ -4342,7 +4342,7 @@ void kmem_obj_info(struct kmem_obj_info *kpp, void *object, struct slab *slab)
|
|||||||
#else
|
#else
|
||||||
objp = objp0;
|
objp = objp0;
|
||||||
#endif
|
#endif
|
||||||
objnr = obj_to_index(s, slab_page(slab), objp);
|
objnr = obj_to_index(s, slab, objp);
|
||||||
kpp->kp_data_offset = (unsigned long)((char *)objp0 - (char *)objp);
|
kpp->kp_data_offset = (unsigned long)((char *)objp0 - (char *)objp);
|
||||||
objp = base + s->size * objnr;
|
objp = base + s->size * objnr;
|
||||||
kpp->kp_objp = objp;
|
kpp->kp_objp = objp;
|
||||||
|
Loading…
Reference in New Issue
Block a user