suppress EINTR in sem_wait and sem_timedwait
[musl] / src / stdlib / wcstod.c
1 #include "shgetc.h"
2 #include "floatscan.h"
3 #include "stdio_impl.h"
4 #include <wchar.h>
5 #include <wctype.h>
6
7 /* This read function heavily cheats. It knows:
8  *  (1) len will always be 1
9  *  (2) non-ascii characters don't matter */
10
11 static size_t do_read(FILE *f, unsigned char *buf, size_t len)
12 {
13         size_t i;
14         const wchar_t *wcs = f->cookie;
15
16         if (!wcs[0]) wcs=L"@";
17         for (i=0; i<f->buf_size && wcs[i]; i++)
18                 f->buf[i] = wcs[i] < 128 ? wcs[i] : '@';
19         f->rpos = f->buf;
20         f->rend = f->buf + i;
21         f->cookie = (void *)(wcs+i);
22
23         if (i && len) {
24                 *buf = *f->rpos++;
25                 return 1;
26         }
27         return 0;
28 }
29
30 static long double wcstox(const wchar_t *s, wchar_t **p, int prec)
31 {
32         wchar_t *t = (wchar_t *)s;
33         unsigned char buf[64];
34         FILE f = {0};
35         f.flags = 0;
36         f.rpos = f.rend = 0;
37         f.buf = buf + 4;
38         f.buf_size = sizeof buf - 4;
39         f.lock = -1;
40         f.read = do_read;
41         while (iswspace(*t)) t++;
42         f.cookie = (void *)t;
43         shlim(&f, 0);
44         long double y = __floatscan(&f, prec, 1);
45         if (p) {
46                 size_t cnt = shcnt(&f);
47                 *p = cnt ? t + cnt : (wchar_t *)s;
48         }
49         return y;
50 }
51
52 float wcstof(const wchar_t *restrict s, wchar_t **restrict p)
53 {
54         return wcstox(s, p, 0);
55 }
56
57 double wcstod(const wchar_t *restrict s, wchar_t **restrict p)
58 {
59         return wcstox(s, p, 1);
60 }
61
62 long double wcstold(const wchar_t *restrict s, wchar_t **restrict p)
63 {
64         return wcstox(s, p, 2);
65 }