LCOV - code coverage report
Current view: top level - include/linux - memory_hotplug.h (source / functions) Hit Total Coverage
Test: combined.info Lines: 8 11 72.7 %
Date: 2022-04-01 14:58:12 Functions: 0 0 -
Branches: 2 4 50.0 %

           Branch data     Line data    Source code
       1                 :            : /* SPDX-License-Identifier: GPL-2.0 */
       2                 :            : #ifndef __LINUX_MEMORY_HOTPLUG_H
       3                 :            : #define __LINUX_MEMORY_HOTPLUG_H
       4                 :            : 
       5                 :            : #include <linux/mmzone.h>
       6                 :            : #include <linux/spinlock.h>
       7                 :            : #include <linux/notifier.h>
       8                 :            : #include <linux/bug.h>
       9                 :            : 
      10                 :            : struct page;
      11                 :            : struct zone;
      12                 :            : struct pglist_data;
      13                 :            : struct mem_section;
      14                 :            : struct memory_block;
      15                 :            : struct resource;
      16                 :            : struct vmem_altmap;
      17                 :            : 
      18                 :            : #ifdef CONFIG_MEMORY_HOTPLUG
      19                 :            : /*
      20                 :            :  * Return page for the valid pfn only if the page is online. All pfn
      21                 :            :  * walkers which rely on the fully initialized page->flags and others
      22                 :            :  * should use this rather than pfn_valid && pfn_to_page
      23                 :            :  */
      24                 :            : #define pfn_to_online_page(pfn)                                    \
      25                 :            : ({                                                                 \
      26                 :            :         struct page *___page = NULL;                               \
      27                 :            :         unsigned long ___pfn = pfn;                                \
      28                 :            :         unsigned long ___nr = pfn_to_section_nr(___pfn);           \
      29                 :            :                                                                    \
      30                 :            :         if (___nr < NR_MEM_SECTIONS && online_section_nr(___nr) && \
      31                 :            :             pfn_valid_within(___pfn))                              \
      32                 :            :                 ___page = pfn_to_page(___pfn);                     \
      33                 :            :         ___page;                                                   \
      34                 :            : })
      35                 :            : 
      36                 :            : /*
      37                 :            :  * Types for free bootmem stored in page->lru.next. These have to be in
      38                 :            :  * some random range in unsigned long space for debugging purposes.
      39                 :            :  */
      40                 :            : enum {
      41                 :            :         MEMORY_HOTPLUG_MIN_BOOTMEM_TYPE = 12,
      42                 :            :         SECTION_INFO = MEMORY_HOTPLUG_MIN_BOOTMEM_TYPE,
      43                 :            :         MIX_SECTION_INFO,
      44                 :            :         NODE_INFO,
      45                 :            :         MEMORY_HOTPLUG_MAX_BOOTMEM_TYPE = NODE_INFO,
      46                 :            : };
      47                 :            : 
      48                 :            : /* Types for control the zone type of onlined and offlined memory */
      49                 :            : enum {
      50                 :            :         MMOP_OFFLINE = -1,
      51                 :            :         MMOP_ONLINE_KEEP,
      52                 :            :         MMOP_ONLINE_KERNEL,
      53                 :            :         MMOP_ONLINE_MOVABLE,
      54                 :            : };
      55                 :            : 
      56                 :            : /*
      57                 :            :  * Restrictions for the memory hotplug:
      58                 :            :  * flags:  MHP_ flags
      59                 :            :  * altmap: alternative allocator for memmap array
      60                 :            :  */
      61                 :            : struct mhp_restrictions {
      62                 :            :         unsigned long flags;
      63                 :            :         struct vmem_altmap *altmap;
      64                 :            : };
      65                 :            : 
      66                 :            : /*
      67                 :            :  * Zone resizing functions
      68                 :            :  *
      69                 :            :  * Note: any attempt to resize a zone should has pgdat_resize_lock()
      70                 :            :  * zone_span_writelock() both held. This ensure the size of a zone
      71                 :            :  * can't be changed while pgdat_resize_lock() held.
      72                 :            :  */
      73                 :            : static inline unsigned zone_span_seqbegin(struct zone *zone)
      74                 :            : {
      75                 :            :         return read_seqbegin(&zone->span_seqlock);
      76                 :            : }
      77                 :            : static inline int zone_span_seqretry(struct zone *zone, unsigned iv)
      78                 :            : {
      79                 :            :         return read_seqretry(&zone->span_seqlock, iv);
      80                 :            : }
      81                 :            : static inline void zone_span_writelock(struct zone *zone)
      82                 :            : {
      83                 :            :         write_seqlock(&zone->span_seqlock);
      84                 :            : }
      85                 :            : static inline void zone_span_writeunlock(struct zone *zone)
      86                 :            : {
      87                 :            :         write_sequnlock(&zone->span_seqlock);
      88                 :            : }
      89                 :            : static inline void zone_seqlock_init(struct zone *zone)
      90                 :            : {
      91                 :            :         seqlock_init(&zone->span_seqlock);
      92                 :            : }
      93                 :            : extern int zone_grow_free_lists(struct zone *zone, unsigned long new_nr_pages);
      94                 :            : extern int zone_grow_waitqueues(struct zone *zone, unsigned long nr_pages);
      95                 :            : extern int add_one_highpage(struct page *page, int pfn, int bad_ppro);
      96                 :            : /* VM interface that may be used by firmware interface */
      97                 :            : extern int online_pages(unsigned long pfn, unsigned long nr_pages,
      98                 :            :                         int online_type, int nid);
      99                 :            : extern struct zone *test_pages_in_a_zone(unsigned long start_pfn,
     100                 :            :                                          unsigned long end_pfn);
     101                 :            : extern unsigned long __offline_isolated_pages(unsigned long start_pfn,
     102                 :            :                                                 unsigned long end_pfn);
     103                 :            : 
     104                 :            : typedef void (*online_page_callback_t)(struct page *page, unsigned int order);
     105                 :            : 
     106                 :            : extern void generic_online_page(struct page *page, unsigned int order);
     107                 :            : extern int set_online_page_callback(online_page_callback_t callback);
     108                 :            : extern int restore_online_page_callback(online_page_callback_t callback);
     109                 :            : 
     110                 :            : extern int try_online_node(int nid);
     111                 :            : 
     112                 :            : extern int arch_add_memory(int nid, u64 start, u64 size,
     113                 :            :                         struct mhp_restrictions *restrictions);
     114                 :            : extern u64 max_mem_size;
     115                 :            : 
     116                 :            : extern bool memhp_auto_online;
     117                 :            : /* If movable_node boot option specified */
     118                 :            : extern bool movable_node_enabled;
     119                 :            : static inline bool movable_node_is_enabled(void)
     120                 :            : {
     121                 :            :         return movable_node_enabled;
     122                 :            : }
     123                 :            : 
     124                 :            : extern void arch_remove_memory(int nid, u64 start, u64 size,
     125                 :            :                                struct vmem_altmap *altmap);
     126                 :            : extern void __remove_pages(unsigned long start_pfn, unsigned long nr_pages,
     127                 :            :                            struct vmem_altmap *altmap);
     128                 :            : 
     129                 :            : /* reasonably generic interface to expand the physical pages */
     130                 :            : extern int __add_pages(int nid, unsigned long start_pfn, unsigned long nr_pages,
     131                 :            :                        struct mhp_restrictions *restrictions);
     132                 :            : 
     133                 :            : #ifndef CONFIG_ARCH_HAS_ADD_PAGES
     134                 :            : static inline int add_pages(int nid, unsigned long start_pfn,
     135                 :            :                 unsigned long nr_pages, struct mhp_restrictions *restrictions)
     136                 :            : {
     137                 :            :         return __add_pages(nid, start_pfn, nr_pages, restrictions);
     138                 :            : }
     139                 :            : #else /* ARCH_HAS_ADD_PAGES */
     140                 :            : int add_pages(int nid, unsigned long start_pfn, unsigned long nr_pages,
     141                 :            :               struct mhp_restrictions *restrictions);
     142                 :            : #endif /* ARCH_HAS_ADD_PAGES */
     143                 :            : 
     144                 :            : #ifdef CONFIG_NUMA
     145                 :            : extern int memory_add_physaddr_to_nid(u64 start);
     146                 :            : #else
     147                 :            : static inline int memory_add_physaddr_to_nid(u64 start)
     148                 :            : {
     149                 :            :         return 0;
     150                 :            : }
     151                 :            : #endif
     152                 :            : 
     153                 :            : #ifdef CONFIG_HAVE_ARCH_NODEDATA_EXTENSION
     154                 :            : /*
     155                 :            :  * For supporting node-hotadd, we have to allocate a new pgdat.
     156                 :            :  *
     157                 :            :  * If an arch has generic style NODE_DATA(),
     158                 :            :  * node_data[nid] = kzalloc() works well. But it depends on the architecture.
     159                 :            :  *
     160                 :            :  * In general, generic_alloc_nodedata() is used.
     161                 :            :  * Now, arch_free_nodedata() is just defined for error path of node_hot_add.
     162                 :            :  *
     163                 :            :  */
     164                 :            : extern pg_data_t *arch_alloc_nodedata(int nid);
     165                 :            : extern void arch_free_nodedata(pg_data_t *pgdat);
     166                 :            : extern void arch_refresh_nodedata(int nid, pg_data_t *pgdat);
     167                 :            : 
     168                 :            : #else /* CONFIG_HAVE_ARCH_NODEDATA_EXTENSION */
     169                 :            : 
     170                 :            : #define arch_alloc_nodedata(nid)        generic_alloc_nodedata(nid)
     171                 :            : #define arch_free_nodedata(pgdat)       generic_free_nodedata(pgdat)
     172                 :            : 
     173                 :            : #ifdef CONFIG_NUMA
     174                 :            : /*
     175                 :            :  * If ARCH_HAS_NODEDATA_EXTENSION=n, this func is used to allocate pgdat.
     176                 :            :  * XXX: kmalloc_node() can't work well to get new node's memory at this time.
     177                 :            :  *      Because, pgdat for the new node is not allocated/initialized yet itself.
     178                 :            :  *      To use new node's memory, more consideration will be necessary.
     179                 :            :  */
     180                 :            : #define generic_alloc_nodedata(nid)                             \
     181                 :            : ({                                                              \
     182                 :            :         kzalloc(sizeof(pg_data_t), GFP_KERNEL);                 \
     183                 :            : })
     184                 :            : /*
     185                 :            :  * This definition is just for error path in node hotadd.
     186                 :            :  * For node hotremove, we have to replace this.
     187                 :            :  */
     188                 :            : #define generic_free_nodedata(pgdat)    kfree(pgdat)
     189                 :            : 
     190                 :            : extern pg_data_t *node_data[];
     191                 :            : static inline void arch_refresh_nodedata(int nid, pg_data_t *pgdat)
     192                 :            : {
     193                 :            :         node_data[nid] = pgdat;
     194                 :            : }
     195                 :            : 
     196                 :            : #else /* !CONFIG_NUMA */
     197                 :            : 
     198                 :            : /* never called */
     199                 :            : static inline pg_data_t *generic_alloc_nodedata(int nid)
     200                 :            : {
     201                 :            :         BUG();
     202                 :            :         return NULL;
     203                 :            : }
     204                 :            : static inline void generic_free_nodedata(pg_data_t *pgdat)
     205                 :            : {
     206                 :            : }
     207                 :            : static inline void arch_refresh_nodedata(int nid, pg_data_t *pgdat)
     208                 :            : {
     209                 :            : }
     210                 :            : #endif /* CONFIG_NUMA */
     211                 :            : #endif /* CONFIG_HAVE_ARCH_NODEDATA_EXTENSION */
     212                 :            : 
     213                 :            : #ifdef CONFIG_HAVE_BOOTMEM_INFO_NODE
     214                 :            : extern void __init register_page_bootmem_info_node(struct pglist_data *pgdat);
     215                 :            : #else
     216                 :            : static inline void register_page_bootmem_info_node(struct pglist_data *pgdat)
     217                 :            : {
     218                 :            : }
     219                 :            : #endif
     220                 :            : extern void put_page_bootmem(struct page *page);
     221                 :            : extern void get_page_bootmem(unsigned long ingo, struct page *page,
     222                 :            :                              unsigned long type);
     223                 :            : 
     224                 :            : void get_online_mems(void);
     225                 :            : void put_online_mems(void);
     226                 :            : 
     227                 :            : void mem_hotplug_begin(void);
     228                 :            : void mem_hotplug_done(void);
     229                 :            : 
     230                 :            : #else /* ! CONFIG_MEMORY_HOTPLUG */
     231                 :            : #define pfn_to_online_page(pfn)                 \
     232                 :            : ({                                              \
     233                 :            :         struct page *___page = NULL;            \
     234                 :            :         if (pfn_valid(pfn))                     \
     235                 :            :                 ___page = pfn_to_page(pfn);     \
     236                 :            :         ___page;                                \
     237                 :            :  })
     238                 :            : 
     239                 :            : static inline unsigned zone_span_seqbegin(struct zone *zone)
     240                 :            : {
     241                 :            :         return 0;
     242                 :            : }
     243                 :            : static inline int zone_span_seqretry(struct zone *zone, unsigned iv)
     244                 :            : {
     245                 :            :         return 0;
     246                 :            : }
     247                 :            : static inline void zone_span_writelock(struct zone *zone) {}
     248                 :            : static inline void zone_span_writeunlock(struct zone *zone) {}
     249                 :         12 : static inline void zone_seqlock_init(struct zone *zone) {}
     250                 :            : 
     251                 :            : static inline int mhp_notimplemented(const char *func)
     252                 :            : {
     253                 :            :         printk(KERN_WARNING "%s() called, with CONFIG_MEMORY_HOTPLUG disabled\n", func);
     254                 :            :         dump_stack();
     255                 :            :         return -ENOSYS;
     256                 :            : }
     257                 :            : 
     258                 :          3 : static inline void register_page_bootmem_info_node(struct pglist_data *pgdat)
     259                 :            : {
     260                 :          3 : }
     261                 :            : 
     262                 :          0 : static inline int try_online_node(int nid)
     263                 :            : {
     264                 :          0 :         return 0;
     265                 :            : }
     266                 :            : 
     267                 :        546 : static inline void get_online_mems(void) {}
     268                 :        546 : static inline void put_online_mems(void) {}
     269                 :            : 
     270                 :            : static inline void mem_hotplug_begin(void) {}
     271                 :            : static inline void mem_hotplug_done(void) {}
     272                 :            : 
     273                 :       8685 : static inline bool movable_node_is_enabled(void)
     274                 :            : {
     275   [ -  +  -  + ]:       8685 :         return false;
     276                 :            : }
     277                 :            : #endif /* ! CONFIG_MEMORY_HOTPLUG */
     278                 :            : 
     279                 :            : #if defined(CONFIG_MEMORY_HOTPLUG) || defined(CONFIG_DEFERRED_STRUCT_PAGE_INIT)
     280                 :            : /*
     281                 :            :  * pgdat resizing functions
     282                 :            :  */
     283                 :            : static inline
     284                 :            : void pgdat_resize_lock(struct pglist_data *pgdat, unsigned long *flags)
     285                 :            : {
     286                 :            :         spin_lock_irqsave(&pgdat->node_size_lock, *flags);
     287                 :            : }
     288                 :            : static inline
     289                 :            : void pgdat_resize_unlock(struct pglist_data *pgdat, unsigned long *flags)
     290                 :            : {
     291                 :            :         spin_unlock_irqrestore(&pgdat->node_size_lock, *flags);
     292                 :            : }
     293                 :            : static inline
     294                 :            : void pgdat_resize_init(struct pglist_data *pgdat)
     295                 :            : {
     296                 :            :         spin_lock_init(&pgdat->node_size_lock);
     297                 :            : }
     298                 :            : #else /* !(CONFIG_MEMORY_HOTPLUG || CONFIG_DEFERRED_STRUCT_PAGE_INIT) */
     299                 :            : /*
     300                 :            :  * Stub functions for when hotplug is off
     301                 :            :  */
     302                 :            : static inline void pgdat_resize_lock(struct pglist_data *p, unsigned long *f) {}
     303                 :            : static inline void pgdat_resize_unlock(struct pglist_data *p, unsigned long *f) {}
     304                 :          3 : static inline void pgdat_resize_init(struct pglist_data *pgdat) {}
     305                 :            : #endif /* !(CONFIG_MEMORY_HOTPLUG || CONFIG_DEFERRED_STRUCT_PAGE_INIT) */
     306                 :            : 
     307                 :            : #ifdef CONFIG_MEMORY_HOTREMOVE
     308                 :            : 
     309                 :            : extern bool is_mem_section_removable(unsigned long pfn, unsigned long nr_pages);
     310                 :            : extern void try_offline_node(int nid);
     311                 :            : extern int offline_pages(unsigned long start_pfn, unsigned long nr_pages);
     312                 :            : extern int remove_memory(int nid, u64 start, u64 size);
     313                 :            : extern void __remove_memory(int nid, u64 start, u64 size);
     314                 :            : 
     315                 :            : #else
     316                 :            : static inline bool is_mem_section_removable(unsigned long pfn,
     317                 :            :                                         unsigned long nr_pages)
     318                 :            : {
     319                 :            :         return false;
     320                 :            : }
     321                 :            : 
     322                 :          0 : static inline void try_offline_node(int nid) {}
     323                 :            : 
     324                 :            : static inline int offline_pages(unsigned long start_pfn, unsigned long nr_pages)
     325                 :            : {
     326                 :            :         return -EINVAL;
     327                 :            : }
     328                 :            : 
     329                 :            : static inline int remove_memory(int nid, u64 start, u64 size)
     330                 :            : {
     331                 :            :         return -EBUSY;
     332                 :            : }
     333                 :            : 
     334                 :            : static inline void __remove_memory(int nid, u64 start, u64 size) {}
     335                 :            : #endif /* CONFIG_MEMORY_HOTREMOVE */
     336                 :            : 
     337                 :            : extern void set_zone_contiguous(struct zone *zone);
     338                 :            : extern void clear_zone_contiguous(struct zone *zone);
     339                 :            : 
     340                 :            : extern void __ref free_area_init_core_hotplug(int nid);
     341                 :            : extern int __add_memory(int nid, u64 start, u64 size);
     342                 :            : extern int add_memory(int nid, u64 start, u64 size);
     343                 :            : extern int add_memory_resource(int nid, struct resource *resource);
     344                 :            : extern void move_pfn_range_to_zone(struct zone *zone, unsigned long start_pfn,
     345                 :            :                 unsigned long nr_pages, struct vmem_altmap *altmap);
     346                 :            : extern void remove_pfn_range_from_zone(struct zone *zone,
     347                 :            :                                        unsigned long start_pfn,
     348                 :            :                                        unsigned long nr_pages);
     349                 :            : extern bool is_memblock_offlined(struct memory_block *mem);
     350                 :            : extern int sparse_add_section(int nid, unsigned long pfn,
     351                 :            :                 unsigned long nr_pages, struct vmem_altmap *altmap);
     352                 :            : extern void sparse_remove_section(struct mem_section *ms,
     353                 :            :                 unsigned long pfn, unsigned long nr_pages,
     354                 :            :                 unsigned long map_offset, struct vmem_altmap *altmap);
     355                 :            : extern struct page *sparse_decode_mem_map(unsigned long coded_mem_map,
     356                 :            :                                           unsigned long pnum);
     357                 :            : extern bool allow_online_pfn_range(int nid, unsigned long pfn, unsigned long nr_pages,
     358                 :            :                 int online_type);
     359                 :            : extern struct zone *zone_for_pfn_range(int online_type, int nid, unsigned start_pfn,
     360                 :            :                 unsigned long nr_pages);
     361                 :            : #endif /* __LINUX_MEMORY_HOTPLUG_H */

Generated by: LCOV version 1.14