* Added debugger notifications for thread creation/deletion events.
* Now we deal correctly with thread debug flags (child threads inherit the parent ones, save for syscall tracing in which case it needs to be specified explicitly). git-svn-id: file:///srv/svn/repos/haiku/trunk/current@11477 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
2ee487c82a
commit
9c448e2414
@ -309,12 +309,13 @@ static thread_id
|
|||||||
create_thread(const char *name, team_id teamID, thread_entry_func entry,
|
create_thread(const char *name, team_id teamID, thread_entry_func entry,
|
||||||
void *args1, void *args2, int32 priority, bool kernel)
|
void *args1, void *args2, int32 priority, bool kernel)
|
||||||
{
|
{
|
||||||
struct thread *t;
|
struct thread *t, *currentThread;
|
||||||
struct team *team;
|
struct team *team;
|
||||||
cpu_status state;
|
cpu_status state;
|
||||||
char stack_name[B_OS_NAME_LENGTH];
|
char stack_name[B_OS_NAME_LENGTH];
|
||||||
status_t status;
|
status_t status;
|
||||||
bool abort = false;
|
bool abort = false;
|
||||||
|
bool debugNewThread = false;
|
||||||
|
|
||||||
t = create_thread_struct(name);
|
t = create_thread_struct(name);
|
||||||
if (t == NULL)
|
if (t == NULL)
|
||||||
@ -345,6 +346,26 @@ create_thread(const char *name, team_id teamID, thread_entry_func entry,
|
|||||||
state = disable_interrupts();
|
state = disable_interrupts();
|
||||||
GRAB_THREAD_LOCK();
|
GRAB_THREAD_LOCK();
|
||||||
|
|
||||||
|
// If the new thread belongs to the same team as the current thread,
|
||||||
|
// it may inherit some of the thread debug flags.
|
||||||
|
currentThread = thread_get_current_thread();
|
||||||
|
if (currentThread && currentThread->team->id == teamID) {
|
||||||
|
// inherit all user flags...
|
||||||
|
int32 debugFlags = currentThread->debug_info.flags
|
||||||
|
& B_THREAD_DEBUG_USER_FLAG_MASK;
|
||||||
|
|
||||||
|
// ... save the syscall tracing flags, unless explicitely specified
|
||||||
|
if (!(debugFlags & B_THREAD_DEBUG_SYSCALL_TRACE_CHILD_THREADS)) {
|
||||||
|
debugFlags &= ~(B_THREAD_DEBUG_PRE_SYSCALL
|
||||||
|
| B_THREAD_DEBUG_POST_SYSCALL);
|
||||||
|
}
|
||||||
|
|
||||||
|
t->debug_info.flags = debugFlags;
|
||||||
|
|
||||||
|
// stop the new thread, if desired
|
||||||
|
debugNewThread = debugFlags & B_THREAD_DEBUG_STOP_CHILD_THREADS;
|
||||||
|
}
|
||||||
|
|
||||||
// insert into global list
|
// insert into global list
|
||||||
hash_insert(sThreadHash, t);
|
hash_insert(sThreadHash, t);
|
||||||
sUsedThreads++;
|
sUsedThreads++;
|
||||||
@ -353,9 +374,19 @@ create_thread(const char *name, team_id teamID, thread_entry_func entry,
|
|||||||
GRAB_TEAM_LOCK();
|
GRAB_TEAM_LOCK();
|
||||||
// look at the team, make sure it's not being deleted
|
// look at the team, make sure it's not being deleted
|
||||||
team = team_get_team_struct_locked(teamID);
|
team = team_get_team_struct_locked(teamID);
|
||||||
if (team != NULL && team->state != TEAM_STATE_DEATH)
|
if (team != NULL && team->state != TEAM_STATE_DEATH) {
|
||||||
|
// Debug the new thread, if the parent thread required that (see above),
|
||||||
|
// or the respective global team debug flag is set. But only, if a
|
||||||
|
// debugger is installed for the team.
|
||||||
|
debugNewThread
|
||||||
|
|= (team->debug_info.flags & B_TEAM_DEBUG_STOP_NEW_THREADS);
|
||||||
|
if (debugNewThread
|
||||||
|
&& (team->debug_info.flags & B_TEAM_DEBUG_DEBUGGER_INSTALLED)) {
|
||||||
|
t->debug_info.flags |= B_THREAD_DEBUG_STOP;
|
||||||
|
}
|
||||||
|
|
||||||
insert_thread_into_team(team, t);
|
insert_thread_into_team(team, t);
|
||||||
else
|
} else
|
||||||
abort = true;
|
abort = true;
|
||||||
|
|
||||||
RELEASE_TEAM_LOCK();
|
RELEASE_TEAM_LOCK();
|
||||||
@ -691,6 +722,7 @@ struct thread_exit_args {
|
|||||||
cpu_status int_state;
|
cpu_status int_state;
|
||||||
uint32 death_stack;
|
uint32 death_stack;
|
||||||
sem_id death_sem;
|
sem_id death_sem;
|
||||||
|
team_id teamID;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -738,6 +770,10 @@ thread_exit2(void *_args)
|
|||||||
// set the next state to be gone. Will return the thread structure to a ready pool upon reschedule
|
// set the next state to be gone. Will return the thread structure to a ready pool upon reschedule
|
||||||
args.thread->next_state = THREAD_STATE_FREE_ON_RESCHED;
|
args.thread->next_state = THREAD_STATE_FREE_ON_RESCHED;
|
||||||
|
|
||||||
|
// notify the debugger
|
||||||
|
if (args.teamID >= 0 && args.teamID != team_get_kernel_team_id())
|
||||||
|
user_debug_thread_deleted(args.teamID, args.thread->id);
|
||||||
|
|
||||||
// return the death stack and reschedule one last time
|
// return the death stack and reschedule one last time
|
||||||
put_death_stack_and_reschedule(args.death_stack);
|
put_death_stack_and_reschedule(args.death_stack);
|
||||||
|
|
||||||
@ -760,6 +796,7 @@ thread_exit(void)
|
|||||||
sem_id cachedDeathSem, parentDeadSem = -1, groupDeadSem = -1;
|
sem_id cachedDeathSem, parentDeadSem = -1, groupDeadSem = -1;
|
||||||
status_t status;
|
status_t status;
|
||||||
struct thread_debug_info debugInfo;
|
struct thread_debug_info debugInfo;
|
||||||
|
team_id teamID = team->id;
|
||||||
|
|
||||||
if (!are_interrupts_enabled())
|
if (!are_interrupts_enabled())
|
||||||
dprintf("thread_exit() called with interrupts disabled!\n");
|
dprintf("thread_exit() called with interrupts disabled!\n");
|
||||||
@ -924,6 +961,7 @@ thread_exit(void)
|
|||||||
args.old_kernel_stack = thread->kernel_stack_area;
|
args.old_kernel_stack = thread->kernel_stack_area;
|
||||||
args.death_stack = death_stack;
|
args.death_stack = death_stack;
|
||||||
args.death_sem = cachedDeathSem;
|
args.death_sem = cachedDeathSem;
|
||||||
|
args.teamID = teamID;
|
||||||
|
|
||||||
// set the new kernel stack officially to the death stack, wont be really switched until
|
// set the new kernel stack officially to the death stack, wont be really switched until
|
||||||
// the next function is called. This bookkeeping must be done now before a context switch
|
// the next function is called. This bookkeeping must be done now before a context switch
|
||||||
@ -1818,13 +1856,19 @@ thread_id
|
|||||||
_user_spawn_thread(int32 (*entry)(thread_func, void *), const char *userName, int32 priority, void *data1, void *data2)
|
_user_spawn_thread(int32 (*entry)(thread_func, void *), const char *userName, int32 priority, void *data1, void *data2)
|
||||||
{
|
{
|
||||||
char name[B_OS_NAME_LENGTH];
|
char name[B_OS_NAME_LENGTH];
|
||||||
|
thread_id threadID;
|
||||||
|
|
||||||
if (!IS_USER_ADDRESS(entry) || entry == NULL
|
if (!IS_USER_ADDRESS(entry) || entry == NULL
|
||||||
|| !IS_USER_ADDRESS(userName)
|
|| !IS_USER_ADDRESS(userName)
|
||||||
|| user_strlcpy(name, userName, B_OS_NAME_LENGTH) < B_OK)
|
|| user_strlcpy(name, userName, B_OS_NAME_LENGTH) < B_OK)
|
||||||
return B_BAD_ADDRESS;
|
return B_BAD_ADDRESS;
|
||||||
|
|
||||||
return create_thread(name, thread_get_current_thread()->team->id, entry, data1, data2, priority, false);
|
threadID = create_thread(name, thread_get_current_thread()->team->id, entry,
|
||||||
|
data1, data2, priority, false);
|
||||||
|
|
||||||
|
user_debug_thread_created(threadID);
|
||||||
|
|
||||||
|
return threadID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user