1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
|
/* Test of sprintf_min(), invalid format.
$Id: sprintf_min-inv.c,v 1.2.2.1 2008/03/20 21:42:32 joerg_wunsch Exp $ */
#ifndef __AVR__
# define PRINTFLN(line, fmt, ...) \
printf("\nLine %2d: " fmt "\n", line, ##__VA_ARGS__)
# define EXIT(code) exit ((code) < 255 ? (code) : 255)
# define sprintf_P sprintf
#else
# if defined(__AVR_ATmega128__)
/* ATmega128 has enough RAM for sprintf(), print to 0x2000 in XRAM. */
# define PRINTFLN(line, fmt, ...) \
sprintf ((char *)0x2000, "\nLine %d: " fmt "\n", line, ##__VA_ARGS__)
# else
/* small AVR */
# define PRINTFLN(args...)
# endif
# define EXIT exit
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "progmem.h"
void Check (int line,
int explen,
const char *expstr,
int retval, const char *retstr)
{
int code;
if (retval != explen)
code = 1000 + line;
else if (strcmp_P (retstr, expstr))
code = line;
else
return;
PRINTFLN (line, "expect: %3d, \"%s\",\n%8s output: %3d, \"%s\"\n",
explen, expstr, " ", retval, retstr);
#ifdef DEBUG
code = (int)retstr;
#endif
EXIT (code);
}
/* 'vp' is used to avoid gcc warnings about format string. */
#define CHECK(explen, expstr, fmt, ...) do { \
char s[260]; \
int i; \
int (* volatile vp)(char *, const char *, ...) = sprintf_P; \
memset (s, 0, sizeof(s)); \
i = vp (s, PSTR(fmt), ##__VA_ARGS__); \
Check (__LINE__, explen, PSTR(expstr), i, s); \
} while (0)
int main ()
{
/* Format last symbol is absent */
#ifdef __AVR__ /* Glibc return -1, if '%' has not a format spec. */
CHECK (0, "", "%");
CHECK (1, ".", ".%");
CHECK (1, ".", ".%0");
CHECK (6, "12345 ", "%d %+l", 12345);
#endif
/* 'hh' length modifier is possible, but it is ignored in avr-libc */
CHECK (3, "123", "%hhd", 123);
/* Length modifier and conversion mismatch: ignored */
CHECK (1, "a", "%hhc", 'a');
CHECK (1, "b", "%hc", 'b');
CHECK (3, "foo", "%hhs", "foo");
CHECK (3, "foo", "%hs", "foo");
/* If the `space' and `+' flags both appear, the `space' flag
is ignored. */
#ifdef __AVR__
CHECK (5, "10.11", "% +d.%+ d", 10, 11); /* PRINTF_MIN: both ignored */
#else
CHECK (7, "+10.+11", "% +d.%+ d", 10, 11);
#endif
#ifdef __AVR__
/* Unknown format: glibc output is "%b", return value 2. */
CHECK (0, "", "%b");
/* Two precision dots. */
// CHECK (-1, "", "%8..4d", 1);
CHECK (1, "1", "%8..4d", 1); /* PRINTF_MIN: not checked */
/* Long long arg */
CHECK (0, "", "%lld", 1LL);
/* wint_t, wchar_t: ignore */
CHECK (1, "c", "%lc", 'c');
CHECK (3, "foo", "%ls", "foo");
/* The asterisk `*' in width or precision. */
CHECK (0, "", "%*d", 10, 12345);
CHECK (0, "", "%10.*d", 6, 12345);
CHECK (0, "", "%*.*d", 10, 6, 12345);
/* Float numbers: are skipped */
CHECK (1, "?", "%e", 0.0);
CHECK (23, "? e ? f ? g ? E ? F ? G",
"%e %c %e %c %e %c %e %c %e %c %e %c",
1.0, 'e', 2.0, 'f', 3.0, 'g', 4.0, 'E', 5.0, 'F', 6.0, 'G');
CHECK (14, "? 10 ? 11 ? 12",
"%+0e %d %-10f %d % 10.5g %d",
10.0, 10, 11.0, 11, 12.0, 12);
#endif
return 0;
}
|