mirror of
https://github.com/fluencelabs/musl
synced 2025-06-23 11:41:57 +00:00
previously, stdio used spinlocks, which would be unacceptable if we ever add support for thread priorities, and which yielded pathologically bad performance if an application attempted to use flockfile on a key file as a major/primary locking mechanism. i had held off on making this change for fear that it would hurt performance in the non-threaded case, but actually support for recursive locking had already inflicted that cost. by having the internal locking functions store a flag indicating whether they need to perform unlocking, rather than using the actual recursive lock counter, i was able to combine the conditionals at unlock time, eliminating any additional cost, and also avoid a nasty corner case where a huge number of calls to ftrylockfile could cause deadlock later at the point of internal locking. this commit also fixes some issues with usage of pthread_self conflicting with __attribute__((const)) which resulted in crashes with some compiler versions/optimizations, mainly in flockfile prior to pthread_create.
51 lines
892 B
C
51 lines
892 B
C
#include "stdio_impl.h"
|
|
|
|
wint_t __fgetwc_unlocked(FILE *f)
|
|
{
|
|
mbstate_t st = { 0 };
|
|
wchar_t wc;
|
|
int c;
|
|
unsigned char b;
|
|
size_t l;
|
|
|
|
f->mode |= f->mode+1;
|
|
|
|
/* Convert character from buffer if possible */
|
|
if (f->rpos < f->rend) {
|
|
l = mbrtowc(&wc, (void *)f->rpos, f->rend - f->rpos, &st);
|
|
if (l+2 >= 2) {
|
|
f->rpos += l + !l; /* l==0 means 1 byte, null */
|
|
return wc;
|
|
}
|
|
if (l == -1) {
|
|
f->rpos++;
|
|
return WEOF;
|
|
}
|
|
} else l = -2;
|
|
|
|
/* Convert character byte-by-byte */
|
|
while (l == -2) {
|
|
b = c = getc_unlocked(f);
|
|
if (c < 0) {
|
|
if (!mbsinit(&st)) errno = EILSEQ;
|
|
return WEOF;
|
|
}
|
|
l = mbrtowc(&wc, (void *)&b, 1, &st);
|
|
if (l == -1) return WEOF;
|
|
}
|
|
|
|
return wc;
|
|
}
|
|
|
|
wint_t fgetwc(FILE *f)
|
|
{
|
|
wint_t c;
|
|
FLOCK(f);
|
|
c = __fgetwc_unlocked(f);
|
|
FUNLOCK(f);
|
|
return c;
|
|
}
|
|
|
|
weak_alias(__fgetwc_unlocked, fgetwc_unlocked);
|
|
weak_alias(__fgetwc_unlocked, getwc_unlocked);
|