time, string, spawn tests from libc-testsuit
authornsz <nsz@port70.net>
Fri, 5 Aug 2011 00:51:42 +0000 (02:51 +0200)
committernsz <nsz@port70.net>
Fri, 5 Aug 2011 00:51:42 +0000 (02:51 +0200)
src/process/Makefile [new file with mode: 0644]
src/process/spawn.c [new file with mode: 0644]
src/string/Makefile [new file with mode: 0644]
src/string/string.c [new file with mode: 0644]
src/time/Makefile [new file with mode: 0644]
src/time/time.c [new file with mode: 0644]

diff --git a/src/process/Makefile b/src/process/Makefile
new file mode 100644 (file)
index 0000000..ee4552b
--- /dev/null
@@ -0,0 +1 @@
+include ../../Makefile.inc
diff --git a/src/process/spawn.c b/src/process/spawn.c
new file mode 100644 (file)
index 0000000..18b8e27
--- /dev/null
@@ -0,0 +1,38 @@
+#define _XOPEN_SOURCE 700
+#include <stdlib.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <errno.h>
+#include <string.h>
+#include <spawn.h>
+#include <sys/wait.h>
+#include "test.h"
+
+#define TEST(r, f, x, m) ( \
+       ((r) = (f)) == (x) || \
+       (error("%s failed (" m ")\n", #f, r, x), 0) )
+
+#define TEST_E(f) ( (errno = 0), (f) || \
+       (error("%s failed (errno = %d \"%s\")\n", #f, errno, strerror(errno)), 0) )
+
+void test_spawn(void) {
+       int r;
+       char foo[10];
+       int p[2];
+       pid_t pid;
+       int status;
+       posix_spawnattr_t attr;
+       posix_spawn_file_actions_t fa;
+
+       TEST_E(!pipe(p));
+       TEST(r, posix_spawn_file_actions_init(&fa), 0, "%d != %d");
+       TEST(r, posix_spawn_file_actions_addclose(&fa, p[0]), 0, "%d != %d");
+       TEST(r, posix_spawn_file_actions_adddup2(&fa, p[1], 1), 0, "%d != %d");
+       TEST(r, posix_spawn_file_actions_addclose(&fa, p[1]), 0, "%d != %d");
+       TEST(r, posix_spawnp(&pid, "echo", &fa, 0, (char *[]){"echo","hello",0}, 0), 0, "%d != %d");
+       close(p[1]);
+       TEST(r, waitpid(pid, &status, 0), pid, "%d != %d");
+       TEST(r, read(p[0], foo, sizeof foo), 6, "%d != %d");
+       close(p[0]);
+       TEST(r, posix_spawn_file_actions_destroy(&fa), 0, "%d != %d");
+}
diff --git a/src/string/Makefile b/src/string/Makefile
new file mode 100644 (file)
index 0000000..ee4552b
--- /dev/null
@@ -0,0 +1 @@
+include ../../Makefile.inc
diff --git a/src/string/string.c b/src/string/string.c
new file mode 100644 (file)
index 0000000..3489ba0
--- /dev/null
@@ -0,0 +1,114 @@
+#define _BSD_SOURCE
+#include <stdio.h>
+#include <string.h>
+#include "test.h"
+
+/* r = place to store result
+ * f = function call to test (or any expression)
+ * x = expected result
+ * m = message to print on failure (with formats for r & x)
+**/
+
+#define TEST(r, f, x, m) ( \
+       ((r) = (f)) == (x) || \
+       (error("%s failed (" m ")\n", #f, r, x), 0) )
+
+#define TEST_S(s, x, m) ( \
+       !strcmp((s),(x)) || \
+       (error("[%s] != [%s] (%s)\n", s, x, m), 0) )
+
+void test_string(void) {
+       char b[32];
+       char *s;
+       int i;
+
+       b[16]='a'; b[17]='b'; b[18]='c'; b[19]=0;
+       TEST(s, strcpy(b, b+16), b, "wrong return %p != %p");
+       TEST_S(s, "abc", "strcpy gave incorrect string");
+       TEST(s, strcpy(b+1, b+16), b+1, "wrong return %p != %p");
+       TEST_S(s, "abc", "strcpy gave incorrect string");
+       TEST(s, strcpy(b+2, b+16), b+2, "wrong return %p != %p");
+       TEST_S(s, "abc", "strcpy gave incorrect string");
+       TEST(s, strcpy(b+3, b+16), b+3, "wrong return %p != %p");
+       TEST_S(s, "abc", "strcpy gave incorrect string");
+
+       TEST(s, strcpy(b+1, b+17), b+1, "wrong return %p != %p");
+       TEST_S(s, "bc", "strcpy gave incorrect string");
+       TEST(s, strcpy(b+2, b+18), b+2, "wrong return %p != %p");
+       TEST_S(s, "c", "strcpy gave incorrect string");
+       TEST(s, strcpy(b+3, b+19), b+3, "wrong return %p != %p");
+       TEST_S(s, "", "strcpy gave incorrect string");
+
+       TEST(s, memset(b, 'x', sizeof b), b, "wrong return %p != %p");
+       TEST(s, strncpy(b, "abc", sizeof b - 1), b, "wrong return %p != %p");
+       TEST(i, memcmp(b, "abc\0\0\0\0", 8), 0, "strncpy fails to zero-pad dest");
+       TEST(i, b[sizeof b - 1], 'x', "strncpy overruns buffer when n > strlen(src)");
+
+       b[3] = 'x'; b[4] = 0;
+       strncpy(b, "abc", 3);
+       TEST(i, b[2], 'c', "strncpy fails to copy last byte: %hhu != %hhu");
+       TEST(i, b[3], 'x', "strncpy overruns buffer to null-terminate: %hhu != %hhu");
+
+       TEST(i, !strncmp("abcd", "abce", 3), 1, "strncmp compares past n");
+       TEST(i, !!strncmp("abc", "abd", 3), 1, "strncmp fails to compare n-1st byte");
+
+       strcpy(b, "abc");
+       TEST(s, strncat(b, "123456", 3), b, "%p != %p");
+       TEST(i, b[6], 0, "strncat failed to null-terminate (%d)");
+       TEST_S(s, "abc123", "strncat gave incorrect string");
+
+       strcpy(b, "aaababccdd0001122223");
+       TEST(s, strchr(b, 'b'), b+3, "%p != %p");
+       TEST(s, strrchr(b, 'b'), b+5, "%p != %p");
+       TEST(i, strspn(b, "abcd"), 10, "%d != %d");
+       TEST(i, strcspn(b, "0123"), 10, "%d != %d");
+       TEST(s, strpbrk(b, "0123"), b+10, "%d != %d");
+
+       strcpy(b, "abc   123; xyz; foo");
+       TEST(s, strtok(b, " "), b, "%p != %p");
+       TEST_S(s, "abc", "strtok result");
+
+       TEST(s, strtok(NULL, ";"), b+4, "%p != %p");
+       TEST_S(s, "  123", "strtok result");
+
+       TEST(s, strtok(NULL, " ;"), b+11, "%p != %p");
+       TEST_S(s, "xyz", "strtok result");
+
+       TEST(s, strtok(NULL, " ;"), b+16, "%p != %p");
+       TEST_S(s, "foo", "strtok result");
+
+#ifdef HAVE_BSD_STRL
+       memset(b, 'x', sizeof b);
+       TEST(i, strlcpy(b, "abc", sizeof b - 1), 3, "length %d != %d");
+       TEST(i, b[3], 0, "strlcpy did not null-terminate short string (%d)");
+       TEST(i, b[4], 'x', "strlcpy wrote extra bytes (%d)");
+
+       memset(b, 'x', sizeof b);
+       TEST(i, strlcpy(b, "abc", 2), 3, "length %d != %d");
+       TEST(i, b[0], 'a', "strlcpy did not copy character %d");
+       TEST(i, b[1], 0, "strlcpy did not null-terminate long string (%d)");
+
+       memset(b, 'x', sizeof b);
+       TEST(i, strlcpy(b, "abc", 3), 3, "length %d != %d");
+       TEST(i, b[2], 0, "strlcpy did not null-terminate l-length string (%d)");
+
+       TEST(i, strlcpy(NULL, "abc", 0), 3, "length %d != %d");
+
+       memcpy(b, "abc\0\0\0x\0", 8);
+       TEST(i, strlcat(b, "123", sizeof b), 6, "length %d != %d");
+       TEST_S(b, "abc123", "strlcat result");
+
+       memcpy(b, "abc\0\0\0x\0", 8);
+       TEST(i, strlcat(b, "123", 6), 6, "length %d != %d");
+       TEST_S(b, "abc12", "strlcat result");
+       TEST(i, b[6], 'x', "strlcat wrote past string %d != %d");
+
+       memcpy(b, "abc\0\0\0x\0", 8);
+       TEST(i, strlcat(b, "123", 4), 6, "length %d != %d");
+       TEST_S(b, "abc", "strlcat result");
+
+       memcpy(b, "abc\0\0\0x\0", 8);
+       TEST(i, strlcat(b, "123", 3), 6, "length %d != %d");
+       TEST_S(b, "abc", "strlcat result");
+#endif
+}
diff --git a/src/time/Makefile b/src/time/Makefile
new file mode 100644 (file)
index 0000000..ee4552b
--- /dev/null
@@ -0,0 +1 @@
+include ../../Makefile.inc
diff --git a/src/time/time.c b/src/time/time.c
new file mode 100644 (file)
index 0000000..a06aac7
--- /dev/null
@@ -0,0 +1,74 @@
+#define _XOPEN_SOURCE 700
+#include <stdlib.h>
+#include <stdio.h>
+#include <time.h>
+#include "test.h"
+
+/* We use this instead of memcmp because some broken C libraries
+ * add additional nonstandard fields to struct tm... */
+int tm_cmp(struct tm tm1, struct tm tm2)
+{
+       return  tm1.tm_sec  != tm2.tm_sec  ||
+               tm1.tm_min  != tm2.tm_min  ||
+               tm1.tm_hour != tm2.tm_hour ||
+               tm1.tm_mday != tm2.tm_mday ||
+               tm1.tm_mon  != tm2.tm_mon  ||
+               tm1.tm_year != tm2.tm_year ||
+               tm1.tm_wday != tm2.tm_wday ||
+               tm1.tm_yday != tm2.tm_yday ||
+               tm1.tm_isdst!= tm2.tm_isdst;
+}
+
+char *tm_str(struct tm tm)
+{
+       static int i;
+       static char b[4][64];
+       i = (i+1)%4;
+       snprintf(b[i], sizeof b[i],
+               "s=%02d m=%02d h=%02d mday=%02d mon=%02d year=%04d wday=%d yday=%d isdst=%d",
+               tm.tm_sec, tm.tm_min, tm.tm_hour,
+               tm.tm_mday, tm.tm_mon, tm.tm_year,
+               tm.tm_wday, tm.tm_yday, tm.tm_isdst);
+       return b[i];
+}
+
+#define TM(ss,mm,hh,md,mo,yr,wd,yd,dst) (struct tm){ \
+       .tm_sec = ss, .tm_min = mm, .tm_hour = hh,    \
+       .tm_mday = md, .tm_mon = mo, .tm_year = yr,    \
+       .tm_wday = wd, .tm_yday = yd, .tm_isdst = dst }
+
+#define TM_EPOCH    TM(0,0,0,1,0,70,4,0,0)
+#define TM_Y2038_1S TM(7,14,3,19,0,138,2,18,0)
+#define TM_Y2038    TM(8,14,3,19,0,138,2,18,0)
+
+#define TEST_TM(r,x,m) (!tm_cmp((r),(x)) || \
+       (error("%s failed:\n\tresult: %s\n\texpect: %s\n", \
+              m, tm_str(r), tm_str(x)), 0) )
+
+#define TEST(r, f, x, m) ( \
+       ((r) = (f)) == (x) || \
+       (error("%s failed (" m ")\n", #f, r, x), 0) )
+
+void test_time(void) {
+       struct tm tm, *tm_p;
+       time_t t;
+
+       putenv("TZ=GMT");
+       tzset();
+
+       t=0; tm_p = gmtime(&t);
+       TEST_TM(*tm_p, TM_EPOCH, "gmtime(0)");
+
+       tm = TM_Y2038_1S;
+       t = mktime(&tm);
+       tm = *(gmtime(&t));
+       TEST_TM(*tm_p, TM_Y2038_1S, "mktime/gmtime(Y2038-1)");
+
+       tm = TM_Y2038;
+       t = mktime(&tm);
+       tm = *(gmtime(&t));
+       TEST_TM(*tm_p, TM_Y2038, "mktime/gmtime(Y2038)");
+
+       /* FIXME: set a TZ var and check DST boundary conditions */
+}