summaryrefslogtreecommitdiff
path: root/ccan/io/test/run-09-connect.c
blob: 124b6a1f70e715f08a3014a8cee72372611ce575 (plain)
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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#include <ccan/io/io.h>
/* Include the C files directly. */
#include <ccan/io/poll.c>
#include <ccan/io/io.c>
#include <ccan/tap/tap.h>
#include <sys/wait.h>
#include <stdio.h>

#ifdef DEBUG_CONN
#define PORT "64009"
#else
#define PORT "65009"
#endif

static struct io_listener *l;
static struct data *d2;

struct data {
	int state;
	char buf[10];
};

static struct io_plan *closer(struct io_conn *conn, struct data *d)
{
	d->state++;
	return io_close(conn);
}

static struct io_plan *connected(struct io_conn *conn, struct data *d2)
{
	ok1(d2->state == 0);
	d2->state++;
	return io_read(conn, d2->buf, sizeof(d2->buf), closer, d2);
}

static struct io_plan *init_conn(struct io_conn *conn, struct data *d)
{
#ifdef DEBUG_CONN
	io_set_debug(conn, true);
#endif
	ok1(d->state == 0);
	d->state++;
	io_close_listener(l);

	return io_write(conn, d->buf, sizeof(d->buf), closer, d);
}

static int make_listen_fd(const char *port, struct addrinfo **info)
{
	int fd, on = 1;
	struct addrinfo *addrinfo, hints;

	memset(&hints, 0, sizeof(hints));
	hints.ai_family = AF_UNSPEC;
	hints.ai_socktype = SOCK_STREAM;
	hints.ai_flags = AI_PASSIVE;
	hints.ai_protocol = 0;

	if (getaddrinfo(NULL, port, &hints, &addrinfo) != 0)
		return -1;

	fd = socket(addrinfo->ai_family, addrinfo->ai_socktype,
		    addrinfo->ai_protocol);
	if (fd < 0)
		return -1;

	setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
	if (bind(fd, addrinfo->ai_addr, addrinfo->ai_addrlen) != 0) {
		close(fd);
		return -1;
	}
	if (listen(fd, 1) != 0) {
		close(fd);
		return -1;
	}
	*info = addrinfo;
	return fd;
}

static struct io_plan *setup_connect(struct io_conn *conn,
				     struct addrinfo *addrinfo)
{
	d2 = malloc(sizeof(*d2));
	d2->state = 0;
	return io_connect(conn, addrinfo, connected, d2);
}

int main(void)
{
	struct data *d = malloc(sizeof(*d));
	struct addrinfo *addrinfo;
	int fd;

	/* This is how many tests you plan to run */
	plan_tests(8);
	d->state = 0;
	memset(d->buf, 'a', sizeof(d->buf));
	fd = make_listen_fd(PORT, &addrinfo);
	ok1(fd >= 0);
	l = io_new_listener(NULL, fd, init_conn, d);
	ok1(l);

	fd = socket(addrinfo->ai_family, addrinfo->ai_socktype,
		    addrinfo->ai_protocol);
	ok1(io_new_conn(NULL, fd, setup_connect, addrinfo));

	ok1(io_loop(NULL, NULL) == NULL);
	ok1(d->state == 2);
	ok1(d2->state == 2);

	freeaddrinfo(addrinfo);
	free(d);
	free(d2);

	/* This exits depending on whether all tests passed */
	return exit_status();
}