《Linux内核分析》

第二章 操作系统是如何工作的

2.1 函数调用堆栈

  • 3个关键性的方法机制(3个法宝)

    • 存储程序计算机
    • 函数调用堆栈机制
    • 中断
  • 堆栈相关的寄存器
    • ESP:堆栈指针(stack pointer)
    • EBP:基址指针(base pointrer),在C语言中用作记录当前函数调用基址。
  • 堆栈操作
    • push:栈顶地址减少4个字节(32位),并将操作数放入栈顶存储单元。
    • pop:栈顶地址增加4个字节(32位),并将栈顶存储单元的内容放入操作数。
  • 栈是从搞地质向低地址增加的。
  • 其他关键寄存器
    • CS:EIP总是指向下一条的指令地址(CS就是代码段寄存器,EIP总是指向下一条的指令地址)。

      • 顺序执行:总是指向地址连续的下一条指令。
      • 跳转/分支:执行这样的指令是,CS:EIP的值会根据程序需要被修改。
      • call:将当前CS:EIP的值压入栈顶,CS:EIP指向被调用函数的入口地址。
      • ret:从栈顶弹出原来保存在这里的CS:EIP的值,放入CS:EIP中。
    • 堆栈是CPU指令集的一部分。
  • 函数是如何传递返回值的
    • 保存返回值,就是程序用EAX寄存器来保存返回值。
    • 多个返回值,EAX寄存器返回的是一个内存地址。
  • 函数体内的局部变量是通过堆栈来存储的。

