summaryrefslogtreecommitdiff
path: root/popen.c
blob: 8a5843aebaf8e27bbb1ac79691aa760d9f2f21b7 (plain) (blame)
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
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <signal.h>

#define TEST(r, f, x, m) ( \
((r) = (f)) == (x) || \
(printf(__FILE__ ":%d: %s failed (" m ")\n", __LINE__, #f, r, x), err++, 0) )

#define TEST_E(f) ( (errno = 0), (f) || \
(printf(__FILE__ ":%d: %s failed (errno = %d)\n", __LINE__, #f, errno), err++, 0) )

#define TEST_S(s, x, m) ( \
!strcmp((s),(x)) || \
(printf(__FILE__ ":%d: [%s] != [%s] (%s)\n", __LINE__, s, x, m), err++, 0) )

static sig_atomic_t got_sig;

static void handler(int sig)
{
	got_sig = 1;
}

int test_popen(void)
{
	int i;
	char foo[6];
	char cmd[64];
	int err = 0;
	FILE *f;

	TEST_E(f = popen("echo hello", "r"));
	if (f) {
		TEST_E(fgets(foo, sizeof foo, f));
		TEST_S(foo, "hello", "child process did not say hello");
		TEST(i, pclose(f), 0, "exit status %04x != %04x");
	}

	signal(SIGUSR1, handler);
	snprintf(cmd, sizeof cmd, "read a ; test \"x$a\" = xhello && kill -USR1 %d", getpid());
	TEST_E(f = popen(cmd, "w"));
	if (f) {
		TEST_E(fputs("hello", f) >= 0);
		TEST(i, pclose(f), 0, "exit status %04x != %04x");
		TEST(i, got_sig, 1, "child process did not send signal");
	}
	signal(SIGUSR1, SIG_DFL);

	return err;
}