184 lines
3.7 KiB
C
184 lines
3.7 KiB
C
|
#include <u.h>
|
||
|
#include <libc.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
/*
|
||
|
* try all combination of flags and float conversions
|
||
|
* with some different widths & precisions
|
||
|
*/
|
||
|
|
||
|
#define Njust 2
|
||
|
#define Nplus 3
|
||
|
#define Nalt 2
|
||
|
#define Nzero 2
|
||
|
#define Nspec 5
|
||
|
#define Nwidth 5
|
||
|
#define Nprec 5
|
||
|
|
||
|
static double fmtvals[] = {
|
||
|
3.1415925535897932e15,
|
||
|
3.1415925535897932e14,
|
||
|
3.1415925535897932e13,
|
||
|
3.1415925535897932e12,
|
||
|
3.1415925535897932e11,
|
||
|
3.1415925535897932e10,
|
||
|
3.1415925535897932e9,
|
||
|
3.1415925535897932e8,
|
||
|
3.1415925535897932e7,
|
||
|
3.1415925535897932e6,
|
||
|
3.1415925535897932e5,
|
||
|
3.1415925535897932e4,
|
||
|
3.1415925535897932e3,
|
||
|
3.1415925535897932e2,
|
||
|
3.1415925535897932e1,
|
||
|
3.1415925535897932e0,
|
||
|
3.1415925535897932e-1,
|
||
|
3.1415925535897932e-2,
|
||
|
3.1415925535897932e-3,
|
||
|
3.1415925535897932e-4,
|
||
|
3.1415925535897932e-5,
|
||
|
3.1415925535897932e-6,
|
||
|
3.1415925535897932e-7,
|
||
|
3.1415925535897932e-8,
|
||
|
3.1415925535897932e-9,
|
||
|
3.1415925535897932e-10,
|
||
|
3.1415925535897932e-11,
|
||
|
3.1415925535897932e-12,
|
||
|
3.1415925535897932e-13,
|
||
|
3.1415925535897932e-14,
|
||
|
3.1415925535897932e-15,
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* are the numbers close?
|
||
|
* used to compare long numbers where the last few digits are garbage
|
||
|
* due to precision problems
|
||
|
*/
|
||
|
static int
|
||
|
numclose(char *num1, char *num2)
|
||
|
{
|
||
|
int ndig;
|
||
|
double d1, d2;
|
||
|
enum { MAXDIG = 15 };
|
||
|
|
||
|
d1 = fmtstrtod(num1, 0);
|
||
|
d2 = fmtstrtod(num2, 0);
|
||
|
if(d1 != d2)
|
||
|
return 0;
|
||
|
|
||
|
ndig = 0;
|
||
|
while (*num1) {
|
||
|
if (*num1 >= '0' && *num1 <= '9') {
|
||
|
ndig++;
|
||
|
if (ndig > MAXDIG) {
|
||
|
if (!(*num2 >= '0' && *num2 <= '9')) {
|
||
|
return 0;
|
||
|
}
|
||
|
} else if (*num1 != *num2) {
|
||
|
return 0;
|
||
|
}
|
||
|
} else if (*num1 != *num2) {
|
||
|
return 0;
|
||
|
} else if (*num1 == 'e' || *num1 == 'E') {
|
||
|
ndig = 0;
|
||
|
}
|
||
|
num1++;
|
||
|
num2++;
|
||
|
}
|
||
|
if (*num1 || !num2)
|
||
|
return 0;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
doit(int just, int plus, int alt, int zero, int width, int prec, int spec)
|
||
|
{
|
||
|
char format[256];
|
||
|
char *p;
|
||
|
const char *s;
|
||
|
int i;
|
||
|
|
||
|
p = format;
|
||
|
*p++ = '%';
|
||
|
if (just > 0)
|
||
|
*p++ = "-"[just - 1];
|
||
|
if (plus > 0)
|
||
|
*p++ = "+ "[plus - 1];
|
||
|
if (alt > 0)
|
||
|
*p++ = "#"[alt - 1];
|
||
|
if (zero > 0)
|
||
|
*p++ = "0"[zero - 1];
|
||
|
|
||
|
s = "";
|
||
|
switch (width) {
|
||
|
case 1: s = "1"; break;
|
||
|
case 2: s = "5"; break;
|
||
|
case 3: s = "10"; break;
|
||
|
case 4: s = "15"; break;
|
||
|
}
|
||
|
strcpy(p, s);
|
||
|
|
||
|
s = "";
|
||
|
switch (prec) {
|
||
|
case 1: s = ".0"; break;
|
||
|
case 2: s = ".2"; break;
|
||
|
case 3: s = ".5"; break;
|
||
|
case 4: s = ".15"; break;
|
||
|
}
|
||
|
strcat(p, s);
|
||
|
|
||
|
p = strchr(p, '\0');
|
||
|
*p++ = "efgEG"[spec];
|
||
|
*p = '\0';
|
||
|
|
||
|
for (i = 0; i < sizeof(fmtvals) / sizeof(fmtvals[0]); i++) {
|
||
|
char ref[1024], buf[1024];
|
||
|
Rune rbuf[1024];
|
||
|
double d1, d2;
|
||
|
|
||
|
sprintf(ref, format, fmtvals[i]);
|
||
|
snprint(buf, sizeof(buf), format, fmtvals[i]);
|
||
|
if (strcmp(ref, buf) != 0
|
||
|
&& !numclose(ref, buf)) {
|
||
|
d1 = fmtstrtod(ref, 0);
|
||
|
d2 = fmtstrtod(buf, 0);
|
||
|
fprintf(stderr, "%s: ref='%s'%s fmt='%s'%s\n",
|
||
|
format,
|
||
|
ref, d1==fmtvals[i] ? "" : " (ref is inexact!)",
|
||
|
buf, d2==fmtvals[i] ? "" : " (fmt is inexact!)");
|
||
|
// exits("oops");
|
||
|
}
|
||
|
|
||
|
/* Check again with output to rune string */
|
||
|
runesnprint(rbuf, 1024, format, fmtvals[i]);
|
||
|
snprint(buf, sizeof(buf), "%S", rbuf);
|
||
|
if (strcmp(ref, buf) != 0
|
||
|
&& !numclose(ref, buf)) {
|
||
|
d1 = fmtstrtod(ref, 0);
|
||
|
d2 = fmtstrtod(buf, 0);
|
||
|
fprintf(stderr, "%s: ref='%s'%s fmt='%s'%s\n",
|
||
|
format,
|
||
|
ref, d1==fmtvals[i] ? "" : " (ref is inexact!)",
|
||
|
buf, d2==fmtvals[i] ? "" : " (fmt is inexact!)");
|
||
|
// exits("oops");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
main(int argc, char **argv)
|
||
|
{
|
||
|
int just, plus, alt, zero, width, prec, spec;
|
||
|
|
||
|
for (just = 0; just < Njust; just++)
|
||
|
for (plus = 0; plus < Nplus; plus++)
|
||
|
for (alt = 0; alt < Nalt; alt++)
|
||
|
for (zero = 0; zero < Nzero; zero++)
|
||
|
for (width = 0; width < Nwidth; width++)
|
||
|
for (prec = 0; prec < Nprec; prec++)
|
||
|
for (spec = 0; spec < Nspec; spec++)
|
||
|
doit(just, plus, alt, zero, width, prec, spec);
|
||
|
|
||
|
exits(0);
|
||
|
}
|