进程的5种状态:创建,就绪,运行,阻塞,完成
在用户编写程序使用fork()
函数可以创建子进程

用户是调用了内核中fork.c中的函数(linux-0.11/kernel/fork.c),其中主要copy_process()函数起作用(复制进程)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
|
int copy_process (int nr, long ebp, long edi, long esi, long gs, long none, long ebx, long ecx, long edx, long fs, long es, long ds, long eip, long cs, long eflags, long esp, long ss) { struct task_struct *p; int i; struct file *f; struct i387_struct *p_i387; p = (struct task_struct *) get_free_page (); if (!p) return -EAGAIN; task[nr] = p;
*p = *current; p->state = TASK_UNINTERRUPTIBLE; p->pid = last_pid; p->father = current->pid; p->counter = p->priority; p->signal = 0; p->alarm = 0; p->leader = 0;
p->utime = p->stime = 0; p->cutime = p->cstime = 0; p->start_time = jiffies;
p->tss.back_link = 0; p->tss.esp0 = PAGE_SIZE + (long) p;
p->tss.ss0 = 0x10; p->tss.eip = eip; p->tss.eflags = eflags; p->tss.eax = 0; p->tss.ecx = ecx; p->tss.edx = edx; p->tss.ebx = ebx; p->tss.esp = esp; p->tss.ebp = ebp; p->tss.esi = esi; p->tss.edi = edi; p->tss.es = es & 0xffff; p->tss.cs = cs & 0xffff; p->tss.ss = ss & 0xffff; p->tss.ds = ds & 0xffff; p->tss.fs = fs & 0xffff; p->tss.gs = gs & 0xffff; p->tss.ldt = _LDT (nr); p->tss.trace_bitmap = 0x80000000;
p_i387 = &p->tss.i387; if (last_task_used_math == current) _asm{ mov ebx, p_i387 clts fnsave [p_i387] }
if (copy_mem (nr, p)) { task[nr] = NULL; free_page ((long) p); return -EAGAIN; }
for (i = 0; i < NR_OPEN; i++) if (f = p->filp[i]) f->f_count++;
if (current->pwd) current->pwd->i_count++; if (current->root) current->root->i_count++; if (current->executable) current->executable->i_count++;
set_tss_desc (gdt + (nr << 1) + FIRST_TSS_ENTRY, &(p->tss)); set_ldt_desc (gdt + (nr << 1) + FIRST_LDT_ENTRY, &(p->ldt)); p->state = TASK_RUNNING;
return last_pid; }
|
创建 ——> 就绪
现在进程在就绪队列中等待分配CPU资源运行。
当一个进程在CPU结束运行后如何在就绪队列中选择进程运行呢?
- 根据等待的顺序来选择
- 根据优先级高低来选择
- 根据短任务优先(分配的时间片短)
在Linux0.11中 linux-0.11/kernel/sched.c调度算法如下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
|
void schedule (void) { int i, next, c; struct task_struct **p;
for (p = &LAST_TASK; p > &FIRST_TASK; --p) if (*p) {
if ((*p)->alarm && (*p)->alarm < jiffies) { (*p)->signal |= (1 << (SIGALRM - 1)); (*p)->alarm = 0; }
if (((*p)->signal & ~(_BLOCKABLE & (*p)->blocked)) && (*p)->state == TASK_INTERRUPTIBLE) (*p)->state = TASK_RUNNING; }
while (1) { c = -1; next = 0; i = NR_TASKS; p = &task[NR_TASKS];
while (--i) { if (!*--p) continue;
if ((*p)->state == TASK_RUNNING && (*p)->counter > c) c = (*p)->counter, next = i; } if (c) break;
for (p = &LAST_TASK; p > &FIRST_TASK; --p) if (*p) (*p)->counter = ((*p)->counter >> 1) + (*p)->priority; } switch_to (next); }
|
注意到这里的counter和priority在Linux0.11设置为15,15的
就绪态 ——> 运行态
当时间片使用完后,该进程还没有执行完需要再分配时间片。
也就是上述代码所示:目前的进程在与下一个进程切换,目前的进程只在时钟中断、int0x80中断和进程结束才会切换到下一个进程。而在schedule()
函数中只涉及时间片,则判断当前进程的时间片使用完了。如果当前的进程还需要运行则把运行态 ——> 就绪态,等待下一次的调用。而下一个进程从就绪态 ——> 运行态。
一个进程除了在CPU在执行任务,也会读取磁盘上的信息,这个时候进程就会释放CPU的资源,让给其他的进程。这个时候该进程就阻塞了,相对与CPU来说。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
int sys_pause(void) { current->state = TASK_INTERRUPTIBLE; if(current->pid != 0) { } schedule(); return 0; }
|
当一个运行的进程可能发生不可中断睡眠状态和可中断睡眠状态时,会释放CPU资源,则进程状态:运行态 ——> 阻塞态。之后调用shedule()函数把CPU资源给下一个进程使用。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
|
void sleep_on(struct task_struct **p) { struct task_struct *tmp;
if (!p) return; if (current == &(init_task.task)) panic("task[0] trying to sleep"); tmp = *p; *p = current; current->state = TASK_UNINTERRUPTIBLE; schedule();
** if (tmp) { tmp->state=0; } }
void interruptible_sleep_on (struct task_struct **p) { struct task_struct *tmp;
if (!p) return; if (current == &(init_task.task)) panic ("task[0] trying to sleep"); tmp = *p; *p = current; repeat: current->state = TASK_INTERRUPTIBLE; schedule ();
if (*p && *p != current) { (**p).state = 0; goto repeat; }
*p = NULL; if (tmp) tmp->state = 0; }
|

