首页 > 其他 > 详细

完成一个简单的时间片轮转多道程序内核代码

时间:2016-03-05 16:07:58      阅读:256      评论:0      收藏:0      [点我收藏+]

学号:20135114

姓名:王朝宪

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

三个法宝

①存储程序计算机工作模型,计算机系统最最基础性的逻辑结构;

②函数调用堆栈,高级语言得以运行的基础,只有机器语言和汇编语言的时候堆栈机制对于计算机来说并不那么重要,但有了高级语言及函数,堆栈成为了计算机的基础功能;

③中断,多道程序操作系统的基点,没有中断机制程序只能从头一直运行结束才有可能开始运行其他程序。

函数堆栈框架的形成

  1. call xxx
  • 执行call之前
  • 执行call时,cs : eip原来的值指向call下一条指令,该值被保存到栈顶,然后cs : eip的值指向xxx的入口地址

  2. 进入xxx

  • 第一条指令: pushl %ebp
  • 第二条指令: movl %esp, %ebp
  • 函数体中的常规操作,可能会压栈、出栈

  3. 退出xxx

  • movl %ebp,%esp
  • popl %ebp
  • ret

C代码中嵌入汇编代码

 技术分享

实验要求:

使用实验楼的虚拟机打开shell
cd LinuxKernel/linux-3.9.4
qemu -kernel arch/x86/boot/bzImage

然后cd mykernel 您可以看到qemu窗口输出的内容的代码mymain.c和myinterrupt.c

使用自己的Linux系统环境搭建过程参见mykernel,其中也可以找到一个简单的时间片轮转多道程序内核代码

实验要求:
•完成一个简单的时间片轮转多道程序内核代码,代码见视频中或从mykernel找。
•详细分析该精简内核的源代码并给出实验截图,撰写一篇署名博客,并在博客文章中注明“真实姓名(与最后申请证书的姓名务必一致) + 原创作品转载请注明出处 + 《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000 ”,博客内容的具体要求如下:?题目自拟,内容围绕操作系统是如何工作的进行;
?博客中需要使用实验截图
?博客内容中需要仔细分析进程的启动和进程的切换机制
?总结部分需要阐明自己对“操作系统是如何工作的”理解。

一、实验截图

技术分享

技术分享

二、代码分析

技术分享

技术分享

 

mypcb.h

#define MAX_TASK_NUM        4
#define KERNEL_STACK_SIZE   1024*8

/* CPU-specific state of this task */
struct Thread {
    unsigned long       ip;//用于eip的保存
    unsigned long       sp;//用于esp的保存
};

typedef struct PCB{//用于表示一个进程,定义了进程管理相关的数据结构
    int pid;
    volatile long state;    /* -1 unrunnable, 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);//调用了my_schedule,表示调度器#chaper3

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;//初始化一个进程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 = (unsigned long)my_process;
    //定义进程0的入口为my_process
    task[pid].thread.sp = (unsigned long)&task[pid].stack[KERNEL_STACK_SIZE-1];
    task[pid].next = &task[pid];
    //因为一开始系统里只有进程0,所以这一行代码表示的是pid的next还是指向自己
    /*fork more process */
    //创建更多其他的进程,在初始化这些进程的时候可以直接拷贝0号进程的代码
    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 = (unsigned long)&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 d mean %ecx/%edx*/
    );
}
    /*  %0表示参数thread.ip,%1表示参数thread.sp。
        movl %1,%%esp表示把参数thread.sp放到esp中;
        接下来push %1,又因为当前栈为空,esp=ebp,所以等价于push ebp;
        然后push thread.ip;ret等价于pop thread.ip;最后pop ebp  */ 

void my_process(void)//定义所有进程的工作,if语句表示循环1000万次才有机会判断是否需要调度。
{
    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);
        }     
    }
}

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.
     * it runs in the name of current running process,
     * so it 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_handler here<<<\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 */
        {
            /* 进程切换跳转到下一进程 */
            asm volatile(   
                "pushl %%ebp\n\t"       /* 保存当前ebp */
                "movl %%esp,%0\n\t"     /* 保存当前esp */
                "movl %2,%%esp\n\t"     /* 重新记录要跳转进程的 esp,%2为 next->thread.sp*/
                "movl $1f,%1\n\t"       /* 保存当前 eip ,%1为prev->thread.ip*/   
                "pushl %3\n\t" 
                "ret\n\t"               /* 记录要跳转进程的  eip,%3为 next->thread.ip*/
                "1:\t"                  /* 下一个进程开始执行 */
                "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"       /* 保存当前 ebp */
                "movl %%esp,%0\n\t"     /* 保存当前 esp */
                "movl %2,%%esp\n\t"     /* 重新记录要跳转进程的 esp ,%2为 next->thread.sp*/
                "movl %2,%%ebp\n\t"     /* 重新记录要跳转进程的  ebp,%2为 next->thread.sp */
                "movl $1f,%1\n\t"       /* 保存当前  eip ,%1为prev->thread.ip,%1f就是指标号1:的代码在内存中存储的地址*/   
                "pushl %3\n\t" 
                "ret\n\t"               /* 重新记录要跳转进程的  eip,%3为 next->thread.ip */
                : "=m" (prev->thread.sp),"=m" (prev->thread.ip)
                : "m" (next->thread.sp),"m" (next->thread.ip)
            );          
        }   
        return; 
    }

总结:

调度的实现需要保存当前task/process的现场(ebp/eip),然后配合时钟中断,对第一次被调度和再次被调度分情况处理。

操作系统作用:

1.保存现场与恢复现场:中断上下文处理程序

2.进程上下文的切换

完成一个简单的时间片轮转多道程序内核代码

原文:http://www.cnblogs.com/Justtry/p/5244814.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!