Linux内核分析实验二

Posted YoungX0701

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux内核分析实验二相关的知识,希望对你有一定的参考价值。

<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<

刘旸 + 原创作品转载请注明出处 + 《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000

>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

       操作系统(Operating System,简称OS)是管理和控制计算机硬件与软件资源的计算机程序。操作系统在进行进程管理时通常采用分时的概念,将计算机的系统资源(尤其是 CPU时间)进行时间上的分割,每个时间段称为一个时间片。操作系统以时间片为单位,轮流为每个进程服务。时间片轮转的方式使得多个进程能在同一台计算机同时运行。

       下面我们通过实验来模拟一个小型的操作系统,理解它采用时间片分配来进行进程管理的过程。

       1. 在实验楼的虚拟机中打开终端,输入以下命令:

           cd LinuxKernel/linux-3.9.4

           qemu -kernel arch/x86/boot/bzImage

 

           mykernel启动后效果如下:

 

           不难看出该程序是在模拟一个操作系统,不断地打印指定的信息。


       2.打开LinuxKernel/linux-3.9.4/mykernel文件夹,找到并打开mymain.c和myinterrupt.c

 




       3.可以看到在mymain.c中有一个_initmy_start_kernel函数,我们在运行mykernel之后见到的一部分信息——“my_start_kernel here”就是由这个函数中的循环打印的。再观察myinterrupt.c文件,其中有一个名为my_timer_handler的函数,会周期性地被中断调用,从而输出另一部分信息“>>>my_timer_handler here <<<”。

 

       到这里我们可以得出一个结论,mykernel启动之后将会做两件事:

        1) 调用my_start_kernel函数

        2) 周期性调用my_timer_handler函数

       因此只要通过编写这两个函数,完成进程的初始化和进程的时间片轮转调度,就可以写出一个简单的操作系统了。下面我们就来研究分析一个简单的时间片轮转多道程序。


        4.从https://github.com/mengning/mykernel获取实验用的源代码,主要是下面的三个文件:mypcb.h,myinterrupt.c和mymain.c。用下载的后两个文件的内容取代实验楼虚拟机的mykernel文件夹下相应同名文件的内容,然后在mykernel文件夹下自行新建一个mypcb.h文件,将下载的mypcb.h的内容拷贝进去并保存。


 

        5.打开终端定位到LinuxKernel/linux-3.9.4,并执行以下命令:

           make allnoconfig

           make

           qemu -kernel arch/x86/boot/bzImage

           此时mykernel的运行效果如下:

 

           可以看到隔一段时间后,系统会自动运行下一个进程。

 

代码分析

首先我们来看mypcb.h

/*
 * linux/mykernel/mypcb.h
 *
 * Kernel internal PCB types
 *
 * Copyright (C) 2013  Mengning
 *
 */ 
 
#define MAX_TASK_NUM        4 
#define KERNEL_STACK_SIZE   1024*8 
 
/* CPU-specific state of this task */ 
struct Thread  
   unsigned long       ip; 
   unsigned long       sp; 
; 
 
typedef struct PCB 
   int pid; 
   volatile long state;    /* -1unrunnable, 0 runnable, >0 stopped */ 
   char stack[KERNEL_STACK_SIZE]; 
   /* CPU-specific state of this task */ 
   struct Thread thread; 
   unsigned long   task_entry; 
   struct PCB *next; 
tPCB; 
 
void my_schedule(void);

        该头文件:

        1) 定义了Thread结构体,其中,ip、sp分别代表ip寄存器和sp寄存器,用于存储现场。

        2) 定义了PCB结构体,其中pid为进程的进程号,state为进程状态,stack为分配给进程的栈空间,thread为线程信息,task_entry为进程的入口函数,next指针指向下一个PCB。

        3) 声明了my_schedule函数。其具体实现会放在my_interrupt.c中分析。mymain.c中的各个进程函数会根据一个全局变量的状态来决定是否调用它,从而实现主动调度。

 

 接下来我们继续分析mymain.c

/*
 * linux/mykernel/mymain.c
 *
 * Kernel internal my_start_kernel
 *
 * Copyright (C) 2013  Mengning
 *
 */ 
#include <linux/types.h> 
#include <linux/string.h> 
#include <linux/ctype.h> 
#include <linux/tty.h> 
#include <linux/vmalloc.h> 
 
 
#include "mypcb.h" 
 
tPCB task[MAX_TASK_NUM]; 
tPCB * my_current_task = NULL; 
volatile int my_need_sched = 0; 
 
void my_process(void); 
 
 
void __init my_start_kernel(void) 
 
   int pid = 0; 
   int i; 
   /* Initialize process 0*/ 
   task[pid].pid = pid; 
   task[pid].state = 0;/* -1 unrunnable, 0 runnable, >0 stopped */ 
   task[pid].task_entry = task[pid].thread.ip = (unsignedlong)my_process; 
   task[pid].thread.sp = (unsignedlong)&task[pid].stack[KERNEL_STACK_SIZE-1]; 
   task[pid].next = &task[pid]; 
   /*fork more process */ 
   for(i=1;i<MAX_TASK_NUM;i++) 
    
       memcpy(&task[i],&task[0],sizeof(tPCB)); 
       task[i].pid = i; 
       task[i].state = -1; 
       task[i].thread.sp = (unsignedlong)&task[i].stack[KERNEL_STACK_SIZE-1]; 
       task[i].next = task[i-1].next; 
       task[i-1].next = &task[i]; 
    
   /* start process 0 by task[0] */ 
   pid = 0; 
   my_current_task = &task[pid]; 
   asm volatile( 
       "movl %1,%%esp\\n\\t"    /* set task[pid].thread.sp to esp */ 
       "pushl %1\\n\\t"         /* push ebp */ 
       "pushl %0\\n\\t"         /* push task[pid].thread.ip */ 
       "ret\\n\\t"              /* pop task[pid].thread.ip to eip */ 
       "popl %%ebp\\n\\t" 
       :  
       : "c" (task[pid].thread.ip),"d"(task[pid].thread.sp)   /* input c or dmean %ecx/%edx*/ 
   ); 
    
