| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235 |
- /* tst-strptime2 - Test strptime %z timezone offset specifier. */
- #include <limits.h>
- #include <stdbool.h>
- #include <stdio.h>
- #include <time.h>
- #include <libc-diag.h>
- /* Dummy string is used to match strptime's %s specifier. */
- static const char dummy_string[] = "1113472456";
- /* buffer_size contains the maximum test string length, including
- trailing NUL. */
- enum
- {
- buffer_size = 20,
- };
- /* Verbose execution, set with --verbose command line option. */
- static bool verbose;
- /* mkbuf - Write a test string for strptime with the specified time
- value and number of digits into the supplied buffer, and return
- the expected strptime test result.
- The test string, buf, is written with the following content:
- a dummy string matching strptime "%s" format specifier,
- whitespace matching strptime " " format specifier, and
- timezone string matching strptime "%z" format specifier.
- Note that a valid timezone string is either "Z" or contains the
- following fields:
- Sign field consisting of a '+' or '-' sign,
- Hours field in two decimal digits, and
- optional Minutes field in two decimal digits. Optionally,
- a ':' is used to separate hours and minutes.
- This function may write test strings with minutes values outside
- the valid range 00-59. These are invalid strings and useful for
- testing strptime's rejection of invalid strings.
- The ndigits parameter is used to limit the number of timezone
- string digits to be written and may range from 0 to 4. Note that
- only 2 and 4 digit strings are valid input to strptime; strings
- with 0, 1 or 3 digits are invalid and useful for testing strptime's
- rejection of invalid strings.
- This function returns the behavior expected of strptime resulting
- from parsing the the test string. For valid strings, the function
- returns the expected tm_gmtoff value. For invalid strings,
- LONG_MAX is returned. LONG_MAX indicates the expectation that
- strptime will return NULL; for example, if the number of digits
- are not correct, or minutes part of the time is outside the valid
- range of 00 to 59. */
- static long int
- mkbuf (char *buf, bool neg, bool colon, unsigned int hhmm, size_t ndigits)
- {
- const int mm_max = 59;
- char sign = neg ? '-' : '+';
- int i;
- unsigned int hh = hhmm / 100;
- unsigned int mm = hhmm % 100;
- long int expect = LONG_MAX;
- i = sprintf (buf, "%s %c", dummy_string, sign);
- #if __GNUC_PREREQ (7, 0)
- /* GCC issues a warning when it thinks the snprintf buffer may be too short.
- This test is explicitly using short buffers to force snprintf to truncate
- the output so we ignore the warnings. */
- DIAG_PUSH_NEEDS_COMMENT;
- DIAG_IGNORE_NEEDS_COMMENT (7.0, "-Wformat-truncation");
- #endif
- if (colon)
- snprintf (buf + i, ndigits + 2, "%02u:%02u", hh, mm);
- else
- snprintf (buf + i, ndigits + 1, "%04u", hhmm);
- #if __GNUC_PREREQ (7, 0)
- DIAG_POP_NEEDS_COMMENT;
- #endif
- if (mm <= mm_max && (ndigits == 2 || ndigits == 4))
- {
- long int tm_gmtoff = hh * 3600 + mm * 60;
- expect = neg ? -tm_gmtoff : tm_gmtoff;
- }
- return expect;
- }
- /* Write a description of expected or actual test result to stdout. */
- static void
- describe (bool string_valid, long int tm_gmtoff)
- {
- if (string_valid)
- printf ("valid, tm.tm_gmtoff %ld", tm_gmtoff);
- else
- printf ("invalid, return value NULL");
- }
- /* Using buffer buf, run strptime. Compare results against expect,
- the expected result. Report failures and verbose results to stdout.
- Update the result counts. Return 1 if test failed, 0 if passed. */
- static int
- compare (const char *buf, long int expect, unsigned int *nresult)
- {
- struct tm tm;
- char *p;
- bool test_string_valid;
- long int test_result;
- bool fail;
- int result;
- p = strptime (buf, "%s %z", &tm);
- test_string_valid = p != NULL;
- test_result = test_string_valid ? tm.tm_gmtoff : LONG_MAX;
- fail = test_result != expect;
- if (fail || verbose)
- {
- bool expect_string_valid = expect != LONG_MAX;
- printf ("%s: input \"%s\", expected: ", fail ? "FAIL" : "PASS", buf);
- describe (expect_string_valid, expect);
- if (fail)
- {
- printf (", got: ");
- describe (test_string_valid, test_result);
- }
- printf ("\n");
- }
- result = fail ? 1 : 0;
- nresult[result]++;
- return result;
- }
- static int
- do_test (void)
- {
- char buf[buffer_size];
- long int expect;
- int result = 0;
- /* Number of tests run with passing (index==0) and failing (index==1)
- results. */
- unsigned int nresult[2];
- unsigned int ndigits;
- unsigned int step;
- unsigned int hhmm;
- nresult[0] = 0;
- nresult[1] = 0;
- /* Create and test input string with no sign and four digits input
- (invalid format). */
- sprintf (buf, "%s 1030", dummy_string);
- expect = LONG_MAX;
- result |= compare (buf, expect, nresult);
- /* Create and test input string with "Z" input (valid format).
- Expect tm_gmtoff of 0. */
- sprintf (buf, "%s Z", dummy_string);
- expect = 0;
- result |= compare (buf, expect, nresult);
- /* Create and test input strings with sign and digits:
- 0 digits (invalid format),
- 1 digit (invalid format),
- 2 digits (valid format),
- 3 digits (invalid format),
- 4 digits (valid format if and only if minutes is in range 00-59,
- otherwise invalid).
- If format is valid, the returned tm_gmtoff is checked. */
- for (ndigits = 0, step = 10000; ndigits <= 4; ndigits++, step /= 10)
- for (hhmm = 0; hhmm <= 9999; hhmm += step)
- {
- /* Test both positive and negative signs. */
- expect = mkbuf (buf, false, false, hhmm, ndigits);
- result |= compare (buf, expect, nresult);
- expect = mkbuf (buf, true, false, hhmm, ndigits);
- result |= compare (buf, expect, nresult);
- /* Test with colon as well. */
- if (ndigits >= 3)
- {
- expect = mkbuf (buf, false, true, hhmm, ndigits);
- result |= compare (buf, expect, nresult);
- expect = mkbuf (buf, true, true, hhmm, ndigits);
- result |= compare (buf, expect, nresult);
- }
- }
- if (result > 0 || verbose)
- printf ("%s: %u input strings: %u fail, %u pass\n",
- result > 0 ? "FAIL" : "PASS",
- nresult[1] + nresult[0], nresult[1], nresult[0]);
- return result;
- }
- /* Add a "--verbose" command line option to test-skeleton.c. */
- #define OPT_VERBOSE 10000
- #define CMDLINE_OPTIONS \
- { "verbose", no_argument, NULL, OPT_VERBOSE, },
- #define CMDLINE_PROCESS \
- case OPT_VERBOSE: \
- verbose = true; \
- break;
- #define TEST_FUNCTION do_test ()
- #include "../test-skeleton.c"
|