Linux内核中的current_thread_info
2014-04-07 21:30
274 查看
current_thread_info的定义在include/asm/thread_info.h中:
[align=left]/* Given a task stack pointer, you can find it's task structure[/align]
[align=left] * just by masking it to the 8K boundary.[/align]
[align=left] */[/align]
static
inline struct thread_info *current_thread_info(void)
[align=left]{[/align]
[align=left] struct thread_info *ti;[/align]
__asm__("%0 = sp;":
"=&d"(ti):
[align=left] );[/align]
[align=left] return (struct thread_info *)((long)ti & ~((long)THREAD_SIZE-1));[/align]
[align=left]}[/align]
它的作用在于取得运行当前代码的线程的信息,此信息保存在thread_info结构体中。
在上述函数中,THREAD_SIZE定义为:
[align=left] [/align]
[align=left]/*[/align]
[align=left] * Size of kernel stack for each process. This must be a power of 2...[/align]
[align=left] */[/align]
#define THREAD_SIZE 8192
/* 2 pages */
也就是说,上述函数将取SP指针,并将此指针按照8K对齐,即2页对齐,这就是线程信息的指针了。
记得在处理head.s的时候,有一段代码:
/*
* load the current thread pointer and stack
*/
r1.l = _init_thread_union;
r1.h = _init_thread_union;
r2.l = 0x2000; // 8192字节
r2.h = 0x0000;
r1 = r1 + r2;
sp = r1;
usp = sp;
fp = sp;
其中init_thread_union的定义为:
[align=left]/*[/align]
[align=left] * Initial thread structure.[/align]
[align=left] *[/align]
[align=left] * We need to make sure that this is 8192-byte aligned due to the[/align]
[align=left] * way process stacks are handled. This is done by having a special[/align]
[align=left] * "init_task" linker map entry.[/align]
[align=left] */[/align]
[align=left]union thread_union init_thread_union[/align]
[align=left] __attribute__ ((__section__(".data.init_task"))) = {[/align]
[align=left]INIT_THREAD_INFO(init_task)};[/align]
thread_union的定义在include/linux/shed.h中:
[align=left] [/align]
[align=left]union thread_union {[/align]
[align=left] struct thread_info thread_info;[/align]
unsigned
long stack[THREAD_SIZE/sizeof(long)];
[align=left]};[/align]
这个union的结构体大小为THREAD_SIZE,也就是8K,在初始化的时候,将FP和SP都指向了初始线程stack的最高位置。
那么这个线程信息是如何保证以8K对齐的呢?答案在.data.init_task这个段,在LDF文件中,有这样的语句:
.data
{
/* make sure the init_task is aligned to the
* kernel thread size so we can locate the kernel
* stack properly and quickly.
*/
__sdata = .;
INPUT_SECTION_ALIGN(8192)
INPUT_SECTIONS($LIBRARIES_CORE_A(.data.init_task))
…
} > MEM_SDRAM
从而保证了init_thread_union是以8192对齐的。
对于一般的线程信息,uclinux内核是如何做到这一点的呢?在thread_info.h中有一个宏定义:
[align=left] [/align]
[align=left]/* thread information allocation */[/align]
[align=left]#define alloc_thread_info(tsk) ((struct thread_info *) /[/align]
[align=left] __get_free_pages(GFP_KERNEL, 1))[/align]
#define free_thread_info(ti) free_pages((unsigned
long) (ti), 1)
猜测是通过__get_free_pages这个函数来达到目的的,不过目前还没有涉及到内存分配与管理这一块,暂且放一放。
转至:http://blog.csdn.net/lights_joy/article/details/2436899
[align=left]/* Given a task stack pointer, you can find it's task structure[/align]
[align=left] * just by masking it to the 8K boundary.[/align]
[align=left] */[/align]
static
inline struct thread_info *current_thread_info(void)
[align=left]{[/align]
[align=left] struct thread_info *ti;[/align]
__asm__("%0 = sp;":
"=&d"(ti):
[align=left] );[/align]
[align=left] return (struct thread_info *)((long)ti & ~((long)THREAD_SIZE-1));[/align]
[align=left]}[/align]
它的作用在于取得运行当前代码的线程的信息,此信息保存在thread_info结构体中。
在上述函数中,THREAD_SIZE定义为:
[align=left] [/align]
[align=left]/*[/align]
[align=left] * Size of kernel stack for each process. This must be a power of 2...[/align]
[align=left] */[/align]
#define THREAD_SIZE 8192
/* 2 pages */
也就是说,上述函数将取SP指针,并将此指针按照8K对齐,即2页对齐,这就是线程信息的指针了。
记得在处理head.s的时候,有一段代码:
/*
* load the current thread pointer and stack
*/
r1.l = _init_thread_union;
r1.h = _init_thread_union;
r2.l = 0x2000; // 8192字节
r2.h = 0x0000;
r1 = r1 + r2;
sp = r1;
usp = sp;
fp = sp;
其中init_thread_union的定义为:
[align=left]/*[/align]
[align=left] * Initial thread structure.[/align]
[align=left] *[/align]
[align=left] * We need to make sure that this is 8192-byte aligned due to the[/align]
[align=left] * way process stacks are handled. This is done by having a special[/align]
[align=left] * "init_task" linker map entry.[/align]
[align=left] */[/align]
[align=left]union thread_union init_thread_union[/align]
[align=left] __attribute__ ((__section__(".data.init_task"))) = {[/align]
[align=left]INIT_THREAD_INFO(init_task)};[/align]
thread_union的定义在include/linux/shed.h中:
[align=left] [/align]
[align=left]union thread_union {[/align]
[align=left] struct thread_info thread_info;[/align]
unsigned
long stack[THREAD_SIZE/sizeof(long)];
[align=left]};[/align]
这个union的结构体大小为THREAD_SIZE,也就是8K,在初始化的时候,将FP和SP都指向了初始线程stack的最高位置。
那么这个线程信息是如何保证以8K对齐的呢?答案在.data.init_task这个段,在LDF文件中,有这样的语句:
.data
{
/* make sure the init_task is aligned to the
* kernel thread size so we can locate the kernel
* stack properly and quickly.
*/
__sdata = .;
INPUT_SECTION_ALIGN(8192)
INPUT_SECTIONS($LIBRARIES_CORE_A(.data.init_task))
…
} > MEM_SDRAM
从而保证了init_thread_union是以8192对齐的。
对于一般的线程信息,uclinux内核是如何做到这一点的呢?在thread_info.h中有一个宏定义:
[align=left] [/align]
[align=left]/* thread information allocation */[/align]
[align=left]#define alloc_thread_info(tsk) ((struct thread_info *) /[/align]
[align=left] __get_free_pages(GFP_KERNEL, 1))[/align]
#define free_thread_info(ti) free_pages((unsigned
long) (ti), 1)
猜测是通过__get_free_pages这个函数来达到目的的,不过目前还没有涉及到内存分配与管理这一块,暂且放一放。
转至:http://blog.csdn.net/lights_joy/article/details/2436899
相关文章推荐
- linux下sed的使用(下)
- linux下sed的使用(中)
- linux下sed的使用(上)
- arm-Linux汇编语法及它和ADS汇编转换的问题
- linux中mv命令使用详解(移动文件或者将文件改名)
- Linux常用vi命令
- linux中rmdir命令使用详解(删除空目录)
- linux内核线程的创建与销毁
- FS_S5PC100平台上Linux Camera驱动开发详解
- 如何查看linux的文件系统block size
- linux驱动头文件位置的说明
- (转自鸟哥的Linux私房菜)Linux中使用ls指令时total的意义
- linux rm 命令(删除文件和目录) 使用详解
- halt和shutdown
- Linux下C语言编程资料
- Linux下C语言Mysql数据库使用范例
- Linux C之atoi()函数
- [arm驱动]linux内核时钟 推荐
- Linux C语言操作MySQL
- Linux学习之CentOS(十三)--CentOS6.4下Mysql数据库的安装与配置