#include <linux/errno.h>
#include <linux/kernel.h>
#include <asm/segment.h>
#include <linux/mm.h>        /* defines GFP_KERNEL */
#include <linux/string.h>
#include <linux/module.h>
#include <linux/sched.h>
#include <linux/malloc.h>

struct module *module_list = NULL;
int freeing_modules;        /* true if some modules are marked for deletion */

struct module *find_module( const char *name);
int get_mod_name( char *user_name, char *buf);
int free_modules( void);

/*
 * Allocate space for a module.
 */
//系统调用,创建module,为module分配空间
asmlinkage int
sys_create_module(char *module_name, unsigned long size)
{
    int npages;
    void* addr;
    int len;
    char name[MOD_MAX_NAME];
    char *savename;
    struct module *mp;
    int error;

//如果不是超级用户root,则退出
    if (!suser())
        return -EPERM;
    //如果模块的名称为空或者长度为0,则退出
    if (module_name == NULL || size == 0)
        return -EINVAL;
    //根据模块的名称,获取name
    if ((error = get_mod_name(module_name, name)) != 0)
        return error;
    //搜索module
    if (find_module(name) != NULL) {
        return -EEXIST;
    }
    len = strlen(name) + 1;
    //分配空间
    if ((savename = (char*) kmalloc(len, GFP_KERNEL)) == NULL)
        return -ENOMEM;
    //拷贝名称
    memcpy(savename, name, len);
    //申请空间
    if ((mp = (struct module*) kmalloc(sizeof *mp, GFP_KERNEL)) == NULL) {
        kfree(savename);
        return -ENOMEM;
    }
    //计算申请的页数
    npages = (size + sizeof (int) + 4095) / 4096;
    if ((addr = vmalloc(npages * 4096)) == 0) {
        kfree_s(mp, sizeof *mp);
        kfree(savename);
        return -ENOMEM;
    }
    mp->name = savename;
    mp->size = npages;
    mp->addr = addr;
    mp->state = MOD_UNINITIALIZED;
    * (int *) addr = 0;        /* set use count to zero */
    mp->cleanup = NULL;
    mp->next = module_list;
    module_list = mp;
    printk("module `%s' (%lu pages @ 0x%08lx) created\n",
        mp->name, (unsigned long) mp->size, (unsigned long) mp->addr);
    return (int) addr;
}

/*
 * Initialize a module.
 */
//初始化module
asmlinkage int
sys_init_module(char *module_name, char *code, unsigned codesize,
        struct mod_routines *routines)
{
    struct module *mp;
    char name[MOD_MAX_NAME];
    int error;
    struct mod_routines rt;

if (!suser())
        return -EPERM;
    /*
     * First reclaim any memory from dead modules that where not
     * freed when deleted. Should I think be done by timers when
     * the module was deleted - Jon.
     */
    free_modules();

if ((error = get_mod_name(module_name, name)) != 0)
        return error;
    printk( "initializing module `%s', %d (0x%x) bytes\n",
        name, codesize, codesize);
    memcpy_fromfs(&rt, routines, sizeof rt);
    if ((mp = find_module(name)) == NULL)
        return -ENOENT;
    if ((codesize + sizeof (int) + 4095) / 4096 > mp->size)
        return -EINVAL;
    memcpy_fromfs((char *)mp->addr + sizeof (int), code, codesize);
    memset((char *)mp->addr + sizeof (int) + codesize, 0,
        mp->size * 4096 - (codesize + sizeof (int)));
    printk( "  init entry @ 0x%08lx, cleanup entry @ 0x%08lx\n",
        (unsigned long) rt.init, (unsigned long) rt.cleanup);
    mp->cleanup = rt.cleanup;
    if ((*rt.init)() != 0)
        return -EBUSY;
    mp->state = MOD_RUNNING;
    return 0;
}

//删除module
asmlinkage int
sys_delete_module(char *module_name)
{
    struct module *mp;
    char name[MOD_MAX_NAME];
    int error;

if (!suser())
        return -EPERM;
    if (module_name != NULL) {
        if ((error = get_mod_name(module_name, name)) != 0)
            return error;
        if ((mp = find_module(name)) == NULL)
            return -ENOENT;
        if (mp->state == MOD_RUNNING)
            (*mp->cleanup)();
        mp->state = MOD_DELETED;
    }
    free_modules();
    return 0;
}

