1*c9945492SAndroid Build Coastguard Worker #include "stdio_impl.h"
2*c9945492SAndroid Build Coastguard Worker #include <limits.h>
3*c9945492SAndroid Build Coastguard Worker #include <errno.h>
4*c9945492SAndroid Build Coastguard Worker #include <stdint.h>
5*c9945492SAndroid Build Coastguard Worker #include <stdlib.h>
6*c9945492SAndroid Build Coastguard Worker #include <wchar.h>
7*c9945492SAndroid Build Coastguard Worker
8*c9945492SAndroid Build Coastguard Worker struct cookie {
9*c9945492SAndroid Build Coastguard Worker wchar_t *ws;
10*c9945492SAndroid Build Coastguard Worker size_t l;
11*c9945492SAndroid Build Coastguard Worker };
12*c9945492SAndroid Build Coastguard Worker
sw_write(FILE * f,const unsigned char * s,size_t l)13*c9945492SAndroid Build Coastguard Worker static size_t sw_write(FILE *f, const unsigned char *s, size_t l)
14*c9945492SAndroid Build Coastguard Worker {
15*c9945492SAndroid Build Coastguard Worker size_t l0 = l;
16*c9945492SAndroid Build Coastguard Worker int i = 0;
17*c9945492SAndroid Build Coastguard Worker struct cookie *c = f->cookie;
18*c9945492SAndroid Build Coastguard Worker if (s!=f->wbase && sw_write(f, f->wbase, f->wpos-f->wbase)==-1)
19*c9945492SAndroid Build Coastguard Worker return -1;
20*c9945492SAndroid Build Coastguard Worker while (c->l && l && (i=mbtowc(c->ws, (void *)s, l))>=0) {
21*c9945492SAndroid Build Coastguard Worker if (!i) i=1;
22*c9945492SAndroid Build Coastguard Worker s+=i;
23*c9945492SAndroid Build Coastguard Worker l-=i;
24*c9945492SAndroid Build Coastguard Worker c->l--;
25*c9945492SAndroid Build Coastguard Worker c->ws++;
26*c9945492SAndroid Build Coastguard Worker }
27*c9945492SAndroid Build Coastguard Worker *c->ws = 0;
28*c9945492SAndroid Build Coastguard Worker if (i < 0) {
29*c9945492SAndroid Build Coastguard Worker f->wpos = f->wbase = f->wend = 0;
30*c9945492SAndroid Build Coastguard Worker f->flags |= F_ERR;
31*c9945492SAndroid Build Coastguard Worker return i;
32*c9945492SAndroid Build Coastguard Worker }
33*c9945492SAndroid Build Coastguard Worker f->wend = f->buf + f->buf_size;
34*c9945492SAndroid Build Coastguard Worker f->wpos = f->wbase = f->buf;
35*c9945492SAndroid Build Coastguard Worker return l0;
36*c9945492SAndroid Build Coastguard Worker }
37*c9945492SAndroid Build Coastguard Worker
vswprintf(wchar_t * restrict s,size_t n,const wchar_t * restrict fmt,va_list ap)38*c9945492SAndroid Build Coastguard Worker int vswprintf(wchar_t *restrict s, size_t n, const wchar_t *restrict fmt, va_list ap)
39*c9945492SAndroid Build Coastguard Worker {
40*c9945492SAndroid Build Coastguard Worker int r;
41*c9945492SAndroid Build Coastguard Worker unsigned char buf[256];
42*c9945492SAndroid Build Coastguard Worker struct cookie c = { s, n-1 };
43*c9945492SAndroid Build Coastguard Worker FILE f = {
44*c9945492SAndroid Build Coastguard Worker .lbf = EOF,
45*c9945492SAndroid Build Coastguard Worker .write = sw_write,
46*c9945492SAndroid Build Coastguard Worker .lock = -1,
47*c9945492SAndroid Build Coastguard Worker .buf = buf,
48*c9945492SAndroid Build Coastguard Worker .buf_size = sizeof buf,
49*c9945492SAndroid Build Coastguard Worker .cookie = &c,
50*c9945492SAndroid Build Coastguard Worker };
51*c9945492SAndroid Build Coastguard Worker
52*c9945492SAndroid Build Coastguard Worker if (!n) {
53*c9945492SAndroid Build Coastguard Worker return -1;
54*c9945492SAndroid Build Coastguard Worker }
55*c9945492SAndroid Build Coastguard Worker r = vfwprintf(&f, fmt, ap);
56*c9945492SAndroid Build Coastguard Worker sw_write(&f, 0, 0);
57*c9945492SAndroid Build Coastguard Worker return r>=n ? -1 : r;
58*c9945492SAndroid Build Coastguard Worker }
59