1 /* tst-strptime2 - Test strptime %z timezone offset specifier. */
7 #include <libc-internal.h>
9 /* Dummy string is used to match strptime's %s specifier. */
11 static const char dummy_string
[] = "1113472456";
13 /* buffer_size contains the maximum test string length, including
21 /* Verbose execution, set with --verbose command line option. */
26 /* mkbuf - Write a test string for strptime with the specified time
27 value and number of digits into the supplied buffer, and return
28 the expected strptime test result.
30 The test string, buf, is written with the following content:
31 a dummy string matching strptime "%s" format specifier,
32 whitespace matching strptime " " format specifier, and
33 timezone string matching strptime "%z" format specifier.
35 Note that a valid timezone string is either "Z" or contains the
37 Sign field consisting of a '+' or '-' sign,
38 Hours field in two decimal digits, and
39 optional Minutes field in two decimal digits. Optionally,
40 a ':' is used to seperate hours and minutes.
42 This function may write test strings with minutes values outside
43 the valid range 00-59. These are invalid strings and useful for
44 testing strptime's rejection of invalid strings.
46 The ndigits parameter is used to limit the number of timezone
47 string digits to be written and may range from 0 to 4. Note that
48 only 2 and 4 digit strings are valid input to strptime; strings
49 with 0, 1 or 3 digits are invalid and useful for testing strptime's
50 rejection of invalid strings.
52 This function returns the behavior expected of strptime resulting
53 from parsing the the test string. For valid strings, the function
54 returns the expected tm_gmtoff value. For invalid strings,
55 LONG_MAX is returned. LONG_MAX indicates the expectation that
56 strptime will return NULL; for example, if the number of digits
57 are not correct, or minutes part of the time is outside the valid
61 mkbuf (char *buf
, bool neg
, bool colon
, unsigned int hhmm
, size_t ndigits
)
63 const int mm_max
= 59;
64 char sign
= neg
? '-' : '+';
66 unsigned int hh
= hhmm
/ 100;
67 unsigned int mm
= hhmm
% 100;
68 long int expect
= LONG_MAX
;
70 i
= sprintf (buf
, "%s %c", dummy_string
, sign
);
71 #if __GNUC_PREREQ (7, 0)
72 /* GCC issues a warning when it thinks the snprintf buffer may be too short.
73 This test is explicitly using short buffers to force snprintf to truncate
74 the output so we ignore the warnings. */
75 DIAG_PUSH_NEEDS_COMMENT
;
76 DIAG_IGNORE_NEEDS_COMMENT (7.0, "-Wformat-length");
79 snprintf (buf
+ i
, ndigits
+ 2, "%02u:%02u", hh
, mm
);
81 snprintf (buf
+ i
, ndigits
+ 1, "%04u", hhmm
);
82 #if __GNUC_PREREQ (7, 0)
83 DIAG_POP_NEEDS_COMMENT
;
86 if (mm
<= mm_max
&& (ndigits
== 2 || ndigits
== 4))
88 long int tm_gmtoff
= hh
* 3600 + mm
* 60;
90 expect
= neg
? -tm_gmtoff
: tm_gmtoff
;
97 /* Write a description of expected or actual test result to stdout. */
100 describe (bool string_valid
, long int tm_gmtoff
)
103 printf ("valid, tm.tm_gmtoff %ld", tm_gmtoff
);
105 printf ("invalid, return value NULL");
109 /* Using buffer buf, run strptime. Compare results against expect,
110 the expected result. Report failures and verbose results to stdout.
111 Update the result counts. Return 1 if test failed, 0 if passed. */
114 compare (const char *buf
, long int expect
, unsigned int *nresult
)
118 bool test_string_valid
;
119 long int test_result
;
123 p
= strptime (buf
, "%s %z", &tm
);
124 test_string_valid
= p
!= NULL
;
125 test_result
= test_string_valid
? tm
.tm_gmtoff
: LONG_MAX
;
126 fail
= test_result
!= expect
;
130 bool expect_string_valid
= expect
!= LONG_MAX
;
132 printf ("%s: input \"%s\", expected: ", fail
? "FAIL" : "PASS", buf
);
133 describe (expect_string_valid
, expect
);
138 describe (test_string_valid
, test_result
);
144 result
= fail
? 1 : 0;
154 char buf
[buffer_size
];
157 /* Number of tests run with passing (index==0) and failing (index==1)
159 unsigned int nresult
[2];
160 unsigned int ndigits
;
167 /* Create and test input string with no sign and four digits input
170 sprintf (buf
, "%s 1030", dummy_string
);
172 result
|= compare (buf
, expect
, nresult
);
174 /* Create and test input string with "Z" input (valid format).
175 Expect tm_gmtoff of 0. */
177 sprintf (buf
, "%s Z", dummy_string
);
179 result
|= compare (buf
, expect
, nresult
);
181 /* Create and test input strings with sign and digits:
182 0 digits (invalid format),
183 1 digit (invalid format),
184 2 digits (valid format),
185 3 digits (invalid format),
186 4 digits (valid format if and only if minutes is in range 00-59,
188 If format is valid, the returned tm_gmtoff is checked. */
190 for (ndigits
= 0, step
= 10000; ndigits
<= 4; ndigits
++, step
/= 10)
191 for (hhmm
= 0; hhmm
<= 9999; hhmm
+= step
)
193 /* Test both positive and negative signs. */
195 expect
= mkbuf (buf
, false, false, hhmm
, ndigits
);
196 result
|= compare (buf
, expect
, nresult
);
198 expect
= mkbuf (buf
, true, false, hhmm
, ndigits
);
199 result
|= compare (buf
, expect
, nresult
);
201 /* Test with colon as well. */
205 expect
= mkbuf (buf
, false, true, hhmm
, ndigits
);
206 result
|= compare (buf
, expect
, nresult
);
208 expect
= mkbuf (buf
, true, true, hhmm
, ndigits
);
209 result
|= compare (buf
, expect
, nresult
);
213 if (result
> 0 || verbose
)
214 printf ("%s: %u input strings: %u fail, %u pass\n",
215 result
> 0 ? "FAIL" : "PASS",
216 nresult
[1] + nresult
[0], nresult
[1], nresult
[0]);
222 /* Add a "--verbose" command line option to test-skeleton.c. */
224 #define OPT_VERBOSE 10000
226 #define CMDLINE_OPTIONS \
227 { "verbose", no_argument, NULL, OPT_VERBOSE, },
229 #define CMDLINE_PROCESS \
234 #define TEST_FUNCTION do_test ()
235 #include "../test-skeleton.c"