#pragma GCC diagnostic ignored "-Wpointer-sign" #pragma GCC diagnostic ignored "-Wunused-value" #include "patches.h" #include "string.h" #include "stdarg.h" #include "xstdio.h" char* strchr(const char* s, int c) { const char ch = c; while (*s != ch) { if (*s == 0) { return NULL; } s++; } return (char*)s; } #define isdigit(x) ((x >= '0' && x <= '9')) #define LDSIGN(x) (((unsigned short *)&(x))[0] & 0x8000) #define ATOI(dst, src) \ for (dst = 0; isdigit(*src); ++src) \ { \ if (dst < 999) \ dst = dst * 10 + *src - '0'; \ } #define MAX_PAD ((sizeof(spaces) - 1)) #define PAD(s, n) \ if (0 < (n)) \ { \ int i, j = (n); \ for (; 0 < j; j -= i) \ { \ i = MAX_PAD < (unsigned int)j ? (int)MAX_PAD : j; \ PUT(s, i); \ } \ } #define PUT(s, n) \ if (0 < (n)) \ { \ if ((arg = (*pfn)(arg, s, n)) != NULL) \ x.nchar += (n); \ else \ return x.nchar; \ } static char spaces[] = " "; static char zeroes[] = "00000000000000000000000000000000"; void _Putfld(_Pft *px, va_list *pap, char code, char *ac); int _Printf(void* pfn(void*,const char*,size_t), void *arg, const char *fmt, va_list ap) { _Pft x; x.nchar = 0; while (1) { const char *s; char c; const char *t; static const char fchar[] = {' ', '+', '-', '#', '0', '\0'}; static const unsigned int fbit[] = {FLAGS_SPACE, FLAGS_PLUS, FLAGS_MINUS, FLAGS_HASH, FLAGS_ZERO, 0}; char ac[32]; s = fmt; for (c = *s; c != 0 && c != '%';) { c = *++s; } PUT(fmt, s - fmt); if (c == 0) { return x.nchar; } fmt = ++s; for (x.flags = 0; (t = strchr(fchar, *s)) != NULL; s++) { x.flags |= fbit[t - fchar]; } if (*s == '*') { x.width = va_arg(ap, int); if (x.width < 0) { x.width = -x.width; x.flags |= FLAGS_MINUS; } s++; } else ATOI(x.width, s); if (*s != '.') { x.prec = -1; } else if (*++s == '*') { x.prec = va_arg(ap, int); ++s; } else for (x.prec = 0; isdigit(*s); s++) { if (x.prec < 999) x.prec = x.prec * 10 + *s - '0'; } x.qual = strchr("hlL", *s) ? *s++ : '\0'; if (x.qual == 'l' && *s == 'l') { x.qual = 'L'; ++s; } _Putfld(&x, &ap, *s, ac); x.width -= x.n0 + x.nz0 + x.n1 + x.nz1 + x.n2 + x.nz2; { if (!(x.flags & FLAGS_MINUS)) { int i, j; if (0 < (x.width)) { i, j = x.width; for (; 0 < j; j -= i) { i = MAX_PAD < (unsigned int)j ? (int)MAX_PAD : j; PUT(spaces, i); } } } PUT(ac, x.n0); PAD(zeroes, x.nz0) PUT(x.s, x.n1); PAD(zeroes, x.nz1); PUT(x.s + x.n1, x.n2); PAD(zeroes, x.nz2); if (x.flags & FLAGS_MINUS) { PAD(spaces, x.width); } } fmt = s + 1; } return 0; } void _Putfld(_Pft *px, va_list *pap, char code, char *ac) { px->n0 = px->nz0 = px->n1 = px->nz1 = px->n2 = px->nz2 = 0; switch (code) { case 'c': ac[px->n0++] = va_arg(*pap, int); break; case 'd': case 'i': if (px->qual == 'l') { px->v.ll = va_arg(*pap, long); } else if (px->qual == 'L') { px->v.ll = va_arg(*pap, long long); } else { px->v.ll = va_arg(*pap, int); } if (px->qual == 'h') { px->v.ll = (short)px->v.ll; } if (px->v.ll < 0) { ac[px->n0++] = '-'; } else if (px->flags & FLAGS_PLUS) { ac[px->n0++] = '+'; } else if (px->flags & FLAGS_SPACE) { ac[px->n0++] = ' '; } px->s = (char *)&ac[px->n0]; _Litob(px, code); break; case 'x': case 'X': case 'u': case 'o': if (px->qual == 'l') { px->v.ll = va_arg(*pap, long); } else if (px->qual == 'L') { px->v.ll = va_arg(*pap, long long); } else { px->v.ll = va_arg(*pap, int); } if (px->qual == 'h') { px->v.ll = (unsigned short)px->v.ll; } else if (px->qual == 0) { px->v.ll = (unsigned int)px->v.ll; } if (px->flags & FLAGS_HASH) { ac[px->n0++] = '0'; if (code == 'x' || code == 'X') { ac[px->n0++] = code; } } px->s = (char *)&ac[px->n0]; _Litob(px, code); break; case 'e': case 'f': case 'g': case 'E': case 'G': px->v.ld = px->qual == 'L' ? va_arg(*pap, ldouble) : va_arg(*pap, double); if (LDSIGN(px->v.ld)) ac[px->n0++] = '-'; else if (px->flags & FLAGS_PLUS) ac[px->n0++] = '+'; else if (px->flags & FLAGS_SPACE) ac[px->n0++] = ' '; px->s = (char *)&ac[px->n0]; _Ldtob(px, code); break; case 'n': if (px->qual == 'h') { *va_arg(*pap, unsigned short *) = px->nchar; } else if (px->qual == 'l') { *va_arg(*pap, unsigned long *) = px->nchar; } else if (px->qual == 'L') { *va_arg(*pap, unsigned long long *) = px->nchar; } else { *va_arg(*pap, unsigned int *) = px->nchar; } break; case 'p': px->v.ll = (long)va_arg(*pap, void *); px->s = (char *)&ac[px->n0]; _Litob(px, 'x'); break; case 's': px->s = va_arg(*pap, char *); px->n1 = strlen(px->s); if (px->prec >= 0 && px->prec < px->n1) { px->n1 = px->prec; } break; case '%': ac[px->n0++] = '%'; break; default: ac[px->n0++] = code; break; } }