在阻塞态的进程等待除CPU资源外的资源,资源到了后需要唤醒进程
1 2 3 4 5 6 7 8 9
| void wake_up (struct task_struct **p) { if (p && *p) { (**p).state = 0; *p = NULL; } }
|
这时就把阻塞态 ——> 就绪态
比如父进程等待子进程都结束后才开始运行,该父进程就存在调用系统等待
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
| int sys_waitpid(pid_t pid,unsigned long * stat_addr, int options) { int flag, code; struct task_struct ** p;
verify_area(stat_addr,4); repeat: flag=0; for(p = &LAST_TASK ; p > &FIRST_TASK ; --p) { if (!*p || *p == current) continue; if ((*p)->father != current->pid) continue; if (pid>0) { if ((*p)->pid != pid) continue; } else if (!pid) { if ((*p)->pgrp != current->pgrp) continue; } else if (pid != -1) { if ((*p)->pgrp != -pid) continue; } switch ((*p)->state) { case TASK_STOPPED: if (!(options & WUNTRACED)) continue; put_fs_long(0x7f,stat_addr); return (*p)->pid; case TASK_ZOMBIE: current->cutime += (*p)->utime; current->cstime += (*p)->stime; flag = (*p)->pid; code = (*p)->exit_code; release(*p); put_fs_long(code,stat_addr); return flag; default: flag=1; continue; } } if (flag) { if (options & WNOHANG) return 0; current->state=TASK_INTERRUPTIBLE; fprintk(3,"%d\tW\t%d\n",current->pid,jiffies); schedule(); if (!(current->signal &= ~(1<<(SIGCHLD-1)))) goto repeat; else return -EINTR; } return -ECHILD; }
|
运行态 ——> 结束
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| int do_exit(long code) { int i; free_page_tables(get_base(current->ldt[1]),get_limit(0x0f)); free_page_tables(get_base(current->ldt[2]),get_limit(0x17)); for (i=0 ; i<NR_TASKS ; i++) if (task[i] && task[i]->father == current->pid) { task[i]->father = 1; if (task[i]->state == TASK_ZOMBIE) (void) send_sig(SIGCHLD, task[1], 1); } for (i=0 ; i<NR_OPEN ; i++) if (current->filp[i]) sys_close(i); iput(current->pwd); current->pwd=NULL; iput(current->root); current->root=NULL; iput(current->executable); current->executable=NULL; if (current->leader && current->tty >= 0) tty_table[current->tty].pgrp = 0; if (last_task_used_math == current) last_task_used_math = NULL; if (current->leader) kill_session(); **current->state = TASK_ZOMBIE;** fprintk(3,"%d\tE\t%d\n",current->pid,jiffies); current->exit_code = code; tell_father(current->father); schedule(); return (-1); }
|