alloc_bootmem_low_pages()定义在:include/linux/bootmem.h中,定义如下:

#define alloc_bootmem_low_pages(x) \

__alloc_bootmem_low(x, PAGE_SIZE, 0)

__alloc_bootmem_low()定义在:mm/bootmem.c,定义如下:

void * __init __alloc_bootmem_low(unsigned long size, unsigned long align,
                  unsigned long goal)
{
    bootmem_data_t *bdata;
    void *ptr;
    list_for_each_entry(bdata, &bdata_list, list) {
        ptr = __alloc_bootmem_core(bdata, size, align, goal,
                        ARCH_LOW_ADDRESS_LIMIT);
        if (ptr)
            return ptr;
    }
    printk(KERN_ALERT "low bootmem alloc of %lu bytes failed!\n", size);
    panic("Out of low memory");
    return NULL;
}

查找bdata链表上的每个节点,这每个节点代表一个bitmap,当然x86默认配置上只有一个节点,该节点在bootmem allocator初始化时创建,并添加到这个链表中。

核心函数还是__alloc_bootmem_core(),该函数定义于:mm/bootmem.c,定义如下:

void * __init
__alloc_bootmem_core(struct bootmem_data *bdata, unsigned long size,
          unsigned long align, unsigned long goal, unsigned long limit)
{
    unsigned long offset, remaining_size, areasize, preferred;
    unsigned long i, start = 0, incr, eidx, end_pfn;
    void *ret;

if (!size) {
        printk("__alloc_bootmem_core(): zero-sized request\n");
        BUG();
    }
    BUG_ON(align & (align-1));

if (limit && bdata->node_boot_start >= limit)
        return NULL;

/* on nodes without memory - bootmem_map is NULL */
    if (!bdata->node_bootmem_map)
        return NULL;

end_pfn = bdata->node_low_pfn;      // max_low_pfn
    limit = PFN_DOWN(limit);            // limit = 0xffffffff
    if (limit && end_pfn > limit)
        end_pfn = limit;

/*
     * 处理node_boot_start不对齐的情况。
     * @offset: offset + PFN_DOWN(bdata->node_boot_start)是align对齐的
     */
    eidx = end_pfn - PFN_DOWN(bdata->node_boot_start); // eidx = max_low_pfn
    offset = 0;
    if (align && (bdata->node_boot_start & (align - 1UL)) != 0)
        offset = align - (bdata->node_boot_start & (align - 1UL));
    offset = PFN_DOWN(offset);

/*
     * We try to allocate bootmem pages above 'goal'
     * first, then we try to allocate lower pages.
     */
    if (goal && goal >= bdata->node_boot_start && PFN_DOWN(goal) < end_pfn) {
        // 相对于node_boot_start的偏移量,只不过node_boot_start=0
        // 如果不是0,这里会出现歧义,因为在下面preferred将可能被赋值为last_success
        // 而这个success很显然不是某种偏移量,而是物理地址。
        preferred = goal - bdata->node_boot_start;

if (bdata->last_success >= preferred)
            if (!limit || (limit && limit > bdata->last_success))
                // last_success中保存的是上次成功时的起始地址
                preferred = bdata->last_success;
    } else
        preferred = 0;

/* 根据对齐调整要求,调整preferred, preferred是pfn */
    preferred = PFN_DOWN(ALIGN(preferred, align)) + offset;
    areasize = (size + PAGE_SIZE-1) / PAGE_SIZE;
    /* incr表示在查找内存时每次叠加几个页面 */
    incr = align >> PAGE_SHIFT ? : 1;

restart_scan:
    for (i = preferred; i < eidx; i += incr) {
        unsigned long j;
        i = find_next_zero_bit(bdata->node_bootmem_map, eidx, i);
        i = ALIGN(i, incr);
        if (i >= eidx)
            break;
        if (test_bit(i, bdata->node_bootmem_map))
            continue;

/* for循环中判断从i是否有areasize个连续的空闲页面,
         * 如果没有,则跳转到fail_block继续下一次的搜索. */
        for (j = i + 1; j < i + areasize; ++j) {
            if (j >= eidx)
                goto fail_block;
            if (test_bit(j, bdata->node_bootmem_map))
                goto fail_block;
        }
        start = i; /*如果在for循环中没有退出,则说明从第i个页面起一共有areasize个连续的页面可供使用*/
        goto found;
    fail_block:
        i = ALIGN(j, incr);
    }
    if (preferred > offset) {
        preferred = offset;
        /* 如果大于goal的内存不符合要求,转到从offset开始的内存开始搜索 */
        goto restart_scan;
    }
    return NULL;

found:
    bdata->last_success = PFN_PHYS(start);
    BUG_ON(start >= eidx);

/*找到合适页面后并没有马上把写页面返回,而是尝试着把这次申请的页面和上次申请的页面进行合并,
 * 以减少这两次内存之间的空隙(内存碎片),这样做的前提是:上次申请内存的最后一页和这次申请
 * 内存的第一页是连续的,在下面的if里就是判断这个前提,如果不连续,就不能进行合并了*/    
/* if语句判断这次找到和上次找到的是相邻的两页,这样才可能合并,否则进入else */
    if (align < PAGE_SIZE &&
        bdata->last_offset && bdata->last_pos+1 == start) {
        offset = ALIGN(bdata->last_offset, align);
        BUG_ON(offset > PAGE_SIZE);
        remaining_size = PAGE_SIZE - offset;
        if (size < remaining_size) { /*上次申请页面剩余的内存足够这次使用*/
            areasize = 0;
            /* last_pos unchanged */
            bdata->last_offset = offset + size;
            ret = phys_to_virt(bdata->last_pos * PAGE_SIZE +
                       offset +
                       bdata->node_boot_start);
        } else {                 /*上次申请页面剩余的内存不够这次使用*/
            remaining_size = size - remaining_size;
            areasize = (remaining_size + PAGE_SIZE-1) / PAGE_SIZE;
            ret = phys_to_virt(bdata->last_pos * PAGE_SIZE +
                       offset +
                       bdata->node_boot_start);
            bdata->last_pos = start + areasize - 1;
            bdata->last_offset = remaining_size;
        }
        bdata->last_offset &= ~PAGE_MASK;
    } else {
        bdata->last_pos = start + areasize - 1;
        bdata->last_offset = size & ~PAGE_MASK;
        ret = phys_to_virt(start * PAGE_SIZE + bdata->node_boot_start);
    }

for (i = start; i < start + areasize; i++)
        if (unlikely(test_and_set_bit(i, bdata->node_bootmem_map)))
            BUG();
    memset(ret, 0, size);
    return ret;
}

