【 声明:版权所有,欢迎转载,请勿用于商业用途。  联系信箱:feixiaoxing @163.com】

虽然写操作系统的博客要比写普通的技术点要麻烦一些,但是心中还是挺开心的。一方面,通过几行代码就可以说明一些问题,把理论实践化,这本身就很具有挑战性;另外一方面还锻炼自己的沟通能力,让更多的人明白你的想法,认可你的想法。

其实,通过上面一篇博客,我们就已经清楚任务的创建是怎么一回事,但是我们还是愿意就这个问题讲得更细一点,说得更多一点。系统本身是多线程的,那说明所有线程的地址空间都是共享的。由于资源都是操作系统本身提供的,所以线程本身的要求就很低,函数名、堆栈、入口点、堆栈大小、优先级,大体上也就是这么多。至于这个堆栈是哪里的内存,其实已经不太重要了。为了简单起见,我们对原来的初始化函数 稍微修改了一下,

void task_init()
{
UINT32 unit = STACK_LENGTH;
memset((void*)data, 0, STACK_LENGTH * sizeof(UINT32));
data[unit -1] = (UINT32) hello;
data[unit -2] = 0;
data[unit -3] = 0;
data[unit -4] = 0;
data[unit -5] = 0;
data[unit -6] = 0;
data[unit -7] = 0;
data[unit -8] = 0;
data[unit -9] = 0;
data[unit -10] = (UINT32) &data[unit - 9];
new = (UINT32) &data[unit -10];
}

上面的操作比较简陋,只是对堆栈进行了设置。这是线程初始化的时候必须要做的一步。当然,这里的hello就是我们的函数入口点。因为这里用SIGALRM代替的时钟中断是没有办法做到抢占的,所以我们可以人为多设置一些调度点,比如象这样,

void hello()
{
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
quit = 1;
swap(&new, &old);
}

在编写程序的时候,最恐怖的事情就是堆栈溢出了。但是在操作系统中,我们完全可以自己判断当前的堆栈是否已经溢出。因为我们知道,在线程调度的时候,保存的堆栈esp永远指向最低的那个地址。

int check_stack_overflow(unsigned int base, unsigned int current)
{
assert(0 != base && 0 != current);
return (current < base) ? 1 :0;
}

当然,这些说的都是线程调度的事,你也可以编写输入输出命令,实现对嵌入式操作系统的某种控制。要打印什么,设置什么,保存什么,都可以通过你的输入命令来解析执行,这些都是和signal处理是分开来的。后面这部分还要详细讨论,这里可以稍微添加一下,

int main()
{
char val;
task_init();
set_timer();
signal(SIGALRM, signal_handler);
while(1)
{
scanf("%c", &val);
}
exit(0);
return 1;
}

最后,还是老规矩,附上详细的代码。虽然这一过程有点繁琐和冗余,但是至少看上去更完整一些。

#include <stdio.h>
#include <time.h>
#include <stdlib.h>
#include <signal.h>
#include <assert.h>
#include <sys/time.h>
#define UINT32 unsigned int
#define STACK_LENGTH  512
static struct itimerval oldtv;
UINT32 old = 0;
UINT32 new = 0;
UINT32 count = 0;
UINT32 data[STACK_LENGTH] = {0};
UINT32 quit = 0;
void set_timer()
{
struct itimerval itv;
itv.it_interval.tv_sec = 1;
itv.it_interval.tv_usec = 0;
itv.it_value.tv_sec = 1;
itv.it_value.tv_usec = 0;
setitimer(ITIMER_REAL, &itv, &oldtv);
}
void swap(UINT32* prev, UINT32* next)
{
__asm("push %%eax\n\t"
"push %%ebx\n\t"
"push %%ecx\n\t"
"push %%edx\n\t"
"push %%esi\n\t"
"push %%edi\n\t"
"push %%ebp\n\t"
"push %%esp\n\t"
"lea 0x8(%%ebp), %%eax\n\t"
"mov (%%eax), %%eax\n\t"
"mov %%esp, (%%eax)\n\t"
"lea 0xc(%%ebp), %%eax\n\t"
"mov (%%eax), %%eax\n\t"
"mov (%%eax), %%esp\n\t"
"pop %%esp\n\t"
"pop %%ebp\n\t"
"pop %%edi\n\t"
"pop %%esi\n\t"
"pop %%edx\n\t"
"pop %%ecx\n\t"
"pop %%ebx\n\t"
"pop %%eax\n\t"
::);
}
void hello()
{
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
swap(&new, &old);
printf("count = %d in sub!\n", count ++);
quit = 1;
swap(&new, &old);
}
void task_init()
{
UINT32 unit = STACK_LENGTH;
memset((void*)data, 0, STACK_LENGTH * sizeof(UINT32));
data[unit -1] = (UINT32) hello;
data[unit -2] = 0;
data[unit -3] = 0;
data[unit -4] = 0;
data[unit -5] = 0;
data[unit -6] = 0;
data[unit -7] = 0;
data[unit -8] = 0;
data[unit -9] = 0;
data[unit -10] = (UINT32) &data[unit - 9];
new = (UINT32) &data[unit -10];
}
int check_stack_overflow(unsigned int base, unsigned int current)
{
assert(0 != base && 0 != current);
return (current < base) ? 1 :0;
}
void signal_handler(int m)
{
if(0 == quit)
{
swap(&old, &new);
assert(0 == check_stack_overflow(data,new));
return;
}
printf("count = %d in main!\n", count ++);
}
int main()
{
char val;
task_init();
set_timer();
signal(SIGALRM, signal_handler);
while(1)
{
scanf("%c", &val);
}
exit(0);
return 1;
}

