aboutsummaryrefslogtreecommitdiffstats
path: root/main/musl
diff options
context:
space:
mode:
Diffstat (limited to 'main/musl')
-rw-r--r--main/musl/0005-4674809b-to-3e936ce8.patch3400
-rw-r--r--main/musl/APKBUILD11
-rw-r--r--main/musl/alignas.patch17
3 files changed, 3406 insertions, 22 deletions
diff --git a/main/musl/0005-4674809b-to-3e936ce8.patch b/main/musl/0005-4674809b-to-3e936ce8.patch
new file mode 100644
index 0000000000..9180e7eab6
--- /dev/null
+++ b/main/musl/0005-4674809b-to-3e936ce8.patch
@@ -0,0 +1,3400 @@
+diff --git a/arch/arm/bits/alltypes.h.in b/arch/arm/bits/alltypes.h.in
+index 183c4c4..bbe3ce6 100644
+--- a/arch/arm/bits/alltypes.h.in
++++ b/arch/arm/bits/alltypes.h.in
+@@ -20,6 +20,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/i386/bits/alltypes.h.in b/arch/i386/bits/alltypes.h.in
+index 8ba8f6f..646922b 100644
+--- a/arch/i386/bits/alltypes.h.in
++++ b/arch/i386/bits/alltypes.h.in
+@@ -27,13 +27,19 @@ TYPEDEF long double float_t;
+ TYPEDEF long double double_t;
+ #endif
+
++#ifdef __cplusplus
++TYPEDEF struct { alignas(8) long long __ll; long double __ld; } max_align_t;
++#else
+ TYPEDEF struct { _Alignas(8) long long __ll; long double __ld; } max_align_t;
++#endif
+
+ TYPEDEF long time_t;
+ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/microblaze/bits/alltypes.h.in b/arch/microblaze/bits/alltypes.h.in
+index a03e1b8..95f1ebb 100644
+--- a/arch/microblaze/bits/alltypes.h.in
++++ b/arch/microblaze/bits/alltypes.h.in
+@@ -20,6 +20,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/mips/bits/alltypes.h.in b/arch/mips/bits/alltypes.h.in
+index a03e1b8..95f1ebb 100644
+--- a/arch/mips/bits/alltypes.h.in
++++ b/arch/mips/bits/alltypes.h.in
+@@ -20,6 +20,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/or1k/bits/alltypes.h.in b/arch/or1k/bits/alltypes.h.in
+index 183c4c4..bbe3ce6 100644
+--- a/arch/or1k/bits/alltypes.h.in
++++ b/arch/or1k/bits/alltypes.h.in
+@@ -20,6 +20,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/powerpc/bits/alltypes.h.in b/arch/powerpc/bits/alltypes.h.in
+index ee7f137..7446ab4 100644
+--- a/arch/powerpc/bits/alltypes.h.in
++++ b/arch/powerpc/bits/alltypes.h.in
+@@ -20,6 +20,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/sh/bits/alltypes.h.in b/arch/sh/bits/alltypes.h.in
+index ee7f137..7446ab4 100644
+--- a/arch/sh/bits/alltypes.h.in
++++ b/arch/sh/bits/alltypes.h.in
+@@ -20,6 +20,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[9]; unsigned __s[9]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[6]; volatile void *volatile __p[6]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[12]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[8]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[5]; void *__p[5]; } __u; } pthread_barrier_t;
+diff --git a/arch/x32/bits/alltypes.h.in b/arch/x32/bits/alltypes.h.in
+index 8e396c9..a469d5c 100644
+--- a/arch/x32/bits/alltypes.h.in
++++ b/arch/x32/bits/alltypes.h.in
+@@ -25,6 +25,8 @@ TYPEDEF long long suseconds_t;
+
+ TYPEDEF struct { union { int __i[14]; unsigned long __s[7]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[10]; volatile void *volatile __p[5]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[10]; volatile void *volatile __p[5]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[6]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[6]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[14]; void *__p[7]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[4]; } __u; } pthread_barrier_t;
+diff --git a/arch/x86_64/bits/alltypes.h.in b/arch/x86_64/bits/alltypes.h.in
+index 7b4f3e7..2ce8e4a 100644
+--- a/arch/x86_64/bits/alltypes.h.in
++++ b/arch/x86_64/bits/alltypes.h.in
+@@ -25,6 +25,8 @@ TYPEDEF long suseconds_t;
+
+ TYPEDEF struct { union { int __i[14]; unsigned long __s[7]; } __u; } pthread_attr_t;
+ TYPEDEF struct { union { int __i[10]; volatile void *volatile __p[5]; } __u; } pthread_mutex_t;
++TYPEDEF struct { union { int __i[10]; volatile void *volatile __p[5]; } __u; } mtx_t;
+ TYPEDEF struct { union { int __i[12]; void *__p[6]; } __u; } pthread_cond_t;
++TYPEDEF struct { union { int __i[12]; void *__p[6]; } __u; } cnd_t;
+ TYPEDEF struct { union { int __i[14]; void *__p[7]; } __u; } pthread_rwlock_t;
+ TYPEDEF struct { union { int __i[8]; void *__p[4]; } __u; } pthread_barrier_t;
+diff --git a/include/fcntl.h b/include/fcntl.h
+index 2d8fa6e..ff9fcb9 100644
+--- a/include/fcntl.h
++++ b/include/fcntl.h
+@@ -46,6 +46,10 @@ int posix_fallocate(int, off_t, off_t);
+ #define O_WRONLY 01
+ #define O_RDWR 02
+
++#define F_OFD_GETLK 36
++#define F_OFD_SETLK 37
++#define F_OFD_SETLKW 38
++
+ #define F_DUPFD_CLOEXEC 1030
+
+ #define F_RDLCK 0
+diff --git a/include/features.h b/include/features.h
+index 294c61d..3cc3e57 100644
+--- a/include/features.h
++++ b/include/features.h
+@@ -1,10 +1,14 @@
+ #ifndef _FEATURES_H
+ #define _FEATURES_H
+
+-#ifdef _ALL_SOURCE
++#if defined(_ALL_SOURCE) && !defined(_GNU_SOURCE)
+ #define _GNU_SOURCE 1
+ #endif
+
++#if defined(_DEFAULT_SOURCE) && !defined(_BSD_SOURCE)
++#define _BSD_SOURCE 1
++#endif
++
+ #if !defined(_POSIX_SOURCE) && !defined(_POSIX_C_SOURCE) \
+ && !defined(_XOPEN_SOURCE) && !defined(_GNU_SOURCE) \
+ && !defined(_BSD_SOURCE) && !defined(__STRICT_ANSI__)
+diff --git a/include/glob.h b/include/glob.h
+index 9fbbaa6..76f6c1c 100644
+--- a/include/glob.h
++++ b/include/glob.h
+@@ -39,6 +39,7 @@ void globfree(glob_t *);
+ #if defined(_LARGEFILE64_SOURCE) || defined(_GNU_SOURCE)
+ #define glob64 glob
+ #define globfree64 globfree
++#define glob64_t glob_t
+ #endif
+
+ #ifdef __cplusplus
+diff --git a/include/signal.h b/include/signal.h
+index c36e4d5..87301ba 100644
+--- a/include/signal.h
++++ b/include/signal.h
+@@ -196,7 +196,7 @@ void psignal(int, const char *);
+
+ #endif
+
+-#if defined(_XOPEN_SOURCE) || defined(_GNU_SOURCE)
++#if defined(_XOPEN_SOURCE) || defined(_BSD_SOURCE) || defined(_GNU_SOURCE)
+ int killpg(pid_t, int);
+ int sigaltstack(const stack_t *__restrict, stack_t *__restrict);
+ int sighold(int);
+diff --git a/include/stdlib.h b/include/stdlib.h
+index db569d9..97ce5a7 100644
+--- a/include/stdlib.h
++++ b/include/stdlib.h
+@@ -114,9 +114,6 @@ long int random (void);
+ void srandom (unsigned int);
+ char *initstate (unsigned int, char *, size_t);
+ char *setstate (char *);
+-#endif
+-
+-#if defined(_XOPEN_SOURCE) || defined(_GNU_SOURCE)
+ int putenv (char *);
+ int posix_openpt (int);
+ int grantpt (int);
+diff --git a/include/sys/stat.h b/include/sys/stat.h
+index c6abab5..82a6490 100644
+--- a/include/sys/stat.h
++++ b/include/sys/stat.h
+@@ -100,8 +100,9 @@ int lchmod(const char *, mode_t);
+ #define fstat64 fstat
+ #define lstat64 lstat
+ #define fstatat64 fstatat
+-#define blksize64_t blksize_t
+ #define blkcnt64_t blkcnt_t
++#define fsblkcnt64_t fsblkcnt_t
++#define fsfilcnt64_t fsfilcnt_t
+ #define ino64_t ino_t
+ #define off64_t off_t
+ #endif
+diff --git a/include/sys/types.h b/include/sys/types.h
+index f00db03..75e489c 100644
+--- a/include/sys/types.h
++++ b/include/sys/types.h
+@@ -73,7 +73,6 @@ typedef unsigned long long u_quad_t;
+ #endif
+
+ #if defined(_LARGEFILE64_SOURCE) || defined(_GNU_SOURCE)
+-#define blksize64_t blksize_t
+ #define blkcnt64_t blkcnt_t
+ #define fsblkcnt64_t fsblkcnt_t
+ #define fsfilcnt64_t fsfilcnt_t
+diff --git a/include/threads.h b/include/threads.h
+new file mode 100644
+index 0000000..0e5836c
+--- /dev/null
++++ b/include/threads.h
+@@ -0,0 +1,85 @@
++#ifndef _THREADS_H
++#define _THREADS_H
++
++#include <features.h>
++#include <time.h>
++
++#ifdef __cplusplus
++extern "C" {
++typedef unsigned long thrd_t;
++#else
++typedef struct __pthread *thrd_t;
++#define thread_local _Thread_local
++#endif
++
++typedef int once_flag;
++typedef unsigned tss_t;
++typedef int (*thrd_start_t)(void *);
++typedef void (*tss_dtor_t)(void *);
++
++#define __NEED_cnd_t
++#define __NEED_mtx_t
++
++#include <bits/alltypes.h>
++
++#define TSS_DTOR_ITERATIONS 4
++
++enum {
++ thrd_success = 0,
++ thrd_busy = 1,
++ thrd_error = 2,
++ thrd_nomem = 3,
++ thrd_timedout = 4,
++};
++
++enum {
++ mtx_plain = 0,
++ mtx_recursive = 1,
++ mtx_timed = 2,
++};
++
++#define ONCE_FLAG_INIT 0
++
++int thrd_create(thrd_t *, thrd_start_t, void *);
++_Noreturn void thrd_exit(int);
++
++int thrd_detach(thrd_t);
++int thrd_join(thrd_t, int *);
++
++int thrd_sleep(const struct timespec *, struct timespec *);
++void thrd_yield(void);
++
++thrd_t thrd_current(void);
++int thrd_equal(thrd_t, thrd_t);
++#define thrd_equal(A, B) ((A) == (B))
++
++void call_once(once_flag *, void (*)(void));
++
++int mtx_init(mtx_t *, int);
++void mtx_destroy(mtx_t *);
++
++int mtx_lock(mtx_t *);
++int mtx_timedlock(mtx_t *__restrict, const struct timespec *__restrict);
++int mtx_trylock(mtx_t *);
++int mtx_unlock(mtx_t *);
++
++int cnd_init(cnd_t *);
++void cnd_destroy(cnd_t *);
++
++int cnd_broadcast(cnd_t *);
++int cnd_signal(cnd_t *);
++
++int cnd_timedwait(cnd_t *__restrict, mtx_t *__restrict, const struct timespec *__restrict);
++int cnd_wait(cnd_t *, mtx_t *);
++
++int tss_create(tss_t *, tss_dtor_t);
++void tss_delete(tss_t key);
++
++int tss_set(tss_t, void *);
++void *tss_get(tss_t);
++
++#ifdef __cplusplus
++}
++#endif
++
++#endif
+diff --git a/include/time.h b/include/time.h
+index dc88070..16ec08a 100644
+--- a/include/time.h
++++ b/include/time.h
+@@ -17,11 +17,11 @@ extern "C" {
+ #define __NEED_size_t
+ #define __NEED_time_t
+ #define __NEED_clock_t
++#define __NEED_struct_timespec
+
+ #if defined(_POSIX_SOURCE) || defined(_POSIX_C_SOURCE) \
+ || defined(_XOPEN_SOURCE) || defined(_GNU_SOURCE) \
+ || defined(_BSD_SOURCE)
+-#define __NEED_struct_timespec
+ #define __NEED_clockid_t
+ #define __NEED_timer_t
+ #define __NEED_pid_t
+@@ -59,9 +59,11 @@ struct tm *gmtime (const time_t *);
+ struct tm *localtime (const time_t *);
+ char *asctime (const struct tm *);
+ char *ctime (const time_t *);
++int timespec_get(struct timespec *, int);
+
+ #define CLOCKS_PER_SEC 1000000L
+
++#define TIME_UTC 1
+
+ #if defined(_POSIX_SOURCE) || defined(_POSIX_C_SOURCE) \
+ || defined(_XOPEN_SOURCE) || defined(_GNU_SOURCE) \
+@@ -114,7 +116,7 @@ int timer_getoverrun (timer_t);
+ #endif
+
+
+-#if defined(_XOPEN_SOURCE) || defined(_GNU_SOURCE)
++#if defined(_XOPEN_SOURCE) || defined(_BSD_SOURCE) || defined(_GNU_SOURCE)
+ char *strptime (const char *__restrict, const char *__restrict, struct tm *__restrict);
+ extern int daylight;
+ extern long timezone;
+diff --git a/include/unistd.h b/include/unistd.h
+index ac6055a..0fe75d5 100644
+--- a/include/unistd.h
++++ b/include/unistd.h
+@@ -140,9 +140,6 @@ int lockf(int, int, off_t);
+ long gethostid(void);
+ int nice(int);
+ void sync(void);
+-#endif
+-
+-#if defined(_XOPEN_SOURCE) || defined(_GNU_SOURCE)
+ pid_t setpgrp(void);
+ char *crypt(const char *, const char *);
+ void encrypt(char *, int);
+diff --git a/src/aio/aio_cancel.c b/src/aio/aio_cancel.c
+index 5a753b1..16fc431 100644
+--- a/src/aio/aio_cancel.c
++++ b/src/aio/aio_cancel.c
+@@ -1,6 +1,7 @@
+ #include <aio.h>
+ #include <pthread.h>
+ #include <errno.h>
++#include "libc.h"
+
+ int aio_cancel(int fd, struct aiocb *cb)
+ {
+@@ -14,3 +15,5 @@ int aio_cancel(int fd, struct aiocb *cb)
+ }
+ return cb->__err==EINPROGRESS ? AIO_NOTCANCELED : AIO_ALLDONE;
+ }
++
++LFS64(aio_cancel);
+diff --git a/src/aio/aio_error.c b/src/aio/aio_error.c
+index a780fd3..fd42ea1 100644
+--- a/src/aio/aio_error.c
++++ b/src/aio/aio_error.c
+@@ -1,6 +1,9 @@
+ #include <aio.h>
++#include "libc.h"
+
+ int aio_error(const struct aiocb *cb)
+ {
+ return cb->__err;
+ }
++
++LFS64(aio_error);
+diff --git a/src/aio/aio_fsync.c b/src/aio/aio_fsync.c
+index 0ac6ea8..6e1a70a 100644
+--- a/src/aio/aio_fsync.c
++++ b/src/aio/aio_fsync.c
+@@ -1,5 +1,6 @@
+ #include <aio.h>
+ #include <errno.h>
++#include "libc.h"
+
+ int aio_fsync(int op, struct aiocb *cb)
+ {
+@@ -7,3 +8,5 @@ int aio_fsync(int op, struct aiocb *cb)
+ errno = EINVAL;
+ return -1;
+ }
++
++LFS64(aio_fsync);
+diff --git a/src/aio/aio_readwrite.c b/src/aio/aio_readwrite.c
+index 2278226..8753ffd 100644
+--- a/src/aio/aio_readwrite.c
++++ b/src/aio/aio_readwrite.c
+@@ -105,3 +105,6 @@ int aio_write(struct aiocb *cb)
+ cb->aio_lio_opcode = LIO_WRITE;
+ return new_req(cb);
+ }
++
++LFS64(aio_read);
++LFS64(aio_write);
+diff --git a/src/aio/aio_return.c b/src/aio/aio_return.c
+index df10bdb..c1ce450 100644
+--- a/src/aio/aio_return.c
++++ b/src/aio/aio_return.c
+@@ -1,6 +1,9 @@
+ #include <aio.h>
++#include "libc.h"
+
+ ssize_t aio_return(struct aiocb *cb)
+ {
+ return cb->__ret;
+ }
++
++LFS64(aio_return);
+diff --git a/src/aio/aio_suspend.c b/src/aio/aio_suspend.c
+index 39a1d3a..dcdf601 100644
+--- a/src/aio/aio_suspend.c
++++ b/src/aio/aio_suspend.c
+@@ -1,6 +1,7 @@
+ #include <aio.h>
+ #include <errno.h>
+ #include "pthread_impl.h"
++#include "libc.h"
+
+ /* Due to the requirement that aio_suspend be async-signal-safe, we cannot
+ * use any locks, wait queues, etc. that would make it more efficient. The
+@@ -55,3 +56,5 @@ int aio_suspend(const struct aiocb *const cbs[], int cnt, const struct timespec
+ }
+ }
+ }
++
++LFS64(aio_suspend);
+diff --git a/src/aio/lio_listio.c b/src/aio/lio_listio.c
+index 75ed225..bd37767 100644
+--- a/src/aio/lio_listio.c
++++ b/src/aio/lio_listio.c
+@@ -141,3 +141,4 @@ int lio_listio(int mode, struct aiocb *restrict const *restrict cbs, int cnt, st
+ return 0;
+ }
+
++LFS64(lio_listio);
+diff --git a/src/conf/fpathconf.c b/src/conf/fpathconf.c
+index 28c4345..8eb037e 100644
+--- a/src/conf/fpathconf.c
++++ b/src/conf/fpathconf.c
+@@ -27,7 +27,7 @@ long fpathconf(int fd, int name)
+ [_PC_SYMLINK_MAX] = SYMLINK_MAX,
+ [_PC_2_SYMLINKS] = 1
+ };
+- if (name > sizeof(values)/sizeof(values[0])) {
++ if (name >= sizeof(values)/sizeof(values[0])) {
+ errno = EINVAL;
+ return -1;
+ }
+diff --git a/src/dirent/versionsort.c b/src/dirent/versionsort.c
+index 9769610..410cb70 100644
+--- a/src/dirent/versionsort.c
++++ b/src/dirent/versionsort.c
+@@ -1,8 +1,12 @@
+ #define _GNU_SOURCE
+ #include <string.h>
+ #include <dirent.h>
++#include "libc.h"
+
+ int versionsort(const struct dirent **a, const struct dirent **b)
+ {
+ return strverscmp((*a)->d_name, (*b)->d_name);
+ }
++
++#undef versionsort64
++LFS64(versionsort);
+diff --git a/src/internal/intscan.c b/src/internal/intscan.c
+index 69350ef..65d497e 100644
+--- a/src/internal/intscan.c
++++ b/src/internal/intscan.c
+@@ -83,6 +83,7 @@ unsigned long long __intscan(FILE *f, unsigned base, int pok, unsigned long long
+ for (; val[c]<base; c=shgetc(f));
+ errno = ERANGE;
+ y = lim;
++ if (lim&1) neg = 0;
+ }
+ done:
+ shunget(f);
+diff --git a/src/internal/pthread_impl.h b/src/internal/pthread_impl.h
+index 74c62cc..ae6e60b 100644
+--- a/src/internal/pthread_impl.h
++++ b/src/internal/pthread_impl.h
+@@ -128,4 +128,6 @@ void __restore_sigs(void *);
+ #define DEFAULT_STACK_SIZE 81920
+ #define DEFAULT_GUARD_SIZE PAGE_SIZE
+
++#define __ATTRP_C11_THREAD ((void*)(uintptr_t)-1)
++
+ #endif
+diff --git a/src/locale/__lctrans.c b/src/locale/__lctrans.c
+index 2769c08..15994c5 100644
+--- a/src/locale/__lctrans.c
++++ b/src/locale/__lctrans.c
+@@ -2,7 +2,7 @@
+ #include "locale_impl.h"
+ #include "libc.h"
+
+-const char *dummy(const char *msg, const struct __locale_map *lm)
++static const char *dummy(const char *msg, const struct __locale_map *lm)
+ {
+ return msg;
+ }
+diff --git a/src/math/__polevll.c b/src/math/__polevll.c
+index a272865..ce1a840 100644
+--- a/src/math/__polevll.c
++++ b/src/math/__polevll.c
+@@ -56,6 +56,8 @@
+
+ #include "libm.h"
+
++#if LDBL_MANT_DIG == 53 && LDBL_MAX_EXP == 1024
++#else
+ /*
+ * Polynomial evaluator:
+ * P[0] x^n + P[1] x^(n-1) + ... + P[n]
+@@ -88,3 +90,4 @@ long double __p1evll(long double x, const long double *P, int n)
+
+ return y;
+ }
++#endif
+diff --git a/src/math/exp10.c b/src/math/exp10.c
+index 16d704a..9f5e3c2 100644
+--- a/src/math/exp10.c
++++ b/src/math/exp10.c
+@@ -1,5 +1,6 @@
+ #define _GNU_SOURCE
+ #include <math.h>
++#include <stdint.h>
+ #include "libc.h"
+
+ double exp10(double x)
+@@ -11,7 +12,9 @@ double exp10(double x)
+ 1e10, 1e11, 1e12, 1e13, 1e14, 1e15
+ };
+ double n, y = modf(x, &n);
+- if (fabs(n) < 16) {
++ union {double f; uint64_t i;} u = {n};
++ /* fabs(n) < 16 without raising invalid on nan */
++ if ((u.i>>52 & 0x7ff) < 0x3ff+4) {
+ if (!y) return p10[(int)n+15];
+ y = exp2(3.32192809488736234787031942948939 * y);
+ return y * p10[(int)n+15];
+diff --git a/src/math/exp10f.c b/src/math/exp10f.c
+index 5fd1af9..7a8d447 100644
+--- a/src/math/exp10f.c
++++ b/src/math/exp10f.c
+@@ -1,5 +1,6 @@
+ #define _GNU_SOURCE
+ #include <math.h>
++#include <stdint.h>
+ #include "libc.h"
+
+ float exp10f(float x)
+@@ -9,7 +10,9 @@ float exp10f(float x)
+ 1, 1e1, 1e2, 1e3, 1e4, 1e5, 1e6, 1e7
+ };
+ float n, y = modff(x, &n);
+- if (fabsf(n) < 8) {
++ union {float f; uint32_t i;} u = {n};
++ /* fabsf(n) < 8 without raising invalid on nan */
++ if ((u.i>>23 & 0xff) < 0x7f+3) {
+ if (!y) return p10[(int)n+7];
+ y = exp2f(3.32192809488736234787031942948939f * y);
+ return y * p10[(int)n+7];
+diff --git a/src/math/exp10l.c b/src/math/exp10l.c
+index 22a4636..b758ebf 100644
+--- a/src/math/exp10l.c
++++ b/src/math/exp10l.c
+@@ -1,7 +1,15 @@
+ #define _GNU_SOURCE
++#include <float.h>
+ #include <math.h>
+ #include "libc.h"
++#include "libm.h"
+
++#if LDBL_MANT_DIG == 53 && LDBL_MAX_EXP == 1024
++long double exp10l(long double x)
++{
++ return exp10(x);
++}
++#elif (LDBL_MANT_DIG == 64 || LDBL_MANT_DIG == 113) && LDBL_MAX_EXP == 16384
+ long double exp10l(long double x)
+ {
+ static const long double p10[] = {
+@@ -11,12 +19,15 @@ long double exp10l(long double x)
+ 1e10, 1e11, 1e12, 1e13, 1e14, 1e15
+ };
+ long double n, y = modfl(x, &n);
+- if (fabsl(n) < 16) {
++ union ldshape u = {n};
++ /* fabsl(n) < 16 without raising invalid on nan */
++ if ((u.i.se & 0x7fff) < 0x3fff+4) {
+ if (!y) return p10[(int)n+15];
+ y = exp2l(3.32192809488736234787031942948939L * y);
+ return y * p10[(int)n+15];
+ }
+ return powl(10.0, x);
+ }
++#endif
+
+ weak_alias(exp10l, pow10l);
+diff --git a/src/mman/mprotect.c b/src/mman/mprotect.c
+index f488486..535787b 100644
+--- a/src/mman/mprotect.c
++++ b/src/mman/mprotect.c
+@@ -2,10 +2,12 @@
+ #include "libc.h"
+ #include "syscall.h"
+
+-int mprotect(void *addr, size_t len, int prot)
++int __mprotect(void *addr, size_t len, int prot)
+ {
+ size_t start, end;
+ start = (size_t)addr & -PAGE_SIZE;
+ end = (size_t)((char *)addr + len + PAGE_SIZE-1) & -PAGE_SIZE;
+ return syscall(SYS_mprotect, start, end-start, prot);
+ }
++
++weak_alias(__mprotect, mprotect);
+diff --git a/src/network/getnameinfo.c b/src/network/getnameinfo.c
+index 2ba66e3..3484fc6 100644
+--- a/src/network/getnameinfo.c
++++ b/src/network/getnameinfo.c
+@@ -96,7 +96,7 @@ static void reverse_services(char *buf, int port, int dgram)
+ if ((p=strchr(line, '#'))) *p++='\n', *p=0;
+
+ for (p=line; *p && !isspace(*p); p++);
+- if (!p) continue;
++ if (!*p) continue;
+ *p++ = 0;
+ svport = strtoul(p, &z, 10);
+
+diff --git a/src/network/lookup_serv.c b/src/network/lookup_serv.c
+index bf4cba0..a9be0f3 100644
+--- a/src/network/lookup_serv.c
++++ b/src/network/lookup_serv.c
+@@ -52,7 +52,6 @@ int __lookup_serv(struct service buf[static MAXSERVS], const char *name, int pro
+
+ /* Skip past canonical name at beginning of line */
+ for (p=line; *p && !isspace(*p); p++);
+- if (!p) continue;
+
+ port = strtoul(p, &z, 10);
+ if (port > 65535 || z==p) continue;
+diff --git a/src/regex/regcomp.c b/src/regex/regcomp.c
+index d907627..4cdaa1e 100644
+--- a/src/regex/regcomp.c
++++ b/src/regex/regcomp.c
+@@ -34,6 +34,7 @@
+ #include <regex.h>
+ #include <limits.h>
+ #include <stdint.h>
++#include <ctype.h>
+
+ #include "tre.h"
+
+@@ -135,108 +136,88 @@ typedef struct {
+ tre_ast_node_t *right;
+ } tre_union_t;
+
+-static tre_ast_node_t *
+-tre_ast_new_node(tre_mem_t mem, tre_ast_type_t type, size_t size);
+-
+-static tre_ast_node_t *
+-tre_ast_new_literal(tre_mem_t mem, int code_min, int code_max, int position);
+-
+-static tre_ast_node_t *
+-tre_ast_new_iter(tre_mem_t mem, tre_ast_node_t *arg, int min, int max,
+- int minimal);
+-
+-static tre_ast_node_t *
+-tre_ast_new_union(tre_mem_t mem, tre_ast_node_t *left, tre_ast_node_t *right);
+-
+-static tre_ast_node_t *
+-tre_ast_new_catenation(tre_mem_t mem, tre_ast_node_t *left,
+- tre_ast_node_t *right);
+-
+
+ static tre_ast_node_t *
+-tre_ast_new_node(tre_mem_t mem, tre_ast_type_t type, size_t size)
++tre_ast_new_node(tre_mem_t mem, int type, void *obj)
+ {
+- tre_ast_node_t *node;
+-
+- node = tre_mem_calloc(mem, sizeof(*node));
+- if (!node)
+- return NULL;
+- node->obj = tre_mem_calloc(mem, size);
+- if (!node->obj)
+- return NULL;
+- node->type = type;
+- node->nullable = -1;
+- node->submatch_id = -1;
+-
+- return node;
++ tre_ast_node_t *node = tre_mem_calloc(mem, sizeof *node);
++ if (!node || !obj)
++ return 0;
++ node->obj = obj;
++ node->type = type;
++ node->nullable = -1;
++ node->submatch_id = -1;
++ return node;
+ }
+
+ static tre_ast_node_t *
+ tre_ast_new_literal(tre_mem_t mem, int code_min, int code_max, int position)
+ {
+- tre_ast_node_t *node;
+- tre_literal_t *lit;
+-
+- node = tre_ast_new_node(mem, LITERAL, sizeof(tre_literal_t));
+- if (!node)
+- return NULL;
+- lit = node->obj;
+- lit->code_min = code_min;
+- lit->code_max = code_max;
+- lit->position = position;
+-
+- return node;
++ tre_ast_node_t *node;
++ tre_literal_t *lit;
++
++ lit = tre_mem_calloc(mem, sizeof *lit);
++ node = tre_ast_new_node(mem, LITERAL, lit);
++ if (!node)
++ return 0;
++ lit->code_min = code_min;
++ lit->code_max = code_max;
++ lit->position = position;
++ return node;
+ }
+
+ static tre_ast_node_t *
+-tre_ast_new_iter(tre_mem_t mem, tre_ast_node_t *arg, int min, int max,
+- int minimal)
++tre_ast_new_iter(tre_mem_t mem, tre_ast_node_t *arg, int min, int max, int minimal)
+ {
+- tre_ast_node_t *node;
+- tre_iteration_t *iter;
+-
+- node = tre_ast_new_node(mem, ITERATION, sizeof(tre_iteration_t));
+- if (!node)
+- return NULL;
+- iter = node->obj;
+- iter->arg = arg;
+- iter->min = min;
+- iter->max = max;
+- iter->minimal = minimal;
+- node->num_submatches = arg->num_submatches;
+-
+- return node;
++ tre_ast_node_t *node;
++ tre_iteration_t *iter;
++
++ iter = tre_mem_calloc(mem, sizeof *iter);
++ node = tre_ast_new_node(mem, ITERATION, iter);
++ if (!node)
++ return 0;
++ iter->arg = arg;
++ iter->min = min;
++ iter->max = max;
++ iter->minimal = minimal;
++ node->num_submatches = arg->num_submatches;
++ return node;
+ }
+
+ static tre_ast_node_t *
+ tre_ast_new_union(tre_mem_t mem, tre_ast_node_t *left, tre_ast_node_t *right)
+ {
+- tre_ast_node_t *node;
+-
+- node = tre_ast_new_node(mem, UNION, sizeof(tre_union_t));
+- if (node == NULL)
+- return NULL;
+- ((tre_union_t *)node->obj)->left = left;
+- ((tre_union_t *)node->obj)->right = right;
+- node->num_submatches = left->num_submatches + right->num_submatches;
+-
+- return node;
++ tre_ast_node_t *node;
++ tre_union_t *un;
++
++ if (!left)
++ return right;
++ un = tre_mem_calloc(mem, sizeof *un);
++ node = tre_ast_new_node(mem, UNION, un);
++ if (!node || !right)
++ return 0;
++ un->left = left;
++ un->right = right;
++ node->num_submatches = left->num_submatches + right->num_submatches;
++ return node;
+ }
+
+ static tre_ast_node_t *
+-tre_ast_new_catenation(tre_mem_t mem, tre_ast_node_t *left,
+- tre_ast_node_t *right)
++tre_ast_new_catenation(tre_mem_t mem, tre_ast_node_t *left, tre_ast_node_t *right)
+ {
+- tre_ast_node_t *node;
+-
+- node = tre_ast_new_node(mem, CATENATION, sizeof(tre_catenation_t));
+- if (node == NULL)
+- return NULL;
+- ((tre_catenation_t *)node->obj)->left = left;
+- ((tre_catenation_t *)node->obj)->right = right;
+- node->num_submatches = left->num_submatches + right->num_submatches;
+-
+- return node;
++ tre_ast_node_t *node;
++ tre_catenation_t *cat;
++
++ if (!left)
++ return right;
++ cat = tre_mem_calloc(mem, sizeof *cat);
++ node = tre_ast_new_node(mem, CATENATION, cat);
++ if (!node)
++ return 0;
++ cat->left = left;
++ cat->right = right;
++ node->num_submatches = left->num_submatches + right->num_submatches;
++ return node;
+ }
+
+
+@@ -412,1077 +393,654 @@ define_popf(voidptr, void *)
+
+ /* Parse context. */
+ typedef struct {
+- /* Memory allocator. The AST is allocated using this. */
+- tre_mem_t mem;
+- /* Stack used for keeping track of regexp syntax. */
+- tre_stack_t *stack;
+- /* The parse result. */
+- tre_ast_node_t *result;
+- /* The regexp to parse and its length. */
+- const char *re;
+- /* The first character of the entire regexp. */
+- const char *re_start;
+- /* Current submatch ID. */
+- int submatch_id;
+- /* Current position (number of literal). */
+- int position;
+- /* The highest back reference or -1 if none seen so far. */
+- int max_backref;
+- /* This flag is set if the regexp uses approximate matching. */
+- int have_approx;
+- /* Compilation flags. */
+- int cflags;
+- /* If this flag is set the top-level submatch is not captured. */
+- int nofirstsub;
++ /* Memory allocator. The AST is allocated using this. */
++ tre_mem_t mem;
++ /* Stack used for keeping track of regexp syntax. */
++ tre_stack_t *stack;
++ /* The parsed node after a parse function returns. */
++ tre_ast_node_t *n;
++ /* Position in the regexp pattern after a parse function returns. */
++ const char *s;
++ /* The first character of the regexp. */
++ const char *re;
++ /* Current submatch ID. */
++ int submatch_id;
++ /* Current position (number of literal). */
++ int position;
++ /* The highest back reference or -1 if none seen so far. */
++ int max_backref;
++ /* Compilation flags. */
++ int cflags;
+ } tre_parse_ctx_t;
+
+-/* Parses a wide character regexp pattern into a syntax tree. This parser
+- handles both syntaxes (BRE and ERE), including the TRE extensions. */
+-static reg_errcode_t
+-tre_parse(tre_parse_ctx_t *ctx);
+-
+-
+-/*
+- This parser is just a simple recursive descent parser for POSIX.2
+- regexps. The parser supports both the obsolete default syntax and
+- the "extended" syntax, and some nonstandard extensions.
+-*/
+-
+-/* Characters with special meanings in regexp syntax. */
+-#define CHAR_PIPE '|'
+-#define CHAR_LPAREN '('
+-#define CHAR_RPAREN ')'
+-#define CHAR_LBRACE '{'
+-#define CHAR_RBRACE '}'
+-#define CHAR_LBRACKET '['
+-#define CHAR_RBRACKET ']'
+-#define CHAR_MINUS '-'
+-#define CHAR_STAR '*'
+-#define CHAR_QUESTIONMARK '?'
+-#define CHAR_PLUS '+'
+-#define CHAR_PERIOD '.'
+-#define CHAR_COLON ':'
+-#define CHAR_EQUAL '='
+-#define CHAR_COMMA ','
+-#define CHAR_CARET '^'
+-#define CHAR_DOLLAR '$'
+-#define CHAR_BACKSLASH '\\'
+-#define CHAR_HASH '#'
+-#define CHAR_TILDE '~'
+-
+-
+ /* Some macros for expanding \w, \s, etc. */
+-static const struct tre_macro_struct {
+- const char c;
+- const char *expansion;
+-} tre_macros[] =
+- { {'t', "\t"}, {'n', "\n"}, {'r', "\r"},
+- {'f', "\f"}, {'a', "\a"}, {'e', "\033"},
+- {'w', "[[:alnum:]_]"}, {'W', "[^[:alnum:]_]"}, {'s', "[[:space:]]"},
+- {'S', "[^[:space:]]"}, {'d', "[[:digit:]]"}, {'D', "[^[:digit:]]"},
+- { 0, NULL }
+- };
+-
++static const struct {
++ char c;
++ const char *expansion;
++} tre_macros[] = {
++ {'t', "\t"}, {'n', "\n"}, {'r', "\r"},
++ {'f', "\f"}, {'a', "\a"}, {'e', "\033"},
++ {'w', "[[:alnum:]_]"}, {'W', "[^[:alnum:]_]"}, {'s', "[[:space:]]"},
++ {'S', "[^[:space:]]"}, {'d', "[[:digit:]]"}, {'D', "[^[:digit:]]"},
++ { 0, 0 }
++};
+
+ /* Expands a macro delimited by `regex' and `regex_end' to `buf', which
+ must have at least `len' items. Sets buf[0] to zero if the there
+ is no match in `tre_macros'. */
+-static const char *
+-tre_expand_macro(const char *regex)
++static const char *tre_expand_macro(const char *s)
+ {
+- int i;
+-
+- if (!*regex)
+- return 0;
+-
+- for (i = 0; tre_macros[i].expansion && tre_macros[i].c != *regex; i++);
+- return tre_macros[i].expansion;
++ int i;
++ for (i = 0; tre_macros[i].c && tre_macros[i].c != *s; i++);
++ return tre_macros[i].expansion;
+ }
+
+-static reg_errcode_t
+-tre_new_item(tre_mem_t mem, int min, int max, int *i, int *max_i,
+- tre_ast_node_t ***items)
++static int
++tre_compare_lit(const void *a, const void *b)
+ {
+- reg_errcode_t status;
+- tre_ast_node_t **array = *items;
+- /* Allocate more space if necessary. */
+- if (*i >= *max_i)
+- {
+- tre_ast_node_t **new_items;
+- /* If the array is already 1024 items large, give up -- there's
+- probably an error in the regexp (e.g. not a '\0' terminated
+- string and missing ']') */
+- if (*max_i > 1024)
+- return REG_ESPACE;
+- *max_i *= 2;
+- new_items = xrealloc(array, sizeof(*array) * *max_i);
+- if (new_items == NULL)
+- return REG_ESPACE;
+- *items = array = new_items;
+- }
+- array[*i] = tre_ast_new_literal(mem, min, max, -1);
+- status = array[*i] == NULL ? REG_ESPACE : REG_OK;
+- (*i)++;
+- return status;
++ const tre_literal_t *const *la = a;
++ const tre_literal_t *const *lb = b;
++ /* assumes the range of valid code_min is < INT_MAX */
++ return la[0]->code_min - lb[0]->code_min;
+ }
+
++struct literals {
++ tre_mem_t mem;
++ tre_literal_t **a;
++ int len;
++ int cap;
++};
+
+-static int
+-tre_compare_items(const void *a, const void *b)
++static tre_literal_t *tre_new_lit(struct literals *p)
+ {
+- const tre_ast_node_t *node_a = *(tre_ast_node_t * const *)a;
+- const tre_ast_node_t *node_b = *(tre_ast_node_t * const *)b;
+- tre_literal_t *l_a = node_a->obj, *l_b = node_b->obj;
+- int a_min = l_a->code_min, b_min = l_b->code_min;
+-
+- if (a_min < b_min)
+- return -1;
+- else if (a_min > b_min)
+- return 1;
+- else
+- return 0;
++ tre_literal_t **a;
++ if (p->len >= p->cap) {
++ if (p->cap >= 1<<15)
++ return 0;
++ p->cap *= 2;
++ a = xrealloc(p->a, p->cap * sizeof *p->a);
++ if (!a)
++ return 0;
++ p->a = a;
++ }
++ a = p->a + p->len++;
++ *a = tre_mem_calloc(p->mem, sizeof **a);
++ return *a;
+ }
+
+-/* Maximum number of character classes that can occur in a negated bracket
+- expression. */
+-#define MAX_NEG_CLASSES 64
+-
+-/* Maximum length of character class names. */
+-#define MAX_CLASS_NAME
+-
+-static reg_errcode_t
+-tre_parse_bracket_items(tre_parse_ctx_t *ctx, int negate,
+- tre_ctype_t neg_classes[], int *num_neg_classes,
+- tre_ast_node_t ***items, int *num_items,
+- int *items_size)
++static int add_icase_literals(struct literals *ls, int min, int max)
+ {
+- const char *re = ctx->re;
+- reg_errcode_t status = REG_OK;
+- tre_ctype_t class = (tre_ctype_t)0;
+- int i = *num_items;
+- int max_i = *items_size;
+- int skip;
+-
+- /* Build an array of the items in the bracket expression. */
+- while (status == REG_OK)
+- {
+- skip = 0;
+- if (!*re)
+- {
+- status = REG_EBRACK;
+- }
+- else if (*re == CHAR_RBRACKET && re > ctx->re)
+- {
+- re++;
+- break;
++ tre_literal_t *lit;
++ int b, e, c;
++ for (c=min; c<=max; ) {
++ /* assumes islower(c) and isupper(c) are exclusive
++ and toupper(c)!=c if islower(c).
++ multiple opposite case characters are not supported */
++ if (tre_islower(c)) {
++ b = e = tre_toupper(c);
++ for (c++, e++; c<=max; c++, e++)
++ if (tre_toupper(c) != e) break;
++ } else if (tre_isupper(c)) {
++ b = e = tre_tolower(c);
++ for (c++, e++; c<=max; c++, e++)
++ if (tre_tolower(c) != e) break;
++ } else {
++ c++;
++ continue;
++ }
++ lit = tre_new_lit(ls);
++ if (!lit)
++ return -1;
++ lit->code_min = b;
++ lit->code_max = e-1;
++ lit->position = -1;
+ }
+- else
+- {
+- tre_cint_t min = 0, max = 0;
+- wchar_t wc;
+- int clen = mbtowc(&wc, re, -1);
++ return 0;
++}
+
+- if (clen<0) clen=1, wc=WEOF;
+
+- class = (tre_ctype_t)0;
+- if (*(re + clen) == CHAR_MINUS && *(re + clen + 1) != CHAR_RBRACKET)
+- {
+- min = wc;
+- re += clen+1;
+- clen = mbtowc(&wc, re, -1);
+- if (clen<0) clen=1, wc=WEOF;
+- max = wc;
+- re += clen;
+- /* XXX - Should use collation order instead of encoding values
+- in character ranges. */
+- if (min > max)
+- status = REG_ERANGE;
+- }
+- else if (*re == CHAR_LBRACKET && *(re + 1) == CHAR_PERIOD)
+- status = REG_ECOLLATE;
+- else if (*re == CHAR_LBRACKET && *(re + 1) == CHAR_EQUAL)
+- status = REG_ECOLLATE;
+- else if (*re == CHAR_LBRACKET && *(re + 1) == CHAR_COLON)
+- {
+- char tmp_str[64];
+- const char *endptr = re + 2;
+- int len;
+- while (*endptr && *endptr != CHAR_COLON)
+- endptr++;
+- if (*endptr)
+- {
+- len = MIN(endptr - re - 2, 63);
+- strncpy(tmp_str, re + 2, len);
+- tmp_str[len] = '\0';
+- class = tre_ctype(tmp_str);
+- if (!class)
+- status = REG_ECTYPE;
+- re = endptr + 2;
+- }
+- else
+- status = REG_ECTYPE;
+- min = 0;
+- max = TRE_CHAR_MAX;
+- }
+- else
+- {
+- if (*re == CHAR_MINUS && *(re + 1) != CHAR_RBRACKET
+- && ctx->re != re)
+- /* Two ranges are not allowed to share and endpoint. */
+- status = REG_ERANGE;
+- min = max = wc;
+- re += clen;
+- }
++/* Maximum number of character classes in a negated bracket expression. */
++#define MAX_NEG_CLASSES 64
+
+- if (status != REG_OK)
+- break;
++struct neg {
++ int negate;
++ int len;
++ tre_ctype_t a[MAX_NEG_CLASSES];
++};
+
+- if (class && negate)
+- if (*num_neg_classes >= MAX_NEG_CLASSES)
+- status = REG_ESPACE;
+- else
+- neg_classes[(*num_neg_classes)++] = class;
+- else if (!skip)
+- {
+- status = tre_new_item(ctx->mem, min, max, &i, &max_i, items);
+- if (status != REG_OK)
+- break;
+- ((tre_literal_t*)((*items)[i-1])->obj)->class = class;
+- }
++// TODO: parse bracket into a set of non-overlapping [lo,hi] ranges
+
+- /* Add opposite-case counterpoints if REG_ICASE is present.
+- This is broken if there are more than two "same" characters. */
+- if (ctx->cflags & REG_ICASE && !class && status == REG_OK && !skip)
+- {
+- tre_cint_t cmin, ccurr;
++/*
++bracket grammar:
++Bracket = '[' List ']' | '[^' List ']'
++List = Term | List Term
++Term = Char | Range | Chclass | Eqclass
++Range = Char '-' Char | Char '-' '-'
++Char = Coll | coll_single
++Meta = ']' | '-'
++Coll = '[.' coll_single '.]' | '[.' coll_multi '.]' | '[.' Meta '.]'
++Eqclass = '[=' coll_single '=]' | '[=' coll_multi '=]'
++Chclass = '[:' class ':]'
++
++coll_single is a single char collating element but it can be
++ '-' only at the beginning or end of a List and
++ ']' only at the beginning of a List and
++ '^' anywhere except after the openning '['
++*/
+
+- while (min <= max)
+- {
+- if (tre_islower(min))
+- {
+- cmin = ccurr = tre_toupper(min++);
+- while (tre_islower(min) && tre_toupper(min) == ccurr + 1
+- && min <= max)
+- ccurr = tre_toupper(min++);
+- status = tre_new_item(ctx->mem, cmin, ccurr,
+- &i, &max_i, items);
+- }
+- else if (tre_isupper(min))
+- {
+- cmin = ccurr = tre_tolower(min++);
+- while (tre_isupper(min) && tre_tolower(min) == ccurr + 1
+- && min <= max)
+- ccurr = tre_tolower(min++);
+- status = tre_new_item(ctx->mem, cmin, ccurr,
+- &i, &max_i, items);
+- }
+- else min++;
+- if (status != REG_OK)
+- break;
++static reg_errcode_t parse_bracket_terms(tre_parse_ctx_t *ctx, const char *s, struct literals *ls, struct neg *neg)
++{
++ const char *start = s;
++ tre_ctype_t class;
++ int min, max;
++ wchar_t wc;
++ int len;
++
++ for (;;) {
++ class = 0;
++ len = mbtowc(&wc, s, -1);
++ if (len <= 0)
++ return *s ? REG_BADPAT : REG_EBRACK;
++ if (*s == ']' && s != start) {
++ ctx->s = s+1;
++ return REG_OK;
++ }
++ if (*s == '-' && s != start && s[1] != ']' &&
++ /* extension: [a-z--@] is accepted as [a-z]|[--@] */
++ (s[1] != '-' || s[2] == ']'))
++ return REG_ERANGE;
++ if (*s == '[' && (s[1] == '.' || s[1] == '='))
++ /* collating symbols and equivalence classes are not supported */
++ return REG_ECOLLATE;
++ if (*s == '[' && s[1] == ':') {
++ char tmp[CHARCLASS_NAME_MAX+1];
++ s += 2;
++ for (len=0; len < CHARCLASS_NAME_MAX && s[len]; len++) {
++ if (s[len] == ':') {
++ memcpy(tmp, s, len);
++ tmp[len] = 0;
++ class = tre_ctype(tmp);
++ break;
++ }
++ }
++ if (!class || s[len+1] != ']')
++ return REG_ECTYPE;
++ min = 0;
++ max = TRE_CHAR_MAX;
++ s += len+2;
++ } else {
++ min = max = wc;
++ s += len;
++ if (*s == '-' && s[1] != ']') {
++ s++;
++ len = mbtowc(&wc, s, -1);
++ max = wc;
++ /* XXX - Should use collation order instead of
++ encoding values in character ranges. */
++ if (len <= 0 || min > max)
++ return REG_ERANGE;
++ s += len;
++ }
++ }
++
++ if (class && neg->negate) {
++ if (neg->len >= MAX_NEG_CLASSES)
++ return REG_ESPACE;
++ neg->a[neg->len++] = class;
++ } else {
++ tre_literal_t *lit = tre_new_lit(ls);
++ if (!lit)
++ return REG_ESPACE;
++ lit->code_min = min;
++ lit->code_max = max;
++ lit->class = class;
++ lit->position = -1;
++
++ /* Add opposite-case codepoints if REG_ICASE is present.
++ It seems that POSIX requires that bracket negation
++ should happen before case-folding, but most practical
++ implementations do it the other way around. Changing
++ the order would need efficient representation of
++ case-fold ranges and bracket range sets even with
++ simple patterns so this is ok for now. */
++ if (ctx->cflags & REG_ICASE && !class)
++ if (add_icase_literals(ls, min, max))
++ return REG_ESPACE;
+ }
+- if (status != REG_OK)
+- break;
+- }
+ }
+- }
+- *num_items = i;
+- *items_size = max_i;
+- ctx->re = re;
+- return status;
+ }
+
+-static reg_errcode_t
+-tre_parse_bracket(tre_parse_ctx_t *ctx, tre_ast_node_t **result)
++static reg_errcode_t parse_bracket(tre_parse_ctx_t *ctx, const char *s)
+ {
+- tre_ast_node_t *node = NULL;
+- int negate = 0;
+- reg_errcode_t status = REG_OK;
+- tre_ast_node_t **items, *u, *n;
+- int i = 0, j, max_i = 32, curr_max, curr_min;
+- tre_ctype_t neg_classes[MAX_NEG_CLASSES];
+- int num_neg_classes = 0;
+-
+- /* Start off with an array of `max_i' elements. */
+- items = xmalloc(sizeof(*items) * max_i);
+- if (items == NULL)
+- return REG_ESPACE;
+-
+- if (*ctx->re == CHAR_CARET)
+- {
+- negate = 1;
+- ctx->re++;
+- }
+-
+- status = tre_parse_bracket_items(ctx, negate, neg_classes, &num_neg_classes,
+- &items, &i, &max_i);
+-
+- if (status != REG_OK)
+- goto parse_bracket_done;
+-
+- /* Sort the array if we need to negate it. */
+- if (negate)
+- qsort(items, (unsigned)i, sizeof(*items), tre_compare_items);
+-
+- curr_max = curr_min = 0;
+- /* Build a union of the items in the array, negated if necessary. */
+- for (j = 0; j < i && status == REG_OK; j++)
+- {
+- int min, max;
+- tre_literal_t *l = items[j]->obj;
+- min = l->code_min;
+- max = l->code_max;
+-
+- if (negate)
+- {
+- if (min < curr_max)
+- {
+- /* Overlap. */
+- curr_max = MAX(max + 1, curr_max);
+- l = NULL;
+- }
+- else
+- {
+- /* No overlap. */
+- curr_max = min - 1;
+- if (curr_max >= curr_min)
+- {
+- l->code_min = curr_min;
+- l->code_max = curr_max;
++ int i, max, min, negmax, negmin;
++ tre_ast_node_t *node = 0, *n;
++ tre_ctype_t *nc = 0;
++ tre_literal_t *lit;
++ struct literals ls;
++ struct neg neg;
++ reg_errcode_t err;
++
++ ls.mem = ctx->mem;
++ ls.len = 0;
++ ls.cap = 32;
++ ls.a = xmalloc(ls.cap * sizeof *ls.a);
++ if (!ls.a)
++ return REG_ESPACE;
++ neg.len = 0;
++ neg.negate = *s == '^';
++ if (neg.negate)
++ s++;
++
++ err = parse_bracket_terms(ctx, s, &ls, &neg);
++ if (err != REG_OK)
++ goto parse_bracket_done;
++
++ if (neg.negate) {
++ /* Sort the array if we need to negate it. */
++ qsort(ls.a, ls.len, sizeof *ls.a, tre_compare_lit);
++ /* extra lit for the last negated range */
++ lit = tre_new_lit(&ls);
++ if (!lit) {
++ err = REG_ESPACE;
++ goto parse_bracket_done;
+ }
+- else
+- {
+- l = NULL;
++ lit->code_min = TRE_CHAR_MAX+1;
++ lit->code_max = TRE_CHAR_MAX+1;
++ lit->position = -1;
++ /* negated classes */
++ if (neg.len) {
++ nc = tre_mem_alloc(ctx->mem, (neg.len+1)*sizeof *neg.a);
++ if (!nc) {
++ err = REG_ESPACE;
++ goto parse_bracket_done;
++ }
++ memcpy(nc, neg.a, neg.len*sizeof *neg.a);
++ nc[neg.len] = 0;
+ }
+- curr_min = curr_max = max + 1;
+- }
+ }
+
+- if (l != NULL)
+- {
+- int k;
+- l->position = ctx->position;
+- if (num_neg_classes > 0)
+- {
+- l->neg_classes = tre_mem_alloc(ctx->mem,
+- (sizeof(*l->neg_classes)
+- * (num_neg_classes + 1)));
+- if (l->neg_classes == NULL)
+- {
+- status = REG_ESPACE;
+- break;
++ /* Build a union of the items in the array, negated if necessary. */
++ negmax = negmin = 0;
++ for (i = 0; i < ls.len; i++) {
++ lit = ls.a[i];
++ min = lit->code_min;
++ max = lit->code_max;
++ if (neg.negate) {
++ if (min <= negmin) {
++ /* Overlap. */
++ negmin = MAX(max + 1, negmin);
++ continue;
++ }
++ negmax = min - 1;
++ lit->code_min = negmin;
++ lit->code_max = negmax;
++ negmin = max + 1;
+ }
+- for (k = 0; k < num_neg_classes; k++)
+- l->neg_classes[k] = neg_classes[k];
+- l->neg_classes[k] = (tre_ctype_t)0;
+- }
+- else
+- l->neg_classes = NULL;
+- if (node == NULL)
+- node = items[j];
+- else
+- {
+- u = tre_ast_new_union(ctx->mem, node, items[j]);
+- if (u == NULL)
+- status = REG_ESPACE;
+- node = u;
+- }
+- }
+- }
+-
+- if (status != REG_OK)
+- goto parse_bracket_done;
+-
+- if (negate)
+- {
+- int k;
+- n = tre_ast_new_literal(ctx->mem, curr_min, TRE_CHAR_MAX, ctx->position);
+- if (n == NULL)
+- status = REG_ESPACE;
+- else
+- {
+- tre_literal_t *l = n->obj;
+- if (num_neg_classes > 0)
+- {
+- l->neg_classes = tre_mem_alloc(ctx->mem,
+- (sizeof(*l->neg_classes)
+- * (num_neg_classes + 1)));
+- if (l->neg_classes == NULL)
+- {
+- status = REG_ESPACE;
+- goto parse_bracket_done;
++ lit->position = ctx->position;
++ lit->neg_classes = nc;
++ n = tre_ast_new_node(ctx->mem, LITERAL, lit);
++ node = tre_ast_new_union(ctx->mem, node, n);
++ if (!node) {
++ err = REG_ESPACE;
++ break;
+ }
+- for (k = 0; k < num_neg_classes; k++)
+- l->neg_classes[k] = neg_classes[k];
+- l->neg_classes[k] = (tre_ctype_t)0;
+- }
+- else
+- l->neg_classes = NULL;
+- if (node == NULL)
+- node = n;
+- else
+- {
+- u = tre_ast_new_union(ctx->mem, node, n);
+- if (u == NULL)
+- status = REG_ESPACE;
+- node = u;
+- }
+ }
+- }
+-
+- if (status != REG_OK)
+- goto parse_bracket_done;
+
+-#ifdef TRE_DEBUG
+- tre_ast_print(node);
+-#endif /* TRE_DEBUG */
+-
+- parse_bracket_done:
+- xfree(items);
+- ctx->position++;
+- *result = node;
+- return status;
++parse_bracket_done:
++ xfree(ls.a);
++ ctx->position++;
++ ctx->n = node;
++ return err;
+ }
+
+-
+-/* Parses a positive decimal integer. Returns -1 if the string does not
+- contain a valid number. */
+-static int
+-tre_parse_int(const char **regex)
++static const char *parse_dup_count(const char *s, int *n)
+ {
+- int num = -1;
+- const char *r = *regex;
+- while (*r-'0'<10U)
+- {
+- if (num < 0)
+- num = 0;
+- num = num * 10 + *r - '0';
+- r++;
+- }
+- *regex = r;
+- return num;
++ *n = -1;
++ if (!isdigit(*s))
++ return s;
++ *n = 0;
++ for (;;) {
++ *n = 10 * *n + (*s - '0');
++ s++;
++ if (!isdigit(*s) || *n > RE_DUP_MAX)
++ break;
++ }
++ return s;
+ }
+
+-
+-static reg_errcode_t
+-tre_parse_bound(tre_parse_ctx_t *ctx, tre_ast_node_t **result)
++static reg_errcode_t parse_dup(tre_parse_ctx_t *ctx, const char *s)
+ {
+- int min, max;
+- const char *r = ctx->re;
+- int minimal = 0;
+-
+- /* Parse number (minimum repetition count). */
+- min = -1;
+- if (*r >= '0' && *r <= '9') {
+- min = tre_parse_int(&r);
+- }
+-
+- /* Parse comma and second number (maximum repetition count). */
+- max = min;
+- if (*r == CHAR_COMMA)
+- {
+- r++;
+- max = tre_parse_int(&r);
+- }
+-
+- /* Check that the repeat counts are sane. */
+- if ((max >= 0 && min > max) || max > RE_DUP_MAX)
+- return REG_BADBR;
+-
+- /* Missing }. */
+- if (!*r)
+- return REG_EBRACE;
+-
+- /* Empty contents of {}. */
+- if (r == ctx->re)
+- return REG_BADBR;
+-
+- /* Parse the ending '}' or '\}'.*/
+- if (ctx->cflags & REG_EXTENDED)
+- {
+- if (*r != CHAR_RBRACE)
+- return REG_BADBR;
+- r++;
+- }
+- else
+- {
+- if (*r != CHAR_BACKSLASH || *(r + 1) != CHAR_RBRACE)
+- return REG_BADBR;
+- r += 2;
+- }
+-
+- /* Create the AST node(s). */
+- if (min == 0 && max == 0)
+- {
+- *result = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
+- if (*result == NULL)
+- return REG_ESPACE;
+- }
+- else
+- {
+- if (min < 0 && max < 0)
+- /* Only approximate parameters set, no repetitions. */
+- min = max = 1;
+-
+- *result = tre_ast_new_iter(ctx->mem, *result, min, max, minimal);
+- if (!*result)
+- return REG_ESPACE;
+- }
+-
+- ctx->re = r;
+- return REG_OK;
++ int min, max;
++
++ s = parse_dup_count(s, &min);
++ if (*s == ',')
++ s = parse_dup_count(s+1, &max);
++ else
++ max = min;
++
++ if (
++ (max < min && max >= 0) ||
++ max > RE_DUP_MAX ||
++ min > RE_DUP_MAX ||
++ min < 0 ||
++ (!(ctx->cflags & REG_EXTENDED) && *s++ != '\\') ||
++ *s++ != '}'
++ )
++ return REG_BADBR;
++
++ if (min == 0 && max == 0)
++ ctx->n = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
++ else
++ ctx->n = tre_ast_new_iter(ctx->mem, ctx->n, min, max, 0);
++ if (!ctx->n)
++ return REG_ESPACE;
++ ctx->s = s;
++ return REG_OK;
+ }
+
+-typedef enum {
+- PARSE_RE = 0,
+- PARSE_ATOM,
+- PARSE_MARK_FOR_SUBMATCH,
+- PARSE_BRANCH,
+- PARSE_PIECE,
+- PARSE_CATENATION,
+- PARSE_POST_CATENATION,
+- PARSE_UNION,
+- PARSE_POST_UNION,
+- PARSE_POSTFIX,
+- PARSE_RESTORE_CFLAGS
+-} tre_parse_re_stack_symbol_t;
+-
+-
+-static reg_errcode_t
+-tre_parse(tre_parse_ctx_t *ctx)
++static int hexval(unsigned c)
+ {
+- tre_ast_node_t *result = NULL;
+- tre_parse_re_stack_symbol_t symbol;
+- reg_errcode_t status = REG_OK;
+- tre_stack_t *stack = ctx->stack;
+- int bottom = tre_stack_num_objects(stack);
+- int depth = 0;
+- wchar_t wc;
+- int clen;
+-
+- if (!ctx->nofirstsub)
+- {
+- STACK_PUSH(stack, int, ctx->submatch_id);
+- STACK_PUSH(stack, int, PARSE_MARK_FOR_SUBMATCH);
+- ctx->submatch_id++;
+- }
+- STACK_PUSH(stack, int, PARSE_RE);
+- ctx->re_start = ctx->re;
+-
+-
+- /* The following is basically just a recursive descent parser. I use
+- an explicit stack instead of recursive functions mostly because of
+- two reasons: compatibility with systems which have an overflowable
+- call stack, and efficiency (both in lines of code and speed). */
+- while (tre_stack_num_objects(stack) > bottom && status == REG_OK)
+- {
+- if (status != REG_OK)
+- break;
+- symbol = tre_stack_pop_int(stack);
+- switch (symbol)
+- {
+- case PARSE_RE:
+- /* Parse a full regexp. A regexp is one or more branches,
+- separated by the union operator `|'. */
+- if (ctx->cflags & REG_EXTENDED)
+- STACK_PUSHX(stack, int, PARSE_UNION);
+- STACK_PUSHX(stack, int, PARSE_BRANCH);
+- break;
+-
+- case PARSE_BRANCH:
+- /* Parse a branch. A branch is one or more pieces, concatenated.
+- A piece is an atom possibly followed by a postfix operator. */
+- STACK_PUSHX(stack, int, PARSE_CATENATION);
+- STACK_PUSHX(stack, int, PARSE_PIECE);
+- break;
+-
+- case PARSE_PIECE:
+- /* Parse a piece. A piece is an atom possibly followed by one
+- or more postfix operators. */
+- STACK_PUSHX(stack, int, PARSE_POSTFIX);
+- STACK_PUSHX(stack, int, PARSE_ATOM);
+- break;
+-
+- case PARSE_CATENATION:
+- /* If the expression has not ended, parse another piece. */
+- {
+- tre_char_t c;
+- if (!*ctx->re)
+- break;
+- c = *ctx->re;
+- if (ctx->cflags & REG_EXTENDED && c == CHAR_PIPE)
+- break;
+- if ((ctx->cflags & REG_EXTENDED
+- && c == CHAR_RPAREN && depth > 0)
+- || (!(ctx->cflags & REG_EXTENDED)
+- && (c == CHAR_BACKSLASH
+- && *(ctx->re + 1) == CHAR_RPAREN)))
+- {
+- if (!(ctx->cflags & REG_EXTENDED) && depth == 0)
+- status = REG_EPAREN;
+- depth--;
+- if (!(ctx->cflags & REG_EXTENDED))
+- ctx->re += 2;
+- break;
+- }
+-
+- {
+- /* Default case, left associative concatenation. */
+- STACK_PUSHX(stack, int, PARSE_CATENATION);
+- STACK_PUSHX(stack, voidptr, result);
+- STACK_PUSHX(stack, int, PARSE_POST_CATENATION);
+- STACK_PUSHX(stack, int, PARSE_PIECE);
+- }
+- break;
+- }
+-
+- case PARSE_POST_CATENATION:
+- {
+- tre_ast_node_t *tree = tre_stack_pop_voidptr(stack);
+- tre_ast_node_t *tmp_node;
+- tmp_node = tre_ast_new_catenation(ctx->mem, tree, result);
+- if (!tmp_node)
+- return REG_ESPACE;
+- result = tmp_node;
+- break;
+- }
+-
+- case PARSE_UNION:
+- switch (*ctx->re)
+- {
+- case CHAR_PIPE:
+- STACK_PUSHX(stack, int, PARSE_UNION);
+- STACK_PUSHX(stack, voidptr, result);
+- STACK_PUSHX(stack, int, PARSE_POST_UNION);
+- STACK_PUSHX(stack, int, PARSE_BRANCH);
+- ctx->re++;
+- break;
+-
+- case CHAR_RPAREN:
+- ctx->re++;
+- break;
+-
+- default:
+- break;
+- }
+- break;
+-
+- case PARSE_POST_UNION:
+- {
+- tre_ast_node_t *tmp_node;
+- tre_ast_node_t *tree = tre_stack_pop_voidptr(stack);
+- tmp_node = tre_ast_new_union(ctx->mem, tree, result);
+- if (!tmp_node)
+- return REG_ESPACE;
+- result = tmp_node;
+- break;
+- }
+-
+- case PARSE_POSTFIX:
+- /* Parse postfix operators. */
+- switch (*ctx->re)
+- {
+- case CHAR_PLUS:
+- case CHAR_QUESTIONMARK:
+- if (!(ctx->cflags & REG_EXTENDED))
+- break;
+- /*FALLTHROUGH*/
+- case CHAR_STAR:
+- {
+- tre_ast_node_t *tmp_node;
+- int minimal = 0;
+- int rep_min = 0;
+- int rep_max = -1;
+-
+- if (*ctx->re == CHAR_PLUS)
+- rep_min = 1;
+- if (*ctx->re == CHAR_QUESTIONMARK)
+- rep_max = 1;
+-
+- ctx->re++;
+- tmp_node = tre_ast_new_iter(ctx->mem, result, rep_min, rep_max,
+- minimal);
+- if (tmp_node == NULL)
+- return REG_ESPACE;
+- result = tmp_node;
+- STACK_PUSHX(stack, int, PARSE_POSTFIX);
+- }
+- break;
+-
+- case CHAR_BACKSLASH:
+- /* "\{" is special without REG_EXTENDED */
+- if (!(ctx->cflags & REG_EXTENDED)
+- && *(ctx->re + 1) == CHAR_LBRACE)
+- {
+- ctx->re++;
+- goto parse_brace;
+- }
+- else
+- break;
+-
+- case CHAR_LBRACE:
+- /* "{" is literal without REG_EXTENDED */
+- if (!(ctx->cflags & REG_EXTENDED))
+- break;
+-
+- parse_brace:
+- ctx->re++;
++ if (c-'0'<10) return c-'0';
++ c |= 32;
++ if (c-'a'<6) return c-'a'+10;
++ return -1;
++}
+
+- status = tre_parse_bound(ctx, &result);
+- if (status != REG_OK)
+- return status;
+- STACK_PUSHX(stack, int, PARSE_POSTFIX);
+- break;
+- }
+- break;
++static reg_errcode_t marksub(tre_parse_ctx_t *ctx, tre_ast_node_t *node, int subid)
++{
++ if (node->submatch_id >= 0) {
++ tre_ast_node_t *n = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
++ if (!n)
++ return REG_ESPACE;
++ n = tre_ast_new_catenation(ctx->mem, n, node);
++ if (!n)
++ return REG_ESPACE;
++ n->num_submatches = node->num_submatches;
++ node = n;
++ }
++ node->submatch_id = subid;
++ node->num_submatches++;
++ ctx->n = node;
++ return REG_OK;
++}
+
+- case PARSE_ATOM:
+- /* Parse an atom. An atom is a regular expression enclosed in `()',
+- an empty set of `()', a bracket expression, `.', `^', `$',
+- a `\' followed by a character, or a single character. */
++/*
++BRE grammar:
++Regex = Branch | '^' | '$' | '^$' | '^' Branch | Branch '$' | '^' Branch '$'
++Branch = Atom | Branch Atom
++Atom = char | quoted_char | '.' | Bracket | Atom Dup | '\(' Branch '\)' | back_ref
++Dup = '*' | '\{' Count '\}' | '\{' Count ',\}' | '\{' Count ',' Count '\}'
+
+- switch (*ctx->re)
+- {
+- case CHAR_LPAREN: /* parenthesized subexpression */
++(leading ^ and trailing $ in a sub expr may be an anchor or literal as well)
+
+- if (ctx->cflags & REG_EXTENDED)
+- {
+- lparen:
+- depth++;
+- {
+- ctx->re++;
+- /* First parse a whole RE, then mark the resulting tree
+- for submatching. */
+- STACK_PUSHX(stack, int, ctx->submatch_id);
+- STACK_PUSHX(stack, int, PARSE_MARK_FOR_SUBMATCH);
+- STACK_PUSHX(stack, int, PARSE_RE);
+- ctx->submatch_id++;
+- }
+- }
+- else
+- goto parse_literal;
+- break;
++ERE grammar:
++Regex = Branch | Regex '|' Branch
++Branch = Atom | Branch Atom
++Atom = char | quoted_char | '.' | Bracket | Atom Dup | '(' Regex ')' | '^' | '$'
++Dup = '*' | '+' | '?' | '{' Count '}' | '{' Count ',}' | '{' Count ',' Count '}'
+
+- case CHAR_LBRACKET: /* bracket expression */
+- ctx->re++;
+- status = tre_parse_bracket(ctx, &result);
+- if (status != REG_OK)
+- return status;
+- break;
++(a*+?, ^*, $+, \X, {, (|a) are unspecified)
++*/
+
+- case CHAR_BACKSLASH:
+- /* If this is "\(" or "\)" chew off the backslash and
+- try again. */
+- if (!(ctx->cflags & REG_EXTENDED) && *(ctx->re + 1) == CHAR_LPAREN)
+- {
+- ctx->re++;
+- goto lparen;
+- }
+- if (!(ctx->cflags & REG_EXTENDED) && *(ctx->re + 1) == CHAR_RPAREN)
+- {
+- goto empty_atom;
++static reg_errcode_t parse_atom(tre_parse_ctx_t *ctx, const char *s)
++{
++ int len, ere = ctx->cflags & REG_EXTENDED;
++ const char *p;
++ tre_ast_node_t *node;
++ wchar_t wc;
++ switch (*s) {
++ case '[':
++ return parse_bracket(ctx, s+1);
++ case '\\':
++ p = tre_expand_macro(s+1);
++ if (p) {
++ /* assume \X expansion is a single atom */
++ reg_errcode_t err = parse_atom(ctx, p);
++ ctx->s = s+2;
++ return err;
+ }
+-
+- /* If a macro is used, parse the expanded macro recursively. */
+- {
+- const char *buf = tre_expand_macro(ctx->re + 1);
+- if (buf)
+- {
+- tre_parse_ctx_t subctx;
+- memcpy(&subctx, ctx, sizeof(subctx));
+- subctx.re = buf;
+- subctx.nofirstsub = 1;
+- status = tre_parse(&subctx);
+- if (status != REG_OK)
+- return status;
+- ctx->re += 2;
+- ctx->position = subctx.position;
+- result = subctx.result;
+- break;
+- }
+- }
+-
+- if (!ctx->re[1])
+- /* Trailing backslash. */
+- return REG_EESCAPE;
+-
+- ctx->re++;
+- switch (*ctx->re)
+- {
++ /* extensions: \b, \B, \<, \>, \xHH \x{HHHH} */
++ switch (*++s) {
++ case 0:
++ return REG_EESCAPE;
+ case 'b':
+- result = tre_ast_new_literal(ctx->mem, ASSERTION,
+- ASSERT_AT_WB, -1);
+- ctx->re++;
+- break;
++ node = tre_ast_new_literal(ctx->mem, ASSERTION, ASSERT_AT_WB, -1);
++ break;
+ case 'B':
+- result = tre_ast_new_literal(ctx->mem, ASSERTION,
+- ASSERT_AT_WB_NEG, -1);
+- ctx->re++;
+- break;
++ node = tre_ast_new_literal(ctx->mem, ASSERTION, ASSERT_AT_WB_NEG, -1);
++ break;
+ case '<':
+- result = tre_ast_new_literal(ctx->mem, ASSERTION,
+- ASSERT_AT_BOW, -1);
+- ctx->re++;
+- break;
++ node = tre_ast_new_literal(ctx->mem, ASSERTION, ASSERT_AT_BOW, -1);
++ break;
+ case '>':
+- result = tre_ast_new_literal(ctx->mem, ASSERTION,
+- ASSERT_AT_EOW, -1);
+- ctx->re++;
+- break;
++ node = tre_ast_new_literal(ctx->mem, ASSERTION, ASSERT_AT_EOW, -1);
++ break;
+ case 'x':
+- ctx->re++;
+- if (ctx->re[0] != CHAR_LBRACE)
+- {
+- /* 8 bit hex char. */
+- char tmp[3] = {0, 0, 0};
+- long val;
+-
+- if (tre_isxdigit(ctx->re[0]))
+- {
+- tmp[0] = (char)ctx->re[0];
+- ctx->re++;
++ s++;
++ int i, v = 0, c;
++ len = 2;
++ if (*s == '{') {
++ len = 8;
++ s++;
+ }
+- if (tre_isxdigit(ctx->re[0]))
+- {
+- tmp[1] = (char)ctx->re[0];
+- ctx->re++;
++ for (i=0; i<len && v<0x110000; i++) {
++ c = hexval(s[i]);
++ if (c < 0) break;
++ v = 16*v + c;
+ }
+- val = strtol(tmp, NULL, 16);
+- result = tre_ast_new_literal(ctx->mem, (int)val,
+- (int)val, ctx->position);
+- ctx->position++;
+- break;
+- }
+- else if (*ctx->re)
+- {
+- /* Wide char. */
+- char tmp[32];
+- long val;
+- int i = 0;
+- ctx->re++;
+- while (*ctx->re && i < sizeof tmp)
+- {
+- if (ctx->re[0] == CHAR_RBRACE)
+- break;
+- if (tre_isxdigit(ctx->re[0]))
+- {
+- tmp[i] = (char)ctx->re[0];
+- i++;
+- ctx->re++;
+- continue;
+- }
+- return REG_EBRACE;
++ s += i;
++ if (len == 8) {
++ if (*s != '}')
++ return REG_EBRACE;
++ s++;
+ }
+- ctx->re++;
+- tmp[i] = 0;
+- val = strtol(tmp, NULL, 16);
+- result = tre_ast_new_literal(ctx->mem, (int)val, (int)val,
+- ctx->position);
+- ctx->position++;
+- break;
+- }
+- /*FALLTHROUGH*/
+-
++ node = tre_ast_new_literal(ctx->mem, v, v, ctx->position);
++ ctx->position++;
++ s--;
++ break;
+ default:
+- if (tre_isdigit(*ctx->re))
+- {
+- /* Back reference. */
+- int val = *ctx->re - '0';
+- result = tre_ast_new_literal(ctx->mem, BACKREF, val,
+- ctx->position);
+- if (result == NULL)
+- return REG_ESPACE;
+- ctx->position++;
+- ctx->max_backref = MAX(val, ctx->max_backref);
+- ctx->re++;
+- }
+- else
+- {
+- /* Escaped character. */
+- result = tre_ast_new_literal(ctx->mem, *ctx->re, *ctx->re,
+- ctx->position);
+- ctx->position++;
+- ctx->re++;
+- }
+- break;
+- }
+- if (result == NULL)
+- return REG_ESPACE;
+- break;
+-
+- case CHAR_PERIOD: /* the any-symbol */
+- if (ctx->cflags & REG_NEWLINE)
+- {
+- tre_ast_node_t *tmp1;
+- tre_ast_node_t *tmp2;
+- tmp1 = tre_ast_new_literal(ctx->mem, 0, '\n' - 1,
+- ctx->position);
+- if (!tmp1)
+- return REG_ESPACE;
+- tmp2 = tre_ast_new_literal(ctx->mem, '\n' + 1, TRE_CHAR_MAX,
+- ctx->position + 1);
+- if (!tmp2)
+- return REG_ESPACE;
+- result = tre_ast_new_union(ctx->mem, tmp1, tmp2);
+- if (!result)
+- return REG_ESPACE;
+- ctx->position += 2;
+- }
+- else
+- {
+- result = tre_ast_new_literal(ctx->mem, 0, TRE_CHAR_MAX,
+- ctx->position);
+- if (!result)
+- return REG_ESPACE;
+- ctx->position++;
++ if (isdigit(*s)) {
++ /* back reference */
++ int val = *s - '0';
++ node = tre_ast_new_literal(ctx->mem, BACKREF, val, ctx->position);
++ ctx->max_backref = MAX(val, ctx->max_backref);
++ } else {
++ /* extension: accept unknown escaped char
++ as a literal */
++ node = tre_ast_new_literal(ctx->mem, *s, *s, ctx->position);
++ }
++ ctx->position++;
+ }
+- ctx->re++;
+- break;
+-
+- case CHAR_CARET: /* beginning of line assertion */
+- /* '^' has a special meaning everywhere in EREs, and at
+- beginning of BRE. */
+- if (ctx->cflags & REG_EXTENDED
+- || ctx->re == ctx->re_start)
+- {
+- if (!(ctx->cflags & REG_EXTENDED))
+- STACK_PUSHX(stack, int, PARSE_CATENATION);
+- result = tre_ast_new_literal(ctx->mem, ASSERTION,
+- ASSERT_AT_BOL, -1);
+- if (result == NULL)
+- return REG_ESPACE;
+- ctx->re++;
++ s++;
++ break;
++ case '.':
++ if (ctx->cflags & REG_NEWLINE) {
++ tre_ast_node_t *tmp1, *tmp2;
++ tmp1 = tre_ast_new_literal(ctx->mem, 0, '\n'-1, ctx->position++);
++ tmp2 = tre_ast_new_literal(ctx->mem, '\n'+1, TRE_CHAR_MAX, ctx->position++);
++ if (tmp1 && tmp2)
++ node = tre_ast_new_union(ctx->mem, tmp1, tmp2);
++ else
++ node = 0;
++ } else {
++ node = tre_ast_new_literal(ctx->mem, 0, TRE_CHAR_MAX, ctx->position++);
+ }
+- else
+- goto parse_literal;
+- break;
+-
+- case CHAR_DOLLAR: /* end of line assertion. */
+- /* '$' is special everywhere in EREs, and in the end of the
+- string in BREs. */
+- if (ctx->cflags & REG_EXTENDED
+- || !*(ctx->re + 1))
+- {
+- result = tre_ast_new_literal(ctx->mem, ASSERTION,
+- ASSERT_AT_EOL, -1);
+- if (result == NULL)
+- return REG_ESPACE;
+- ctx->re++;
++ s++;
++ break;
++ case '^':
++ /* '^' has a special meaning everywhere in EREs, and at beginning of BRE. */
++ if (!ere && s != ctx->re)
++ goto parse_literal;
++ node = tre_ast_new_literal(ctx->mem, ASSERTION, ASSERT_AT_BOL, -1);
++ s++;
++ break;
++ case '$':
++ /* '$' is special everywhere in EREs, and in the end of the string in BREs. */
++ if (!ere && s[1])
++ goto parse_literal;
++ node = tre_ast_new_literal(ctx->mem, ASSERTION, ASSERT_AT_EOL, -1);
++ s++;
++ break;
++ case '*':
++ case '|':
++ case '{':
++ case '+':
++ case '?':
++ if (!ere)
++ goto parse_literal;
++ case 0:
++ node = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
++ break;
++ default:
++parse_literal:
++ len = mbtowc(&wc, s, -1);
++ if (len < 0)
++ return REG_BADPAT;
++ if (ctx->cflags & REG_ICASE && (tre_isupper(wc) || tre_islower(wc))) {
++ tre_ast_node_t *tmp1, *tmp2;
++ /* multiple opposite case characters are not supported */
++ tmp1 = tre_ast_new_literal(ctx->mem, tre_toupper(wc), tre_toupper(wc), ctx->position);
++ tmp2 = tre_ast_new_literal(ctx->mem, tre_tolower(wc), tre_tolower(wc), ctx->position);
++ if (tmp1 && tmp2)
++ node = tre_ast_new_union(ctx->mem, tmp1, tmp2);
++ else
++ node = 0;
++ } else {
++ node = tre_ast_new_literal(ctx->mem, wc, wc, ctx->position);
+ }
+- else
+- goto parse_literal;
+- break;
+-
+- case CHAR_RPAREN:
+- if (!depth)
+- goto parse_literal;
+- case CHAR_STAR:
+- case CHAR_PIPE:
+- case CHAR_LBRACE:
+- case CHAR_PLUS:
+- case CHAR_QUESTIONMARK:
+- if (!(ctx->cflags & REG_EXTENDED))
+- goto parse_literal;
+-
+- case 0:
+- empty_atom:
+- result = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
+- if (!result)
++ ctx->position++;
++ s += len;
++ break;
++ }
++ if (!node)
+ return REG_ESPACE;
+- break;
++ ctx->n = node;
++ ctx->s = s;
++ return REG_OK;
++}
+
+- default:
+- parse_literal:
++#define PUSHPTR(err, s, v) do { \
++ if ((err = tre_stack_push_voidptr(s, v)) != REG_OK) \
++ return err; \
++} while(0)
+
+- clen = mbtowc(&wc, ctx->re, -1);
+- if (clen<0) clen=1, wc=WEOF;
++#define PUSHINT(err, s, v) do { \
++ if ((err = tre_stack_push_int(s, v)) != REG_OK) \
++ return err; \
++} while(0)
+
+- /* Note that we can't use an tre_isalpha() test here, since there
+- may be characters which are alphabetic but neither upper or
+- lower case. */
+- if (ctx->cflags & REG_ICASE
+- && (tre_isupper(wc) || tre_islower(wc)))
+- {
+- tre_ast_node_t *tmp1;
+- tre_ast_node_t *tmp2;
+-
+- /* XXX - Can there be more than one opposite-case
+- counterpoints for some character in some locale? Or
+- more than two characters which all should be regarded
+- the same character if case is ignored? If yes, there
+- does not seem to be a portable way to detect it. I guess
+- that at least for multi-character collating elements there
+- could be several opposite-case counterpoints, but they
+- cannot be supported portably anyway. */
+- tmp1 = tre_ast_new_literal(ctx->mem, tre_toupper(wc),
+- tre_toupper(wc),
+- ctx->position);
+- if (!tmp1)
+- return REG_ESPACE;
+- tmp2 = tre_ast_new_literal(ctx->mem, tre_tolower(wc),
+- tre_tolower(wc),
+- ctx->position);
+- if (!tmp2)
+- return REG_ESPACE;
+- result = tre_ast_new_union(ctx->mem, tmp1, tmp2);
+- if (!result)
+- return REG_ESPACE;
++static reg_errcode_t tre_parse(tre_parse_ctx_t *ctx)
++{
++ tre_ast_node_t *nbranch=0, *nunion=0;
++ int ere = ctx->cflags & REG_EXTENDED;
++ const char *s = ctx->re;
++ int subid = 0;
++ int depth = 0;
++ reg_errcode_t err;
++ tre_stack_t *stack = ctx->stack;
++
++ PUSHINT(err, stack, subid++);
++ for (;;) {
++ if ((!ere && *s == '\\' && s[1] == '(') ||
++ (ere && *s == '(')) {
++ PUSHPTR(err, stack, nunion);
++ PUSHPTR(err, stack, nbranch);
++ PUSHINT(err, stack, subid++);
++ s++;
++ if (!ere)
++ s++;
++ depth++;
++ nbranch = nunion = 0;
++ continue;
+ }
+- else
+- {
+- result = tre_ast_new_literal(ctx->mem, wc, wc,
+- ctx->position);
+- if (!result)
+- return REG_ESPACE;
++ if ((!ere && *s == '\\' && s[1] == ')') ||
++ (ere && *s == ')' && depth)) {
++ ctx->n = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
++ if (!ctx->n)
++ return REG_ESPACE;
++ } else {
++ err = parse_atom(ctx, s);
++ if (err != REG_OK)
++ return err;
++ s = ctx->s;
+ }
+- ctx->position++;
+- ctx->re += clen;
+- break;
+- }
+- break;
+-
+- case PARSE_MARK_FOR_SUBMATCH:
+- {
+- int submatch_id = tre_stack_pop_int(stack);
+
+- if (result->submatch_id >= 0)
+- {
+- tre_ast_node_t *n, *tmp_node;
+- n = tre_ast_new_literal(ctx->mem, EMPTY, -1, -1);
+- if (n == NULL)
+- return REG_ESPACE;
+- tmp_node = tre_ast_new_catenation(ctx->mem, n, result);
+- if (tmp_node == NULL)
+- return REG_ESPACE;
+- tmp_node->num_submatches = result->num_submatches;
+- result = tmp_node;
+- }
+- result->submatch_id = submatch_id;
+- result->num_submatches++;
+- break;
+- }
+-
+- case PARSE_RESTORE_CFLAGS:
+- ctx->cflags = tre_stack_pop_int(stack);
+- break;
++ parse_iter:
++ /* extension: repetitions are accepted after an empty node
++ eg. (+), ^*, a$?, a|{2} */
++ switch (*s) {
++ case '+':
++ case '?':
++ if (!ere)
++ break;
++ /* fallthrough */
++ case '*':;
++ int min=0, max=-1;
++ if (*s == '+')
++ min = 1;
++ if (*s == '?')
++ max = 1;
++ s++;
++ ctx->n = tre_ast_new_iter(ctx->mem, ctx->n, min, max, 0);
++ if (!ctx->n)
++ return REG_ESPACE;
++ /* extension: multiple consecutive *+?{,} is unspecified,
++ but (a+)+ has to be supported so accepting a++ makes
++ sense, note however that the RE_DUP_MAX limit can be
++ circumvented: (a{255}){255} uses a lot of memory.. */
++ goto parse_iter;
++ case '\\':
++ if (ere || s[1] != '{')
++ break;
++ s++;
++ goto parse_brace;
++ case '{':
++ if (!ere)
++ break;
++ parse_brace:
++ err = parse_dup(ctx, s+1);
++ if (err != REG_OK)
++ return err;
++ s = ctx->s;
++ goto parse_iter;
++ }
+
+- default:
+- assert(0);
+- break;
++ nbranch = tre_ast_new_catenation(ctx->mem, nbranch, ctx->n);
++ if ((ere && *s == '|') ||
++ (ere && *s == ')' && depth) ||
++ (!ere && *s == '\\' && s[1] == ')') ||
++ !*s) {
++ /* extension: empty branch is unspecified (), (|a), (a|)
++ here they are not rejected but match on empty string */
++ int c = *s;
++ nunion = tre_ast_new_union(ctx->mem, nunion, nbranch);
++ nbranch = 0;
++ if (c != '|') {
++ if (c == '\\') {
++ if (!depth) return REG_EPAREN;
++ s+=2;
++ } else if (c == ')')
++ s++;
++ depth--;
++ err = marksub(ctx, nunion, tre_stack_pop_int(stack));
++ if (err != REG_OK)
++ return err;
++ if (!c && depth<0) {
++ ctx->submatch_id = subid;
++ return REG_OK;
++ }
++ if (!c || depth<0)
++ return REG_EPAREN;
++ nbranch = tre_stack_pop_voidptr(stack);
++ nunion = tre_stack_pop_voidptr(stack);
++ goto parse_iter;
++ }
++ s++;
++ }
+ }
+- }
+-
+- /* Check for missing closing parentheses. */
+- if (depth > 0)
+- return REG_EPAREN;
+-
+- if (status == REG_OK)
+- ctx->result = result;
+-
+- return status;
+ }
+
+
+-
+ /***********************************************************************
+ from tre-compile.c
+ ***********************************************************************/
+@@ -3122,12 +2680,7 @@ regcomp(regex_t *restrict preg, const char *restrict regex, int cflags)
+ if (errcode != REG_OK)
+ ERROR_EXIT(errcode);
+ preg->re_nsub = parse_ctx.submatch_id - 1;
+- tree = parse_ctx.result;
+-
+- /* Back references and approximate matching cannot currently be used
+- in the same regexp. */
+- if (parse_ctx.max_backref >= 0 && parse_ctx.have_approx)
+- ERROR_EXIT(REG_BADPAT);
++ tree = parse_ctx.n;
+
+ #ifdef TRE_DEBUG
+ tre_ast_print(tree);
+@@ -3142,7 +2695,7 @@ regcomp(regex_t *restrict preg, const char *restrict regex, int cflags)
+ if (tnfa == NULL)
+ ERROR_EXIT(REG_ESPACE);
+ tnfa->have_backrefs = parse_ctx.max_backref >= 0;
+- tnfa->have_approx = parse_ctx.have_approx;
++ tnfa->have_approx = 0;
+ tnfa->num_submatches = parse_ctx.submatch_id;
+
+ /* Set up tags for submatch addressing. If REG_NOSUB is set and the
+diff --git a/src/regex/regexec.c b/src/regex/regexec.c
+index 2e35b83..16c5d0a 100644
+--- a/src/regex/regexec.c
++++ b/src/regex/regexec.c
+@@ -52,7 +52,7 @@ tre_fill_pmatch(size_t nmatch, regmatch_t pmatch[], int cflags,
+ #define GET_NEXT_WCHAR() do { \
+ prev_c = next_c; pos += pos_add_next; \
+ if ((pos_add_next = mbtowc(&next_c, str_byte, MB_LEN_MAX)) <= 0) { \
+- if (pos_add_next < 0) return REG_NOMATCH; \
++ if (pos_add_next < 0) { ret = REG_NOMATCH; goto error_exit; } \
+ else pos_add_next++; \
+ } \
+ str_byte += pos_add_next; \
+@@ -181,6 +181,7 @@ tre_tnfa_run_parallel(const tre_tnfa_t *tnfa, const void *string,
+ int reg_notbol = eflags & REG_NOTBOL;
+ int reg_noteol = eflags & REG_NOTEOL;
+ int reg_newline = tnfa->cflags & REG_NEWLINE;
++ reg_errcode_t ret;
+
+ char *buf;
+ tre_tnfa_transition_t *trans_i;
+@@ -439,11 +440,11 @@ tre_tnfa_run_parallel(const tre_tnfa_t *tnfa, const void *string,
+ reach_next_i->state = NULL;
+ }
+
+- if (buf)
+- xfree(buf);
+-
+ *match_end_ofs = match_eo;
+- return match_eo >= 0 ? REG_OK : REG_NOMATCH;
++ ret = match_eo >= 0 ? REG_OK : REG_NOMATCH;
++error_exit:
++ xfree(buf);
++ return ret;
+ }
+
+
+diff --git a/src/stdio/ftrylockfile.c b/src/stdio/ftrylockfile.c
+index 6f9a4b8..eb13c83 100644
+--- a/src/stdio/ftrylockfile.c
++++ b/src/stdio/ftrylockfile.c
+@@ -34,6 +34,7 @@ int ftrylockfile(FILE *f)
+ f->lockcount = 1;
+ f->prev_locked = 0;
+ f->next_locked = self->stdio_locks;
++ if (f->next_locked) f->next_locked->prev_locked = f;
+ self->stdio_locks = f;
+ return 0;
+ }
+diff --git a/src/thread/__timedwait.c b/src/thread/__timedwait.c
+index d6f1233..c9ec70c 100644
+--- a/src/thread/__timedwait.c
++++ b/src/thread/__timedwait.c
+@@ -4,6 +4,9 @@
+ #include "futex.h"
+ #include "syscall.h"
+
++int __pthread_setcancelstate(int, int *);
++int __clock_gettime(clockid_t, struct timespec *);
++
+ int __timedwait(volatile int *addr, int val,
+ clockid_t clk, const struct timespec *at,
+ void (*cleanup)(void *), void *arg, int priv)
+@@ -15,7 +18,7 @@ int __timedwait(volatile int *addr, int val,
+
+ if (at) {
+ if (at->tv_nsec >= 1000000000UL) return EINVAL;
+- if (clock_gettime(clk, &to)) return EINVAL;
++ if (__clock_gettime(clk, &to)) return EINVAL;
+ to.tv_sec = at->tv_sec - to.tv_sec;
+ if ((to.tv_nsec = at->tv_nsec - to.tv_nsec) < 0) {
+ to.tv_sec--;
+@@ -25,7 +28,7 @@ int __timedwait(volatile int *addr, int val,
+ top = &to;
+ }
+
+- if (!cleanup) pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);
++ if (!cleanup) __pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);
+ pthread_cleanup_push(cleanup, arg);
+
+ r = -__syscall_cp(SYS_futex, addr, FUTEX_WAIT|priv, val, top);
+@@ -33,7 +36,7 @@ int __timedwait(volatile int *addr, int val,
+ if (r != EINTR && r != ETIMEDOUT) r = 0;
+
+ pthread_cleanup_pop(0);
+- if (!cleanup) pthread_setcancelstate(cs, 0);
++ if (!cleanup) __pthread_setcancelstate(cs, 0);
+
+ return r;
+ }
+diff --git a/src/thread/call_once.c b/src/thread/call_once.c
+new file mode 100644
+index 0000000..a7bc935
+--- /dev/null
++++ b/src/thread/call_once.c
+@@ -0,0 +1,8 @@
++#include <threads.h>
++
++int __pthread_once(once_flag *, void (*)(void));
++
++void call_once(once_flag *flag, void (*func)(void))
++{
++ __pthread_once(flag, func);
++}
+diff --git a/src/thread/cnd_broadcast.c b/src/thread/cnd_broadcast.c
+new file mode 100644
+index 0000000..85d4d3e
+--- /dev/null
++++ b/src/thread/cnd_broadcast.c
+@@ -0,0 +1,10 @@
++#include <threads.h>
++
++int __private_cond_signal(cnd_t *, int);
++
++int cnd_broadcast(cnd_t *c)
++{
++ /* This internal function never fails, and always returns zero,
++ * which matches the value thrd_success is defined with. */
++ return __private_cond_signal(c, -1);
++}
+diff --git a/src/thread/cnd_destroy.c b/src/thread/cnd_destroy.c
+new file mode 100644
+index 0000000..453c90b
+--- /dev/null
++++ b/src/thread/cnd_destroy.c
+@@ -0,0 +1,6 @@
++#include <threads.h>
++
++void cnd_destroy(cnd_t *c)
++{
++ /* For private cv this is a no-op */
++}
+diff --git a/src/thread/cnd_init.c b/src/thread/cnd_init.c
+new file mode 100644
+index 0000000..18c5085
+--- /dev/null
++++ b/src/thread/cnd_init.c
+@@ -0,0 +1,7 @@
++#include <threads.h>
++
++int cnd_init(cnd_t *c)
++{
++ *c = (cnd_t){ 0 };
++ return thrd_success;
++}
+diff --git a/src/thread/cnd_signal.c b/src/thread/cnd_signal.c
+new file mode 100644
+index 0000000..1211260
+--- /dev/null
++++ b/src/thread/cnd_signal.c
+@@ -0,0 +1,10 @@
++#include <threads.h>
++
++int __private_cond_signal(cnd_t *, int);
++
++int cnd_signal(cnd_t *c)
++{
++ /* This internal function never fails, and always returns zero,
++ * which matches the value thrd_success is defined with. */
++ return __private_cond_signal(c, 1);
++}
+diff --git a/src/thread/cnd_timedwait.c b/src/thread/cnd_timedwait.c
+new file mode 100644
+index 0000000..5997679
+--- /dev/null
++++ b/src/thread/cnd_timedwait.c
+@@ -0,0 +1,15 @@
++#include <threads.h>
++#include <errno.h>
++
++int __pthread_cond_timedwait(cnd_t *restrict, mtx_t *restrict, const struct timespec *restrict);
++
++int cnd_timedwait(cnd_t *restrict c, mtx_t *restrict m, const struct timespec *restrict ts)
++{
++ int ret = __pthread_cond_timedwait(c, m, ts);
++ switch (ret) {
++ /* May also return EINVAL or EPERM. */
++ default: return thrd_error;
++ case 0: return thrd_success;
++ case ETIMEDOUT: return thrd_timedout;
++ }
++}
+diff --git a/src/thread/cnd_wait.c b/src/thread/cnd_wait.c
+new file mode 100644
+index 0000000..602796f
+--- /dev/null
++++ b/src/thread/cnd_wait.c
+@@ -0,0 +1,9 @@
++#include <threads.h>
++
++int cnd_wait(cnd_t *c, mtx_t *m)
++{
++ /* Calling cnd_timedwait with a null pointer is an extension.
++ * It is convenient here to avoid duplication of the logic
++ * for return values. */
++ return cnd_timedwait(c, m, 0);
++}
+diff --git a/src/thread/mtx_destroy.c b/src/thread/mtx_destroy.c
+new file mode 100644
+index 0000000..40a0899
+--- /dev/null
++++ b/src/thread/mtx_destroy.c
+@@ -0,0 +1,5 @@
++#include <threads.h>
++
++void mtx_destroy(mtx_t *mtx)
++{
++}
+diff --git a/src/thread/mtx_init.c b/src/thread/mtx_init.c
+new file mode 100644
+index 0000000..4826f76
+--- /dev/null
++++ b/src/thread/mtx_init.c
+@@ -0,0 +1,10 @@
++#include "pthread_impl.h"
++#include <threads.h>
++
++int mtx_init(mtx_t *m, int type)
++{
++ *m = (mtx_t){
++ ._m_type = ((type&mtx_recursive) ? PTHREAD_MUTEX_RECURSIVE : PTHREAD_MUTEX_NORMAL),
++ };
++ return thrd_success;
++}
+diff --git a/src/thread/mtx_lock.c b/src/thread/mtx_lock.c
+new file mode 100644
+index 0000000..5c2415c
+--- /dev/null
++++ b/src/thread/mtx_lock.c
+@@ -0,0 +1,12 @@
++#include "pthread_impl.h"
++#include <threads.h>
++
++int mtx_lock(mtx_t *m)
++{
++ if (m->_m_type == PTHREAD_MUTEX_NORMAL && !a_cas(&m->_m_lock, 0, EBUSY))
++ return thrd_success;
++ /* Calling mtx_timedlock with a null pointer is an extension.
++ * It is convenient, here to avoid duplication of the logic
++ * for return values. */
++ return mtx_timedlock(m, 0);
++}
+diff --git a/src/thread/mtx_timedlock.c b/src/thread/mtx_timedlock.c
+new file mode 100644
+index 0000000..bcc152c
+--- /dev/null
++++ b/src/thread/mtx_timedlock.c
+@@ -0,0 +1,14 @@
++#include <threads.h>
++#include <errno.h>
++
++int __pthread_mutex_timedlock(mtx_t *restrict, const struct timespec *restrict);
++
++int mtx_timedlock(mtx_t *restrict m, const struct timespec *restrict ts)
++{
++ int ret = __pthread_mutex_timedlock(m, ts);
++ switch (ret) {
++ default: return thrd_error;
++ case 0: return thrd_success;
++ case ETIMEDOUT: return thrd_timedout;
++ }
++}
+diff --git a/src/thread/mtx_trylock.c b/src/thread/mtx_trylock.c
+new file mode 100644
+index 0000000..61e7694
+--- /dev/null
++++ b/src/thread/mtx_trylock.c
+@@ -0,0 +1,17 @@
++#include "pthread_impl.h"
++#include <threads.h>
++
++int __pthread_mutex_trylock(mtx_t *);
++
++int mtx_trylock(mtx_t *m)
++{
++ if (m->_m_type == PTHREAD_MUTEX_NORMAL)
++ return (a_cas(&m->_m_lock, 0, EBUSY) & EBUSY) ? thrd_busy : thrd_success;
++
++ int ret = __pthread_mutex_trylock(m);
++ switch (ret) {
++ default: return thrd_error;
++ case 0: return thrd_success;
++ case EBUSY: return thrd_busy;
++ }
++}
+diff --git a/src/thread/mtx_unlock.c b/src/thread/mtx_unlock.c
+new file mode 100644
+index 0000000..5033ace
+--- /dev/null
++++ b/src/thread/mtx_unlock.c
+@@ -0,0 +1,11 @@
++#include <threads.h>
++
++int __pthread_mutex_unlock(mtx_t *);
++
++int mtx_unlock(mtx_t *mtx)
++{
++ /* The only cases where pthread_mutex_unlock can return an
++ * error are undefined behavior for C11 mtx_unlock, so we can
++ * assume it does not return an error and simply tail call. */
++ return __pthread_mutex_unlock(mtx);
++}
+diff --git a/src/thread/pthread_cond_timedwait.c b/src/thread/pthread_cond_timedwait.c
+index 2d192b0..0a80d30 100644
+--- a/src/thread/pthread_cond_timedwait.c
++++ b/src/thread/pthread_cond_timedwait.c
+@@ -1,5 +1,9 @@
+ #include "pthread_impl.h"
+
++void __pthread_testcancel(void);
++int __pthread_mutex_lock(pthread_mutex_t *);
++int __pthread_mutex_unlock(pthread_mutex_t *);
++
+ /*
+ * struct waiter
+ *
+@@ -119,7 +123,7 @@ static void unwait(void *arg)
+ }
+ }
+
+-int pthread_cond_timedwait(pthread_cond_t *restrict c, pthread_mutex_t *restrict m, const struct timespec *restrict ts)
++int __pthread_cond_timedwait(pthread_cond_t *restrict c, pthread_mutex_t *restrict m, const struct timespec *restrict ts)
+ {
+ struct waiter node = { .cond = c, .mutex = m };
+ int e, seq, *fut, clock = c->_c_clock;
+@@ -130,7 +134,7 @@ int pthread_cond_timedwait(pthread_cond_t *restrict c, pthread_mutex_t *restrict
+ if (ts && ts->tv_nsec >= 1000000000UL)
+ return EINVAL;
+
+- pthread_testcancel();
++ __pthread_testcancel();
+
+ if (c->_c_shared) {
+ node.shared = 1;
+@@ -151,7 +155,7 @@ int pthread_cond_timedwait(pthread_cond_t *restrict c, pthread_mutex_t *restrict
+ unlock(&c->_c_lock);
+ }
+
+- pthread_mutex_unlock(m);
++ __pthread_mutex_unlock(m);
+
+ do e = __timedwait(fut, seq, clock, ts, unwait, &node, !node.shared);
+ while (*fut==seq && (!e || e==EINTR));
+@@ -197,3 +201,5 @@ int __private_cond_signal(pthread_cond_t *c, int n)
+
+ return 0;
+ }
++
++weak_alias(__pthread_cond_timedwait, pthread_cond_timedwait);
+diff --git a/src/thread/pthread_create.c b/src/thread/pthread_create.c
+index e441bda..1a47ed1 100644
+--- a/src/thread/pthread_create.c
++++ b/src/thread/pthread_create.c
+@@ -5,6 +5,10 @@
+ #include <sys/mman.h>
+ #include <string.h>
+
++void *__mmap(void *, size_t, int, int, int, off_t);
++int __munmap(void *, size_t);
++int __mprotect(void *, size_t, int);
++
+ static void dummy_0()
+ {
+ }
+@@ -14,7 +18,7 @@ weak_alias(dummy_0, __pthread_tsd_run_dtors);
+ weak_alias(dummy_0, __do_private_robust_list);
+ weak_alias(dummy_0, __do_orphaned_stdio_locks);
+
+-_Noreturn void pthread_exit(void *result)
++_Noreturn void __pthread_exit(void *result)
+ {
+ pthread_t self = __pthread_self();
+ sigset_t set;
+@@ -115,7 +119,15 @@ static int start(void *p)
+ if (self->unblock_cancel)
+ __syscall(SYS_rt_sigprocmask, SIG_UNBLOCK,
+ SIGPT_SET, 0, _NSIG/8);
+- pthread_exit(self->start(self->start_arg));
++ __pthread_exit(self->start(self->start_arg));
++ return 0;
++}
++
++static int start_c11(void *p)
++{
++ pthread_t self = p;
++ int (*start)(void*) = (int(*)(void*)) self->start;
++ __pthread_exit((void *)(uintptr_t)start(self->start_arg));
+ return 0;
+ }
+
+@@ -139,9 +151,9 @@ static void init_file_lock(FILE *f)
+
+ void *__copy_tls(unsigned char *);
+
+-int pthread_create(pthread_t *restrict res, const pthread_attr_t *restrict attrp, void *(*entry)(void *), void *restrict arg)
++int __pthread_create(pthread_t *restrict res, const pthread_attr_t *restrict attrp, void *(*entry)(void *), void *restrict arg)
+ {
+- int ret;
++ int ret, c11 = (attrp == __ATTRP_C11_THREAD);
+ size_t size, guard;
+ struct pthread *self, *new;
+ unsigned char *map = 0, *stack = 0, *tsd = 0, *stack_limit;
+@@ -163,7 +175,7 @@ int pthread_create(pthread_t *restrict res, const pthread_attr_t *restrict attrp
+ self->tsd = (void **)__pthread_tsd_main;
+ libc.threaded = 1;
+ }
+- if (attrp) attr = *attrp;
++ if (attrp && !c11) attr = *attrp;
+
+ __acquire_ptc();
+
+@@ -191,14 +203,14 @@ int pthread_create(pthread_t *restrict res, const pthread_attr_t *restrict attrp
+
+ if (!tsd) {
+ if (guard) {
+- map = mmap(0, size, PROT_NONE, MAP_PRIVATE|MAP_ANON, -1, 0);
++ map = __mmap(0, size, PROT_NONE, MAP_PRIVATE|MAP_ANON, -1, 0);
+ if (map == MAP_FAILED) goto fail;
+- if (mprotect(map+guard, size-guard, PROT_READ|PROT_WRITE)) {
+- munmap(map, size);
++ if (__mprotect(map+guard, size-guard, PROT_READ|PROT_WRITE)) {
++ __munmap(map, size);
+ goto fail;
+ }
+ } else {
+- map = mmap(0, size, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANON, -1, 0);
++ map = __mmap(0, size, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANON, -1, 0);
+ if (map == MAP_FAILED) goto fail;
+ }
+ tsd = map + size - __pthread_tsd_size;
+@@ -230,7 +242,7 @@ int pthread_create(pthread_t *restrict res, const pthread_attr_t *restrict attrp
+ new->canary = self->canary;
+
+ a_inc(&libc.threads_minus_1);
+- ret = __clone(start, stack, flags, new, &new->tid, TP_ADJ(new), &new->tid);
++ ret = __clone((c11 ? start_c11 : start), stack, flags, new, &new->tid, TP_ADJ(new), &new->tid);
+
+ __release_ptc();
+
+@@ -240,7 +252,7 @@ int pthread_create(pthread_t *restrict res, const pthread_attr_t *restrict attrp
+
+ if (ret < 0) {
+ a_dec(&libc.threads_minus_1);
+- if (map) munmap(map, size);
++ if (map) __munmap(map, size);
+ return EAGAIN;
+ }
+
+@@ -258,3 +270,6 @@ fail:
+ __release_ptc();
+ return EAGAIN;
+ }
++
++weak_alias(__pthread_exit, pthread_exit);
++weak_alias(__pthread_create, pthread_create);
+diff --git a/src/thread/pthread_detach.c b/src/thread/pthread_detach.c
+index 651c38e..ed77f74 100644
+--- a/src/thread/pthread_detach.c
++++ b/src/thread/pthread_detach.c
+@@ -1,11 +1,17 @@
+ #include "pthread_impl.h"
++#include <threads.h>
+
+-int pthread_detach(pthread_t t)
++int __pthread_join(pthread_t, void **);
++
++static int __pthread_detach(pthread_t t)
+ {
+ /* Cannot detach a thread that's already exiting */
+ if (a_swap(t->exitlock, 1))
+- return pthread_join(t, 0);
++ return __pthread_join(t, 0);
+ t->detached = 2;
+ __unlock(t->exitlock);
+ return 0;
+ }
++
++weak_alias(__pthread_detach, pthread_detach);
++weak_alias(__pthread_detach, thrd_detach);
+diff --git a/src/thread/pthread_equal.c b/src/thread/pthread_equal.c
+index 3e3df4f..7c31482 100644
+--- a/src/thread/pthread_equal.c
++++ b/src/thread/pthread_equal.c
+@@ -1,6 +1,11 @@
+ #include <pthread.h>
++#include <threads.h>
++#include "libc.h"
+
+-int (pthread_equal)(pthread_t a, pthread_t b)
++static int __pthread_equal(pthread_t a, pthread_t b)
+ {
+ return a==b;
+ }
++
++weak_alias(__pthread_equal, pthread_equal);
++weak_alias(__pthread_equal, thrd_equal);
+diff --git a/src/thread/pthread_getspecific.c b/src/thread/pthread_getspecific.c
+index b2a282c..d9342a5 100644
+--- a/src/thread/pthread_getspecific.c
++++ b/src/thread/pthread_getspecific.c
+@@ -1,7 +1,11 @@
+ #include "pthread_impl.h"
++#include <threads.h>
+
+-void *pthread_getspecific(pthread_key_t k)
++static void *__pthread_getspecific(pthread_key_t k)
+ {
+ struct pthread *self = __pthread_self();
+ return self->tsd[k];
+ }
++
++weak_alias(__pthread_getspecific, pthread_getspecific);
++weak_alias(__pthread_getspecific, tss_get);
+diff --git a/src/thread/pthread_join.c b/src/thread/pthread_join.c
+index abd2d66..19e6b54 100644
+--- a/src/thread/pthread_join.c
++++ b/src/thread/pthread_join.c
+@@ -1,16 +1,20 @@
+ #include "pthread_impl.h"
+ #include <sys/mman.h>
+
++int __munmap(void *, size_t);
++
+ static void dummy(void *p)
+ {
+ }
+
+-int pthread_join(pthread_t t, void **res)
++int __pthread_join(pthread_t t, void **res)
+ {
+ int tmp;
+ pthread_testcancel();
+ while ((tmp = t->tid)) __timedwait(&t->tid, tmp, 0, 0, dummy, 0, 0);
+ if (res) *res = t->result;
+- if (t->map_base) munmap(t->map_base, t->map_size);
++ if (t->map_base) __munmap(t->map_base, t->map_size);
+ return 0;
+ }
++
++weak_alias(__pthread_join, pthread_join);
+diff --git a/src/thread/pthread_key_create.c b/src/thread/pthread_key_create.c
+index a9187f7..bfcd597 100644
+--- a/src/thread/pthread_key_create.c
++++ b/src/thread/pthread_key_create.c
+@@ -9,7 +9,7 @@ static void nodtor(void *dummy)
+ {
+ }
+
+-int pthread_key_create(pthread_key_t *k, void (*dtor)(void *))
++int __pthread_key_create(pthread_key_t *k, void (*dtor)(void *))
+ {
+ unsigned i = (uintptr_t)&k / 16 % PTHREAD_KEYS_MAX;
+ unsigned j = i;
+@@ -31,7 +31,7 @@ int pthread_key_create(pthread_key_t *k, void (*dtor)(void *))
+ return EAGAIN;
+ }
+
+-int pthread_key_delete(pthread_key_t k)
++int __pthread_key_delete(pthread_key_t k)
+ {
+ keys[k] = 0;
+ return 0;
+@@ -53,3 +53,6 @@ void __pthread_tsd_run_dtors()
+ }
+ }
+ }
++
++weak_alias(__pthread_key_delete, pthread_key_delete);
++weak_alias(__pthread_key_create, pthread_key_create);
+diff --git a/src/thread/pthread_mutex_lock.c b/src/thread/pthread_mutex_lock.c
+index 2a9a3aa..d0c93ca 100644
+--- a/src/thread/pthread_mutex_lock.c
++++ b/src/thread/pthread_mutex_lock.c
+@@ -1,10 +1,14 @@
+ #include "pthread_impl.h"
+
+-int pthread_mutex_lock(pthread_mutex_t *m)
++int __pthread_mutex_timedlock(pthread_mutex_t *restrict, const struct timespec *restrict);
++
++int __pthread_mutex_lock(pthread_mutex_t *m)
+ {
+ if ((m->_m_type&15) == PTHREAD_MUTEX_NORMAL
+ && !a_cas(&m->_m_lock, 0, EBUSY))
+ return 0;
+
+- return pthread_mutex_timedlock(m, 0);
++ return __pthread_mutex_timedlock(m, 0);
+ }
++
++weak_alias(__pthread_mutex_lock, pthread_mutex_lock);
+diff --git a/src/thread/pthread_mutex_timedlock.c b/src/thread/pthread_mutex_timedlock.c
+index ae883f9..16241ee 100644
+--- a/src/thread/pthread_mutex_timedlock.c
++++ b/src/thread/pthread_mutex_timedlock.c
+@@ -1,6 +1,6 @@
+ #include "pthread_impl.h"
+
+-int pthread_mutex_timedlock(pthread_mutex_t *restrict m, const struct timespec *restrict at)
++int __pthread_mutex_timedlock(pthread_mutex_t *restrict m, const struct timespec *restrict at)
+ {
+ if ((m->_m_type&15) == PTHREAD_MUTEX_NORMAL
+ && !a_cas(&m->_m_lock, 0, EBUSY))
+@@ -30,3 +30,5 @@ int pthread_mutex_timedlock(pthread_mutex_t *restrict m, const struct timespec *
+ }
+ return r;
+ }
++
++weak_alias(__pthread_mutex_timedlock, pthread_mutex_timedlock);
+diff --git a/src/thread/pthread_mutex_trylock.c b/src/thread/pthread_mutex_trylock.c
+index e851517..cb93565 100644
+--- a/src/thread/pthread_mutex_trylock.c
++++ b/src/thread/pthread_mutex_trylock.c
+@@ -50,9 +50,11 @@ int __pthread_mutex_trylock_owner(pthread_mutex_t *m)
+ return 0;
+ }
+
+-int pthread_mutex_trylock(pthread_mutex_t *m)
++int __pthread_mutex_trylock(pthread_mutex_t *m)
+ {
+ if ((m->_m_type&15) == PTHREAD_MUTEX_NORMAL)
+ return a_cas(&m->_m_lock, 0, EBUSY) & EBUSY;
+ return __pthread_mutex_trylock_owner(m);
+ }
++
++weak_alias(__pthread_mutex_trylock, pthread_mutex_trylock);
+diff --git a/src/thread/pthread_mutex_unlock.c b/src/thread/pthread_mutex_unlock.c
+index 46761d9..a7f39c7 100644
+--- a/src/thread/pthread_mutex_unlock.c
++++ b/src/thread/pthread_mutex_unlock.c
+@@ -3,7 +3,7 @@
+ void __vm_lock_impl(int);
+ void __vm_unlock_impl(void);
+
+-int pthread_mutex_unlock(pthread_mutex_t *m)
++int __pthread_mutex_unlock(pthread_mutex_t *m)
+ {
+ pthread_t self;
+ int waiters = m->_m_waiters;
+@@ -36,3 +36,5 @@ int pthread_mutex_unlock(pthread_mutex_t *m)
+ __wake(&m->_m_lock, 1, priv);
+ return 0;
+ }
++
++weak_alias(__pthread_mutex_unlock, pthread_mutex_unlock);
+diff --git a/src/thread/pthread_once.c b/src/thread/pthread_once.c
+index 2eb0f93..05ebe69 100644
+--- a/src/thread/pthread_once.c
++++ b/src/thread/pthread_once.c
+@@ -6,7 +6,7 @@ static void undo(void *control)
+ __wake(control, 1, 1);
+ }
+
+-int pthread_once(pthread_once_t *control, void (*init)(void))
++int __pthread_once(pthread_once_t *control, void (*init)(void))
+ {
+ static int waiters;
+
+@@ -34,3 +34,5 @@ int pthread_once(pthread_once_t *control, void (*init)(void))
+ return 0;
+ }
+ }
++
++weak_alias(__pthread_once, pthread_once);
+diff --git a/src/thread/pthread_self.c b/src/thread/pthread_self.c
+index 5f9e651..241a620 100644
+--- a/src/thread/pthread_self.c
++++ b/src/thread/pthread_self.c
+@@ -1,6 +1,11 @@
+ #include "pthread_impl.h"
++#include <threads.h>
++#include "libc.h"
+
+-pthread_t pthread_self()
++static pthread_t __pthread_self_internal()
+ {
+ return __pthread_self();
+ }
++
++weak_alias(__pthread_self_internal, pthread_self);
++weak_alias(__pthread_self_internal, thrd_current);
+diff --git a/src/thread/pthread_setcancelstate.c b/src/thread/pthread_setcancelstate.c
+index 060bcdc..2268217 100644
+--- a/src/thread/pthread_setcancelstate.c
++++ b/src/thread/pthread_setcancelstate.c
+@@ -1,6 +1,6 @@
+ #include "pthread_impl.h"
+
+-int pthread_setcancelstate(int new, int *old)
++int __pthread_setcancelstate(int new, int *old)
+ {
+ if (new > 1U) return EINVAL;
+ if (!libc.has_thread_pointer) return ENOSYS;
+@@ -9,3 +9,5 @@ int pthread_setcancelstate(int new, int *old)
+ self->canceldisable = new;
+ return 0;
+ }
++
++weak_alias(__pthread_setcancelstate, pthread_setcancelstate);
+diff --git a/src/thread/pthread_testcancel.c b/src/thread/pthread_testcancel.c
+index ba5f7c6..ee48e6d 100644
+--- a/src/thread/pthread_testcancel.c
++++ b/src/thread/pthread_testcancel.c
+@@ -7,7 +7,9 @@ static void dummy()
+
+ weak_alias(dummy, __testcancel);
+
+-void pthread_testcancel()
++void __pthread_testcancel()
+ {
+ __testcancel();
+ }
++
++weak_alias(__pthread_testcancel, pthread_testcancel);
+diff --git a/src/thread/thrd_create.c b/src/thread/thrd_create.c
+new file mode 100644
+index 0000000..e033669
+--- /dev/null
++++ b/src/thread/thrd_create.c
+@@ -0,0 +1,14 @@
++#include "pthread_impl.h"
++#include <threads.h>
++
++int __pthread_create(pthread_t *restrict, const pthread_attr_t *restrict, void *(*)(void *), void *restrict);
++
++int thrd_create(thrd_t *thr, thrd_start_t func, void *arg)
++{
++ int ret = __pthread_create(thr, __ATTRP_C11_THREAD, (void *(*)(void *))func, arg);
++ switch (ret) {
++ case 0: return thrd_success;
++ case EAGAIN: return thrd_nomem;
++ default: return thrd_error;
++ }
++}
+diff --git a/src/thread/thrd_exit.c b/src/thread/thrd_exit.c
+new file mode 100644
+index 0000000..b66bd99
+--- /dev/null
++++ b/src/thread/thrd_exit.c
+@@ -0,0 +1,9 @@
++#include "pthread_impl.h"
++#include <threads.h>
++
++_Noreturn void __pthread_exit(void *);
++
++_Noreturn void thrd_exit(int result)
++{
++ __pthread_exit((void*)(intptr_t)result);
++}
+diff --git a/src/thread/thrd_join.c b/src/thread/thrd_join.c
+new file mode 100644
+index 0000000..ac66789
+--- /dev/null
++++ b/src/thread/thrd_join.c
+@@ -0,0 +1,12 @@
++#include <stdint.h>
++#include <threads.h>
++
++int __pthread_join(thrd_t, void**);
++
++int thrd_join(thrd_t t, int *res)
++{
++ void *pthread_res;
++ __pthread_join(t, &pthread_res);
++ if (res) *res = (int)(intptr_t)pthread_res;
++ return thrd_success;
++}
+diff --git a/src/thread/thrd_sleep.c b/src/thread/thrd_sleep.c
+new file mode 100644
+index 0000000..e8dfe40
+--- /dev/null
++++ b/src/thread/thrd_sleep.c
+@@ -0,0 +1,13 @@
++#include <threads.h>
++#include <errno.h>
++#include "syscall.h"
++
++int thrd_sleep(const struct timespec *req, struct timespec *rem)
++{
++ int ret = __syscall(SYS_nanosleep, req, rem);
++ switch (ret) {
++ case 0: return 0;
++ case -EINTR: return -1; /* value specified by C11 */
++ default: return -2;
++ }
++}
+diff --git a/src/thread/thrd_yield.c b/src/thread/thrd_yield.c
+new file mode 100644
+index 0000000..f7ad132
+--- /dev/null
++++ b/src/thread/thrd_yield.c
+@@ -0,0 +1,7 @@
++#include <threads.h>
++#include "syscall.h"
++
++void thrd_yield()
++{
++ __syscall(SYS_sched_yield);
++}
+diff --git a/src/thread/tss_create.c b/src/thread/tss_create.c
+new file mode 100644
+index 0000000..251d22b
+--- /dev/null
++++ b/src/thread/tss_create.c
+@@ -0,0 +1,11 @@
++#include <threads.h>
++
++int __pthread_key_create(tss_t *, void (*)(void *));
++
++int tss_create(tss_t *tss, tss_dtor_t dtor)
++{
++ /* Different error returns are possible. C glues them together into
++ * just failure notification. Can't be optimized to a tail call,
++ * unless thrd_error equals EAGAIN. */
++ return __pthread_key_create(tss, dtor) ? thrd_error : thrd_success;
++}
+diff --git a/src/thread/tss_delete.c b/src/thread/tss_delete.c
+new file mode 100644
+index 0000000..35db103
+--- /dev/null
++++ b/src/thread/tss_delete.c
+@@ -0,0 +1,8 @@
++#include <threads.h>
++
++int __pthread_key_delete(tss_t k);
++
++void tss_delete(tss_t key)
++{
++ __pthread_key_delete(key);
++}
+diff --git a/src/thread/tss_set.c b/src/thread/tss_set.c
+new file mode 100644
+index 0000000..70c4fb7
+--- /dev/null
++++ b/src/thread/tss_set.c
+@@ -0,0 +1,13 @@
++#include "pthread_impl.h"
++#include <threads.h>
++
++int tss_set(tss_t k, void *x)
++{
++ struct pthread *self = __pthread_self();
++ /* Avoid unnecessary COW */
++ if (self->tsd[k] != x) {
++ self->tsd[k] = x;
++ self->tsd_used = 1;
++ }
++ return thrd_success;
++}
+diff --git a/src/time/timespec_get.c b/src/time/timespec_get.c
+new file mode 100644
+index 0000000..03c5a77
+--- /dev/null
++++ b/src/time/timespec_get.c
+@@ -0,0 +1,12 @@
++#include <time.h>
++
++int __clock_gettime(clockid_t, struct timespec *);
++
++/* There is no other implemented value than TIME_UTC; all other values
++ * are considered erroneous. */
++int timespec_get(struct timespec * ts, int base)
++{
++ if (base != TIME_UTC) return 0;
++ int ret = __clock_gettime(CLOCK_REALTIME, ts);
++ return ret < 0 ? 0 : base;
++}
diff --git a/main/musl/APKBUILD b/main/musl/APKBUILD
index 6cf0c3143e..1998acee67 100644
--- a/main/musl/APKBUILD
+++ b/main/musl/APKBUILD
@@ -2,7 +2,7 @@
# Maintainer: Timo Teräs <timo.teras@iki.fi>
pkgname=musl
pkgver=1.1.4
-pkgrel=7
+pkgrel=8
pkgdesc="the musl c library (libc) implementation"
url="http://www.musl-libc.org/"
arch="all"
@@ -17,10 +17,11 @@ source="http://www.musl-libc.org/releases/musl-$pkgver.tar.gz
0002-d86af2a0-to-4992ace9.patch
0003-4992ace9-to-6e2bb7ac.patch
0004-6e2bb7ac-to-4674809b.patch
+ 0005-4674809b-to-3e936ce8.patch
+
1001-add-basic-dns-record-parsing-functions.patch
1002-check-lockcount-in-funlockfile.patch
1003-remove-ulimit-fiddling-from-setxid.patch
- alignas.patch
ldconfig
getopt_long.c
@@ -128,10 +129,10 @@ md5sums="f18f3bdbe088438cd64a5313c19a7312 musl-1.1.4.tar.gz
04eef5b44c0b56b79055340bee1febc2 0002-d86af2a0-to-4992ace9.patch
6710102c768f5a971b93100219de3fbb 0003-4992ace9-to-6e2bb7ac.patch
294d3d382098977ea0770731a4803322 0004-6e2bb7ac-to-4674809b.patch
+5e7769c1f65acbdb9decf3012a70b411 0005-4674809b-to-3e936ce8.patch
2371eb1ce057fcb709a0e6a81f0d356c 1001-add-basic-dns-record-parsing-functions.patch
8a763b1853ee16d034abe038a0c44641 1002-check-lockcount-in-funlockfile.patch
71b2a4dcc39c436a6b89173943424043 1003-remove-ulimit-fiddling-from-setxid.patch
-20014a2b6bbf5e1e7ad95de338a66562 alignas.patch
830d01f7821b978df770b06db3790921 ldconfig
61c6c1e84ed1df82abbe6d75e90cf21c getopt_long.c
0df687757221bbb0fc1aa67f1bd646f9 __stack_chk_fail_local.c
@@ -143,10 +144,10 @@ sha256sums="658c65ad3c3a9b281a96c5281e75720c758d91fcaae35ab987f2fdfb4f88f5cd mu
c7521464f3096befb5b44d303819e9169a43a6a5a3c93566ac1cdf523a6a389d 0002-d86af2a0-to-4992ace9.patch
7a716e03084724e3776300d8ed498e835b2cad4321b638a3eec5a9d8f28de8d9 0003-4992ace9-to-6e2bb7ac.patch
e727b50aa65886563960c10449f60e229ffb94dd3476df1098e877ce599358c4 0004-6e2bb7ac-to-4674809b.patch
+1672b904cf69c9c9b349e7b07f534663cd2083a878778db188f74104bfc051b5 0005-4674809b-to-3e936ce8.patch
75053a31f6b84a64846d92c0ec631c76d7f747a9c0dc92a6dc1aa1bddfe2ea76 1001-add-basic-dns-record-parsing-functions.patch
eebc0afce600cbe2f41055404c7735f1a7be6e262df602dbcbfde7321633bdd9 1002-check-lockcount-in-funlockfile.patch
fb542c2bd5081ff2f601c519edb3dac8f54ca5c888f44bc6cfb84e6565472025 1003-remove-ulimit-fiddling-from-setxid.patch
-91a20ddcbbeb1d14e805afe45d5914dda5dea51a945f9f53afd589a60a3ab9ba alignas.patch
b4a2c06db38742e8c42c3c9838b285a7d8cdac6c091ff3df5ff9a15f1e41b9c7 ldconfig
d9b644ec20bc33e81a7c52b9fcf7973d835923a69faf50f03db45534b811bd96 getopt_long.c
299a7d75a09de3e2e11e7fb4acc3182e4a14e868093d2f30938fce9bfcff13da __stack_chk_fail_local.c
@@ -158,10 +159,10 @@ sha512sums="a46fb1db23f518beaa959e9bebcb3bf0574e583c197792d30dcd52b3974e3c285594
0e892a0d0e2abf09cb52d27759769a4cdd39ebd87fb1af89f502da00a4c9453309291aeb69862d60d88a8d0a0c2517aabcc92b50214d67279b12113b3c5a0157 0002-d86af2a0-to-4992ace9.patch
af92b766d2ca9fcf58b57c705b65ec0b60ecd9af1d38f1b9697a6b999baa56d8cb0ecf94bc02267022a13763e4a1a13a3f73bc03fdf73c0cd8f80126c2949a13 0003-4992ace9-to-6e2bb7ac.patch
8aa879e31c3eb18f9b00ea98d25872d94cad9120c05a80419965dcf7d3a4e0a69c370b35e2871cb11b639bbcf606842a48d3ad64ec916bc34b2eb21d3c4a3efa 0004-6e2bb7ac-to-4674809b.patch
+7104eb3753135b5cd2a291552c6fbf36034d7169178ae45ef99d634a4055afa30a7e89ce7a01695b120c4a872ec51f1a837b34c4e83d69778b0248ca6c931d48 0005-4674809b-to-3e936ce8.patch
5b8ffa0a50419581adbf6ce2dae5797774022551c6331fa5aa2ff13635eb72b74eedd8a92cb478d45d73e1956af2f588669681ac414f3a255abd4d8ba8579448 1001-add-basic-dns-record-parsing-functions.patch
beb515f46fa7c18eafe72cc612c5e8877dd1fd083e05c427af3a2a4d4281df06c6c1727c39d136d8e3f267be8832ea133362f5088a2b6a26b33c9e184acf1884 1002-check-lockcount-in-funlockfile.patch
dae010b45419fcab64410568466f659cdc874e63113025e2cbc2fbab047b470fec23851ecbef08886505924482a069caf37c16b483b6922535fbd31832f1c4a3 1003-remove-ulimit-fiddling-from-setxid.patch
-3cba68979fddff66d4989290c989c51b1f1b1f45d371ab23ee8622c6caf8a12409eecb3a15ad88c1b1f4d5c9c62380e9c0073b6c45f117e0312c4f50df4a7ad8 alignas.patch
8d3a2d5315fc56fee7da9abb8b89bb38c6046c33d154c10d168fb35bfde6b0cf9f13042a3bceee34daf091bc409d699223735dcf19f382eeee1f6be34154f26f ldconfig
140f3f20d30bd95ebce8c41b8cc7f616c6cbedf4ea06c729c21014e74f6043796825cc40ebc5180620ea38173afdba23f09ebf6d8b11fa05440b14d23764fca9 getopt_long.c
062bb49fa54839010acd4af113e20f7263dde1c8a2ca359b5fb2661ef9ed9d84a0f7c3bc10c25dcfa10bb3c5a4874588dff636ac43d5dbb3d748d75400756d0b __stack_chk_fail_local.c
diff --git a/main/musl/alignas.patch b/main/musl/alignas.patch
deleted file mode 100644
index 61dd221cb7..0000000000
--- a/main/musl/alignas.patch
+++ /dev/null
@@ -1,17 +0,0 @@
-diff --git a/arch/i386/bits/alltypes.h.in b/arch/i386/bits/alltypes.h.in
-index a7882ed..c4ba2c7 100644
---- a/arch/i386/bits/alltypes.h.in
-+++ b/arch/i386/bits/alltypes.h.in
-@@ -27,7 +27,11 @@ TYPEDEF long double float_t;
- TYPEDEF long double double_t;
- #endif
-
-+#if __cplusplus >= 201103L
-+TYPEDEF struct { alignas(8) long long __ll; long double __ld; } max_align_t;
-+#else
- TYPEDEF struct { _Alignas(8) long long __ll; long double __ld; } max_align_t;
-+#endif
-
- TYPEDEF long time_t;
- TYPEDEF long suseconds_t;
-