linux 内核 内存管理 bootmem alloctor 申请内存相关推荐

  1. 【Linux 内核 内存管理】内存管理架构 ② ( 用户空间内存管理 | malloc | ptmalloc | 内核空间内存管理 | sys_brk | sys_mmap | sys_munmap)

    文章目录 一.用户空间内存管理 ( malloc / free / ptmalloc / jemalloc / tcmalloc ) 二.内核空间内存管理 1.内核内存管理系统调用 ( sys_brk ...

  2. linux内存管理子系统采用基于内存区域,Linux 内存管理之highmem简介

    一.Linux内核地址空间 一般来说Linux 内核按照 3:1 的比率来划分虚拟内存(X86等):3 GB 的虚拟内存用于用户空间,1GB 的内存用于内核空间.当然有些体系结构如MIPS使用2:2 ...

  3. Linux内核页表管理-那些鲜为人知的秘密

    1.开场白 环境: 处理器架构:arm64 内核源码:linux-5.11 ubuntu版本:20.04.1 代码阅读工具:vim+ctags+cscope 通用操作系统,通常都会开启mmu来支持虚拟 ...

  4. Linux内核-进程管理

    Linux内核-进程管理 引言 本文主要介绍Linux内核进程管理相关知识,包括进程描述符.进程创建.销毁.状态.线程的实现以及Linux进程相关命令等. 进程描述符 内核把进程的列表存放在叫做任务队 ...

  5. JVM内存管理:深入Java内存区域与OOM

    Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述: 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝 ...

  6. Java内存管理:深入Java内存区域

    Java内存管理:深入Java内存区域 本文引用自:深入理解Java虚拟机的第2章内容 Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述 ...

  7. Linux 驱动开发 三十五:Linux 内核时钟管理

    参考: linux时间管理,时钟中断,系统节拍_u010936265的博客-CSDN博客_系统节拍时钟中断 Linux内核时钟系统和定时器实现_anonymalias的专栏-CSDN博客_linux内 ...

  8. 操作系统内存管理_操作系统6内存管理基础

    引言 花了一段时间才把之前的笔记整理了一部分,平时太忙也没啥时间.今天开始整理内存管理部分的,内存管理部分大致分为三部分笔记,第一部分就是本篇内存管理基础,第二部分是虚拟内存,第三部分高速缓存. 一个 ...

  9. JVM内存管理------JAVA语言的内存管理概述

    转载自  JVM内存管理------JAVA语言的内存管理概述 引言 内存管理一直是JAVA语言自豪与骄傲的资本,它让JAVA程序员基本上可以彻底忽略与内存管理相关的细节,只专注于业务逻辑.不过世界上 ...

  10. 挑战360无死角讲解Linux内核 进程管理,调度器的5种实现丨C++后端开发丨C/C++Linux服务器开发丨内核开发丨网络编程

    挑战360无死角讲解 进程管理,调度器的5种实现 1. 8500行 CFS是什么 2. RT调度器使用场景 3. IDLE/Dealine调度器 视频讲解如下,点击观看: 挑战360无死角讲解Linu ...

最新文章

  1. 字节流 system.in
  2. PostgreSQL参数优化对比性能测试
  3. 成功解决IndexError: shape mismatch: indexing arrays could not be broadcast together with shapes (100,)
  4. codeblocks全屏模式怎么退出_IntelliJ IDEA 2020.1 EAP2 发布:新增禅模式和 LightEdit 模式...
  5. POI的入门:绘制图形
  6. Android之android.os.NewWorkOnMainThreadException解决办法
  7. java的构造函数格式_来自模板化对象的Java 8函数构造函数
  8. 二分查找(递归和非递归实现)
  9. set, unordered_set模板类
  10. mysql做wp网站_mysql做wp网站
  11. foreach语句使用总结
  12. 苹果mac 3D游戏动画开发软件:Unity Pro
  13. mac上配置java jdk环境
  14. 国际c语言乱码大赛PDF,国际C 语言乱码大赛(IOCCC)获奖作品
  15. 苹果 开发者账号如下
  16. 国产无线充电宝哪个牌子好?国产无线充电宝品牌排行
  17. cocos2d-x传智播客_10年和超过520集播客-科技是一场马拉松,而不是短跑
  18. getvod.php_PHPvod模板开发手册PHPvod模板开发手册.pdf
  19. 最先进的微型计算机,微型计算机中的先进计算机技术.ppt
  20. Windows 10 1809 MSDN 版本下载 ed2k 链接

热门文章

  1. ]flume高并发优化——(1)load_balance
  2. React服务端渲染Next.js 8发布,新增无服务器功能
  3. NetBeans在Apache基金会取得的进展
  4. SpringMVC一路总结(一)
  5. 进程间通信之管道与有名管道
  6. LINUX 命令手册
  7. lvs-rrd 监控LVS
  8. HP刀片带外管理系统OA各功能实例示范
  9. redis源码编译和调试
  10. 已经有了Thread为什么还要Runnable