2.2 借助Linux内核部分源代码模拟存储程序计算机工作模型及时钟中断

  • 2.2.1 内嵌汇编

    • 语法:
  1. _asm_ _volatile_(
  2. 汇编语句模板:
  3. 输出部分:
  4. 输入部分:
  5. 破坏描述部分



答案是0,1

  • 2.2.2 虚拟一个x86的CPU硬件平台

    -首先搭建一个虚拟的平台,虚拟一个x86的CPU,然后使用Linux的源代码把CPU初始化配置好,并配置好整个系统,开始执行编写的程序。

    • 用到的命令:
  1. //注意路径是区分大小的
  2. $ cd ~/LinuxKernel/linux-3.9.4
  3. $ rm -rf mykernel
  4. $ patch -p1 < ../mykernel_for_linux3.9.4sc.patch
  5. $ make allnoconfig
  6. //编译内核请耐心等待
  7. $ make
  8. $ qemu -kernel arch/x86/boot/bzImage
  • 搭建起来后的内核启动效果如下:

2.3 在mykernel基础上构造一个简单的操作系统内核

  • 实验代码分析

    • mypcb.h
  1. /*
  2. * linux/mykernel/mypcb.h
  3. *
  4. * Kernel internal PCB types
  5. *
  6. * Copyright (C) 2013 Mengning
  7. *
  8. */
  9. #define MAX_TASK_NUM 4 /*定义最大任务数*/
  10. #define KERNEL_STACK_SIZE 1024*8 /*定义堆栈大小*/
  11. /* CPU-specific state of this task */
  12. struct Thread {
  13. unsigned long ip;
  14. unsigned long sp;
  15. };
  16. typedef struct PCB{ /*定义进程控制块*/
  17. int pid; /*进程的ID*/
  18. volatile long state; /* -1 unrunnable, 0 runnable, >0 stopped */
  19. char stack[KERNEL_STACK_SIZE]; /*当前进的堆栈*/
  20. /* CPU-specific state of this task */
  21. struct Thread thread;
  22. unsigned long task_entry; /*入口*/
  23. struct PCB *next;
  24. }tPCB;
  25. void my_schedule(void); /*调度器*/
  • mymain.c
  1. /*
  2. * linux/mykernel/mymain.c
  3. *
  4. * Kernel internal my_start_kernel
  5. *
  6. * Copyright (C) 2013 Mengning
  7. *
  8. */
  9. #include <linux/types.h>
  10. #include <linux/string.h>
  11. #include <linux/ctype.h>
  12. #include <linux/tty.h>
  13. #include <linux/vmalloc.h>
  14. #include "mypcb.h"
  15. tPCB task[MAX_TASK_NUM]; /*数组*/
  16. tPCB * my_current_task = NULL;
  17. volatile int my_need_sched = 0; /*是否需要调度的标志*/
  18. void my_process(void);
  19. void __init my_start_kernel(void) /*内核入口。初始化并启动0号进程*/
  20. {
  21. int pid = 0;
  22. int i;
  23. /* Initialize process 0*/
  24. task[pid].pid = pid;
  25. task[pid].state = 0;/* -1 unrunnable, 0 runnable, >0 stopped */
  26. task[pid].task_entry = task[pid].thread.ip = (unsigned long)my_process;
  27. task[pid].thread.sp = (unsigned long)&task[pid].stack[KERNEL_STACK_SIZE-1];
  28. task[pid].next = &task[pid];
  29. /*fork more process */
  30. for(i=1;i<MAX_TASK_NUM;i++)
  31. {
  32. memcpy(&task[i],&task[0],sizeof(tPCB));
  33. task[i].pid = i;
  34. task[i].state = -1;
  35. task[i].thread.sp = (unsigned long)&task[i].stack[KERNEL_STACK_SIZE-1];
  36. task[i].next = task[i-1].next; /*将创建的进程加到进程列表的尾部*/
  37. task[i-1].next = &task[i];
  38. }
  39. /* start process 0 by task[0] */
  40. pid = 0;
  41. my_current_task = &task[pid];
  42. asm volatile(
  43. "movl %1,%%esp\n\t" /* set task[pid].thread.sp to esp */
  44. "pushl %1\n\t" /* push ebp */
  45. "pushl %0\n\t" /* push task[pid].thread.ip */
  46. "ret\n\t" /* pop task[pid].thread.ip to eip */
  47. "popl %%ebp\n\t"
  48. :
  49. : "c" (task[pid].thread.ip),"d" (task[pid].thread.sp) /* input c or d mean %ecx/%edx*/
  50. );
  51. }
  52. void my_process(void)
  53. {
  54. int i = 0;
  55. while(1)
  56. {
  57. i++;
  58. if(i%10000000 == 0) /*循环1000万次才有一次机会判断是否需要调度*/
  59. {
  60. printk(KERN_NOTICE "this is process %d -\n",my_current_task->pid);
  61. if(my_need_sched == 1)
  62. {
  63. my_need_sched = 0;
  64. my_schedule();
  65. }
  66. printk(KERN_NOTICE "this is process %d +\n",my_current_task->pid);
  67. }
  68. }
  69. }
  • myinterrupt.c
  1. /*
  2. * linux/mykernel/myinterrupt.c
  3. *
  4. * Kernel internal my_timer_handler
  5. *
  6. * Copyright (C) 2013 Mengning
  7. *
  8. */
  9. #include <linux/types.h>
  10. #include <linux/string.h>
  11. #include <linux/ctype.h>
  12. #include <linux/tty.h>
  13. #include <linux/vmalloc.h>
  14. #include "mypcb.h"
  15. extern tPCB task[MAX_TASK_NUM];
  16. extern tPCB * my_current_task;
  17. extern volatile int my_need_sched;
  18. volatile int time_count = 0;
  19. /*
  20. * Called by timer interrupt.
  21. * it runs in the name of current running process,
  22. * so it use kernel stack of current running process
  23. */
  24. void my_timer_handler(void) /*设置时间片大小,时间片用完时设置一下调度标志*/
  25. {
  26. #if 1
  27. if(time_count%1000 == 0 && my_need_sched != 1)
  28. {
  29. printk(KERN_NOTICE ">>>my_timer_handler here<<<\n");
  30. my_need_sched = 1; /*调度执行my_schedule(void)*/
  31. }
  32. time_count ++ ;
  33. #endif
  34. return;
  35. }
  36. void my_schedule(void) /*进程上下文的切换*/
  37. {
  38. tPCB * next;
  39. tPCB * prev;
  40. if(my_current_task == NULL
  41. || my_current_task->next == NULL)
  42. {
  43. return;
  44. }
  45. printk(KERN_NOTICE ">>>my_schedule<<<\n");
  46. /* schedule */
  47. next = my_current_task->next;
  48. prev = my_current_task;
  49. if(next->state == 0)/* -1 unrunnable, 0 runnable, >0 stopped */
  50. {
  51. my_current_task = next;
  52. printk(KERN_NOTICE ">>>switch %d to %d<<<\n",prev->pid,next->pid);
  53. /* switch to next process */
  54. asm volatile(
  55. "pushl %%ebp\n\t" /* save ebp */
  56. "movl %%esp,%0\n\t" /* save esp */
  57. "movl %2,%%esp\n\t" /* restore esp */
  58. "movl $1f,%1\n\t" /* save eip */
  59. "pushl %3\n\t"
  60. "ret\n\t" /* restore eip */
  61. "1:\t" /* next process start here */
  62. "popl %%ebp\n\t"
  63. : "=m" (prev->thread.sp),"=m" (prev->thread.ip)
  64. : "m" (next->thread.sp),"m" (next->thread.ip)
  65. );
  66. }
  67. else
  68. {
  69. next->state = 0;
  70. my_current_task = next;
  71. printk(KERN_NOTICE ">>>switch %d to %d<<<\n",prev->pid,next->pid);
  72. /* switch to new process */
  73. asm volatile(
  74. "pushl %%ebp\n\t" /* save ebp */
  75. "movl %%esp,%0\n\t" /* save esp */
  76. "movl %2,%%esp\n\t" /* restore esp */
  77. "movl %2,%%ebp\n\t" /* restore ebp */
  78. "movl $1f,%1\n\t" /* save eip */
  79. "pushl %3\n\t"
  80. "ret\n\t" /* restore eip */
  81. : "=m" (prev->thread.sp),"=m" (prev->thread.ip)
  82. : "m" (next->thread.sp),"m" (next->thread.ip)
  83. );
  84. }
  85. return;
  86. }

2019-2020-1 20199308《Linux内核原理与分析》第三周作业的更多相关文章

  1. 2019-2020-1 20199329《Linux内核原理与分析》第九周作业

    <Linux内核原理与分析>第九周作业 一.本周内容概述: 阐释linux操作系统的整体构架 理解linux系统的一般执行过程和进程调度的时机 理解linux系统的中断和进程上下文切换 二 ...

  2. 2019-2020-1 20199329《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 一.上周问题总结: 未能及时整理笔记 Linux还需要多用 markdown格式不熟练 发布博客时间超过规定期限 二.本周学习内容: <庖丁解 ...

  3. 20169212《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 这一周学习了MOOCLinux内核分析的第一讲,计算机是如何工作的?由于本科对相关知识的不熟悉,所以感觉有的知识理解起来了有一定的难度,不过多查查资 ...

  4. 20169210《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 本周作业分为两部分:第一部分为观看学习视频并完成实验楼实验一:第二部分为看<Linux内核设计与实现>1.2.18章并安装配置内核. 第 ...

  5. 2018-2019-1 20189221 《Linux内核原理与分析》第九周作业

    2018-2019-1 20189221 <Linux内核原理与分析>第九周作业 实验八 理理解进程调度时机跟踪分析进程调度与进程切换的过程 进程调度 进度调度时机: 1.中断处理过程(包 ...

  6. 2017-2018-1 20179215《Linux内核原理与分析》第二周作业

    20179215<Linux内核原理与分析>第二周作业 这一周主要了解了计算机是如何工作的,包括现在存储程序计算机的工作模型.X86汇编指令包括几种内存地址的寻址方式和push.pop.c ...

  7. 2019-2020-1 20209313《Linux内核原理与分析》第二周作业

    2019-2020-1 20209313<Linux内核原理与分析>第二周作业 零.总结 阐明自己对"计算机是如何工作的"理解. 一.myod 步骤 复习c文件处理内容 ...

  8. 2018-2019-1 20189221《Linux内核原理与分析》第一周作业

    Linux内核原理与分析 - 第一周作业 实验1 Linux系统简介 Linux历史 1991 年 10 月,Linus Torvalds想在自己的电脑上运行UNIX,可是 UNIX 的商业版本非常昂 ...

  9. 《Linux内核原理与分析》第一周作业 20189210

    实验一 Linux系统简介 这一节主要学习了Linux的历史,Linux有关的重要人物以及学习Linux的方法,Linux和Windows的区别.其中学到了LInux中的应用程序大都为开源自由的软件, ...

  10. 2018-2019-1 20189221《Linux内核原理与分析》第二周作业

    读书报告 <庖丁解牛Linux内核分析> 第 1 章 计算工作原理 1.1 存储程序计算机工作模型 1.2 x86-32汇编基础 1.3汇编一个简单的C语言程序并分析其汇编指令执行过程 因 ...

随机推荐

  1. springBoot集成zuul路由forward,设置setSendZuulResponse无效

    正确书写方式如下: RequestContext ctx = RequestContext.getCurrentContext(); ctx.setSendZuulResponse(false); c ...

  2. html中的字幕滚动marquee属性

    该标签不是HTML3.2的一部分,并且只支持MSIE3以后内核,所以如果你使用非IE内核浏览器(如:Netscape)可能无法看到下面一些很有意思的效果,该标签是个容器标签. 语法: <marq ...

  3. Java 连接数据库总是报错

    mysql账号密码是正确的,但是一直报账号密码错误. 报错信息: java.sql.SQLException: Access denied for user 'root'@'localhost' (u ...

  4. 1034 Head of a Gang (30分)(dfs 利用map)

    One way that the police finds the head of a gang is to check people's phone calls. If there is a pho ...

  5. Xshell连接Centos7

    13:53:10 2019-08-05 一个月暑假开始 学习搭建一个自己的博客 我是用阿里云的服务器搭建自己的博客 先利用XShell连接我的服务器 XShell下载地址:https://www.ne ...

  6. git 从另一个分支融合部分文件

    # git checkout master # git checkout anotherBranch -- abc ./etc # git commit -m "merge some fil ...

  7. spring中BeanPostProcessor之二:CommonAnnotationBeanPostProcessor(01)

    在上篇博客中分享了InstantiationAwareBeanPostProcessor接口中的四个方法,分别对其进行了详细的介绍,在文末留下了一个问题,那就是postProcessPropertie ...

  8. php--MongoDB的使用

    添加 $collection = (new MongoDB\Client)->test->users; // 增加一条 $insertOneResult = $collection-> ...

  9. 配置spark历史服务(spark二)

    1. 编辑spark-defaults.conf位置文件 添加spark.eventLog.enabled和spark.eventLog.dir的配置修改spark.eventLog.dir为我们之前 ...

  10. app扫描二维码登陆

    先说明一下实现原理: 如同微信扫描登陆一样,就是一种pc的自动登陆授权.在网站首页得有切换登陆的选项:密码登陆 扫码登陆 当用户切换到扫码登陆时,向服务器请求一次获得一个唯一的uukey 利用这个uu ...