嵌入式操作系统内核原理和开发(任务创建和堆栈溢出检查)相关推荐

  1. 嵌入式操作系统内核原理和开发

    嵌入式操作系统内核原理和开发(开篇) 操作系统是很多人每天必须打交道的东西,因为在你打开电脑的一刹那,随着bios自检结束,你的windows系统已经开始运行了.如果问大家操作系统是什么?可能有的人会 ...

  2. 嵌入式操作系统内核原理和开发(总结篇)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 很多朋友都喜欢嵌入式操作系统的内容,但是如何实现和仿真这样一个系统一直是困扰我们的难题.现在郑 ...

  3. 嵌入式操作系统内核原理和开发(信号量)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 之前因为工作的原因,操作系统这块一直没有继续写下去.一方面是自己没有这方面的经历,另外一方面就 ...

  4. 嵌入式操作系统内核原理和开发(地址空间)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 不管是什么样的嵌入式cpu,它必然有自己的访问地址空间.至于这个具体的访问空间是什么,那cpu ...

  5. 嵌入式操作系统内核原理和开发(头文件调整)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 很长一段时间,我个人对头文件的功能了解得不是很明白.虽然在平时的开发中,对于头文件也没有犯过什 ...

  6. 嵌入式操作系统内核原理和开发(实时系统中的定时器)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 关于定时器的内容,其实我们之前也讨论过,也书写过相应的代码,但是表达得比较晦涩,效率也比较低. ...

  7. 嵌入式操作系统内核原理和开发(基础)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 在编写我们的操作系统之前,我们需要明确一些事情.比如说,这个系统的运行环境是什么?怎么编译?基 ...

  8. 嵌入式操作系统内核原理和开发(cpu的那些事)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] cpu是数字处理系统中的一个重要环节.在我看来,单片机.微处理器.dsp都可以称作是cpu,只 ...

  9. 嵌入式操作系统内核原理和开发(开篇)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 操作系统是很多人每天必须打交道的东西,因为在你打开电脑的一刹那,随着bios自检结束,你的wi ...

最新文章

  1. linux系统怎么查找 文件是否存在,在Linux上,如何知道一个可执行文件是否包含调试信息呢?...
  2. 如何把讨厌的人踢出局域网(kickthemout)
  3. 给老婆普及计算机知识
  4. 1、创建数据表(CREATE TABLE语句)
  5. ERROR 2002 (HY000): Can't connect to local MySQL server through socket '/var/lib/mysql/mysq
  6. Codeforces 685C Optimal Point (二分、不同类型距离的相互转换)
  7. wcf简单教程(10) ajax调用,wcf简单教程(10) ajax调用
  8. QT的Q3DBars类的使用
  9. android确认密码代码,Android自定义View实现验证码or密码输入框
  10. 44 SD配置-销售凭证设置-定义状态参数文件
  11. QT [005] 数据库设计 - 一个被忽略的数据库show类 - ConnectionWidget Class
  12. oracle mos账号金额,mos账号注正册步骤+证书查询+注意事项.pptx
  13. 如何在Ubuntu系统中使用github
  14. CVS update常用技巧
  15. 双指针算法基础——输出单词
  16. 数据库知识点总结归纳
  17. HCIE大师之路——Lab讲解
  18. 最小项标准式和卡诺图化简
  19. 如何在matlab中绘制障碍物,Turtlebot与Matlab入门教程-避开障碍物
  20. Vuex入门及进阶笔记

热门文章

  1. 调用语音验证码的接口与图形验证码框架
  2. 使用swiftenv管理swift版本
  3. JUnit编写单元测试代码注意点小结
  4. android aapt 用法 -- ApkReader
  5. Week 1 Team Homework #3 from Z.XML-软件工程在北航
  6. springboot整合nacos配置实现实时更新
  7. nginx经过多层代理后获取真实来源ip
  8. 在URL中使用另一个url作为参数时会被``截断的问题
  9. 使用jQuery获取GridView的数据行的数量
  10. Java实现HTML页面转PDF解决方案(转)