mykernel-调度上下文ebp作用

mykernel开源项目见于github:mykernel

假设有如下任务结构:任务A和任务B:
void a()
{
    ……
};
void b()
{
    ……
    if(当前处于B任务)
        Schedule到A;
    ……
};

void c()
{
    ……
};

任务A 
{
    a();
    Schedule到B;
    b();
    c();
};

任务B
{
    b();
};

首先:一个任务只能有一个上下文环境,即便是反复被调度。

有如下推断:

  • 首先,A运行,然后执行Schedule到B;调度到B,A的运行环境保存(第一次保存A任务上下文),恢复为B的运行环境,B运行。然后调度到A的时候,B的运行环境保存,恢复为之前A的运行环境。

  • A接着上次的位置调用函数b(),此时就是函数嵌套调用了。那么,在A的任务栈保存ebp,esp,返回地址(所有的这些都是函数相关的堆栈指针,是有栈帧结构的,而且和之前的esp并不冲突,因为这些是编译器自动生成的代码,而且是保存到栈里面了。),然后跳转到b()。因为处于任务A中,经过判断,无法调度到A。b()做完该做的事情,栈撤除,返回。
    注意:在这个过程中,由于并没有任务切换,故A任务上下文环境并没有变化。

  • 接着执行c();,会自然地在栈中保存返回地址,形成自己的栈结构,因为会有如下的代码被执行:

pushl %ebp
movl %ebp,%ebp```
- 假如在执行的过程中,不幸时间片到了,被调度到B,A的任务上下文环境保存(**第二次保存A任务上下文**),但是已经被改变了(eip肯定指向了c()中将要执行的某一指令,esp,ebp由于栈帧的形成也变化了)。

- 恢复到B任务执行,B在执行的时候,`b()`会执行调度到A。

- 恢复为A任务上下文环境的时候,此时A中的esp位置由于之前调用`c()`的时候向下(x86栈的增长方式是是满减)移动了几个位置,***那么恢复的时候,无法找到我们之前(第一次A被调度出来,A中eip保存的是`b();`的地址)保存的上下文环境(因为sp,ip均被A中对`c();`的调用破坏掉)。***

***那么看看之前我们在堆栈中又保存了什么呢?***

>实际上,由于我们采用了任务控制块中特殊的变量sp,ip来保存esp,eip,那么实际上任务上下文环境并未保存堆栈中。

***再看看我们恢复的时候又用了堆栈里面的什么?***

>什么也没用到,还是只用到了tcb的特殊变量sp,ip。实际上,在任务切换的时候,堆栈仅仅被我们用来迂回地改变eip,进行函数跳转执行,而这些动作,对esp并无影响。

>这样一来,恢复到任务A的过程是没有问题的。但是要记住的是,恢复到的并不是第一次调度的地方,而是恢复到`c();`中被中断的地方。

- 恢复到A之后,因为`c();`还没有执行完成,所以,要等待`c();`执行完,当`c();`执行完成之后,`c();`用的栈撤除,返回。

- 至此A执行完毕,可见,在任务的栈sp,ip被函数的调用或者其他破坏掉的时候,也不会影响任务的正常执行流程。

*任务会有很多个调度点,但是这很多个调度点是按照时间的先后顺序来的,而且后面的调度点相对于前面的调度点也完成了很多有意义的事情,所以我们只需要恢复到最近的一个任务调度点就可以了,恢复到以前的任务调度点反而是错误的。*

*之所以这样,我是这样理解的,任务保存的esp、eip终归是为任务中的函数服务的,对于任务切换来说,并没有实际的意义—怎么变化都没关系,只要能找到将要执行的指令,并且栈地址无误,而这一点被编译器良好的栈的建立和撤除保证。*

在这个过程中,任务的切换并没有用到ebp,函数调用准确返回,任务也能完好的切换。
所以,个人觉得在任务切换中,并不需要保存和恢复ebp。

###===============UPDATE================
*神逆转来了。。。*

g:
pushl %ebp
movl %esp, %ebp
movl 8(%ebp), %eax
addl $3, %eax
popl %ebp
ret

f:
pushl %ebp
movl %esp, %ebp
subl $4, %esp
movl 8(%ebp), %eax
movl %eax, (%esp)
call g
leave
ret

main:
pushl %ebp
movl %esp, %ebp
subl 4, %esp movl8, (%esp)
call f
addl $1, %eax
leave
ret


千算万算,没有考虑到上面的这种情况,如果两个任务的入口都是上面的这个函数,如果在上下文切换的时候,不对ebp保存、恢复,那么会有如下分析:
>A任务执行完下面这几句(ebp内为值a),突然被调度到B,B任务也执行完下面这几句(ebp内为值b),然后被调度回A。此时值得注意的是,B任务已经更新了ebp内容,a肯定不等于b。

g:
pushl %ebp
movl %esp, %ebp

切换回去的时候,因为ebp的值已经错乱掉了,那么在`movl 8(%ebp), %eax`引用参数的时候,肯定会引用错误(可能非法访问不属于自己的内存)。
这种情况,可以自己写代码模拟下。
>总结下来,ebp的保存还是有必要的,只不过自己的理解不够深入,随着自己慢慢的挖掘,可能会发现,会有更多没想到的地方。

***--update于2017/3/8 22:45***
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容