1602e473dSmrg#include <check.h>
2b9526c6aSmrg#include <stdio.h>
3602e473dSmrg#include <string.h>
4602e473dSmrg#include <stdlib.h>
5b9526c6aSmrg#ifdef __unix__
6b9526c6aSmrg#include <unistd.h>
7b9526c6aSmrg#endif
8602e473dSmrg#include "check_suites.h"
9602e473dSmrg#include "xcb.h"
10602e473dSmrg#include "xcbext.h"
11602e473dSmrg
12602e473dSmrg/* xcb_parse_display tests {{{ */
13602e473dSmrg
14602e473dSmrgtypedef enum test_type_t {
15602e473dSmrg	TEST_ARGUMENT, TEST_ENVIRONMENT, TEST_END
16602e473dSmrg} test_type_t;
17602e473dSmrgstatic const char *const test_string[] = { "", "via $DISPLAY " };
18602e473dSmrg
19b9526c6aSmrg/* putenv(3) takes a pointer to a writable string that it adds directly
20b9526c6aSmrg   to the environment, so it must be in persistent memory, not on the stack */
21b9526c6aSmrgstatic char display_env[] = "DISPLAY=";
22b9526c6aSmrg
23602e473dSmrgstatic void parse_display_pass(const char *name, const char *host, const int display, const int screen)
24602e473dSmrg{
25602e473dSmrg	int success;
26602e473dSmrg	char *got_host;
27602e473dSmrg	int got_display, got_screen;
28602e473dSmrg	const char *argument = 0;
29602e473dSmrg	test_type_t test_type;
30602e473dSmrg
31602e473dSmrg	for(test_type = TEST_ARGUMENT; test_type != TEST_END; test_type++)
32602e473dSmrg	{
33602e473dSmrg		if(test_type == TEST_ARGUMENT)
34602e473dSmrg		{
35602e473dSmrg			argument = name;
36b9526c6aSmrg			putenv(display_env);
37602e473dSmrg		}
38602e473dSmrg		else if(test_type == TEST_ENVIRONMENT)
39602e473dSmrg		{
40602e473dSmrg			argument = 0;
41602e473dSmrg			setenv("DISPLAY", name, 1);
42602e473dSmrg		}
43602e473dSmrg
44602e473dSmrg		got_host = (char *) -1;
45602e473dSmrg		got_display = got_screen = -42;
46602e473dSmrg		mark_point();
47602e473dSmrg		success = xcb_parse_display(argument, &got_host, &got_display, &got_screen);
488ffb90f1Smrg		ck_assert_msg(success, "unexpected parse failure %sfor '%s'", test_string[test_type], name);
498ffb90f1Smrg		ck_assert_msg(strcmp(host, got_host) == 0, "parse %sproduced unexpected hostname '%s' for '%s': expected '%s'", test_string[test_type], got_host, name, host);
508ffb90f1Smrg		ck_assert_msg(display == got_display, "parse %sproduced unexpected display '%d' for '%s': expected '%d'", test_string[test_type], got_display, name, display);
518ffb90f1Smrg		ck_assert_msg(screen == got_screen, "parse %sproduced unexpected screen '%d' for '%s': expected '%d'", test_string[test_type], got_screen, name, screen);
52602e473dSmrg
53602e473dSmrg		got_host = (char *) -1;
54602e473dSmrg		got_display = got_screen = -42;
55602e473dSmrg		mark_point();
56602e473dSmrg		success = xcb_parse_display(argument, &got_host, &got_display, 0);
578ffb90f1Smrg		ck_assert_msg(success, "unexpected screenless parse failure %sfor '%s'", test_string[test_type], name);
588ffb90f1Smrg		ck_assert_msg(strcmp(host, got_host) == 0, "screenless parse %sproduced unexpected hostname '%s' for '%s': expected '%s'", test_string[test_type], got_host, name, host);
598ffb90f1Smrg		ck_assert_msg(display == got_display, "screenless parse %sproduced unexpected display '%d' for '%s': expected '%d'", test_string[test_type], got_display, name, display);
60602e473dSmrg	}
61b9526c6aSmrg	putenv(display_env);
62602e473dSmrg}
63602e473dSmrg
64602e473dSmrgstatic void parse_display_fail(const char *name)
65602e473dSmrg{
66602e473dSmrg	int success;
67602e473dSmrg	char *got_host;
68602e473dSmrg	int got_display, got_screen;
69602e473dSmrg	const char *argument = 0;
70602e473dSmrg	test_type_t test_type;
71602e473dSmrg
72602e473dSmrg	for(test_type = TEST_ARGUMENT; test_type != TEST_END; test_type++)
73602e473dSmrg	{
74602e473dSmrg		if(test_type == TEST_ARGUMENT)
75602e473dSmrg		{
76602e473dSmrg			argument = name;
77b9526c6aSmrg			putenv(display_env);
78602e473dSmrg		}
79602e473dSmrg		else if(test_type == TEST_ENVIRONMENT)
80602e473dSmrg		{
81602e473dSmrg			if (!name) break;
82602e473dSmrg			argument = 0;
83602e473dSmrg			setenv("DISPLAY", name, 1);
84602e473dSmrg		}
85602e473dSmrg
86602e473dSmrg		got_host = (char *) -1;
87602e473dSmrg		got_display = got_screen = -42;
88602e473dSmrg		mark_point();
89602e473dSmrg		success = xcb_parse_display(argument, &got_host, &got_display, &got_screen);
908ffb90f1Smrg		ck_assert_msg(!success, "unexpected parse success %sfor '%s'", test_string[test_type], name);
918ffb90f1Smrg		ck_assert_msg(got_host == (char *) -1, "host changed on parse failure %sfor '%s': got %p", test_string[test_type], name, got_host);
928ffb90f1Smrg		ck_assert_msg(got_display == -42, "display changed on parse failure %sfor '%s': got %d", test_string[test_type], name, got_display);
938ffb90f1Smrg		ck_assert_msg(got_screen == -42, "screen changed on parse failure %sfor '%s': got %d", test_string[test_type], name, got_screen);
94602e473dSmrg
95602e473dSmrg		got_host = (char *) -1;
96602e473dSmrg		got_display = got_screen = -42;
97602e473dSmrg		mark_point();
98602e473dSmrg		success = xcb_parse_display(argument, &got_host, &got_display, 0);
998ffb90f1Smrg		ck_assert_msg(!success, "unexpected screenless parse success %sfor '%s'", test_string[test_type], name);
1008ffb90f1Smrg		ck_assert_msg(got_host == (char *) -1, "host changed on parse failure %sfor '%s': got %p", test_string[test_type], name, got_host);
1018ffb90f1Smrg		ck_assert_msg(got_display == -42, "display changed on parse failure %sfor '%s': got %d", test_string[test_type], name, got_display);
102602e473dSmrg	}
103b9526c6aSmrg	putenv(display_env);
104602e473dSmrg}
105602e473dSmrg
106602e473dSmrgSTART_TEST(parse_display_unix)
107602e473dSmrg{
108b9526c6aSmrg#ifdef __unix__
109b9526c6aSmrg	char buf[sizeof "/tmp/xcb-test.XXXXXXX"];
110b9526c6aSmrg	char buf2[sizeof(buf) + 7];
111b9526c6aSmrg	int r, v;
112b9526c6aSmrg	memcpy(buf, "/tmp/xcb-test.XXXXXXX", sizeof buf);
113b9526c6aSmrg	v = mkstemp(buf);
114b9526c6aSmrg	ck_assert_msg(v >= 0, "cannot create temporary file");
115b9526c6aSmrg	parse_display_pass(buf, buf, 0, 0);
116b9526c6aSmrg	r = snprintf(buf2, sizeof buf2, "unix:%s", buf);
117b9526c6aSmrg	if (r < 5 || r >= (int)sizeof buf2) {
118b9526c6aSmrg		ck_assert_msg(0, "snprintf() failed (return value %d)", r);
119b9526c6aSmrg		unlink(buf);
120b9526c6aSmrg		return;
121b9526c6aSmrg	}
122b9526c6aSmrg	parse_display_pass(buf2, buf, 0, 0);
123b9526c6aSmrg	r = snprintf(buf2, sizeof buf2, "unix:%s.1", buf);
124b9526c6aSmrg	if (r < 7 || r >= (int)sizeof buf2) {
125b9526c6aSmrg		ck_assert_msg(0, "snprintf() failed (return value %d)", r);
126b9526c6aSmrg		unlink(buf);
127b9526c6aSmrg		return;
128b9526c6aSmrg	}
129b9526c6aSmrg	parse_display_pass(buf2, buf, 0, 1);
130b9526c6aSmrg	r = snprintf(buf2, sizeof buf2, "%s.1", buf);
131b9526c6aSmrg	if (r < 2 || r >= (int)sizeof buf2) {
132b9526c6aSmrg		ck_assert_msg(0, "snprintf() failed (return value %d)", r);
133b9526c6aSmrg		unlink(buf);
134b9526c6aSmrg		return;
135b9526c6aSmrg	}
136b9526c6aSmrg	parse_display_pass(buf2, buf, 0, 1);
137b9526c6aSmrg	unlink(buf);
138b9526c6aSmrg#endif
139602e473dSmrg	parse_display_pass(":0", "", 0, 0);
140602e473dSmrg	parse_display_pass(":1", "", 1, 0);
141602e473dSmrg	parse_display_pass(":0.1", "", 0, 1);
142602e473dSmrg}
143602e473dSmrgEND_TEST
144602e473dSmrg
145602e473dSmrgSTART_TEST(parse_display_ip)
146602e473dSmrg{
147602e473dSmrg	parse_display_pass("x.org:0", "x.org", 0, 0);
148602e473dSmrg	parse_display_pass("expo:0", "expo", 0, 0);
149602e473dSmrg	parse_display_pass("bigmachine:1", "bigmachine", 1, 0);
150602e473dSmrg	parse_display_pass("hydra:0.1", "hydra", 0, 1);
151602e473dSmrg}
152602e473dSmrgEND_TEST
153602e473dSmrg
154602e473dSmrgSTART_TEST(parse_display_ipv4)
155602e473dSmrg{
156602e473dSmrg	parse_display_pass("198.112.45.11:0", "198.112.45.11", 0, 0);
157602e473dSmrg	parse_display_pass("198.112.45.11:0.1", "198.112.45.11", 0, 1);
158602e473dSmrg}
159602e473dSmrgEND_TEST
160602e473dSmrg
161602e473dSmrgSTART_TEST(parse_display_ipv6)
162602e473dSmrg{
163602e473dSmrg	parse_display_pass(":::0", "::", 0, 0);
164602e473dSmrg	parse_display_pass("1:::0", "1::", 0, 0);
165602e473dSmrg	parse_display_pass("::1:0", "::1", 0, 0);
166602e473dSmrg	parse_display_pass("::1:0.1", "::1", 0, 1);
167602e473dSmrg	parse_display_pass("::127.0.0.1:0", "::127.0.0.1", 0, 0);
168602e473dSmrg	parse_display_pass("::ffff:127.0.0.1:0", "::ffff:127.0.0.1", 0, 0);
169602e473dSmrg	parse_display_pass("2002:83fc:d052::1:0", "2002:83fc:d052::1", 0, 0);
170602e473dSmrg	parse_display_pass("2002:83fc:d052::1:0.1", "2002:83fc:d052::1", 0, 1);
171602e473dSmrg	parse_display_pass("[::]:0", "[::]", 0, 0);
172602e473dSmrg	parse_display_pass("[1::]:0", "[1::]", 0, 0);
173602e473dSmrg	parse_display_pass("[::1]:0", "[::1]", 0, 0);
174602e473dSmrg	parse_display_pass("[::1]:0.1", "[::1]", 0, 1);
175602e473dSmrg	parse_display_pass("[::127.0.0.1]:0", "[::127.0.0.1]", 0, 0);
176602e473dSmrg	parse_display_pass("[::ffff:127.0.0.1]:0", "[::ffff:127.0.0.1]", 0, 0);
177602e473dSmrg	parse_display_pass("[2002:83fc:d052::1]:0", "[2002:83fc:d052::1]", 0, 0);
178602e473dSmrg	parse_display_pass("[2002:83fc:d052::1]:0.1", "[2002:83fc:d052::1]", 0, 1);
179602e473dSmrg}
180602e473dSmrgEND_TEST
181602e473dSmrg
182602e473dSmrgSTART_TEST(parse_display_decnet)
183602e473dSmrg{
184602e473dSmrg	parse_display_pass("myws::0", "myws:", 0, 0);
185602e473dSmrg	parse_display_pass("big::1", "big:", 1, 0);
186602e473dSmrg	parse_display_pass("hydra::0.1", "hydra:", 0, 1);
187602e473dSmrg}
188602e473dSmrgEND_TEST
189602e473dSmrg
190602e473dSmrgSTART_TEST(parse_display_negative)
191602e473dSmrg{
192602e473dSmrg	parse_display_fail(0);
193602e473dSmrg	parse_display_fail("");
194602e473dSmrg	parse_display_fail(":");
195602e473dSmrg	parse_display_fail("::");
196602e473dSmrg	parse_display_fail(":::");
197602e473dSmrg	parse_display_fail(":.");
198602e473dSmrg	parse_display_fail(":a");
199602e473dSmrg	parse_display_fail(":a.");
200602e473dSmrg	parse_display_fail(":0.");
201602e473dSmrg	parse_display_fail(":.a");
202602e473dSmrg	parse_display_fail(":.0");
203602e473dSmrg	parse_display_fail(":0.a");
204602e473dSmrg	parse_display_fail(":0.0.");
205602e473dSmrg
206602e473dSmrg	parse_display_fail("127.0.0.1");
207602e473dSmrg	parse_display_fail("127.0.0.1:");
208602e473dSmrg	parse_display_fail("127.0.0.1::");
209602e473dSmrg	parse_display_fail("::127.0.0.1");
210602e473dSmrg	parse_display_fail("::127.0.0.1:");
211602e473dSmrg	parse_display_fail("::127.0.0.1::");
212602e473dSmrg	parse_display_fail("::ffff:127.0.0.1");
213602e473dSmrg	parse_display_fail("::ffff:127.0.0.1:");
214602e473dSmrg	parse_display_fail("::ffff:127.0.0.1::");
215602e473dSmrg	parse_display_fail("localhost");
216602e473dSmrg	parse_display_fail("localhost:");
217602e473dSmrg	parse_display_fail("localhost::");
218602e473dSmrg}
219602e473dSmrgEND_TEST
220602e473dSmrg
221602e473dSmrg/* }}} */
222602e473dSmrg
223602e473dSmrgstatic void popcount_eq(uint32_t bits, int count)
224602e473dSmrg{
2258ffb90f1Smrg	ck_assert_msg(xcb_popcount(bits) == count, "unexpected popcount(%08x) != %d", bits, count);
226602e473dSmrg}
227602e473dSmrg
228602e473dSmrgSTART_TEST(popcount)
229602e473dSmrg{
230602e473dSmrg	uint32_t mask;
231602e473dSmrg	int count;
232602e473dSmrg
233602e473dSmrg	for (mask = 0xffffffff, count = 32; count >= 0; mask >>= 1, --count) {
234602e473dSmrg		popcount_eq(mask, count);
235602e473dSmrg	}
236602e473dSmrg	for (mask = 0x80000000; mask; mask >>= 1) {
237602e473dSmrg		popcount_eq(mask, 1);
238602e473dSmrg	}
239602e473dSmrg	for (mask = 0x80000000; mask > 1; mask >>= 1) {
240602e473dSmrg		popcount_eq(mask | 1, 2);
241602e473dSmrg	}
242602e473dSmrg}
243602e473dSmrgEND_TEST
244602e473dSmrg
245602e473dSmrgSuite *public_suite(void)
246602e473dSmrg{
247602e473dSmrg	Suite *s = suite_create("Public API");
248b9526c6aSmrg	putenv(display_env);
249602e473dSmrg	suite_add_test(s, parse_display_unix, "xcb_parse_display unix");
250602e473dSmrg	suite_add_test(s, parse_display_ip, "xcb_parse_display ip");
251602e473dSmrg	suite_add_test(s, parse_display_ipv4, "xcb_parse_display ipv4");
252602e473dSmrg	suite_add_test(s, parse_display_ipv6, "xcb_parse_display ipv6");
253602e473dSmrg	suite_add_test(s, parse_display_decnet, "xcb_parse_display decnet");
254602e473dSmrg	suite_add_test(s, parse_display_negative, "xcb_parse_display negative");
255602e473dSmrg	suite_add_test(s, popcount, "xcb_popcount");
256602e473dSmrg	return s;
257602e473dSmrg}
258