|  | //===-- 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/libc_errno.h" | 
|  | #include "src/__support/macros/config.h" | 
|  | #include "src/__support/macros/optimization.h" | 
|  | #include "src/__support/threads/thread.h" | 
|  |  | 
|  | #include <pthread.h> // 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<LIBC_NAMESPACE::Thread *>(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 |