/*
 * Copy the kernel symbol table to user space.  If the argument is null,
 * just return the size of the table.
 */
//获取内核符号表
asmlinkage int
sys_get_kernel_syms(struct kernel_sym *table)
{
    struct symbol {
        unsigned long addr;
        char *name;
    };
    extern int symbol_table_size;
    extern struct symbol symbol_table[];
    int i;
    struct symbol *from;
    struct kernel_sym *to;
    struct kernel_sym sym;

if (table != NULL) {
        from = symbol_table;
        to = table;
        i = verify_area(VERIFY_WRITE, to, symbol_table_size * sizeof *table);
        if (i)
            return i;
        for (i = symbol_table_size ; --i >= 0 ; ) {
            sym.value = from->addr;
            strncpy(sym.name, from->name, sizeof sym.name);
            memcpy_tofs(to, &sym, sizeof sym);
            from++, to++;
        }
    }
    return symbol_table_size;
}

/*
 * Copy the name of a module from user space.
 */
 //获取mod的名称
int
get_mod_name(char *user_name, char *buf)
{
    int i;

i = 0;
    for (i = 0 ; (buf[i] = get_fs_byte(user_name + i)) != '\0' ; ) {
        if (++i >= MOD_MAX_NAME)
            return -E2BIG;
    }
    return 0;
}

/*
 * Look for a module by name, ignoring modules marked for deletion.
 */
 //搜索module
struct module *
find_module( const char *name)
{
    struct module *mp;

for (mp = module_list ; mp ; mp = mp->next) {
        if (mp->state == MOD_DELETED)
            continue;
        if (!strcmp(mp->name, name))
            break;
    }
    return mp;
}

/*
 * Try to free modules which have been marked for deletion.  Returns nonzero
 * if a module was actually freed.
 */
 //释放module
int
free_modules( void)
{
    struct module *mp;
    struct module **mpp;
    int did_deletion;

did_deletion = 0;
    freeing_modules = 0;
    mpp = &module_list;
    while ((mp = *mpp) != NULL) {
        if (mp->state != MOD_DELETED) {
            mpp = &mp->next;
        } else if (GET_USE_COUNT(mp) != 0) {
            freeing_modules = 1;
            mpp = &mp->next;
        } else {    /* delete it */
            *mpp = mp->next;
            vfree(mp->addr);
            kfree(mp->name);
            kfree_s(mp, sizeof *mp);
            did_deletion = 1;
        }
    }
    return did_deletion;
}

/*
 * Called by the /proc file system to return a current list of modules.
 */
 //获取所有的module
int get_module_list(char *buf)
{
    char *p;
    char *q;
    int i;
    struct module *mp;
    char size[32];

p = buf;
    for (mp = module_list ; mp ; mp = mp->next) {
        if (p - buf > 4096 - 100)
            break;            /* avoid overflowing buffer */
        q = mp->name;
        i = 20;
        while (*q) {
            *p++ = *q++;
            i--;
        }
        sprintf(size, "%d", mp->size);
        i -= strlen(size);
        if (i <= 0)
            i = 1;
        while (--i >= 0)
            *p++ = ' ';
        q = size;
        while (*q)
            *p++ = *q++;
        if (mp->state == MOD_UNINITIALIZED)
            q = "  (uninitialized)";
        else if (mp->state == MOD_RUNNING)
            q = "";
        else if (mp->state == MOD_DELETED)
            q = "  (deleted)";
        else
            q = "  (bad state)";
        while (*q)
            *p++ = *q++;
        *p++ = '\n';
    }
    return p - buf;
}

