mirror of
https://github.com/torvalds/linux.git
synced 2026-05-12 16:18:45 +02:00
Everything: Total patches: 368 Reviews/patch: 1.56 Reviewed rate: 74% Excluding DAMON: Total patches: 316 Reviews/patch: 1.77 Reviewed rate: 81% Excluding DAMON and zram: Total patches: 306 Reviews/patch: 1.81 Reviewed rate: 82% Excluding DAMON, zram and maple_tree: Total patches: 276 Reviews/patch: 2.01 Reviewed rate: 91% Significant patch series in this merge: - The 30 patch series "maple_tree: Replace big node with maple copy" from Liam Howlett is mainly prepararatory work for ongoing development but it does reduce stack usage and is an improvement. - The 12 patch series "mm, swap: swap table phase III: remove swap_map" from Kairui Song offers memory savings by removing the static swap_map. It also yields some CPU savings and implements several cleanups. - The 2 patch series "mm: memfd_luo: preserve file seals" from Pratyush Yadav adds file seal preservation to LUO's memfd code. - The 2 patch series "mm: zswap: add per-memcg stat for incompressible pages" from Jiayuan Chen adds additional userspace stats reportng to zswap. - The 4 patch series "arch, mm: consolidate empty_zero_page" from Mike Rapoport implements some cleanups for our handling of ZERO_PAGE() and zero_pfn. - The 2 patch series "mm/kmemleak: Improve scan_should_stop() implementation" from Zhongqiu Han provides an robustness improvement and some cleanups in the kmemleak code. - The 4 patch series "Improve khugepaged scan logic" from Vernon Yang "improves the khugepaged scan logic and reduces CPU consumption by prioritizing scanning tasks that access memory frequently". - The 2 patch series "Make KHO Stateless" from Jason Miu simplifies Kexec Handover by "transitioning KHO from an xarray-based metadata tracking system with serialization to a radix tree data structure that can be passed directly to the next kernel" - The 3 patch series "mm: vmscan: add PID and cgroup ID to vmscan tracepoints" from Thomas Ballasi and Steven Rostedt enhances vmscan's tracepointing. - The 5 patch series "mm: arch/shstk: Common shadow stack mapping helper and VM_NOHUGEPAGE" from Catalin Marinas is a cleanup for the shadow stack code: remove per-arch code in favour of a generic implementation. - The 2 patch series "Fix KASAN support for KHO restored vmalloc regions" from Pasha Tatashin fixes a WARN() which can be emitted the KHO restores a vmalloc area. - The 4 patch series "mm: Remove stray references to pagevec" from Tal Zussman provides several cleanups, mainly udpating references to "struct pagevec", which became folio_batch three years ago. - The 17 patch series "mm: Eliminate fake head pages from vmemmap optimization" from Kiryl Shutsemau simplifies the HugeTLB vmemmap optimization (HVO) by changing how tail pages encode their relationship to the head page. - The 2 patch series "mm/damon/core: improve DAMOS quota efficiency for core layer filters" from SeongJae Park improves two problematic behaviors of DAMOS that makes it less efficient when core layer filters are used. - The 3 patch series "mm/damon: strictly respect min_nr_regions" from SeongJae Park improves DAMON usability by extending the treatment of the min_nr_regions user-settable parameter. - The 3 patch series "mm/page_alloc: pcp locking cleanup" from Vlastimil Babka is a proper fix for a previously hotfixed SMP=n issue. Code simplifications and cleanups ennsed. - The 16 patch series "mm: cleanups around unmapping / zapping" from David Hildenbrand implements "a bunch of cleanups around unmapping and zapping. Mostly simplifications, code movements, documentation and renaming of zapping functions". - The 6 patch series "support batched checking of the young flag for MGLRU" from Baolin Wang supports batched checking of the young flag for MGLRU. It's part cleanups; one benchmark shows large performance benefits for arm64. - The 5 patch series "memcg: obj stock and slab stat caching cleanups" from Johannes Weiner provides memcg cleanup and robustness improvements. - The 5 patch series "Allow order zero pages in page reporting" from Yuvraj Sakshith enhances page_reporting's free page reporting - it is presently and undesirably order-0 pages when reporting free memory. - The 6 patch series "mm: vma flag tweaks" from Lorenzo Stoakes is cleanup work following from the recent conversion of the VMA flags to a bitmap. - The 10 patch series "mm/damon: add optional debugging-purpose sanity checks" from SeongJae Park adds some more developer-facing debug checks into DAMON core. - The 2 patch series "mm/damon: test and document power-of-2 min_region_sz requirement" from SeongJae Park adds an additional DAMON kunit test and makes some adjustments to the addr_unit parameter handling. - The 3 patch series "mm/damon/core: make passed_sample_intervals comparisons overflow-safe" from SeongJae Park fixes a hard-to-hit time overflow issue in DAMON core. - The 7 patch series "mm/damon: improve/fixup/update ratio calculation, test and documentation" from SeongJae Park is a "batch of misc/minor improvements and fixups" for DAMON. - The 4 patch series "mm: move vma_(kernel|mmu)_pagesize() out of hugetlb.c" from David Hildenbrand fixes a possible issue with dax-device when CONFIG_HUGETLB=n. Some code movement was required. - The 6 patch series "zram: recompression cleanups and tweaks" from Sergey Senozhatsky provides "a somewhat random mix of fixups, recompression cleanups and improvements" in the zram code. - The 11 patch series "mm/damon: support multiple goal-based quota tuning algorithms" from SeongJae Park extend DAMOS quotas goal auto-tuning to support multiple tuning algorithms that users can select. - The 4 patch series "mm: thp: reduce unnecessary start_stop_khugepaged()" from Breno Leitao fixes the khugpaged sysfs handling so we no longer spam the logs with reams of junk when starting/stopping khugepaged. - The 3 patch series "mm: improve map count checks" from Lorenzo Stoakes provides some cleanups and slight fixes in the mremap, mmap and vma code. - The 5 patch series "mm/damon: support addr_unit on default monitoring targets for modules" from SeongJae Park extends the use of DAMON core's addr_unit tunable. - The 5 patch series "mm: khugepaged cleanups and mTHP prerequisites" from Nico Pache provides cleanups in the khugepaged and is a base for Nico's planned khugepaged mTHP support. - The 15 patch series "mm: memory hot(un)plug and SPARSEMEM cleanups" from David Hildenbrand implements code movement and cleanups in the memhotplug and sparsemem code. - The 2 patch series "mm: remove CONFIG_ARCH_ENABLE_MEMORY_HOTREMOVE and cleanup CONFIG_MIGRATION" from David Hildenbrand rationalizes some memhotplug Kconfig support. - The 6 patch series "change young flag check functions to return bool" from Baolin Wang is "a cleanup patchset to change all young flag check functions to return bool". - The 3 patch series "mm/damon/sysfs: fix memory leak and NULL dereference issues" from Josh Law and SeongJae Park fixes a few potential DAMON bugs. - The 25 patch series "mm/vma: convert vm_flags_t to vma_flags_t in vma code" from "converts a lot of the existing use of the legacy vm_flags_t data type to the new vma_flags_t type which replaces it". Mainly in the vma code. - The 21 patch series "mm: expand mmap_prepare functionality and usage" from Lorenzo Stoakes "expands the mmap_prepare functionality, which is intended to replace the deprecated f_op->mmap hook which has been the source of bugs and security issues for some time". Cleanups, documentation, extension of mmap_prepare into filesystem drivers. - The 13 patch series "mm/huge_memory: refactor zap_huge_pmd()" from Lorenzo Stoakes simplifies and cleans up zap_huge_pmd(). Additional cleanups around vm_normal_folio_pmd() and the softleaf functionality are performed. -----BEGIN PGP SIGNATURE----- iHUEABYIAB0WIQTTMBEPP41GrTpTJgfdBJ7gKXxAjgUCad3HDQAKCRDdBJ7gKXxA jrUQAPwNhPk5nPSxnyxjAeQtOBHqgCdnICeEismLajPKd9aYRgEA0s2XAu3tSUYi GrBnWImHG3s4ePQxVcPCegWTsOUrXgQ= =1Q7o -----END PGP SIGNATURE----- Merge tag 'mm-stable-2026-04-13-21-45' of git://git.kernel.org/pub/scm/linux/kernel/git/akpm/mm Pull MM updates from Andrew Morton: - "maple_tree: Replace big node with maple copy" (Liam Howlett) Mainly prepararatory work for ongoing development but it does reduce stack usage and is an improvement. - "mm, swap: swap table phase III: remove swap_map" (Kairui Song) Offers memory savings by removing the static swap_map. It also yields some CPU savings and implements several cleanups. - "mm: memfd_luo: preserve file seals" (Pratyush Yadav) File seal preservation to LUO's memfd code - "mm: zswap: add per-memcg stat for incompressible pages" (Jiayuan Chen) Additional userspace stats reportng to zswap - "arch, mm: consolidate empty_zero_page" (Mike Rapoport) Some cleanups for our handling of ZERO_PAGE() and zero_pfn - "mm/kmemleak: Improve scan_should_stop() implementation" (Zhongqiu Han) A robustness improvement and some cleanups in the kmemleak code - "Improve khugepaged scan logic" (Vernon Yang) Improve khugepaged scan logic and reduce CPU consumption by prioritizing scanning tasks that access memory frequently - "Make KHO Stateless" (Jason Miu) Simplify Kexec Handover by transitioning KHO from an xarray-based metadata tracking system with serialization to a radix tree data structure that can be passed directly to the next kernel - "mm: vmscan: add PID and cgroup ID to vmscan tracepoints" (Thomas Ballasi and Steven Rostedt) Enhance vmscan's tracepointing - "mm: arch/shstk: Common shadow stack mapping helper and VM_NOHUGEPAGE" (Catalin Marinas) Cleanup for the shadow stack code: remove per-arch code in favour of a generic implementation - "Fix KASAN support for KHO restored vmalloc regions" (Pasha Tatashin) Fix a WARN() which can be emitted the KHO restores a vmalloc area - "mm: Remove stray references to pagevec" (Tal Zussman) Several cleanups, mainly udpating references to "struct pagevec", which became folio_batch three years ago - "mm: Eliminate fake head pages from vmemmap optimization" (Kiryl Shutsemau) Simplify the HugeTLB vmemmap optimization (HVO) by changing how tail pages encode their relationship to the head page - "mm/damon/core: improve DAMOS quota efficiency for core layer filters" (SeongJae Park) Improve two problematic behaviors of DAMOS that makes it less efficient when core layer filters are used - "mm/damon: strictly respect min_nr_regions" (SeongJae Park) Improve DAMON usability by extending the treatment of the min_nr_regions user-settable parameter - "mm/page_alloc: pcp locking cleanup" (Vlastimil Babka) The proper fix for a previously hotfixed SMP=n issue. Code simplifications and cleanups ensued - "mm: cleanups around unmapping / zapping" (David Hildenbrand) A bunch of cleanups around unmapping and zapping. Mostly simplifications, code movements, documentation and renaming of zapping functions - "support batched checking of the young flag for MGLRU" (Baolin Wang) Batched checking of the young flag for MGLRU. It's part cleanups; one benchmark shows large performance benefits for arm64 - "memcg: obj stock and slab stat caching cleanups" (Johannes Weiner) memcg cleanup and robustness improvements - "Allow order zero pages in page reporting" (Yuvraj Sakshith) Enhance free page reporting - it is presently and undesirably order-0 pages when reporting free memory. - "mm: vma flag tweaks" (Lorenzo Stoakes) Cleanup work following from the recent conversion of the VMA flags to a bitmap - "mm/damon: add optional debugging-purpose sanity checks" (SeongJae Park) Add some more developer-facing debug checks into DAMON core - "mm/damon: test and document power-of-2 min_region_sz requirement" (SeongJae Park) An additional DAMON kunit test and makes some adjustments to the addr_unit parameter handling - "mm/damon/core: make passed_sample_intervals comparisons overflow-safe" (SeongJae Park) Fix a hard-to-hit time overflow issue in DAMON core - "mm/damon: improve/fixup/update ratio calculation, test and documentation" (SeongJae Park) A batch of misc/minor improvements and fixups for DAMON - "mm: move vma_(kernel|mmu)_pagesize() out of hugetlb.c" (David Hildenbrand) Fix a possible issue with dax-device when CONFIG_HUGETLB=n. Some code movement was required. - "zram: recompression cleanups and tweaks" (Sergey Senozhatsky) A somewhat random mix of fixups, recompression cleanups and improvements in the zram code - "mm/damon: support multiple goal-based quota tuning algorithms" (SeongJae Park) Extend DAMOS quotas goal auto-tuning to support multiple tuning algorithms that users can select - "mm: thp: reduce unnecessary start_stop_khugepaged()" (Breno Leitao) Fix the khugpaged sysfs handling so we no longer spam the logs with reams of junk when starting/stopping khugepaged - "mm: improve map count checks" (Lorenzo Stoakes) Provide some cleanups and slight fixes in the mremap, mmap and vma code - "mm/damon: support addr_unit on default monitoring targets for modules" (SeongJae Park) Extend the use of DAMON core's addr_unit tunable - "mm: khugepaged cleanups and mTHP prerequisites" (Nico Pache) Cleanups to khugepaged and is a base for Nico's planned khugepaged mTHP support - "mm: memory hot(un)plug and SPARSEMEM cleanups" (David Hildenbrand) Code movement and cleanups in the memhotplug and sparsemem code - "mm: remove CONFIG_ARCH_ENABLE_MEMORY_HOTREMOVE and cleanup CONFIG_MIGRATION" (David Hildenbrand) Rationalize some memhotplug Kconfig support - "change young flag check functions to return bool" (Baolin Wang) Cleanups to change all young flag check functions to return bool - "mm/damon/sysfs: fix memory leak and NULL dereference issues" (Josh Law and SeongJae Park) Fix a few potential DAMON bugs - "mm/vma: convert vm_flags_t to vma_flags_t in vma code" (Lorenzo Stoakes) Convert a lot of the existing use of the legacy vm_flags_t data type to the new vma_flags_t type which replaces it. Mainly in the vma code. - "mm: expand mmap_prepare functionality and usage" (Lorenzo Stoakes) Expand the mmap_prepare functionality, which is intended to replace the deprecated f_op->mmap hook which has been the source of bugs and security issues for some time. Cleanups, documentation, extension of mmap_prepare into filesystem drivers - "mm/huge_memory: refactor zap_huge_pmd()" (Lorenzo Stoakes) Simplify and clean up zap_huge_pmd(). Additional cleanups around vm_normal_folio_pmd() and the softleaf functionality are performed. * tag 'mm-stable-2026-04-13-21-45' of git://git.kernel.org/pub/scm/linux/kernel/git/akpm/mm: (369 commits) mm: fix deferred split queue races during migration mm/khugepaged: fix issue with tracking lock mm/huge_memory: add and use has_deposited_pgtable() mm/huge_memory: add and use normal_or_softleaf_folio_pmd() mm: add softleaf_is_valid_pmd_entry(), pmd_to_softleaf_folio() mm/huge_memory: separate out the folio part of zap_huge_pmd() mm/huge_memory: use mm instead of tlb->mm mm/huge_memory: remove unnecessary sanity checks mm/huge_memory: deduplicate zap deposited table call mm/huge_memory: remove unnecessary VM_BUG_ON_PAGE() mm/huge_memory: add a common exit path to zap_huge_pmd() mm/huge_memory: handle buggy PMD entry in zap_huge_pmd() mm/huge_memory: have zap_huge_pmd return a boolean, add kdoc mm/huge: avoid big else branch in zap_huge_pmd() mm/huge_memory: simplify vma_is_specal_huge() mm: on remap assert that input range within the proposed VMA mm: add mmap_action_map_kernel_pages[_full]() uio: replace deprecated mmap hook with mmap_prepare in uio_info drivers: hv: vmbus: replace deprecated mmap hook with mmap_prepare mm: allow handling of stacked mmap_prepare hooks in more drivers ...
735 lines
20 KiB
C
735 lines
20 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef MM_SLAB_H
|
|
#define MM_SLAB_H
|
|
|
|
#include <linux/reciprocal_div.h>
|
|
#include <linux/list_lru.h>
|
|
#include <linux/local_lock.h>
|
|
#include <linux/random.h>
|
|
#include <linux/kobject.h>
|
|
#include <linux/sched/mm.h>
|
|
#include <linux/memcontrol.h>
|
|
#include <linux/kfence.h>
|
|
#include <linux/kasan.h>
|
|
|
|
/*
|
|
* Internal slab definitions
|
|
*/
|
|
|
|
#ifdef CONFIG_64BIT
|
|
# ifdef system_has_cmpxchg128
|
|
# define system_has_freelist_aba() system_has_cmpxchg128()
|
|
# define try_cmpxchg_freelist try_cmpxchg128
|
|
# endif
|
|
typedef u128 freelist_full_t;
|
|
#else /* CONFIG_64BIT */
|
|
# ifdef system_has_cmpxchg64
|
|
# define system_has_freelist_aba() system_has_cmpxchg64()
|
|
# define try_cmpxchg_freelist try_cmpxchg64
|
|
# endif
|
|
typedef u64 freelist_full_t;
|
|
#endif /* CONFIG_64BIT */
|
|
|
|
#if defined(system_has_freelist_aba) && !defined(CONFIG_HAVE_ALIGNED_STRUCT_PAGE)
|
|
#undef system_has_freelist_aba
|
|
#endif
|
|
|
|
/*
|
|
* Freelist pointer and counter to cmpxchg together, avoids the typical ABA
|
|
* problems with cmpxchg of just a pointer.
|
|
*/
|
|
struct freelist_counters {
|
|
union {
|
|
struct {
|
|
void *freelist;
|
|
union {
|
|
unsigned long counters;
|
|
struct {
|
|
unsigned inuse:16;
|
|
unsigned objects:15;
|
|
/*
|
|
* If slab debugging is enabled then the
|
|
* frozen bit can be reused to indicate
|
|
* that the slab was corrupted
|
|
*/
|
|
unsigned frozen:1;
|
|
#ifdef CONFIG_64BIT
|
|
/*
|
|
* Some optimizations use free bits in 'counters' field
|
|
* to save memory. In case ->stride field is not available,
|
|
* such optimizations are disabled.
|
|
*/
|
|
unsigned int stride;
|
|
#endif
|
|
};
|
|
};
|
|
};
|
|
#ifdef system_has_freelist_aba
|
|
freelist_full_t freelist_counters;
|
|
#endif
|
|
};
|
|
};
|
|
|
|
/* Reuses the bits in struct page */
|
|
struct slab {
|
|
memdesc_flags_t flags;
|
|
|
|
struct kmem_cache *slab_cache;
|
|
union {
|
|
struct {
|
|
struct list_head slab_list;
|
|
/* Double-word boundary */
|
|
struct freelist_counters;
|
|
};
|
|
struct rcu_head rcu_head;
|
|
};
|
|
|
|
unsigned int __page_type;
|
|
atomic_t __page_refcount;
|
|
#ifdef CONFIG_SLAB_OBJ_EXT
|
|
unsigned long obj_exts;
|
|
#endif
|
|
};
|
|
|
|
#define SLAB_MATCH(pg, sl) \
|
|
static_assert(offsetof(struct page, pg) == offsetof(struct slab, sl))
|
|
SLAB_MATCH(flags, flags);
|
|
SLAB_MATCH(compound_info, slab_cache); /* Ensure bit 0 is clear */
|
|
SLAB_MATCH(_refcount, __page_refcount);
|
|
#ifdef CONFIG_MEMCG
|
|
SLAB_MATCH(memcg_data, obj_exts);
|
|
#elif defined(CONFIG_SLAB_OBJ_EXT)
|
|
SLAB_MATCH(_unused_slab_obj_exts, obj_exts);
|
|
#endif
|
|
#undef SLAB_MATCH
|
|
static_assert(sizeof(struct slab) <= sizeof(struct page));
|
|
#if defined(system_has_freelist_aba)
|
|
static_assert(IS_ALIGNED(offsetof(struct slab, freelist), sizeof(struct freelist_counters)));
|
|
#endif
|
|
|
|
/**
|
|
* slab_folio - The folio allocated for a slab
|
|
* @s: The slab.
|
|
*
|
|
* Slabs are allocated as folios that contain the individual objects and are
|
|
* using some fields in the first struct page of the folio - those fields are
|
|
* now accessed by struct slab. It is occasionally necessary to convert back to
|
|
* a folio in order to communicate with the rest of the mm. Please use this
|
|
* helper function instead of casting yourself, as the implementation may change
|
|
* in the future.
|
|
*/
|
|
#define slab_folio(s) (_Generic((s), \
|
|
const struct slab *: (const struct folio *)s, \
|
|
struct slab *: (struct folio *)s))
|
|
|
|
/**
|
|
* page_slab - Converts from struct page to its slab.
|
|
* @page: A page which may or may not belong to a slab.
|
|
*
|
|
* Return: The slab which contains this page or NULL if the page does
|
|
* not belong to a slab. This includes pages returned from large kmalloc.
|
|
*/
|
|
static inline struct slab *page_slab(const struct page *page)
|
|
{
|
|
page = compound_head(page);
|
|
if (data_race(page->page_type >> 24) != PGTY_slab)
|
|
page = NULL;
|
|
|
|
return (struct slab *)page;
|
|
}
|
|
|
|
/**
|
|
* slab_page - The first struct page allocated for a slab
|
|
* @s: The slab.
|
|
*
|
|
* A convenience wrapper for converting slab to the first struct page of the
|
|
* underlying folio, to communicate with code not yet converted to folio or
|
|
* struct slab.
|
|
*/
|
|
#define slab_page(s) folio_page(slab_folio(s), 0)
|
|
|
|
static inline void *slab_address(const struct slab *slab)
|
|
{
|
|
return folio_address(slab_folio(slab));
|
|
}
|
|
|
|
static inline int slab_nid(const struct slab *slab)
|
|
{
|
|
return memdesc_nid(slab->flags);
|
|
}
|
|
|
|
static inline pg_data_t *slab_pgdat(const struct slab *slab)
|
|
{
|
|
return NODE_DATA(slab_nid(slab));
|
|
}
|
|
|
|
static inline struct slab *virt_to_slab(const void *addr)
|
|
{
|
|
return page_slab(virt_to_page(addr));
|
|
}
|
|
|
|
static inline int slab_order(const struct slab *slab)
|
|
{
|
|
return folio_order(slab_folio(slab));
|
|
}
|
|
|
|
static inline size_t slab_size(const struct slab *slab)
|
|
{
|
|
return PAGE_SIZE << slab_order(slab);
|
|
}
|
|
|
|
/*
|
|
* Word size structure that can be atomically updated or read and that
|
|
* contains both the order and the number of objects that a slab of the
|
|
* given order would contain.
|
|
*/
|
|
struct kmem_cache_order_objects {
|
|
unsigned int x;
|
|
};
|
|
|
|
struct kmem_cache_per_node_ptrs {
|
|
struct node_barn *barn;
|
|
struct kmem_cache_node *node;
|
|
};
|
|
|
|
/*
|
|
* Slab cache management.
|
|
*/
|
|
struct kmem_cache {
|
|
struct slub_percpu_sheaves __percpu *cpu_sheaves;
|
|
/* Used for retrieving partial slabs, etc. */
|
|
slab_flags_t flags;
|
|
unsigned long min_partial;
|
|
unsigned int size; /* Object size including metadata */
|
|
unsigned int object_size; /* Object size without metadata */
|
|
struct reciprocal_value reciprocal_size;
|
|
unsigned int offset; /* Free pointer offset */
|
|
unsigned int sheaf_capacity;
|
|
struct kmem_cache_order_objects oo;
|
|
|
|
/* Allocation and freeing of slabs */
|
|
struct kmem_cache_order_objects min;
|
|
gfp_t allocflags; /* gfp flags to use on each alloc */
|
|
int refcount; /* Refcount for slab cache destroy */
|
|
void (*ctor)(void *object); /* Object constructor */
|
|
unsigned int inuse; /* Offset to metadata */
|
|
unsigned int align; /* Alignment */
|
|
unsigned int red_left_pad; /* Left redzone padding size */
|
|
const char *name; /* Name (only for display!) */
|
|
struct list_head list; /* List of slab caches */
|
|
#ifdef CONFIG_SYSFS
|
|
struct kobject kobj; /* For sysfs */
|
|
#endif
|
|
#ifdef CONFIG_SLAB_FREELIST_HARDENED
|
|
unsigned long random;
|
|
#endif
|
|
|
|
#ifdef CONFIG_NUMA
|
|
/*
|
|
* Defragmentation by allocating from a remote node.
|
|
*/
|
|
unsigned int remote_node_defrag_ratio;
|
|
#endif
|
|
|
|
#ifdef CONFIG_SLAB_FREELIST_RANDOM
|
|
unsigned int *random_seq;
|
|
#endif
|
|
|
|
#ifdef CONFIG_KASAN_GENERIC
|
|
struct kasan_cache kasan_info;
|
|
#endif
|
|
|
|
#ifdef CONFIG_HARDENED_USERCOPY
|
|
unsigned int useroffset; /* Usercopy region offset */
|
|
unsigned int usersize; /* Usercopy region size */
|
|
#endif
|
|
|
|
#ifdef CONFIG_SLUB_STATS
|
|
struct kmem_cache_stats __percpu *cpu_stats;
|
|
#endif
|
|
|
|
struct kmem_cache_per_node_ptrs per_node[MAX_NUMNODES];
|
|
};
|
|
|
|
/*
|
|
* Every cache has !NULL s->cpu_sheaves but they may point to the
|
|
* bootstrap_sheaf temporarily during init, or permanently for the boot caches
|
|
* and caches with debugging enabled, or all caches with CONFIG_SLUB_TINY. This
|
|
* helper distinguishes whether cache has real non-bootstrap sheaves.
|
|
*/
|
|
static inline bool cache_has_sheaves(struct kmem_cache *s)
|
|
{
|
|
/* Test CONFIG_SLUB_TINY for code elimination purposes */
|
|
return !IS_ENABLED(CONFIG_SLUB_TINY) && s->sheaf_capacity;
|
|
}
|
|
|
|
#if defined(CONFIG_SYSFS) && !defined(CONFIG_SLUB_TINY)
|
|
#define SLAB_SUPPORTS_SYSFS 1
|
|
void sysfs_slab_unlink(struct kmem_cache *s);
|
|
void sysfs_slab_release(struct kmem_cache *s);
|
|
int sysfs_slab_alias(struct kmem_cache *s, const char *name);
|
|
#else
|
|
static inline void sysfs_slab_unlink(struct kmem_cache *s) { }
|
|
static inline void sysfs_slab_release(struct kmem_cache *s) { }
|
|
static inline int sysfs_slab_alias(struct kmem_cache *s, const char *name)
|
|
{ return 0; }
|
|
#endif
|
|
|
|
void *fixup_red_left(struct kmem_cache *s, void *p);
|
|
|
|
static inline void *nearest_obj(struct kmem_cache *cache,
|
|
const struct slab *slab, void *x)
|
|
{
|
|
void *object = x - (x - slab_address(slab)) % cache->size;
|
|
void *last_object = slab_address(slab) +
|
|
(slab->objects - 1) * cache->size;
|
|
void *result = (unlikely(object > last_object)) ? last_object : object;
|
|
|
|
result = fixup_red_left(cache, result);
|
|
return result;
|
|
}
|
|
|
|
/* Determine object index from a given position */
|
|
static inline unsigned int __obj_to_index(const struct kmem_cache *cache,
|
|
void *addr, const void *obj)
|
|
{
|
|
return reciprocal_divide(kasan_reset_tag(obj) - addr,
|
|
cache->reciprocal_size);
|
|
}
|
|
|
|
static inline unsigned int obj_to_index(const struct kmem_cache *cache,
|
|
const struct slab *slab, const void *obj)
|
|
{
|
|
if (is_kfence_address(obj))
|
|
return 0;
|
|
return __obj_to_index(cache, slab_address(slab), obj);
|
|
}
|
|
|
|
static inline int objs_per_slab(const struct kmem_cache *cache,
|
|
const struct slab *slab)
|
|
{
|
|
return slab->objects;
|
|
}
|
|
|
|
/*
|
|
* State of the slab allocator.
|
|
*
|
|
* This is used to describe the states of the allocator during bootup.
|
|
* Allocators use this to gradually bootstrap themselves. Most allocators
|
|
* have the problem that the structures used for managing slab caches are
|
|
* allocated from slab caches themselves.
|
|
*/
|
|
enum slab_state {
|
|
DOWN, /* No slab functionality yet */
|
|
PARTIAL, /* SLUB: kmem_cache_node available */
|
|
UP, /* Slab caches usable but not all extras yet */
|
|
FULL /* Everything is working */
|
|
};
|
|
|
|
extern enum slab_state slab_state;
|
|
|
|
/* The slab cache mutex protects the management structures during changes */
|
|
extern struct mutex slab_mutex;
|
|
|
|
/* The list of all slab caches on the system */
|
|
extern struct list_head slab_caches;
|
|
|
|
/* The slab cache that manages slab cache information */
|
|
extern struct kmem_cache *kmem_cache;
|
|
|
|
/* A table of kmalloc cache names and sizes */
|
|
extern const struct kmalloc_info_struct {
|
|
const char *name[NR_KMALLOC_TYPES];
|
|
unsigned int size;
|
|
} kmalloc_info[];
|
|
|
|
/* Kmalloc array related functions */
|
|
void setup_kmalloc_cache_index_table(void);
|
|
void create_kmalloc_caches(void);
|
|
|
|
extern u8 kmalloc_size_index[24];
|
|
|
|
static inline unsigned int size_index_elem(unsigned int bytes)
|
|
{
|
|
return (bytes - 1) / 8;
|
|
}
|
|
|
|
/*
|
|
* Find the kmem_cache structure that serves a given size of
|
|
* allocation
|
|
*
|
|
* This assumes size is larger than zero and not larger than
|
|
* KMALLOC_MAX_CACHE_SIZE and the caller must check that.
|
|
*/
|
|
static inline struct kmem_cache *
|
|
kmalloc_slab(size_t size, kmem_buckets *b, gfp_t flags, unsigned long caller)
|
|
{
|
|
unsigned int index;
|
|
|
|
if (!b)
|
|
b = &kmalloc_caches[kmalloc_type(flags, caller)];
|
|
if (size <= 192)
|
|
index = kmalloc_size_index[size_index_elem(size)];
|
|
else
|
|
index = fls(size - 1);
|
|
|
|
return (*b)[index];
|
|
}
|
|
|
|
gfp_t kmalloc_fix_flags(gfp_t flags);
|
|
|
|
/* Functions provided by the slab allocators */
|
|
int do_kmem_cache_create(struct kmem_cache *s, const char *name,
|
|
unsigned int size, struct kmem_cache_args *args,
|
|
slab_flags_t flags);
|
|
|
|
void __init kmem_cache_init(void);
|
|
extern void create_boot_cache(struct kmem_cache *, const char *name,
|
|
unsigned int size, slab_flags_t flags,
|
|
unsigned int useroffset, unsigned int usersize);
|
|
|
|
int slab_unmergeable(struct kmem_cache *s);
|
|
bool slab_args_unmergeable(struct kmem_cache_args *args, slab_flags_t flags);
|
|
|
|
slab_flags_t kmem_cache_flags(slab_flags_t flags, const char *name);
|
|
|
|
static inline bool is_kmalloc_cache(struct kmem_cache *s)
|
|
{
|
|
return (s->flags & SLAB_KMALLOC);
|
|
}
|
|
|
|
static inline bool is_kmalloc_normal(struct kmem_cache *s)
|
|
{
|
|
if (!is_kmalloc_cache(s))
|
|
return false;
|
|
return !(s->flags & (SLAB_CACHE_DMA|SLAB_ACCOUNT|SLAB_RECLAIM_ACCOUNT));
|
|
}
|
|
|
|
bool __kfree_rcu_sheaf(struct kmem_cache *s, void *obj);
|
|
void flush_all_rcu_sheaves(void);
|
|
void flush_rcu_sheaves_on_cache(struct kmem_cache *s);
|
|
|
|
#define SLAB_CORE_FLAGS (SLAB_HWCACHE_ALIGN | SLAB_CACHE_DMA | \
|
|
SLAB_CACHE_DMA32 | SLAB_PANIC | \
|
|
SLAB_TYPESAFE_BY_RCU | SLAB_DEBUG_OBJECTS | \
|
|
SLAB_NOLEAKTRACE | SLAB_RECLAIM_ACCOUNT | \
|
|
SLAB_TEMPORARY | SLAB_ACCOUNT | \
|
|
SLAB_NO_USER_FLAGS | SLAB_KMALLOC | SLAB_NO_MERGE)
|
|
|
|
#define SLAB_DEBUG_FLAGS (SLAB_RED_ZONE | SLAB_POISON | SLAB_STORE_USER | \
|
|
SLAB_TRACE | SLAB_CONSISTENCY_CHECKS)
|
|
|
|
#define SLAB_FLAGS_PERMITTED (SLAB_CORE_FLAGS | SLAB_DEBUG_FLAGS)
|
|
|
|
bool __kmem_cache_empty(struct kmem_cache *);
|
|
int __kmem_cache_shutdown(struct kmem_cache *);
|
|
void __kmem_cache_release(struct kmem_cache *);
|
|
int __kmem_cache_shrink(struct kmem_cache *);
|
|
void slab_kmem_cache_release(struct kmem_cache *);
|
|
|
|
struct seq_file;
|
|
struct file;
|
|
|
|
struct slabinfo {
|
|
unsigned long active_objs;
|
|
unsigned long num_objs;
|
|
unsigned long active_slabs;
|
|
unsigned long num_slabs;
|
|
unsigned long shared_avail;
|
|
unsigned int limit;
|
|
unsigned int batchcount;
|
|
unsigned int shared;
|
|
unsigned int objects_per_slab;
|
|
unsigned int cache_order;
|
|
};
|
|
|
|
void get_slabinfo(struct kmem_cache *s, struct slabinfo *sinfo);
|
|
|
|
#ifdef CONFIG_SLUB_DEBUG
|
|
#ifdef CONFIG_SLUB_DEBUG_ON
|
|
DECLARE_STATIC_KEY_TRUE(slub_debug_enabled);
|
|
#else
|
|
DECLARE_STATIC_KEY_FALSE(slub_debug_enabled);
|
|
#endif
|
|
extern void print_tracking(struct kmem_cache *s, void *object);
|
|
long validate_slab_cache(struct kmem_cache *s);
|
|
static inline bool __slub_debug_enabled(void)
|
|
{
|
|
return static_branch_unlikely(&slub_debug_enabled);
|
|
}
|
|
#else
|
|
static inline void print_tracking(struct kmem_cache *s, void *object)
|
|
{
|
|
}
|
|
static inline bool __slub_debug_enabled(void)
|
|
{
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Returns true if any of the specified slab_debug flags is enabled for the
|
|
* cache. Use only for flags parsed by setup_slub_debug() as it also enables
|
|
* the static key.
|
|
*/
|
|
static inline bool kmem_cache_debug_flags(struct kmem_cache *s, slab_flags_t flags)
|
|
{
|
|
if (IS_ENABLED(CONFIG_SLUB_DEBUG))
|
|
VM_WARN_ON_ONCE(!(flags & SLAB_DEBUG_FLAGS));
|
|
if (__slub_debug_enabled())
|
|
return s->flags & flags;
|
|
return false;
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_SLUB_DEBUG) && IS_ENABLED(CONFIG_KUNIT)
|
|
bool slab_in_kunit_test(void);
|
|
#else
|
|
static inline bool slab_in_kunit_test(void) { return false; }
|
|
#endif
|
|
|
|
/*
|
|
* slub is about to manipulate internal object metadata. This memory lies
|
|
* outside the range of the allocated object, so accessing it would normally
|
|
* be reported by kasan as a bounds error. metadata_access_enable() is used
|
|
* to tell kasan that these accesses are OK.
|
|
*/
|
|
static inline void metadata_access_enable(void)
|
|
{
|
|
kasan_disable_current();
|
|
kmsan_disable_current();
|
|
}
|
|
|
|
static inline void metadata_access_disable(void)
|
|
{
|
|
kmsan_enable_current();
|
|
kasan_enable_current();
|
|
}
|
|
|
|
#ifdef CONFIG_SLAB_OBJ_EXT
|
|
|
|
/*
|
|
* slab_obj_exts - get the pointer to the slab object extension vector
|
|
* associated with a slab.
|
|
* @slab: a pointer to the slab struct
|
|
*
|
|
* Returns the address of the object extension vector associated with the slab,
|
|
* or zero if no such vector has been associated yet.
|
|
* Do not dereference the return value directly; use get/put_slab_obj_exts()
|
|
* pair and slab_obj_ext() to access individual elements.
|
|
*
|
|
* Example usage:
|
|
*
|
|
* obj_exts = slab_obj_exts(slab);
|
|
* if (obj_exts) {
|
|
* get_slab_obj_exts(obj_exts);
|
|
* obj_ext = slab_obj_ext(slab, obj_exts, obj_to_index(s, slab, obj));
|
|
* // do something with obj_ext
|
|
* put_slab_obj_exts(obj_exts);
|
|
* }
|
|
*
|
|
* Note that the get/put semantics does not involve reference counting.
|
|
* Instead, it updates kasan/kmsan depth so that accesses to slabobj_ext
|
|
* won't be reported as access violations.
|
|
*/
|
|
static inline unsigned long slab_obj_exts(struct slab *slab)
|
|
{
|
|
unsigned long obj_exts = READ_ONCE(slab->obj_exts);
|
|
|
|
#ifdef CONFIG_MEMCG
|
|
/*
|
|
* obj_exts should be either NULL, a valid pointer with
|
|
* MEMCG_DATA_OBJEXTS bit set or be equal to OBJEXTS_ALLOC_FAIL.
|
|
*/
|
|
VM_BUG_ON_PAGE(obj_exts && !(obj_exts & MEMCG_DATA_OBJEXTS) &&
|
|
obj_exts != OBJEXTS_ALLOC_FAIL, slab_page(slab));
|
|
VM_BUG_ON_PAGE(obj_exts & MEMCG_DATA_KMEM, slab_page(slab));
|
|
#endif
|
|
|
|
return obj_exts & ~OBJEXTS_FLAGS_MASK;
|
|
}
|
|
|
|
static inline void get_slab_obj_exts(unsigned long obj_exts)
|
|
{
|
|
VM_WARN_ON_ONCE(!obj_exts);
|
|
metadata_access_enable();
|
|
}
|
|
|
|
static inline void put_slab_obj_exts(unsigned long obj_exts)
|
|
{
|
|
metadata_access_disable();
|
|
}
|
|
|
|
#ifdef CONFIG_64BIT
|
|
static inline void slab_set_stride(struct slab *slab, unsigned int stride)
|
|
{
|
|
slab->stride = stride;
|
|
}
|
|
static inline unsigned int slab_get_stride(struct slab *slab)
|
|
{
|
|
return slab->stride;
|
|
}
|
|
#else
|
|
static inline void slab_set_stride(struct slab *slab, unsigned int stride)
|
|
{
|
|
VM_WARN_ON_ONCE(stride != sizeof(struct slabobj_ext));
|
|
}
|
|
static inline unsigned int slab_get_stride(struct slab *slab)
|
|
{
|
|
return sizeof(struct slabobj_ext);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* slab_obj_ext - get the pointer to the slab object extension metadata
|
|
* associated with an object in a slab.
|
|
* @slab: a pointer to the slab struct
|
|
* @obj_exts: a pointer to the object extension vector
|
|
* @index: an index of the object
|
|
*
|
|
* Returns a pointer to the object extension associated with the object.
|
|
* Must be called within a section covered by get/put_slab_obj_exts().
|
|
*/
|
|
static inline struct slabobj_ext *slab_obj_ext(struct slab *slab,
|
|
unsigned long obj_exts,
|
|
unsigned int index)
|
|
{
|
|
struct slabobj_ext *obj_ext;
|
|
|
|
VM_WARN_ON_ONCE(obj_exts != slab_obj_exts(slab));
|
|
|
|
obj_ext = (struct slabobj_ext *)(obj_exts +
|
|
slab_get_stride(slab) * index);
|
|
return kasan_reset_tag(obj_ext);
|
|
}
|
|
|
|
int alloc_slab_obj_exts(struct slab *slab, struct kmem_cache *s,
|
|
gfp_t gfp, bool new_slab);
|
|
|
|
#else /* CONFIG_SLAB_OBJ_EXT */
|
|
|
|
static inline unsigned long slab_obj_exts(struct slab *slab)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static inline struct slabobj_ext *slab_obj_ext(struct slab *slab,
|
|
unsigned long obj_exts,
|
|
unsigned int index)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
static inline void slab_set_stride(struct slab *slab, unsigned int stride) { }
|
|
static inline unsigned int slab_get_stride(struct slab *slab) { return 0; }
|
|
|
|
|
|
#endif /* CONFIG_SLAB_OBJ_EXT */
|
|
|
|
static inline enum node_stat_item cache_vmstat_idx(struct kmem_cache *s)
|
|
{
|
|
return (s->flags & SLAB_RECLAIM_ACCOUNT) ?
|
|
NR_SLAB_RECLAIMABLE_B : NR_SLAB_UNRECLAIMABLE_B;
|
|
}
|
|
|
|
#ifdef CONFIG_MEMCG
|
|
bool __memcg_slab_post_alloc_hook(struct kmem_cache *s, struct list_lru *lru,
|
|
gfp_t flags, size_t size, void **p);
|
|
void __memcg_slab_free_hook(struct kmem_cache *s, struct slab *slab,
|
|
void **p, int objects, unsigned long obj_exts);
|
|
#endif
|
|
|
|
void kvfree_rcu_cb(struct rcu_head *head);
|
|
|
|
static inline unsigned int large_kmalloc_order(const struct page *page)
|
|
{
|
|
return page[1].flags.f & 0xff;
|
|
}
|
|
|
|
static inline size_t large_kmalloc_size(const struct page *page)
|
|
{
|
|
return PAGE_SIZE << large_kmalloc_order(page);
|
|
}
|
|
|
|
#ifdef CONFIG_SLUB_DEBUG
|
|
void dump_unreclaimable_slab(void);
|
|
#else
|
|
static inline void dump_unreclaimable_slab(void)
|
|
{
|
|
}
|
|
#endif
|
|
|
|
void ___cache_free(struct kmem_cache *cache, void *x, unsigned long addr);
|
|
|
|
#ifdef CONFIG_SLAB_FREELIST_RANDOM
|
|
int cache_random_seq_create(struct kmem_cache *cachep, unsigned int count,
|
|
gfp_t gfp);
|
|
void cache_random_seq_destroy(struct kmem_cache *cachep);
|
|
#else
|
|
static inline int cache_random_seq_create(struct kmem_cache *cachep,
|
|
unsigned int count, gfp_t gfp)
|
|
{
|
|
return 0;
|
|
}
|
|
static inline void cache_random_seq_destroy(struct kmem_cache *cachep) { }
|
|
#endif /* CONFIG_SLAB_FREELIST_RANDOM */
|
|
|
|
static inline bool slab_want_init_on_alloc(gfp_t flags, struct kmem_cache *c)
|
|
{
|
|
if (static_branch_maybe(CONFIG_INIT_ON_ALLOC_DEFAULT_ON,
|
|
&init_on_alloc)) {
|
|
if (c->ctor)
|
|
return false;
|
|
if (c->flags & (SLAB_TYPESAFE_BY_RCU | SLAB_POISON))
|
|
return flags & __GFP_ZERO;
|
|
return true;
|
|
}
|
|
return flags & __GFP_ZERO;
|
|
}
|
|
|
|
static inline bool slab_want_init_on_free(struct kmem_cache *c)
|
|
{
|
|
if (static_branch_maybe(CONFIG_INIT_ON_FREE_DEFAULT_ON,
|
|
&init_on_free))
|
|
return !(c->ctor ||
|
|
(c->flags & (SLAB_TYPESAFE_BY_RCU | SLAB_POISON)));
|
|
return false;
|
|
}
|
|
|
|
#if defined(CONFIG_DEBUG_FS) && defined(CONFIG_SLUB_DEBUG)
|
|
void debugfs_slab_release(struct kmem_cache *);
|
|
#else
|
|
static inline void debugfs_slab_release(struct kmem_cache *s) { }
|
|
#endif
|
|
|
|
#ifdef CONFIG_PRINTK
|
|
#define KS_ADDRS_COUNT 16
|
|
struct kmem_obj_info {
|
|
void *kp_ptr;
|
|
struct slab *kp_slab;
|
|
void *kp_objp;
|
|
unsigned long kp_data_offset;
|
|
struct kmem_cache *kp_slab_cache;
|
|
void *kp_ret;
|
|
void *kp_stack[KS_ADDRS_COUNT];
|
|
void *kp_free_stack[KS_ADDRS_COUNT];
|
|
};
|
|
void __kmem_obj_info(struct kmem_obj_info *kpp, void *object, struct slab *slab);
|
|
#endif
|
|
|
|
void __check_heap_object(const void *ptr, unsigned long n,
|
|
const struct slab *slab, bool to_user);
|
|
|
|
void defer_free_barrier(void);
|
|
|
|
static inline bool slub_debug_orig_size(struct kmem_cache *s)
|
|
{
|
|
return (kmem_cache_debug_flags(s, SLAB_STORE_USER) &&
|
|
(s->flags & SLAB_KMALLOC));
|
|
}
|
|
|
|
#ifdef CONFIG_SLUB_DEBUG
|
|
void skip_orig_size_check(struct kmem_cache *s, const void *object);
|
|
#endif
|
|
|
|
#endif /* MM_SLAB_H */
|