linux内核堆栈..docVIP

  1. 1、本文档共5页,可阅读全部内容。
  2. 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  5. 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  6. 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  7. 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  8. 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
linux内核堆栈.

在内核2.4中堆栈是这么定义的: union task_union { ??????? struct task_struct task; ????????unsigned long stack[INIT_TASK_SIZE/sizeof(long)]; ??? }; 而INIT_TASK_SIZE只能是8K。 内核为每个进程分配一个task_struct结构时,实际上分配两个连续的物理页面(8192字节),如图所示。底部用作task_struct结构(大小约为1K字节),结构的上面用作内核堆栈(大小约为7K字节)。访问进程自身的task_struct结构,使用宏操作current, 在2.4中定义如下: #define current get_current() static inline struct task_struct * get_current(void) { ?? ?? struct task_struct *current; ?? ?? __asm__(andl %%esp,%0; :=r (current) : (~8191UL)); ?? ?? return current; } ??~8191UL表示最低13位为0, 其余位全为1。 %esp指向内核堆栈中,当屏蔽掉%esp的最低13后,就得到这个”两个连续的物理页面”的开头,而这个开头正好是task_struct的开始,从而得到了指向task_struct的指针。 在内核2.6中堆栈这么定义: union thread_union { ?? ?? struct thread_info thread_info; ?? ?? unsigned long stack[THREAD_SIZE/sizeof(long)]; }; 根据内核的配置,THREAD_SIZE既可以是4K字节(1个页面)也可以是8K字节(2个页面)。thread_info是52个字节长。 下图是当设为8KB时候的内核堆栈:Thread_info在这个内存区的开始处,内核堆栈从末端向下增长。进程描述符不是在这个内存区中,而分别通过task与thread_info指针使thread_info与进程描述符互联。所以获得当前进程描述符的current定义如下: #define current get_current() static inline struct task_struct * get_current(void) { ?? ?? return current_thread_info()-task; } static inline struct thread_info *current_thread_info(void) { ?? ??? struct thread_info *ti; ?? ??? __asm__(andl %%esp,%0; :=r (ti) : (~(THREAD_SIZE - 1))); ?? ??? return ti; } ??? 根据THREAD_SIZE大小,分别屏蔽掉内核栈的12-bit LSB(4K)或13-bit LSB(8K),从而获得内核栈的起始位置。 struct thread_info { ?? ?? struct task_struct ?? *task; ?? ?? /* main task structure */ ?? ?? struct exec_domain ?? *exec_domain; /* execution domain */ ?? ?? unsigned long ?? ?? ??? flags; ?? ?? /* low level flags */ ?? ?? unsigned long ?? ?? ??? status; ?? ?? /* thread-synchronous flags */ ?? ?? ... .. } fork系统调用中调用dup_task_struct,其执行: 1,? alloc_task_struct宏,为新进程获取进程描述符,并将描述符放在局部变量tsk中。 2,? alloc_thread_info宏以获取一块空闲的内存区,用以存放新进程的thread_info结构和内核栈,并将这块内存区字段的地址放在局部变量ti中(8K 或 4K, 可配置)。 3,? current进程描述符的内容复制到tsk所指向的task_struct结构中,然后把tsk-thread_info置为ti。 4,? current进程的thread_info描述符的内容复制到ti中,然后把ti-task置为tsk。 5,? tsk。 ? static struct task_struc

文档评论(0)

bhzs + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档