/*
 *  linux/mm/vmalloc.c
 *
 *  Copyright (C) 1993  Linus Torvalds
 */

#include <asm/system.h>
#include <linux/config.h>

#include <linux/signal.h>
#include <linux/sched.h>
#include <linux/head.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/types.h>
#include <linux/malloc.h>
#include <asm/segment.h>

//虚拟内存结构体定义
struct vm_struct {
    unsigned long flags;          //标志
    void * addr;                  //地址
    unsigned long size;           //长度
    struct vm_struct * next;      //下一个指针
};

static struct vm_struct * vmlist = NULL;

/* Just any arbitrary offset to the start of the vmalloc VM area: the
 * current 8MB value just means that there will be a 8MB "hole" after the
 * physical memory until the kernel virtual memory starts.  That means that
 * any out-of-bounds memory accesses will hopefully be caught.
 * The vmalloc() routines leaves a hole of 4kB between each vmalloced
 * area for the same reason. ;)
 */
#define VMALLOC_OFFSET    (8*1024*1024)

//设置页目录,实际上每个任务都知道其它任务的页目录
static inline void set_pgdir(unsigned long dindex, unsigned long value)
{
    struct task_struct * p;
    //取init任务
    p = &init_task;
    //每个任务中的指定索引设置为指定值
    do {
        ((unsigned long *) p->tss.cr3)[dindex] = value;
        p = p->next_task;
    } while (p != &init_task);
}

//释放区域页面
static int free_area_pages(unsigned long dindex, unsigned long index, unsigned long nr)
{
    unsigned long page, *pte;
    
    //找到页表
    if (!(PAGE_PRESENT & (page = swapper_pg_dir[dindex])))
        return 0;
    //
    page &= PAGE_MASK;
    //页表中索引
    pte = index + (unsigned long *) page;
    //遍历释放内存页
    do {
        unsigned long pg = *pte;
        *pte = 0;
        if (pg & PAGE_PRESENT)
            free_page(pg);
        pte++;
    } while (--nr);
    //指向页表
    pte = (unsigned long *) page;
    //页表中
    for (nr = 0 ; nr < 1024 ; nr++, pte++)
        if (*pte)
            return 0;
    //设置页目录表项
    set_pgdir(dindex,0);
    mem_map[MAP_NR(page)] = 1;
    //释放页表所在页面
    free_page(page);
    //刷新
    invalidate();
    return 0;
}

//分配区域页面
static int alloc_area_pages(unsigned long dindex, unsigned long index, unsigned long nr)
{
    unsigned long page, *pte;
    
    //根据索引从页目录中找到页表
    page = swapper_pg_dir[dindex];
    //如果页面不存在
    if (!page) {
        //从核心内存中申请一页内存
        page = get_free_page(GFP_KERNEL);
        //如果申请不到,则退出
        if (!page)
            return -ENOMEM;
        //如果能够获取到指定页面
        if (swapper_pg_dir[dindex]) {
            //则释放刚刚申请的页面
            free_page(page);
            //使用索引到的页面
            page = swapper_pg_dir[dindex];
        } else {
            //设置页面属性
            mem_map[MAP_NR(page)] = MAP_PAGE_RESERVED;
            set_pgdir(dindex, page | PAGE_SHARED);
        }
    }
    //已经获取到页面了
    //页面边界对齐
    page &= PAGE_MASK;
    //指定内存
    pte = index + (unsigned long *) page;
    //从竞争中移除,设置为共享
    *pte = PAGE_SHARED;        /* remove a race with vfree() */
    //循环申请内存,共计nr页
    do {
        //申请一页内存
        unsigned long pg = get_free_page(GFP_KERNEL);
        
        //申请失败,返回错误
        if (!pg)
            return -ENOMEM;
        //申请成功,设置属性
        *pte = pg | PAGE_SHARED;
        pte++;
    } while (--nr);
    invalidate();
    return 0;
}

