• linux-4.19 内存之页面回收


    在linux 系统中,当有剩余内存,系统会尽可能的使用内存作为文件缓存(page cache)。文件缓存页面会加入到LRU 链表,当内存紧张,会将修改的文件缓存写入磁盘,释放缓存页面回收内存。linux 系统也会将磁盘等存储设备当做交换分区,内核将很少使用的内存换出到交换区,释放出物理内存,称之为页交换(swapping)。这些获取内存的机制统称页面回收(page reclaim)。

    LRU链表

    LRU是最近最少使用,在内存不足,这些页面就会被候选换出。

    kernel-4.19/include/linux/mmzone.h
    206/*
    207 * We do arithmetic on the LRU lists in various places in the code,
    208 * so it is important to keep the active lists LRU_ACTIVE higher in
    209 * the array than the corresponding inactive lists, and to keep
    210 * the *_FILE lists LRU_FILE higher than the corresponding _ANON lists.
    211 *
    212 * This has to be kept in sync with the statistics in zone_stat_item
    213 * above and the descriptions in vmstat_text in mm/vmstat.c
    214 */
    215#define LRU_BASE 0
    216#define LRU_ACTIVE 1
    217#define LRU_FILE 2
    218
    219enum lru_list {
    220    LRU_INACTIVE_ANON = LRU_BASE,
    221    LRU_ACTIVE_ANON = LRU_BASE + LRU_ACTIVE,
    222    LRU_INACTIVE_FILE = LRU_BASE + LRU_FILE,
    223    LRU_ACTIVE_FILE = LRU_BASE + LRU_FILE + LRU_ACTIVE,
    224    LRU_UNEVICTABLE,
    225    NR_LRU_LISTS
    226};

    255struct lruvec {
    256    struct list_head        lists[NR_LRU_LISTS];
    257    struct zone_reclaim_stat    reclaim_stat;
    258    /* Evictions & activations on the inactive file list */
    259    atomic_long_t            inactive_age;
    260    /* Refaults at the time of last reclaim cycle */
    261    unsigned long            refaults;
    262#ifdef CONFIG_MEMCG
    263    struct pglist_data *pgdat;
    264#endif
    265};

    每个zone (后面内核node)有一套LRU,LRU对应5个类型双向链表,分别表示匿名不活跃页面、匿名活跃页面、文件不活跃页面、文件活跃页面、不可回收页面。匿名页面总是需要写入交换区才能被换出,因此内存紧张时,优先换出page cache 页面,除非page cache 页面被修改了才需要写回磁盘。

    LRU 链表里面添加页面

    kernel-4.19/mm/swap.c

    428/**
    429 * lru_cache_add - add a page to a page list
    430 * @page: the page to be added to the LRU.
    431 *
    432 * Queue the page for addition to the LRU via pagevec. The decision on whether
    433 * to add the page to the [in]active [file|anon] list is deferred until the
    434 * pagevec is drained. This gives a chance for the caller of lru_cache_add()
    435 * have the page added to the active list using mark_page_accessed().
    436 */
    437void lru_cache_add(struct page *page)
    438{
    439    VM_BUG_ON_PAGE(PageActive(page) && PageUnevictable(page), page);
    440    VM_BUG_ON_PAGE(PageLRU(page), page);
    441    __lru_cache_add(page);
    442}


    399static void __lru_cache_add(struct page *page)
    400{
    401    struct pagevec *pvec = &get_cpu_var(lru_add_pvec);
    402
    403    get_page(page);
    404    if (!pagevec_add(pvec, page) || PageCompound(page))
    405        __pagevec_lru_add(pvec);
    406    put_cpu_var(lru_add_pvec);
    407}
     

    先获取pagevec 是否有空闲,如果空闲就加入pagerec,如果没有空闲就将之前加入pagevec 的页面加入lru ,将新的页面加入pagevec。

    lru 使用FIFO,另外在活跃跟不活跃lru 链表间切换,不活跃lru 链表尾最可能被换出。

    在lru 算法上,对即将换出的一个页面使用第二次机会算法,即根据页面PG_activi 、PG_referenced、PG_yong(硬件)再次判断是否换出。

    kswapd 内核线程


    kernel-4.19/mm/vmscan.c
    4299static int __init kswapd_init(void)
    4300{
    4301    int nid, ret;
    4302
    4303    swap_setup();
    4304    for_each_node_state(nid, N_MEMORY)
    4305         kswapd_run(nid);
    4306    ret = cpuhp_setup_state_nocalls(CPUHP_AP_ONLINE_DYN,
    4307                    "mm/vmscan:online", kswapd_cpu_online,
    4308                    NULL);
    4309    WARN_ON(ret < 0);
    4310    return 0;
    4311}
    4312
    4313module_init(kswapd_init)

    1
    4262/*
    4263 * This kswapd start function will be called by init and node-hot-add.
    4264 * On node-hot-add, kswapd will moved to proper cpus if cpus are hot-added.
    4265 */
    4266int kswapd_run(int nid)
    4267{
    4268    pg_data_t *pgdat = NODE_DATA(nid);
    4269    int ret = 0;
    4270
    4271    if (pgdat->kswapd)
    4272        return 0;
    4273
    4274    pgdat->kswapd = kthread_run(kswapd, pgdat, "kswapd%d", nid);
    4275    if (IS_ERR(pgdat->kswapd)) {
    4276        /* failure at boot is fatal */
    4277        BUG_ON(system_state < SYSTEM_RUNNING);
    4278        pr_err("Failed to start kswapd on node %d\n", nid);
    4279        ret = PTR_ERR(pgdat->kswapd);
    4280        pgdat->kswapd = NULL;
    4281    }
    4282    return ret;
    4283}

    alloc_page 在分配页面失败(ALLOC_WMARK_LOW),会通过wakeup_kswapd() 唤醒kswapd ,将zone_id 和 页面2^order 传递给kswapd。kswapd 对特定node 下 zone 进行扫描,在zone free 水位达到WMARK_HIGH,就会停止扫描。

    kswapd->balance_pgdat->pgdat_balanced


    3815/*
    3816 * For kswapd, balance_pgdat() will reclaim pages across a node from zones
    3817 * that are eligible for use by the caller until at least one zone is
    3818 * balanced.
    3819 *
    3820 * Returns the order kswapd finished reclaiming at.
    3821 *
    3822 * kswapd scans the zones in the highmem->normal->dma direction.  It skips
    3823 * zones which have free_pages > high_wmark_pages(zone), but once a zone is
    3824 * found to have free_pages <= high_wmark_pages(zone), any page is that zone
    3825 * or lower is eligible for reclaim until at least one usable zone is
    3826 * balanced.
    3827 */
    3828static int balance_pgdat(pg_data_t *pgdat, int order, int classzone_idx)
    3829{
    3830    int i;
    3831    unsigned long nr_soft_reclaimed;
    3832    unsigned long nr_soft_scanned;
    3833    unsigned long pflags;
    3834    struct zone *zone;
    3835    struct scan_control sc = {
    3836        .gfp_mask = GFP_KERNEL,
    3837        .order = order,
    3838        .priority = DEF_PRIORITY,
    3839        .may_writepage = !laptop_mode,
    3840        .may_unmap = 1,
    3841        .may_swap = 1,
    3842    };
    3843

    kswapd scans the zones in the highmem->normal->dma direction

    kswap 按照这个方向(buddy 系统分配方向),找到内存不平衡的zone,这里balanced_pages 平衡页面数大于zone 管理的页面managed_pages 的25% ,就认为平衡。

    再从相反的方向向不平衡zone,这样避免buddy 系统分配和回收锁竞争,调用kswap_shrink_zone 进行页面回收。

    最后调到shrink_zone 对lru 和 slab 交换和回收。

    页面迁移和内存规整

    migrate_pages 函数用于页面迁移,可以将一个进程的页面迁移到另外node 或地方,通常只对用户进程空间,修改页面映射表。

    在内存低于WMARK_LOW 分配失败,就会触发kswapd,如果还不能分配出需要内存,就会触发compact_zone 对页面迁移规整;

    内存规整:有两个方向扫描,一个从zone 头部向尾部zone 扫描,查找哪些是可以迁移的;另一个是从zone 尾部向zone 头部扫描,查找哪些页面是空闲的;当两个方向扫描在zone 中间碰头或者已经满足一个大块连续内存,就退出扫描。

    KSM 

    kernel samepage merging ,将内容相同的页面合并,这样就可以多份避免冗余物理内存。ksm 允许合并通一个进程或者不同进程内容相同的页面,合并到一个自读页面,从而释放物理内存,当修改页面内容时使用copy-on-write 复制。

    kernel-4.19/mm/ksm.c
    3143static int __init ksm_init(void)
    3144{
    3145    struct task_struct *ksm_thread;
    3146    int err;
    3147
    3148    /* The correct value depends on page size and endianness */
    3149    zero_checksum = calc_checksum(ZERO_PAGE(0));
    3150    /* Default to false for backwards compatibility */
    3151    ksm_use_zero_pages = false;
    3152
    3153    err = ksm_slab_init();
    3154    if (err)
    3155        goto out;
    3156
    3157    ksm_thread = kthread_run(ksm_scan_thread, NULL, "ksmd");
    3158    if (IS_ERR(ksm_thread)) {
    3159        pr_err("ksm: creating kthread failed\n");
    3160        err = PTR_ERR(ksm_thread);
    3161        goto out_free;
    3162    }
    3163
    3164#ifdef CONFIG_SYSFS
    3165    err = sysfs_create_group(mm_kobj, &ksm_attr_group);
    3166    if (err) {
    3167        pr_err("ksm: register sysfs failed\n");
    3168        kthread_stop(ksm_thread);
    3169        goto out_free;
    3170    }
    3171#else
    3172    ksm_run = KSM_RUN_MERGE;    /* no way for user to start it */
    3173
    3174#endif /* CONFIG_SYSFS */
    3175
    3176#ifdef CONFIG_MEMORY_HOTREMOVE
    3177    /* There is no significance to this priority 100 */
    3178    hotplug_memory_notifier(ksm_memory_callback, 100);
    3179#endif
    3180    return 0;
    3181
    3182out_free:
    3183    ksm_slab_free();
    3184out:
    3185    return err;
    3186}
    3187subsys_initcall(ksm_init);

    bionic/libc/bionic/mmap.cpp
    static bool kernel_has_MADV_MERGEABLE = true;

    int rc = madvise(result, size, MADV_MERGEABLE);#加入ksm

    madvise(result, size, MADV_UNMERGEABLE); #解除

    对应匿名页面分配或者mmap 分配私有页面,通过madvise 将分配的页面加入到ksm 系统,会唤醒ksmd 内核线程。ksm 当前不考虑 page cache 。

    一个典型程序可以由5 部分内存组成:

    可执行文件的内存映射 (page cache)

    程序分配的匿名页(如共享库)

    进程打开的文件映射

    进程访问文件系统产生的cache

    进程访问内核产生的内核buffer (如slab)

  • 相关阅读:
    【C++】面向对象之宏观部分——类的层次结构与互相关系(has_a、use_a、is_a)
    jmeter 用户自定义变量
    机器学习(四十五):使用Flask构建机器学习web
    【C语言】真假只知道0/1,肤浅了吧(每日小细节005)
    【前端】Vue+Element UI案例:通用后台管理系统-项目总结
    Javaweb之Vue指令的详细解析
    实习面试记录
    关于Django使用Jquery异步刷新
    【数据结构陈越版笔记】基础实验1-2.1:有序数组的插入
    java数组排序练习
  • 原文地址:https://blog.csdn.net/lei7143/article/details/125513944