ws2_32/tests: Add tabular socket option validity tests.

Signed-off-by: Alex Henrie <alexhenrie24@gmail.com>
Signed-off-by: Zebediah Figura <zfigura@codeweavers.com>
Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
Alex Henrie 2021-08-17 23:50:51 -06:00 committed by Alexandre Julliard
parent cf358c9351
commit d949b66dfd
1 changed files with 298 additions and 30 deletions

View File

@ -11429,44 +11429,312 @@ static void test_so_debug(void)
closesocket(s); closesocket(s);
} }
static void test_set_only_options(void) struct sockopt_validity_test
{ {
unsigned int i; int opt;
int ret, len; int get_error;
int value; int set_error;
SOCKET s; BOOL todo;
};
static const struct static void do_sockopt_validity_tests(const char *type, SOCKET sock, int level,
const struct sockopt_validity_test *tests)
{
char value[256];
int count, rc, expected_rc, i;
for (i = 0; tests[i].opt; i++)
{ {
int level; winetest_push_context("%s option %i", type, tests[i].opt);
int option; memset(value, 0, sizeof(value));
count = sizeof(value);
WSASetLastError(0);
rc = getsockopt(sock, level, tests[i].opt, value, &count);
expected_rc = tests[i].get_error ? SOCKET_ERROR : 0;
todo_wine_if(!tests[i].get_error && tests[i].todo)
ok(rc == expected_rc || broken(rc == SOCKET_ERROR && WSAGetLastError() == WSAENOPROTOOPT),
"expected getsockopt to return %i, got %i\n", expected_rc, rc);
todo_wine_if(tests[i].todo)
ok(WSAGetLastError() == tests[i].get_error || broken(rc == SOCKET_ERROR && WSAGetLastError() == WSAENOPROTOOPT),
"expected getsockopt to set error %i, got %i\n", tests[i].get_error, WSAGetLastError());
if (tests[i].get_error)
{
winetest_pop_context();
continue;
} }
tests[] =
{
{IPPROTO_IP, IP_ADD_MEMBERSHIP},
{IPPROTO_IP, IP_DROP_MEMBERSHIP},
{IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP},
{IPPROTO_IPV6, IPV6_DROP_MEMBERSHIP},
};
for (i = 0; i < ARRAY_SIZE(tests); ++i) WSASetLastError(0);
rc = setsockopt(sock, level, tests[i].opt, value, count);
expected_rc = tests[i].set_error ? SOCKET_ERROR : 0;
todo_wine_if(!tests[i].set_error && tests[i].todo)
ok(rc == expected_rc || broken(rc == SOCKET_ERROR && WSAGetLastError() == WSAENOPROTOOPT),
"expected setsockopt to return %i, got %i\n", expected_rc, rc);
todo_wine_if(tests[i].todo)
ok(WSAGetLastError() == tests[i].set_error || broken(rc == SOCKET_ERROR && WSAGetLastError() == WSAENOPROTOOPT),
"expected setsockopt to set error %i, got %i\n", tests[i].set_error, WSAGetLastError());
winetest_pop_context();
}
}
static void test_sockopt_validity(void)
{
static const struct sockopt_validity_test ipv4_tcp_tests[] =
{ {
if (tests[i].level == IPPROTO_IPV6) { -1, WSAENOPROTOOPT },
{ IP_OPTIONS },
{ IP_HDRINCL, WSAEINVAL },
{ IP_TOS },
{ IP_TTL },
{ IP_MULTICAST_IF, WSAEINVAL },
{ IP_MULTICAST_TTL, WSAEINVAL },
{ IP_MULTICAST_LOOP, WSAEINVAL },
{ IP_ADD_MEMBERSHIP, WSAENOPROTOOPT },
{ IP_DROP_MEMBERSHIP, WSAENOPROTOOPT },
{ IP_DONTFRAGMENT },
{ IP_PKTINFO, WSAEINVAL },
{ IP_RECVTTL, WSAEINVAL, 0, TRUE },
{ IP_RECEIVE_BROADCAST, WSAEINVAL, 0, TRUE },
{ IP_RECVIF, WSAEINVAL, 0, TRUE },
{ IP_RECVDSTADDR, WSAEINVAL, 0, TRUE },
{ IP_IFLIST, 0, 0, TRUE },
{ IP_UNICAST_IF },
{ IP_RTHDR, 0, 0, TRUE },
{ IP_GET_IFLIST, WSAEINVAL, 0, TRUE },
{ IP_RECVRTHDR, WSAEINVAL, 0, TRUE },
{ IP_RECVTCLASS, WSAEINVAL, 0, TRUE },
{ IP_ORIGINAL_ARRIVAL_IF, WSAEINVAL, 0, TRUE },
{ IP_ECN, WSAEINVAL, 0, TRUE },
{ IP_PKTINFO_EX, WSAEINVAL, 0, TRUE },
{ IP_WFP_REDIRECT_RECORDS, WSAEINVAL, 0, TRUE },
{ IP_WFP_REDIRECT_CONTEXT, WSAEINVAL, 0, TRUE },
{ IP_MTU_DISCOVER, 0, WSAEINVAL, TRUE },
{ IP_MTU, WSAENOTCONN, 0, TRUE },
{ IP_RECVERR, WSAEINVAL, 0, TRUE },
{ IP_USER_MTU, 0, 0, TRUE },
{}
};
static const struct sockopt_validity_test ipv4_udp_tests[] =
{ {
s = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP); { -1, WSAENOPROTOOPT },
if (s == INVALID_SOCKET) continue; { IP_OPTIONS },
{ IP_HDRINCL, WSAEINVAL },
{ IP_TOS },
{ IP_TTL },
{ IP_MULTICAST_IF },
{ IP_MULTICAST_TTL },
{ IP_MULTICAST_LOOP },
{ IP_ADD_MEMBERSHIP, WSAENOPROTOOPT },
{ IP_DROP_MEMBERSHIP, WSAENOPROTOOPT },
{ IP_DONTFRAGMENT },
{ IP_PKTINFO },
{ IP_RECVTTL, 0, 0, TRUE },
{ IP_RECEIVE_BROADCAST, 0, 0, TRUE },
{ IP_RECVIF, 0, 0, TRUE },
{ IP_RECVDSTADDR, 0, 0, TRUE },
{ IP_IFLIST, 0, 0, TRUE },
{ IP_UNICAST_IF },
{ IP_RTHDR, 0, 0, TRUE },
{ IP_GET_IFLIST, WSAEINVAL, 0, TRUE },
{ IP_RECVRTHDR, 0, 0, TRUE },
{ IP_RECVTCLASS, 0, 0, TRUE },
{ IP_ORIGINAL_ARRIVAL_IF, 0, 0, TRUE },
{ IP_ECN, 0, 0, TRUE },
{ IP_PKTINFO_EX, 0, 0, TRUE },
{ IP_WFP_REDIRECT_RECORDS, 0, 0, TRUE },
{ IP_WFP_REDIRECT_CONTEXT, 0, 0, TRUE },
{ IP_MTU_DISCOVER, 0, WSAEINVAL, TRUE },
{ IP_MTU, WSAENOTCONN, 0, TRUE },
{ IP_RECVERR, 0, 0, TRUE },
{ IP_USER_MTU, 0, 0, TRUE },
{}
};
static const struct sockopt_validity_test ipv4_raw_tests[] =
{
{ -1, WSAENOPROTOOPT },
{ IP_OPTIONS },
{ IP_HDRINCL, },
{ IP_TOS },
{ IP_TTL },
{ IP_MULTICAST_IF },
{ IP_MULTICAST_TTL },
{ IP_MULTICAST_LOOP },
{ IP_ADD_MEMBERSHIP, WSAENOPROTOOPT },
{ IP_DROP_MEMBERSHIP, WSAENOPROTOOPT },
{ IP_DONTFRAGMENT },
{ IP_PKTINFO },
{ IP_RECVTTL, 0, 0, TRUE },
{ IP_RECEIVE_BROADCAST, 0, 0, TRUE },
{ IP_RECVIF, 0, 0, TRUE },
{ IP_RECVDSTADDR, 0, 0, TRUE },
{ IP_IFLIST, 0, 0, TRUE },
{ IP_UNICAST_IF },
{ IP_RTHDR, 0, 0, TRUE },
{ IP_GET_IFLIST, WSAEINVAL, 0, TRUE },
{ IP_RECVRTHDR, 0, 0, TRUE },
{ IP_RECVTCLASS, 0, 0, TRUE },
{ IP_ORIGINAL_ARRIVAL_IF, 0, 0, TRUE },
{ IP_ECN, 0, 0, TRUE },
{ IP_PKTINFO_EX, 0, 0, TRUE },
{ IP_WFP_REDIRECT_RECORDS, 0, 0, TRUE },
{ IP_WFP_REDIRECT_CONTEXT, 0, 0, TRUE },
{ IP_MTU_DISCOVER, 0, WSAEINVAL, TRUE },
{ IP_MTU, WSAENOTCONN, 0, TRUE },
{ IP_RECVERR, WSAEINVAL, 0, TRUE },
{ IP_USER_MTU, 0, 0, TRUE },
{}
};
static const struct sockopt_validity_test ipv6_tcp_tests[] =
{
{ -1, WSAENOPROTOOPT },
{ IPV6_HOPOPTS, 0, 0, TRUE },
{ IPV6_HDRINCL, WSAEINVAL, 0, TRUE },
{ IPV6_UNICAST_HOPS },
{ IPV6_MULTICAST_IF, WSAEINVAL },
{ IPV6_MULTICAST_HOPS, WSAEINVAL },
{ IPV6_MULTICAST_LOOP, WSAEINVAL },
{ IPV6_ADD_MEMBERSHIP, WSAENOPROTOOPT },
{ IPV6_DROP_MEMBERSHIP, WSAENOPROTOOPT },
{ IPV6_DONTFRAG },
{ IPV6_PKTINFO, WSAEINVAL },
{ IPV6_HOPLIMIT, WSAEINVAL },
{ IPV6_PROTECTION_LEVEL },
{ IPV6_RECVIF, WSAEINVAL, 0, TRUE },
{ IPV6_RECVDSTADDR, WSAEINVAL, 0, TRUE },
{ IPV6_V6ONLY },
{ IPV6_IFLIST, 0, 0, TRUE },
{ IPV6_UNICAST_IF },
{ IPV6_RTHDR, 0, 0, TRUE },
{ IPV6_GET_IFLIST, WSAEINVAL, 0, TRUE },
{ IPV6_RECVRTHDR, WSAEINVAL, 0, TRUE },
{ IPV6_RECVTCLASS, WSAEINVAL },
{ IP_ORIGINAL_ARRIVAL_IF, WSAEINVAL, 0, TRUE },
{ IPV6_ECN, WSAEINVAL, 0, TRUE },
{ IPV6_PKTINFO_EX, WSAEINVAL, 0, TRUE },
{ IPV6_WFP_REDIRECT_RECORDS, WSAEINVAL, 0, TRUE },
{ IPV6_WFP_REDIRECT_CONTEXT, WSAEINVAL, 0, TRUE },
{ IPV6_MTU_DISCOVER, 0, WSAEINVAL, TRUE },
{ IPV6_MTU, WSAENOTCONN, 0, TRUE },
{ IPV6_RECVERR, WSAEINVAL, 0, TRUE },
{ IPV6_USER_MTU, 0, 0, TRUE },
{}
};
static const struct sockopt_validity_test ipv6_udp_tests[] =
{
{ -1, WSAENOPROTOOPT },
{ IPV6_HOPOPTS, 0, 0, TRUE },
{ IPV6_HDRINCL, WSAEINVAL, 0, TRUE },
{ IPV6_UNICAST_HOPS },
{ IPV6_MULTICAST_IF },
{ IPV6_MULTICAST_HOPS },
{ IPV6_MULTICAST_LOOP },
{ IPV6_ADD_MEMBERSHIP, WSAENOPROTOOPT },
{ IPV6_DROP_MEMBERSHIP, WSAENOPROTOOPT },
{ IPV6_DONTFRAG },
{ IPV6_PKTINFO },
{ IPV6_HOPLIMIT },
{ IPV6_PROTECTION_LEVEL },
{ IPV6_RECVIF, 0, 0, TRUE },
{ IPV6_RECVDSTADDR, 0, 0, TRUE },
{ IPV6_V6ONLY },
{ IPV6_IFLIST, 0, 0, TRUE },
{ IPV6_UNICAST_IF },
{ IPV6_RTHDR, 0, 0, TRUE },
{ IPV6_GET_IFLIST, WSAEINVAL, 0, TRUE },
{ IPV6_RECVRTHDR, 0, 0, TRUE },
{ IPV6_RECVTCLASS },
{ IP_ORIGINAL_ARRIVAL_IF, 0, 0, TRUE },
{ IPV6_ECN, 0, 0, TRUE },
{ IPV6_PKTINFO_EX, 0, 0, TRUE },
{ IPV6_WFP_REDIRECT_RECORDS, 0, 0, TRUE },
{ IPV6_WFP_REDIRECT_CONTEXT, 0, 0, TRUE },
{ IPV6_MTU_DISCOVER, 0, WSAEINVAL, TRUE },
{ IPV6_MTU, WSAENOTCONN, 0, TRUE },
{ IPV6_RECVERR, 0, 0, TRUE },
{ IPV6_USER_MTU, 0, 0, TRUE },
{}
};
static const struct sockopt_validity_test ipv6_raw_tests[] =
{
{ -1, WSAENOPROTOOPT },
{ IPV6_HOPOPTS, 0, 0, TRUE },
{ IPV6_HDRINCL, 0, 0, TRUE },
{ IPV6_UNICAST_HOPS },
{ IPV6_MULTICAST_IF },
{ IPV6_MULTICAST_HOPS },
{ IPV6_MULTICAST_LOOP },
{ IPV6_ADD_MEMBERSHIP, WSAENOPROTOOPT },
{ IPV6_DROP_MEMBERSHIP, WSAENOPROTOOPT },
{ IPV6_DONTFRAG },
{ IPV6_PKTINFO },
{ IPV6_HOPLIMIT },
{ IPV6_PROTECTION_LEVEL },
{ IPV6_RECVIF, 0, 0, TRUE },
{ IPV6_RECVDSTADDR, 0, 0, TRUE },
{ IPV6_V6ONLY },
{ IPV6_IFLIST, 0, 0, TRUE },
{ IPV6_UNICAST_IF },
{ IPV6_RTHDR, 0, 0, TRUE },
{ IPV6_GET_IFLIST, WSAEINVAL, 0, TRUE },
{ IPV6_RECVRTHDR, 0, 0, TRUE },
{ IPV6_RECVTCLASS },
{ IP_ORIGINAL_ARRIVAL_IF, 0, 0, TRUE },
{ IPV6_ECN, 0, 0, TRUE },
{ IPV6_PKTINFO_EX, 0, 0, TRUE },
{ IPV6_WFP_REDIRECT_RECORDS, 0, 0, TRUE },
{ IPV6_WFP_REDIRECT_CONTEXT, 0, 0, TRUE },
{ IPV6_MTU_DISCOVER, 0, WSAEINVAL, TRUE },
{ IPV6_MTU, WSAENOTCONN, 0, TRUE },
{ IPV6_RECVERR, WSAEINVAL, 0, TRUE },
{ IPV6_USER_MTU, 0, 0, TRUE },
{}
};
SOCKET sock;
sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
ok(sock != INVALID_SOCKET, "failed to create socket, error %u\n", WSAGetLastError());
do_sockopt_validity_tests("IPv4 TCP", sock, IPPROTO_IP, ipv4_tcp_tests);
closesocket(sock);
sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
ok(sock != INVALID_SOCKET, "failed to create socket, error %u\n", WSAGetLastError());
do_sockopt_validity_tests("IPv4 UDP", sock, IPPROTO_IP, ipv4_udp_tests);
closesocket(sock);
sock = socket(AF_INET, SOCK_RAW, IPPROTO_RAW);
if (sock == INVALID_SOCKET && WSAGetLastError() == WSAEACCES)
{
skip("Raw IPv4 sockets are not available\n");
} }
else else
{ {
s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); ok(sock != INVALID_SOCKET, "failed to create socket, error %u\n", WSAGetLastError());
do_sockopt_validity_tests("IPv4 raw", sock, IPPROTO_IP, ipv4_raw_tests);
closesocket(sock);
} }
len = sizeof(value); sock = socket(AF_INET6, SOCK_STREAM, IPPROTO_TCP);
ret = getsockopt(s, tests[i].level, tests[i].option, (char *)&value, &len); ok(sock != INVALID_SOCKET, "failed to create socket, error %u\n", WSAGetLastError());
ok(ret == -1, "expected failure\n"); do_sockopt_validity_tests("IPv6 TCP", sock, IPPROTO_IPV6, ipv6_tcp_tests);
ok(WSAGetLastError() == WSAENOPROTOOPT, "got error %u\n", WSAGetLastError()); closesocket(sock);
closesocket(s); sock = socket(AF_INET6, SOCK_DGRAM, IPPROTO_UDP);
ok(sock != INVALID_SOCKET, "failed to create socket, error %u\n", WSAGetLastError());
do_sockopt_validity_tests("IPv6 UDP", sock, IPPROTO_IPV6, ipv6_udp_tests);
closesocket(sock);
sock = socket(AF_INET6, SOCK_RAW, IPPROTO_RAW);
if (sock == INVALID_SOCKET && WSAGetLastError() == WSAEACCES)
{
skip("Raw IPv6 sockets are not available\n");
}
else
{
ok(sock != INVALID_SOCKET, "failed to create socket, error %u\n", WSAGetLastError());
do_sockopt_validity_tests("IPv6 raw", sock, IPPROTO_IPV6, ipv6_raw_tests);
closesocket(sock);
} }
} }
@ -11487,7 +11755,7 @@ START_TEST( sock )
test_ipv6_cmsg(); test_ipv6_cmsg();
test_extendedSocketOptions(); test_extendedSocketOptions();
test_so_debug(); test_so_debug();
test_set_only_options(); test_sockopt_validity();
for (i = 0; i < ARRAY_SIZE(tests); i++) for (i = 0; i < ARRAY_SIZE(tests); i++)
do_test(&tests[i]); do_test(&tests[i]);