summaryrefslogtreecommitdiff
path: root/time/tst-strptime2.c
diff options
context:
space:
mode:
Diffstat (limited to 'time/tst-strptime2.c')
-rw-r--r--time/tst-strptime2.c237
1 files changed, 201 insertions, 36 deletions
diff --git a/time/tst-strptime2.c b/time/tst-strptime2.c
index a08e6d7cb7..7fe7350024 100644
--- a/time/tst-strptime2.c
+++ b/time/tst-strptime2.c
@@ -1,59 +1,224 @@
+/* tst-strptime2 - Test strptime %z timezone offset specifier. */
+
#include <limits.h>
+#include <stdbool.h>
#include <stdio.h>
#include <time.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;
+
-static const struct
+/* 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 seperate 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 char *fmt;
- long int gmtoff;
-} tests[] =
- {
- { "1113472456 +1000", 36000 },
- { "1113472456 -1000", -36000 },
- { "1113472456 +10", 36000 },
- { "1113472456 -10", -36000 },
- { "1113472456 +1030", 37800 },
- { "1113472456 -1030", -37800 },
- { "1113472456 +0030", 1800 },
- { "1113472456 -0030", -1800 },
- { "1113472456 -1330", LONG_MAX },
- { "1113472456 +1330", LONG_MAX },
- { "1113472456 -1060", LONG_MAX },
- { "1113472456 +1060", LONG_MAX },
- { "1113472456 1030", LONG_MAX },
- };
-#define ntests (sizeof (tests) / sizeof (tests[0]))
+ 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 (colon)
+ snprintf (buf + i, ndigits + 2, "%02u:%02u", hh, mm);
+ else
+ snprintf (buf + i, ndigits + 1, "%04u", hhmm);
+
+ 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
-do_test (void)
+compare (const char *buf, long int expect, unsigned int *nresult)
{
- int result = 0;
+ 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;
- for (int i = 0; i < ntests; ++i)
+ if (fail || verbose)
{
- struct tm tm;
+ bool expect_string_valid = expect != LONG_MAX;
- if (strptime (tests[i].fmt, "%s %z", &tm) == NULL)
- {
- if (tests[i].gmtoff != LONG_MAX)
- {
- printf ("round %d: strptime unexpectedly failed\n", i);
- result = 1;
- }
- continue;
- }
+ printf ("%s: input \"%s\", expected: ", fail ? "FAIL" : "PASS", buf);
+ describe (expect_string_valid, expect);
- if (tm.tm_gmtoff != tests[i].gmtoff)
+ if (fail)
{
- printf ("round %d: tm_gmtoff is %ld\n", i, (long int) tm.tm_gmtoff);
- result = 1;
+ 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"