Files
Jonas Devlieghere a3292e8924 [lldb] Fix TestThreadExit.py flakiness (#190976)
When `pthread_join` returns, the target thread signals its internal
semaphore, but the underlying Mach thread hasn't been removed from the
task yet with `thread_terminate`.

The flakiness is the result of the debugger stopping the process halts
the dying thread mid-termination. There is no Mach API to distinguish a
dying thread from a live one: it appears as `TH_STATE_STOPPED`, like any
other suspended thread.

This PR adds a helper (`wait_for_thread_cleanup`) that polls
`task_threads` to ensure the Mach thread is actually gone before the
breakpoint. Since there might be other tests that are affected by this
race, I put it in a common location so it can be reused.
2026-04-17 14:00:14 -07:00

66 lines
1.5 KiB
C++

// This test verifies the correct handling of child thread exits.
#include "mach_thread.h"
#include "pseudo_barrier.h"
#include <thread>
pseudo_barrier_t g_barrier1;
pseudo_barrier_t g_barrier2;
pseudo_barrier_t g_barrier3;
void *thread1() {
// Synchronize with the main thread.
pseudo_barrier_wait(g_barrier1);
// Synchronize with the main thread and thread2.
pseudo_barrier_wait(g_barrier2);
// Return
return NULL; // Set second breakpoint here
}
void *thread2() {
// Synchronize with thread1 and the main thread.
pseudo_barrier_wait(g_barrier2);
// Synchronize with the main thread.
pseudo_barrier_wait(g_barrier3);
// Return
return NULL;
}
int main() {
pseudo_barrier_init(g_barrier1, 2);
pseudo_barrier_init(g_barrier2, 3);
pseudo_barrier_init(g_barrier3, 2);
// Create a thread.
std::thread thread_1(thread1);
// Wait for thread1 to start.
pseudo_barrier_wait(g_barrier1);
// Create another thread.
std::thread thread_2(thread2); // Set first breakpoint here
// Wait for thread2 to start.
pseudo_barrier_wait(g_barrier2);
// Wait for the first thread to finish
mach_port_t mach_thread_1 = get_mach_thread(thread_1);
thread_1.join();
wait_for_thread_cleanup(mach_thread_1);
// Synchronize with the remaining thread
int dummy = 47; // Set third breakpoint here
pseudo_barrier_wait(g_barrier3);
// Wait for the second thread to finish
mach_port_t mach_thread_2 = get_mach_thread(thread_2);
thread_2.join();
wait_for_thread_cleanup(mach_thread_2);
return 0; // Set fourth breakpoint here
}