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