内容发布更新时间 : 2024/12/23 4:42:37星期一 下面是文章的全部内容请认真阅读。
Linux内核实验报告
实验题目: 动态模块设计实验
实验目的:
Linux 模块是一些可以独立于内核单独编译的内核函数和数据类型
集合,是可增删的
内核部分。模块在内核启动时装载称为静态装载,在内核已经运行时装载称为动态装载。
模块可以扩充内核所期望的任何功能,但通常用于实现设备驱动程序。
通过本实验,将学习到动态模块的设计过程,以及Proc文件系统的部
分知识。
硬件环境:
Pentium(R) Dual-Core CPU T4400 @ 2.20GHz
软件环境: Ubuntu12.04
gcc version 4.6.3 (Ubuntu/Linaro 4.6.3-1ubuntu5)
内核版本:3.0.24
实验步骤:
1、 代码分析
模块初始化和模块卸载的函数都类同,同时读取proc文件的函数在本次实验中没有用到,所以着重描述写驱动函数。 实验A:
在这个proc函数中只是简单得输出jiffies的数值。
实验B:遍历父进程和所有进程,依然是在proc_read中,通过以下代码片段完成功能,注意在这里,我们是通过直接向系统分配的那一个page直接写入来得到的,所以每次不同的进程访问该proc文件的时候得到的结果都不一样 遍历父进程: len += sprintf(page+len,\遍历父进程\\npid\\tppid\\tcomm\\n\while (task != &init_task){ len += sprintf(page + len,\
遍历所有进程,通过for_each_process(task)这个宏来遍历所有进程:
} task = task->parent;
len += sprintf(page + len,\遍历任务队列\\ncomm\\tpid\\n\
实验C:
for_each_process(task){ } return len; len += sprintf(page + len,\在模块开头通过module_param声明了模块参数,在装载模块时通过它来传参 module_param(param,int,0644);
在proc_read函数中通过这个传入的参数来判断应该读取内核的哪些数值,其中标注黄色的一段是从内核实现/proc/loadavg部分中抽取出来的,loops_per_jiffy是每个滴答能执行的循环数,在内核启动的时候计算出来的这个值: if (param == 0){ //获取loadavg
len
+=
sprintf(page
+
len,
\
LOAD_INT(avenrun[0]), LOAD_FRAC(avenrun[0]), LOAD_INT(avenrun[1]), LOAD_FRAC(avenrun[1]), LOAD_INT(avenrun[2]), LOAD_FRAC(avenrun[2]) /*,nr_running(), nr_threads,*/
/*task_active_pid_ns(current)->last_pid*/);
}else if (param == 1){
len += sprintf(page + len,\len
+=
sprintf(page
+