#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 <errno.h>
#include <pthread.h>
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) {
if (LIBC_UNLIKELY(LIBC_NAMESPACE::pthread_attr_init(&default_attr) != 0))
return EINVAL;
attr = &default_attr;
}
void *stack;
size_t stacksize, guardsize;
int detachstate;
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)
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;
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;
}
}