//执行
static int do_area(void * addr, unsigned long size,
    int (*area_fn)(unsigned long,unsigned long,unsigned long))
{
    unsigned long nr, dindex, index;
    
    //内存页数
    nr = size >> PAGE_SHIFT;
    //页目录索引
    dindex = (TASK_SIZE + (unsigned long) addr) >> 22;
    //页表索引
    index = (((unsigned long) addr) >> PAGE_SHIFT) & (PTRS_PER_PAGE-1);
    //遍历处理,执行area_fn功能
    while (nr > 0) {
        unsigned long i = PTRS_PER_PAGE - index;

if (i > nr)
            i = nr;
        nr -= i;
        if (area_fn(dindex, index, i))
            return -1;
        index = 0;
        dindex++;
    }
    return 0;
}

//释放虚拟内存
void vfree(void * addr)
{
    struct vm_struct **p, *tmp;

if (!addr)
        return;
    if ((PAGE_SIZE-1) & (unsigned long) addr) {
        printk("Trying to vfree() bad address (%p)\n", addr);
        return;
    }
    for (p = &vmlist ; (tmp = *p) ; p = &tmp->next) {
        if (tmp->addr == addr) {
            *p = tmp->next;
            do_area(tmp->addr, tmp->size, free_area_pages);
            kfree(tmp);
            return;
        }
    }
    printk("Trying to vfree() nonexistent vm area (%p)\n", addr);
}

//分配虚拟内存
void * vmalloc(unsigned long size)
{
    void * addr;
    struct vm_struct **p, *tmp, *area;

size = PAGE_ALIGN(size);
    if (!size || size > high_memory)
        return NULL;
    area = (struct vm_struct *) kmalloc(sizeof(*area), GFP_KERNEL);
    if (!area)
        return NULL;
    addr = (void *) ((high_memory + VMALLOC_OFFSET) & ~(VMALLOC_OFFSET-1));
    area->size = size + PAGE_SIZE;
    area->next = NULL;
    for (p = &vmlist; (tmp = *p) ; p = &tmp->next) {
        if (size + (unsigned long) addr < (unsigned long) tmp->addr)
            break;
        addr = (void *) (tmp->size + (unsigned long) tmp->addr);
    }
    area->addr = addr;
    area->next = *p;
    *p = area;
    if (do_area(addr, size, alloc_area_pages)) {
        vfree(addr);
        return NULL;
    }
    return addr;
}

//读取
int vread(char *buf, char *addr, int count)
{
    struct vm_struct **p, *tmp;
    char *vaddr, *buf_start = buf;
    int n;

for (p = &vmlist; (tmp = *p) ; p = &tmp->next) {
        vaddr = (char *) tmp->addr;
        while (addr < vaddr) {
            if (count == 0)
                goto finished;
            put_fs_byte('\0', buf++), addr++, count--;
        }
        n = tmp->size - PAGE_SIZE;
        if (addr > vaddr)
            n -= addr - vaddr;
        while (--n >= 0) {
            if (count == 0)
                goto finished;
            put_fs_byte(*addr++, buf++), count--;
        }
    }
finished:
    return buf - buf_start;
}