void my_process(void) 
 
   int i = 0; 
   while(1) 
    
       i++; 
       if(i%10000000 == 0) 
        
           printk(KERN_NOTICE "this is process %d-\\n",my_current_task->pid); 
           if(my_need_sched == 1) 
            
                my_need_sched = 0; 
                my_schedule(); 
            
           printk(KERN_NOTICE "this is process %d+\\n",my_current_task->pid); 
             
    
 

       系统启动后,函数 my_start_kernel首先被调用,它完成了0号进程的初始化和启动,其中启动过程采用内联汇编代码完成。然后创建好其它进程的PCB,用于后面的进程调度。

       而my_process 函数即是每个进程的函数代码(实际中可能不一样,但在该模拟系统中认为都一样),该函数会打印出当前进程的pid,以显示当前哪个进程正在执行。同时,my_process还负责检查一个全局标志变量 my_need_sched,一旦发现其值为 1 ,就调用 my_schedule 完成进程的调度。

 

最后我们来看负责执行中断的myinterrupt.c文件:

/*
 * linux/mykernel/myinterrupt.c
 *
 * Kernel internal my_timer_handler
 *
 * Copyright (C) 2013  Mengning
 *
 */ 
#include <linux/types.h> 
#include <linux/string.h> 
#include <linux/ctype.h> 
#include <linux/tty.h> 
#include <linux/vmalloc.h> 
 
#include "mypcb.h" 
 
extern tPCB task[MAX_TASK_NUM]; 
extern tPCB * my_current_task; 
extern volatile int my_need_sched; 
volatile int time_count = 0; 
 
/*
 *Called by timer interrupt.
 * itruns in the name of current running process,
 * soit use kernel stack of current running process
 */ 
void my_timer_handler(void) 
 
#if 1 
   if(time_count%1000 == 0 && my_need_sched != 1) 
    
       printk(KERN_NOTICE ">>>my_timer_handlerhere<<<\\n"); 
       my_need_sched = 1; 
     
   time_count ++ ;   
#endif 
   return;      
 
 
void my_schedule(void) 
 
    tPCB * next; 
   tPCB * prev; 
 
   if(my_current_task == NULL  
       || my_current_task->next == NULL) 
    
       return; 
    
   printk(KERN_NOTICE">>>my_schedule<<<\\n"); 
   /* schedule */ 
   next = my_current_task->next; 
   prev = my_current_task; 
   if(next->state == 0)/* -1 unrunnable, 0 runnable, >0 stopped*/ 
    
       /* switch to next process */ 
       asm volatile(    
           "pushl %%ebp\\n\\t"      /* save ebp */ 
           "movl %%esp,%0\\n\\t"    /* save esp */ 
           "movl %2,%%esp\\n\\t"    /* restore  esp */ 
           "movl $1f,%1\\n\\t"      /* save eip */   
           "pushl %3\\n\\t"  
           "ret\\n\\t"              /* restore  eip */ 
           "1:\\t"                  /* next process start here */ 
           "popl %%ebp\\n\\t" 
           : "=m" (prev->thread.sp),"=m"(prev->thread.ip) 
           : "m" (next->thread.sp),"m"(next->thread.ip) 
       );  
       my_current_task = next;  
       printk(KERN_NOTICE ">>>switch %d to%d<<<\\n",prev->pid,next->pid);       
    
   else 
    
       next->state = 0; 
       my_current_task = next; 
       printk(KERN_NOTICE ">>>switch %d to%d<<<\\n",prev->pid,next->pid); 
       /* switch to new process */ 
       asm volatile(    
           "pushl %%ebp\\n\\t"      /* save ebp */ 
           "movl %%esp,%0\\n\\t"    /* save esp */ 
           "movl %2,%%esp\\n\\t"    /* restore  esp */ 
           "movl %2,%%ebp\\n\\t"    /* restore  ebp */ 
           "movl $1f,%1\\n\\t"      /* save eip */   
           "pushl %3\\n\\t"  
           "ret\\n\\t"              /* restore  eip */ 
           : "=m" (prev->thread.sp),"=m"(prev->thread.ip) 
           : "m" (next->thread.sp),"m"(next->thread.ip) 
       );           
       
   return;  
 

       之前已经了解到,my_timer_handler 函数会被内核周期性的调用,每调用1000次,就将全局变量my_need_sched的值置为1,并通知正在运行的进程执行在mypcb.h中声明的my_schedule函数。

       my_schedule函数负责完成进程的切换。切换分两种情况:一种情况是下一个进程没有被调度过,那么就直接启动该进程即可;另外一种情况是下一个进程被调度过,那么就需要从上次该进程被中断的地方继续运行。这就需要先通过查看下一个进程的state变量获得其状态,再根据其状态进行相应的处理。进程切换依然是通过内联汇编代码实现,即保存旧进程的eip和堆栈,将新进程的eip和堆栈的值存入对应的寄存器中。

 

 

       通过以上实验我们可以看出,操作系统的核心功能就是进程调度和中断机制,通过与硬件的配合实现多任务处理,再加上上层应用软件的支持,最终变成可以使用户可以很容易操作的计算机系统。

以上是关于Linux内核分析实验二的主要内容,如果未能解决你的问题,请参考以下文章

Linux内核分析实验五

Linux内核分析实验三

Linux内核分析实验五

Linux内核分析实验三

Linux内核分析实验八

Linux内核分析实验七