//===-- Linux implementation of the pthread_create function ---------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "pthread_create.h" #include "pthread_attr_destroy.h" #include "pthread_attr_init.h" #include "pthread_attr_getdetachstate.h" #include "pthread_attr_getguardsize.h" #include "pthread_attr_getstack.h" #include "src/__support/common.h" #include "src/__support/macros/config.h" #include "src/__support/macros/optimization.h" #include "src/__support/threads/thread.h" #include "src/errno/libc_errno.h" #include // For pthread_* type definitions. namespace LIBC_NAMESPACE_DECL { static_assert(sizeof(pthread_t) == sizeof(LIBC_NAMESPACE::Thread), "Mismatch between pthread_t and internal Thread."); LLVM_LIBC_FUNCTION(int, pthread_create, (pthread_t *__restrict th, const pthread_attr_t *__restrict attr, __pthread_start_t func, void *arg)) { pthread_attr_t default_attr; if (attr == nullptr) { // We failed to initialize attributes (should be impossible) if (LIBC_UNLIKELY(LIBC_NAMESPACE::pthread_attr_init(&default_attr) != 0)) return EINVAL; attr = &default_attr; } void *stack; size_t stacksize, guardsize; int detachstate; // As of writing this all the `pthread_attr_get*` functions always succeed. if (LIBC_UNLIKELY( LIBC_NAMESPACE::pthread_attr_getstack(attr, &stack, &stacksize) != 0)) return EINVAL; if (LIBC_UNLIKELY( LIBC_NAMESPACE::pthread_attr_getguardsize(attr, &guardsize) != 0)) return EINVAL; if (LIBC_UNLIKELY( LIBC_NAMESPACE::pthread_attr_getdetachstate(attr, &detachstate) != 0)) return EINVAL; if (attr == &default_attr) // Should we fail here? Its non-issue as the moment as pthread_attr_destroy // can only succeed. if (LIBC_UNLIKELY(LIBC_NAMESPACE::pthread_attr_destroy(&default_attr) != 0)) return EINVAL; if (stacksize && stacksize < PTHREAD_STACK_MIN) return EINVAL; if (guardsize % EXEC_PAGESIZE != 0) return EINVAL; if (detachstate != PTHREAD_CREATE_DETACHED && detachstate != PTHREAD_CREATE_JOINABLE) return EINVAL; // Thread::run will check validity of the `stack` argument (stack alignment is // universal, not sure a pthread requirement). auto *thread = reinterpret_cast(th); int result = thread->run(func, arg, stack, stacksize, guardsize, detachstate == PTHREAD_CREATE_DETACHED); if (result != 0 && result != EPERM && result != EINVAL) return EAGAIN; return result; } } // namespace LIBC_NAMESPACE_DECL