mm/vmalloc.c的更多相关文章

  1. Memory Allocation API In Linux Kernel && Linux Userspace、kmalloc vmalloc Difference、Kernel Large Section Memory Allocation

    目录 . 内核态(ring0)内存申请和用户态(ring3)内存申请 . 内核态(ring0)内存申请:kmalloc/kfree.vmalloc/vfree . 用户态(ring3)内存申请:mal ...

  2. 高端内存映射之vmalloc分配内存中不连续的页--Linux内存管理(十九)

    1 内存中不连续的页的分配 根据上文的讲述, 我们知道物理上连续的映射对内核是最好的, 但并不总能成功地使用. 在分配一大块内存时, 可能竭尽全力也无法找到连续的内存块. 在用户空间中这不是问题,因为 ...

  3. vmalloc详解

    vmalloc是一个接口函数, 内核代码使用它来分配在虚拟内存中连续但在物理内存中不一定连续的内存. 只需要一个参数,以字节为单位. 使用vmalloc的最著名的实例是内核对模块的实现. 因为模块可能 ...

  4. xenomai安装

    一.Linux内核打实时补丁 1.将下载的Linux和xenomai安装包放在/usr/src目录下,并解压文件包,命令如下 tar xjf  Linux-3.8.13.tar.bz2   tar x ...

  5. Linux内核内存管理

    <Linux内核设计与实现>读书笔记(十二)- 内存管理   内核的内存使用不像用户空间那样随意,内核的内存出现错误时也只有靠自己来解决(用户空间的内存错误可以抛给内核来解决). 所有内核 ...

  6. 《Linux内核设计与实现》读书笔记(十二)- 内存管理【转】

    转自:http://www.cnblogs.com/wang_yb/archive/2013/05/23/3095907.html 内核的内存使用不像用户空间那样随意,内核的内存出现错误时也只有靠自己 ...

  7. systemtap 列出所有linux 内核模块与相关函数0

    diskiohttp://blog.163.com/digoal%40126/blog/static/16387704020131015105532435/ [root@localhost linux ...

  8. Linux内核入门到放弃-内存管理-《深入Linux内核架构》笔记

    概述 内存管理的实现涵盖了许多领域: 内存中的物理内存页管理 分配大块内存的伙伴系统 分配较小内存块的slab.slub和slob分配器 分配非连续内存块的vmalloc机制 进程的地址空间 在IA- ...

  9. 一个未完成的2.6.32-220内核踩内存crash分析记录

    遇到一个crash,log如下: BUG: unable to handle kernel NULL pointer dereference at (null) IP: [<ffffffff81 ...

随机推荐

  1. Linux基础※※※※如何使用Git in Linux(二)

    参考资料: 1. http://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000 2. Git-简 ...

  2. js继承之call,apply和prototype随谈

    在js中,call,apply和prototype都可以实现对象的继承,下面我们看一个例子: function FatherObj1() { this.sayhello = "I am jo ...

  3. 利用gitHub搭建博客

    ##1.gitHub Page的的使用我觉得这边博文写的很清楚,方法.步骤.优缺点以及实例,所以就借用一下啦^_^ [搭建一个免费的,无限流量的Blog](http://www.ruanyifeng. ...

  4. iOS - C 基本语法

    1.数据类型 2.常量 1)凡是整数数字常量一律是 int 型. 2)typedef 和 define. typedef 定义类型, #define 只是字符串替换. 3)在编写整形常量时,可以在常量 ...

  5. poj1625Censored!(AC自动机+dp)

    链接 第一次做这种题目,参考了下题解,相当于把树扯直了做DP,估计这一类题都是这个套路吧. 状态方程dp[i][next] = dp[i][next]+dp[i][j] ;dp[i][j]表示长度为i ...

  6. OpenGL的API函数使用手册

    (一)OpenGL函数库 格式: <库前缀><根命令><可选的参数个数><可选的参数类型> 库前缀有 gl.glu.aux.glut.wgl.glx.a ...

  7. AlwaysOn与数据库镜像端点问题

    今天在搭建一个测试环境的时候发现一个问题,我将AlwaysOn环境中某节点上的某个非可用性组里的数据库想实时备份到另外一台服务器上,因此我找了一个没有加域的工作组的服务器,与AlwaysOn主节点去搭 ...

  8. Android平板上开发应用的一点心得——精确适配不同的dpi和屏幕尺寸

    一.引言 Android的开源使厂商无需自行研发OS,大大降低了研发.生产的成本,使得Android平板品牌如雨后春笋般爆发,山寨机厂商们似乎又找到了一丝希望.与此同时带来的是广大开发者的苦不堪言,各 ...

  9. MySQL 循环执行kill语句杀掉连接

    mysql中创建索引时,一直卡住不动,刚开始以为可能是数据量太大导致的,可是一直等了很久还是一样没有反应,看来操作的表被锁住了,执行show processlist 果然发现有大量的lock,使用ki ...

  10. Hadoop技术内幕-第一章 阅读原代码前的准备

    1.1 源代码学习环境 1.1.1 基础软件的下载 JDK-http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads ...