_create_thread() now correctly allocates the stack with REGION_ADDR_EXACT_ADDRESS

from the USER_STACK_REGION (until B_BASE_ADDRESS is available), made it much better
readable. It now grows from the lower USER_STACK_REGION up to the start of the main
thread's stack location.
Calls arch_thread_tls_init() and accounts for the TLS area in the stack region.
Added some comments, some cleanups.


git-svn-id: file:///srv/svn/repos/haiku/trunk/current@2382 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2003-01-07 09:48:01 +00:00
parent 59cbb92fdf
commit 20726b7fb1

View File

@ -34,6 +34,7 @@
#include <atomic.h>
#include <kerrors.h>
#include <syscalls.h>
#include <tls.h>
#define TRACE_THREAD 0
#if TRACE_THREAD
@ -233,6 +234,8 @@ create_thread_struct(const char *name)
t->kernel_stack_base = 0;
t->user_stack_region_id = -1;
t->user_stack_base = 0;
t->user_local_storage = 0;
t->kernel_errno = 0;
t->team_next = NULL;
t->q_next = NULL;
t->priority = -1;
@ -293,51 +296,56 @@ delete_thread_struct(struct thread *t)
}
/** Initializes the thread and jumps to its userspace entry point.
* This function is called at creation time of every user thread,
* but not for the team's main threads.
*/
static int
_create_user_thread_kentry(void)
{
struct thread *t;
t = thread_get_current_thread();
struct thread *thread = thread_get_current_thread();
// a signal may have been delivered here
// thread_atkernel_exit();
t->last_time = system_time();
t->in_kernel = false;
thread->last_time = system_time();
thread->in_kernel = false;
// jump to the entry point in user space
arch_thread_enter_uspace((addr)t->entry, t->args, t->user_stack_base + STACK_SIZE);
arch_thread_enter_uspace(thread, (addr)thread->entry, thread->args);
// never get here
return 0;
}
/** Initializes the thread and calls it kernel space entry point.
*/
static int
_create_kernel_thread_kentry(void)
{
struct thread *thread = thread_get_current_thread();
int (*func)(void *args);
struct thread *t;
t = thread_get_current_thread();
t->last_time = system_time();
thread->last_time = system_time();
// call the entry function with the appropriate args
func = (void *)t->entry;
func = (void *)thread->entry;
return func(t->args);
return func(thread->args);
}
static thread_id
_create_thread(const char *name, team_id pid, addr entry, void *args, int priority, bool kernel)
_create_thread(const char *name, team_id teamID, addr entry, void *args, int priority, bool kernel)
{
struct thread *t;
struct team *p;
struct team *team;
cpu_status state;
char stack_name[64];
bool abort = false;
addr mainThreadStackBase = USER_STACK_REGION + USER_STACK_REGION_SIZE;
t = create_thread_struct(name);
if (t == NULL)
@ -358,12 +366,15 @@ _create_thread(const char *name, team_id pid, addr entry, void *args, int priori
GRAB_TEAM_LOCK();
// look at the team, make sure it's not being deleted
p = team_get_team_struct_locked(pid);
if (p != NULL && p->state != TEAM_STATE_DEATH)
insert_thread_into_team(p, t);
team = team_get_team_struct_locked(teamID);
if (team != NULL && team->state != TEAM_STATE_DEATH)
insert_thread_into_team(team, t);
else
abort = true;
if (!kernel && team->main_thread)
mainThreadStackBase = team->main_thread->user_stack_base;
RELEASE_TEAM_LOCK();
if (abort) {
GRAB_THREAD_LOCK();
@ -388,33 +399,44 @@ _create_thread(const char *name, team_id pid, addr entry, void *args, int priori
if (kernel) {
// this sets up an initial kthread stack that runs the entry
arch_thread_initialize_kthread_stack(t, &_create_kernel_thread_kentry, &thread_kthread_entry, &thread_kthread_exit);
// Note: whatever function wants to set up a user stack later for this thread
// must initialize the TLS for it
arch_thread_init_kthread_stack(t, &_create_kernel_thread_kentry, &thread_kthread_entry, &thread_kthread_exit);
} else {
// create user stack
// ToDo: make this better. For now just keep trying to create a stack
// until we find a spot.
// -> implement B_BASE_ADDRESS for create_area() and use that one!
// (we can then also eliminate the mainThreadStackBase variable)
t->user_stack_base = (USER_STACK_REGION - STACK_SIZE - ENV_SIZE) + USER_STACK_REGION_SIZE;
while (t->user_stack_base > USER_STACK_REGION) {
sprintf(stack_name, "%s_stack%ld", p->name, t->id);
t->user_stack_region_id = vm_create_anonymous_region(p->_aspace_id, stack_name,
// ToDo: should we move the stack creation to _create_user_thread_kentry()?
// the stack will be between USER_STACK_REGION and the main thread stack area
// (the user stack of the main thread is created in team_create_team())
t->user_stack_base = USER_STACK_REGION;
while (t->user_stack_base < mainThreadStackBase) {
sprintf(stack_name, "%s_stack%ld", team->name, t->id);
t->user_stack_region_id = vm_create_anonymous_region(team->_aspace_id, stack_name,
(void **)&t->user_stack_base,
REGION_ADDR_ANY_ADDRESS, STACK_SIZE, REGION_WIRING_LAZY, LOCK_RW);
if (t->user_stack_region_id < 0) {
t->user_stack_base -= STACK_SIZE;
} else {
// we created a region
REGION_ADDR_EXACT_ADDRESS, STACK_SIZE + TLS_SIZE, REGION_WIRING_LAZY, LOCK_RW);
if (t->user_stack_region_id >= 0)
break;
}
t->user_stack_base += STACK_SIZE + TLS_SIZE;
}
if (t->user_stack_region_id < 0)
panic("_create_thread: unable to create user stack!\n");
// now that the TLS area is allocated, initialize TLS
arch_thread_init_tls(t);
// copy the user entry over to the args field in the thread struct
// the function this will call will immediately switch the thread into
// user space.
arch_thread_initialize_kthread_stack(t, &_create_user_thread_kentry, &thread_kthread_entry, &thread_kthread_exit);
arch_thread_init_kthread_stack(t, &_create_user_thread_kentry, &thread_kthread_entry, &thread_kthread_exit);
}
t->state = B_THREAD_SUSPENDED;
@ -424,9 +446,9 @@ _create_thread(const char *name, team_id pid, addr entry, void *args, int priori
thread_id
thread_create_user_thread(char *name, team_id pid, addr entry, void *args)
thread_create_user_thread(char *name, team_id teamID, addr entry, void *args)
{
return _create_thread(name, pid, entry, args, -1, false);
return _create_thread(name, teamID, entry, args, -1, false);
}
@ -438,9 +460,9 @@ thread_create_kernel_thread(const char *name, int (*func)(void *), void *args)
thread_id
thread_create_kernel_thread_etc(const char *name, int (*func)(void *), void *args, struct team *p)
thread_create_kernel_thread_etc(const char *name, int (*func)(void *), void *args, struct team *team)
{
return _create_thread(name, p->id, (addr)func, args, -1, true);
return _create_thread(name, team->id, (addr)func, args, -1, true);
}
@ -743,7 +765,7 @@ thread_init(kernel_args *ka)
hash_insert(thread_hash, t);
insert_thread_into_team(t->team, t);
idle_threads[i] = t;
if(i == 0)
if (i == 0)
arch_thread_set_current_thread(t);
t->cpu = &cpu[i];
}
@ -892,7 +914,7 @@ thread_exit(void)
{
cpu_status state;
struct thread *t = thread_get_current_thread();
struct team *p = t->team;
struct team *team = t->team;
bool delete_team = false;
unsigned int death_stack;
sem_id cached_death_sem;
@ -911,24 +933,24 @@ thread_exit(void)
cancel_timer(&t->alarm);
// delete the user stack region first
if (p->_aspace_id >= 0 && t->user_stack_region_id >= 0) {
if (team->_aspace_id >= 0 && t->user_stack_region_id >= 0) {
region_id rid = t->user_stack_region_id;
t->user_stack_region_id = -1;
vm_delete_region(p->_aspace_id, rid);
vm_delete_region(team->_aspace_id, rid);
}
if (p != team_get_kernel_team()) {
if (team != team_get_kernel_team()) {
// remove this thread from the current team and add it to the kernel
// put the thread into the kernel team until it dies
state = disable_interrupts();
GRAB_TEAM_LOCK();
remove_thread_from_team(p, t);
remove_thread_from_team(team, t);
insert_thread_into_team(team_get_kernel_team(), t);
if (p->main_thread == t) {
if (team->main_thread == t) {
// this was main thread in this team
delete_team = true;
team_remove_team_from_hash(p);
p->state = TEAM_STATE_DEATH;
team_remove_team_from_hash(team);
team->state = TEAM_STATE_DEATH;
}
RELEASE_TEAM_LOCK();
// swap address spaces, to make sure we're running on the kernel's pgdir
@ -938,27 +960,27 @@ thread_exit(void)
// dprintf("thread_exit: thread 0x%x now a kernel thread!\n", t->id);
}
cached_death_sem = p->death_sem;
cached_death_sem = team->death_sem;
// delete the team
if (delete_team) {
if (p->num_threads > 0) {
if (team->num_threads > 0) {
// there are other threads still in this team,
// cycle through and signal kill on each of the threads
// XXX this can be optimized. There's got to be a better solution.
struct thread *temp_thread;
char death_sem_name[SYS_MAX_OS_NAME_LEN];
sprintf(death_sem_name, "team %ld death sem", p->id);
p->death_sem = create_sem(0, death_sem_name);
if (p->death_sem < 0)
panic("thread_exit: cannot init death sem for team %ld\n", p->id);
sprintf(death_sem_name, "team %ld death sem", team->id);
team->death_sem = create_sem(0, death_sem_name);
if (team->death_sem < 0)
panic("thread_exit: cannot init death sem for team %ld\n", team->id);
state = disable_interrupts();
GRAB_TEAM_LOCK();
// we can safely walk the list because of the lock. no new threads can be created
// because of the TEAM_STATE_DEATH flag on the team
temp_thread = p->thread_list;
temp_thread = team->thread_list;
while(temp_thread) {
struct thread *next = temp_thread->team_next;
thread_kill_thread_nowait(temp_thread->id);
@ -967,16 +989,16 @@ thread_exit(void)
RELEASE_TEAM_LOCK();
restore_interrupts(state);
// wait until all threads in team are dead.
acquire_sem_etc(p->death_sem, p->num_threads, 0, 0);
delete_sem(p->death_sem);
acquire_sem_etc(team->death_sem, team->num_threads, 0, 0);
delete_sem(team->death_sem);
}
cached_death_sem = -1;
vm_put_aspace(p->aspace);
vm_delete_aspace(p->_aspace_id);
delete_owned_ports(p->id);
sem_delete_owned_sems(p->id);
vfs_free_io_context(p->io_context);
free(p);
vm_put_aspace(team->aspace);
vm_delete_aspace(team->_aspace_id);
delete_owned_ports(team->id);
sem_delete_owned_sems(team->id);
vfs_free_io_context(team->io_context);
free(team);
}
// delete the sem that others will use to wait on us and get the retcode