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