In rte_thread_create setting affinity after CreateThread may fail. Such
a failure is reported but strands the newly created thread in a
suspended state.

Resolve the above issue by notifying the newly created thread that
it should terminate as soon as it is resumed, while still continuing to
free the ctx.

Fixes: ce6e911d20f6 ("eal: add thread lifetime API")
Cc: sta...@dpdk.org
Cc: roret...@linux.microsoft.com

Signed-off-by: Tyler Retzlaff <roret...@linux.microsoft.com>
---
 lib/eal/windows/rte_thread.c | 19 ++++++++++++++++---
 1 file changed, 16 insertions(+), 3 deletions(-)

diff --git a/lib/eal/windows/rte_thread.c b/lib/eal/windows/rte_thread.c
index 8556a84..e528ac9 100644
--- a/lib/eal/windows/rte_thread.c
+++ b/lib/eal/windows/rte_thread.c
@@ -19,6 +19,7 @@ struct eal_tls_key {
 
 struct thread_routine_ctx {
        rte_thread_func thread_func;
+       bool thread_init_failed;
        void *routine_args;
 };
 
@@ -167,9 +168,13 @@ struct thread_routine_ctx {
 thread_func_wrapper(void *arg)
 {
        struct thread_routine_ctx ctx = *(struct thread_routine_ctx *)arg;
+       const bool thread_exit = __atomic_load_n(&ctx.thread_init_failed, 
__ATOMIC_ACQUIRE);
 
        free(arg);
 
+       if (thread_exit)
+               return 0;
+
        return (DWORD)ctx.thread_func(ctx.routine_args);
 }
 
@@ -183,6 +188,7 @@ struct thread_routine_ctx {
        HANDLE thread_handle = NULL;
        GROUP_AFFINITY thread_affinity;
        struct thread_routine_ctx *ctx;
+       bool thread_exit = false;
 
        ctx = calloc(1, sizeof(*ctx));
        if (ctx == NULL) {
@@ -192,6 +198,7 @@ struct thread_routine_ctx {
        }
        ctx->routine_args = args;
        ctx->thread_func = thread_func;
+       ctx->thread_init_failed = false;
 
        thread_handle = CreateThread(NULL, 0, thread_func_wrapper, ctx,
                CREATE_SUSPENDED, &tid);
@@ -209,23 +216,29 @@ struct thread_routine_ctx {
                                                        );
                        if (ret != 0) {
                                RTE_LOG(DEBUG, EAL, "Unable to convert cpuset 
to thread affinity\n");
-                               goto cleanup;
+                               thread_exit = true;
+                               goto resume_thread;
                        }
 
                        if (!SetThreadGroupAffinity(thread_handle,
                                                    &thread_affinity, NULL)) {
                                ret = 
thread_log_last_error("SetThreadGroupAffinity()");
-                               goto cleanup;
+                               thread_exit = true;
+                               goto resume_thread;
                        }
                }
                ret = rte_thread_set_priority(*thread_id,
                                thread_attr->priority);
                if (ret != 0) {
                        RTE_LOG(DEBUG, EAL, "Unable to set thread priority\n");
-                       goto cleanup;
+                       thread_exit = true;
+                       goto resume_thread;
                }
        }
 
+resume_thread:
+       __atomic_store_n(&ctx->thread_init_failed, thread_exit, 
__ATOMIC_RELEASE);
+
        if (ResumeThread(thread_handle) == (DWORD)-1) {
                ret = thread_log_last_error("ResumeThread()");
                goto cleanup;
-- 
1.8.3.1

Reply via email to