kernel/module.c的更多相关文章

  1. The vboxdrv kernel module is not loaded

    背景: 在没有关虚拟机的情况下, 直接关了电脑, 我的电脑系统是Centos 6 错误的提示: 在终端执行virtualbox -v 时提示 The vboxdrv kernel module is ...

  2. Xamarin.Android模拟器提示HAX kernel module is not Installed

    Xamarin.Android模拟器提示HAX kernel module is not Installed 错误信息:emulator : ERROR : x86 emulation current ...

  3. HAX kernel module is not installed

    dev.android.emulator.haxm 运行emulator -avd xxx来启动名为xxx的模拟器,但报如下错误: emulator: ERROR: x86 emulation cur ...

  4. emulator: ERROR: x86 emulation currently requires hardware acceleration!Please ensure Intel HAXM is properly installed and usable.CPU acceleration status: HAX kernel module is not installed!

    Android Studio 1.0 已经放出来了,以后的Android平台开发激昂逐步从Eclipse向Android Studio迁移,为了能不落伍我也特意从Google下载了Android St ...

  5. Compiling a kernel module for the raspberry pi 2 via Ubuntu host

    Compiling a kernel module for the raspberry pi 2 via Ubuntu host Normally compiling a kernel module ...

  6. Virtualbox报错------>make sure the kernel module has been loaded successfully

    错误描述 很久没有用virtualbox了,今天打算在virtualbox上安装一个Ubuntu系统的时候,新建好Ubuntu后启动的时候,直接报错: Cannot access the kernel ...

  7. qemu 出现Could not access KVM kernel module: No such file or directory failed to initialize KVM: No such file or directory

    使用qemu命令 qemu-system-x86_64 -hda image/ubuntu-test.img -cdrom ubuntu-16.04.2-server-amd64.iso -m 102 ...

  8. 如何处理VirtualBox启动错误消息:The vboxdrv kernel module is not loaded

    我在启动minikube时,遇到如下错误消息: Starting local Kubernetes v1.10.0 cluster... Starting VM... E1010 03:27:37.9 ...

  9. OpenWrt Kernel Module Creation Howto

    OpenWrt Kernel Module Creation Howto About OpenWrt Kernel Module Compilation You are planning to com ...

随机推荐

  1. ZooKeeper 编程(一)

    Zookeeper的节点都是存放在内存中的,所以读写速度很快.更新日志被记录到了磁盘中,以便用于恢复数据.在更新内在中节点数之前,会先序列化到磁盘中. 为避免单点失效,zookeeper的数据是在多个 ...

  2. Oracle PL/SQL实战代码下载

    最近在学习PL/SQL编程,算是一个进阶吧,书没带光盘,所以按照书中的地址去下载样例,无法下载,到图灵官网找到了源代码下载地址,无法下载的留邮箱,我发给大家 下载地址: http://www.itur ...

  3. PHP 小方法之 显示 今天 昨天 上周 上月 近三月 的时间

    if(!function_exists('get_date_array')){ function get_date_array(){ $_NOW = time(); $today_begin = st ...

  4. python内置函数 3

    int( [x[, radix]])转换为字符串或数字为纯整数.如果参数是一个字符串,它必须包含一个可能有符号的十进制数作为一个 Python 整 数,可能嵌入空格.以 radix 参数给出的基数为基 ...

  5. POJ----(3974 )Palindrome [最长回文串]

    Time Limit: 15000MS   Memory Limit: 65536K Total Submissions: 5121   Accepted: 1834 Description Andy ...

  6. sans-serif

    sans-serif无衬线字体,是一类字体,它在操作系统或者浏览器里是可以设置的,你可以把它设置成宋体,也可以设置成微软雅黑,而设置的这种字体肯定是当前系统里存在的字体,所以使用这个字体就一肯能显示出 ...

  7. 如何根据不同的浏览器写不同的css样式达到兼容

    做前端静态页面的时候总是发现,ie的兼容性很差,总会出点问题.然后就去改代码 ,改完以后 又发现 火狐 谷歌又挂了,这可咋整.     后来发现做个判断吧 哪里有问题哪里就做个判断呗 ,咋判断呢,这么 ...

  8. Project中分清楚挣值项

    在项目管理非常重要的挣值管理,有一些关键项,像PV,EV,AC,BAC,EAC,ETC等等这些都是关键项,如果这个没分清楚,计算出很多东西都是错的,下面两个图是我一个项目快要完成的报表. 图1 图2 ...

  9. 项目二(业务GO)——跨域上传图片(请求接口)

    之前,就听过“跨域上传”图片的问题,只是疏于研究,也就一再搁置,直至今天再次遇见这个不能避免的“坑”,才不得不思考一下,怎么“跨域上传”图片或者文件? 问题来源: 何为“跨域”? ——就是给你一个接口 ...

  10. iOS开发拓展篇—xib中关于拖拽手势的潜在错误

    iOS开发拓展篇—xib中关于拖拽手势的潜在错误 一.错误说明 自定义一个用来封装工具条的类 搭建xib,并添加一个拖拽的手势. 主控制器的代码:加载工具条 封装工具条以及手势拖拽的监听事件 此时运行 ...