2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
#include "sc_sock.h"
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-02-01 07:11:12 +03:00
|
|
|
#include <synchapi.h>
|
|
|
|
#define sleep(n) (Sleep(n * 1000))
|
2020-11-25 09:25:38 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
#include <windows.h>
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
struct sc_thread
|
|
|
|
{
|
|
|
|
HANDLE id;
|
2021-02-01 07:11:12 +03:00
|
|
|
void *(*fn)(void *);
|
|
|
|
void *arg;
|
|
|
|
void *ret;
|
2020-11-25 09:25:38 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
#include <errno.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
struct sc_thread
|
|
|
|
{
|
|
|
|
pthread_t id;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void sc_thread_init(struct sc_thread *thread);
|
|
|
|
int sc_thread_term(struct sc_thread *thread);
|
|
|
|
int sc_thread_start(struct sc_thread *thread, void *(*fn)(void *), void *arg);
|
2021-02-05 20:46:59 +03:00
|
|
|
int sc_thread_join(struct sc_thread *thread, void **ret);
|
2020-11-25 09:25:38 +03:00
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void sc_thread_init(struct sc_thread *thread)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
|
|
|
thread->id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2021-02-01 07:11:12 +03:00
|
|
|
#include <process.h>
|
2020-11-25 09:25:38 +03:00
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
unsigned int __stdcall sc_thread_fn(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-01 07:11:12 +03:00
|
|
|
struct sc_thread *thread = arg;
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
thread->ret = thread->fn(thread->arg);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
int sc_thread_start(struct sc_thread *thread, void *(*fn)(void *), void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
thread->fn = fn;
|
|
|
|
thread->arg = arg;
|
|
|
|
|
|
|
|
thread->id =
|
2021-02-01 07:11:12 +03:00
|
|
|
(HANDLE) _beginthreadex(NULL, 0, sc_thread_fn, thread, 0, NULL);
|
2020-11-25 09:25:38 +03:00
|
|
|
rc = thread->id == 0 ? -1 : 0;
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2021-02-05 20:46:59 +03:00
|
|
|
int sc_thread_join(struct sc_thread *thread, void **ret)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
DWORD rv;
|
|
|
|
BOOL brc;
|
|
|
|
|
|
|
|
if (thread->id == 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = WaitForSingleObject(thread->id, INFINITE);
|
|
|
|
if (rv == WAIT_FAILED) {
|
|
|
|
rc = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
brc = CloseHandle(thread->id);
|
|
|
|
if (!brc) {
|
|
|
|
rc = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
thread->id = 0;
|
|
|
|
if (ret != NULL) {
|
|
|
|
*ret = thread->ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
int sc_thread_start(struct sc_thread *thread, void *(*fn)(void *), void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
pthread_attr_t hndl;
|
|
|
|
|
|
|
|
rc = pthread_attr_init(&hndl);
|
|
|
|
if (rc != 0) {
|
2020-12-28 06:18:18 +03:00
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This may only fail with EINVAL.
|
|
|
|
pthread_attr_setdetachstate(&hndl, PTHREAD_CREATE_JOINABLE);
|
|
|
|
|
|
|
|
rc = pthread_create(&thread->id, &hndl, fn, arg);
|
|
|
|
|
|
|
|
// This may only fail with EINVAL.
|
|
|
|
pthread_attr_destroy(&hndl);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2021-02-05 20:46:59 +03:00
|
|
|
int sc_thread_join(struct sc_thread *thread, void **ret)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
|
|
|
int rc;
|
2021-02-01 07:11:12 +03:00
|
|
|
void *val;
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
if (thread->id == 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = pthread_join(thread->id, &val);
|
|
|
|
thread->id = 0;
|
|
|
|
|
|
|
|
if (ret != NULL) {
|
|
|
|
*ret = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
int sc_thread_term(struct sc_thread *thread)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-05 20:46:59 +03:00
|
|
|
return sc_thread_join(thread, NULL);
|
2020-11-25 09:25:38 +03:00
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *server_ip4(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
char buf[5];
|
|
|
|
char tmp[128];
|
|
|
|
struct sc_sock sock, accepted;
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_listen(&sock, "127.0.0.1", "8004") == 0);
|
|
|
|
sc_sock_print(&sock, tmp, sizeof(tmp));
|
|
|
|
assert(strcmp(tmp, "Local(127.0.0.1:8004), Remote() ") == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_accept(&sock, &accepted) == 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_recv(&accepted, buf, 5, 0) == 5);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(strcmp("test", buf) == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
assert(sc_sock_term(&accepted) == 0);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *client_ip4(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
int rc;
|
|
|
|
char tmp[128];
|
|
|
|
struct sc_sock sock;
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_set_sndtimeo(&sock, 10000) != 0);
|
|
|
|
printf("%s \n", sc_sock_error(&sock));
|
|
|
|
assert(sc_sock_set_rcvtimeo(&sock, 10000) != 0);
|
|
|
|
printf("%s \n", sc_sock_error(&sock));
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
for (int i = 0; i < 5; i++) {
|
|
|
|
rc = sc_sock_connect(&sock, "127.0.0.1", "8004", NULL, NULL);
|
|
|
|
if (rc == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(rc == 0);
|
|
|
|
sc_sock_print(&sock, tmp, sizeof(tmp));
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_set_sndtimeo(&sock, 10000) == 0);
|
|
|
|
assert(sc_sock_set_rcvtimeo(&sock, 10000) == 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_send(&sock, "test", 5, 0) == 5);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_ip4()
|
|
|
|
{
|
|
|
|
struct sc_thread thread1;
|
|
|
|
struct sc_thread thread2;
|
|
|
|
|
|
|
|
sc_thread_init(&thread1);
|
|
|
|
sc_thread_init(&thread2);
|
|
|
|
|
|
|
|
assert(sc_thread_start(&thread1, server_ip4, NULL) == 0);
|
|
|
|
assert(sc_thread_start(&thread2, client_ip4, NULL) == 0);
|
|
|
|
|
|
|
|
assert(sc_thread_term(&thread1) == 0);
|
|
|
|
assert(sc_thread_term(&thread2) == 0);
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *server_ip6(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
2020-11-25 09:25:38 +03:00
|
|
|
char buf[5];
|
|
|
|
char tmp[128];
|
|
|
|
struct sc_sock sock, accepted;
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET6);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_listen(&sock, "::1", "8006") == 0);
|
|
|
|
sc_sock_print(&sock, tmp, sizeof(tmp));
|
|
|
|
assert(strcmp(tmp, "Local(::1:8006), Remote() ") == 0);
|
|
|
|
assert(sc_sock_accept(&sock, &accepted) == 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_recv(&accepted, buf, 5, 0) == 5);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(strcmp("test", buf) == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
assert(sc_sock_term(&accepted) == 0);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *client_ip6(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
2020-11-25 09:25:38 +03:00
|
|
|
int rc;
|
|
|
|
struct sc_sock sock;
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET6);
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
for (int i = 0; i < 5; i++) {
|
|
|
|
rc = sc_sock_connect(&sock, "::1", "8006", NULL, NULL);
|
|
|
|
if (rc == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep(1);
|
|
|
|
}
|
|
|
|
assert(rc == 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_send(&sock, "test", 5, 0) == 5);
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_ip6()
|
|
|
|
{
|
|
|
|
struct sc_thread thread1;
|
|
|
|
struct sc_thread thread2;
|
|
|
|
|
|
|
|
sc_thread_init(&thread1);
|
|
|
|
sc_thread_init(&thread2);
|
|
|
|
|
|
|
|
assert(sc_thread_start(&thread1, server_ip6, NULL) == 0);
|
|
|
|
assert(sc_thread_start(&thread2, client_ip6, NULL) == 0);
|
|
|
|
|
|
|
|
assert(sc_thread_term(&thread1) == 0);
|
|
|
|
assert(sc_thread_term(&thread2) == 0);
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *server_unix(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
2020-11-25 09:25:38 +03:00
|
|
|
char buf[5];
|
|
|
|
char tmp[128];
|
|
|
|
struct sc_sock sock, accepted;
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_UNIX);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_listen(&sock, "x.sock", NULL) == 0);
|
|
|
|
sc_sock_print(&sock, tmp, sizeof(tmp));
|
|
|
|
assert(strcmp(tmp, "Local(x.sock), Remote() ") == 0);
|
|
|
|
int rc = sc_sock_accept(&sock, &accepted);
|
|
|
|
if (rc != 0) {
|
|
|
|
printf("error(%d) : %s\n", errno, sock.err);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(rc == 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_recv(&accepted, buf, 5, 0) == 5);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(strcmp("test", buf) == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
assert(sc_sock_term(&accepted) == 0);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *client_unix(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
2020-11-25 09:25:38 +03:00
|
|
|
int rc;
|
|
|
|
struct sc_sock sock;
|
|
|
|
|
|
|
|
sleep(3);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_UNIX);
|
2020-11-25 09:25:38 +03:00
|
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
printf("Will connect to x.sock \n");
|
|
|
|
rc = sc_sock_connect(&sock, "x.sock", NULL, NULL, NULL);
|
|
|
|
if (rc == 0) {
|
|
|
|
printf("Connected to x.sock \n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("Failed to connect to x.sock \n");
|
|
|
|
sleep(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(rc == 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_send(&sock, "test", 5, 0) == 5);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void test_unix()
|
|
|
|
{
|
|
|
|
struct sc_thread thread1;
|
|
|
|
struct sc_thread thread2;
|
|
|
|
|
|
|
|
sc_thread_init(&thread1);
|
|
|
|
sc_thread_init(&thread2);
|
|
|
|
|
|
|
|
assert(sc_thread_start(&thread1, server_unix, NULL) == 0);
|
|
|
|
assert(sc_thread_start(&thread2, client_unix, NULL) == 0);
|
|
|
|
|
|
|
|
assert(sc_thread_term(&thread1) == 0);
|
|
|
|
assert(sc_thread_term(&thread2) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void test1()
|
|
|
|
{
|
|
|
|
char tmp[5];
|
|
|
|
struct sc_sock sock;
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, false, SC_SOCK_INET);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_connect(&sock, "3127.0.0.1", "2131", NULL, NULL) != 0);
|
|
|
|
assert(sc_sock_finish_connect(&sock) != 0);
|
|
|
|
assert(sc_sock_connect(&sock, "3127.0.0.1", "2131", "127.90.1.1", "50") !=
|
|
|
|
0);
|
|
|
|
assert(sc_sock_finish_connect(&sock) != 0);
|
2020-12-28 11:03:24 +03:00
|
|
|
assert(sc_sock_send(&sock, "test", 5, 0) == -1);
|
|
|
|
assert(sc_sock_recv(&sock, tmp, sizeof(tmp), 0) == -1);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_connect(&sock, "131s::1", "2131", "::1", "50") != 0);
|
|
|
|
assert(sc_sock_finish_connect(&sock) != 0);
|
|
|
|
assert(sc_sock_connect(&sock, "d::1", "2131", "dsad", "50") != 0);
|
|
|
|
assert(sc_sock_finish_connect(&sock) != 0);
|
|
|
|
assert(sc_sock_connect(&sock, "dsadas", "2131", "::1", "50") != 0);
|
|
|
|
|
|
|
|
sc_sock_term(&sock);
|
|
|
|
}
|
|
|
|
|
2021-02-07 20:42:08 +03:00
|
|
|
void test_poll_mass(void)
|
|
|
|
{
|
|
|
|
struct sc_sock_poll poll;
|
|
|
|
struct sc_sock_pipe pipe[100];
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
assert(sc_sock_poll_add(&poll, &pipe[i].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[i]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
assert(sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[i]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
assert(sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[i]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
assert(sc_sock_poll_add(&poll, &pipe[i].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[i].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[i]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
}
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
void test_pipe(void)
|
|
|
|
{
|
|
|
|
char buf[5];
|
|
|
|
struct sc_sock_pipe pipe;
|
|
|
|
|
|
|
|
sc_sock_pipe_init(&pipe, 0);
|
|
|
|
sc_sock_pipe_write(&pipe, "test", 5);
|
|
|
|
sc_sock_pipe_read(&pipe, buf, 5);
|
|
|
|
sc_sock_pipe_term(&pipe);
|
|
|
|
|
|
|
|
assert(strcmp("test", buf) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SC_HAVE_WRAP
|
|
|
|
|
2021-02-07 20:42:08 +03:00
|
|
|
bool fail_malloc = false;
|
|
|
|
void *__real_malloc(size_t n);
|
|
|
|
void *__wrap_malloc(size_t n)
|
|
|
|
{
|
|
|
|
if (fail_malloc) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return __real_malloc(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fail_realloc = false;
|
|
|
|
void *__real_realloc(void *p, size_t size);
|
|
|
|
void *__wrap_realloc(void *p, size_t n)
|
|
|
|
{
|
|
|
|
if (fail_realloc) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return __real_realloc(p, n);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fail_epollcreate1 = false;
|
|
|
|
int __real_epoll_create1(int flags);
|
|
|
|
int __wrap_epoll_create1(int flags)
|
|
|
|
{
|
|
|
|
if (fail_epollcreate1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return __real_epoll_create1(flags);
|
|
|
|
}
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
bool fail_close = false;
|
|
|
|
int __real_close(int fd);
|
|
|
|
int __wrap_close(int fd)
|
|
|
|
{
|
|
|
|
if (fail_close) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return __real_close(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fail_pipe = false;
|
|
|
|
int __real_pipe(int __pipedes[2]);
|
|
|
|
int __wrap_pipe(int __pipedes[2])
|
|
|
|
{
|
|
|
|
if (fail_pipe) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return __real_pipe(__pipedes);
|
|
|
|
}
|
|
|
|
|
2021-02-07 20:42:08 +03:00
|
|
|
void poll_fail_test()
|
|
|
|
{
|
|
|
|
bool fail;
|
|
|
|
int i;
|
|
|
|
struct sc_sock_poll poll;
|
|
|
|
struct sc_sock_pipe pipe[200];
|
|
|
|
|
|
|
|
fail_malloc = true;
|
|
|
|
assert(sc_sock_poll_init(&poll) == -1);
|
|
|
|
fail_malloc = false;
|
|
|
|
|
|
|
|
fail_epollcreate1 = true;
|
|
|
|
assert(sc_sock_poll_init(&poll) == -1);
|
|
|
|
fail_epollcreate1 = false;
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
fail = false;
|
|
|
|
fail_realloc = true;
|
|
|
|
for (i = 0; i < 200; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
fail = sc_sock_poll_add(&poll, &pipe[i].fdt, SC_SOCK_READ, NULL);
|
|
|
|
if (fail) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(fail);
|
|
|
|
for (int j = 0; j <= i; j++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[j]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
fail_realloc = false;
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
fail = false;
|
|
|
|
fail_realloc = true;
|
|
|
|
for (i = 0; i < 200; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
fail = sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL);
|
|
|
|
if (fail) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(fail);
|
|
|
|
for (int j = 0; j <= i; j++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[j]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
fail_realloc = false;
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
fail = false;
|
|
|
|
fail_realloc = true;
|
|
|
|
for (i = 0; i < 200; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
fail = sc_sock_poll_add(&poll, &pipe[i].fdt, SC_SOCK_WRITE, NULL);
|
|
|
|
if (fail) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(fail);
|
|
|
|
for (int j = 0; j <= i; j++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[j]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
fail_realloc = false;
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
fail = false;
|
|
|
|
fail_realloc = true;
|
|
|
|
for (i = 0; i < 200; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
fail = sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL);
|
|
|
|
if (fail) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(fail);
|
|
|
|
for (int j = 0; j <= i; j++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[j]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
fail_realloc = false;
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
fail = false;
|
|
|
|
fail_realloc = true;
|
|
|
|
for (i = 0; i < 200; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
fail = sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL);
|
|
|
|
if (fail) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(fail);
|
|
|
|
for (int j = 0; j <= i; j++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[j]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
fail_realloc = false;
|
|
|
|
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
fail = false;
|
|
|
|
fail_realloc = true;
|
|
|
|
for (i = 0; i < 200; i++) {
|
|
|
|
assert(sc_sock_pipe_init(&pipe[i], 0) == 0);
|
|
|
|
fail = sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL);
|
|
|
|
if (fail) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
assert(sc_sock_poll_add(&poll, &pipe[i].fdt,
|
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
}
|
|
|
|
assert(fail);
|
|
|
|
for (int j = 0; j <= i; j++) {
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_WRITE, NULL) == 0);
|
|
|
|
assert(sc_sock_poll_del(&poll, &pipe[j].fdt, SC_SOCK_READ, NULL) == 0);
|
|
|
|
assert(sc_sock_pipe_term(&pipe[j]) == 0);
|
|
|
|
}
|
|
|
|
assert(poll.count == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
fail_realloc = false;
|
|
|
|
}
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
void pipe_fail_test()
|
|
|
|
{
|
|
|
|
struct sc_sock_pipe pipe;
|
|
|
|
fail_pipe = true;
|
|
|
|
assert(sc_sock_pipe_init(&pipe, 0) != 0);
|
2021-02-04 02:35:34 +03:00
|
|
|
assert(*sc_sock_pipe_err(&pipe) != '\0');
|
2020-11-25 09:25:38 +03:00
|
|
|
fail_pipe = false;
|
|
|
|
assert(sc_sock_pipe_init(&pipe, 0) == 0);
|
|
|
|
fail_close = true;
|
|
|
|
assert(sc_sock_pipe_term(&pipe) == -1);
|
|
|
|
fail_close = false;
|
|
|
|
assert(sc_sock_pipe_term(&pipe) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
2021-02-07 20:42:08 +03:00
|
|
|
void poll_fail_test()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
void pipe_fail_test()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *server(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
2020-11-25 09:25:38 +03:00
|
|
|
int write_triggered = 2;
|
|
|
|
int rc, received = 0;
|
|
|
|
struct sc_sock server;
|
|
|
|
struct sc_sock accepted;
|
|
|
|
struct sc_sock_poll poll;
|
|
|
|
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
2021-02-04 02:35:34 +03:00
|
|
|
assert(*sc_sock_poll_err(&poll) == '\0');
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
assert(sc_sock_poll_init(&poll) == 0);
|
|
|
|
|
|
|
|
sc_sock_init(&server, 0, true, SC_SOCK_INET);
|
|
|
|
rc = sc_sock_listen(&server, "127.0.0.1", "11000");
|
|
|
|
assert(rc == 0);
|
|
|
|
|
|
|
|
rc = sc_sock_poll_add(&poll, &server.fdt, SC_SOCK_READ, &server);
|
|
|
|
assert(rc == 0);
|
|
|
|
bool done = false;
|
|
|
|
|
|
|
|
while (!done) {
|
|
|
|
int count = sc_sock_poll_wait(&poll, -1);
|
|
|
|
assert(rc != -1);
|
|
|
|
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
int ev = sc_sock_poll_event(&poll, i);
|
2021-02-01 07:11:12 +03:00
|
|
|
struct sc_sock *sock = sc_sock_poll_data(&poll, i);
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
if (ev == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sock == &server) {
|
|
|
|
if (ev & SC_SOCK_READ) {
|
|
|
|
rc = sc_sock_accept(&server, &accepted);
|
|
|
|
printf("accepted \n");
|
|
|
|
|
|
|
|
assert(rc == 0);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_poll_add(&poll, &accepted.fdt, SC_SOCK_WRITE,
|
2020-11-25 09:25:38 +03:00
|
|
|
&accepted) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ev & SC_SOCK_WRITE) {
|
|
|
|
assert(false);
|
|
|
|
}
|
2021-02-01 07:11:12 +03:00
|
|
|
} else {
|
2020-11-25 09:25:38 +03:00
|
|
|
if (ev & SC_SOCK_READ) {
|
|
|
|
char buf[8];
|
2020-12-28 11:03:24 +03:00
|
|
|
rc = sc_sock_recv(&accepted, buf, sizeof(buf), 0);
|
2020-11-25 09:25:38 +03:00
|
|
|
if (rc == 8) {
|
|
|
|
assert(strcmp(buf, "dataxxx") == 0);
|
|
|
|
received++;
|
2021-02-01 07:11:12 +03:00
|
|
|
} else if (rc == 0 || rc < 0) {
|
2020-11-25 09:25:38 +03:00
|
|
|
rc = sc_sock_poll_del(&poll, &accepted.fdt,
|
2021-02-01 07:11:12 +03:00
|
|
|
SC_SOCK_READ | SC_SOCK_WRITE,
|
2020-11-25 09:25:38 +03:00
|
|
|
&accepted);
|
|
|
|
assert(rc == 0);
|
|
|
|
rc = sc_sock_term(&accepted);
|
|
|
|
assert(rc == 0);
|
|
|
|
done = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (write_triggered > 0) {
|
|
|
|
write_triggered--;
|
|
|
|
if (write_triggered == 0) {
|
2021-02-01 07:11:12 +03:00
|
|
|
rc = sc_sock_poll_del(&poll, &accepted.fdt,
|
|
|
|
SC_SOCK_WRITE, &accepted);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(rc == 0);
|
2021-02-01 07:11:12 +03:00
|
|
|
} else {
|
|
|
|
rc = sc_sock_poll_add(&poll, &accepted.fdt,
|
|
|
|
SC_SOCK_READ, &accepted);
|
2020-11-25 09:25:38 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(write_triggered == 0);
|
|
|
|
assert(sc_sock_term(&server) == 0);
|
|
|
|
assert(sc_sock_poll_term(&poll) == 0);
|
|
|
|
|
|
|
|
printf("Received :%d \n", received);
|
|
|
|
assert(received == 10000);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void *client(void *arg)
|
2020-11-25 09:25:38 +03:00
|
|
|
{
|
2021-02-07 22:31:04 +03:00
|
|
|
(void) arg;
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
int rc;
|
|
|
|
struct sc_sock sock;
|
|
|
|
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
|
|
|
|
|
|
|
rc = sc_sock_connect(&sock, "127.0.0.1", "11000", NULL, NULL);
|
|
|
|
if (rc != 0) {
|
|
|
|
assert(sc_sock_term(&sock) == 0);
|
|
|
|
sleep(1);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int j = 0; j < 10000; j++) {
|
2020-12-28 11:03:24 +03:00
|
|
|
rc = sc_sock_send(&sock, "dataxxx", 8, 0);
|
2020-11-25 09:25:38 +03:00
|
|
|
assert(rc == 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = sc_sock_term(&sock);
|
|
|
|
assert(rc == 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_poll()
|
|
|
|
{
|
|
|
|
struct sc_thread thread1;
|
|
|
|
struct sc_thread thread2;
|
|
|
|
|
|
|
|
sc_thread_init(&thread1);
|
|
|
|
sc_thread_init(&thread2);
|
|
|
|
|
|
|
|
assert(sc_thread_start(&thread1, server, NULL) == 0);
|
|
|
|
assert(sc_thread_start(&thread2, client, NULL) == 0);
|
|
|
|
|
|
|
|
assert(sc_thread_term(&thread1) == 0);
|
|
|
|
assert(sc_thread_term(&thread2) == 0);
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:11:12 +03:00
|
|
|
void test_err()
|
|
|
|
{
|
|
|
|
struct sc_sock sock;
|
|
|
|
char buf[128];
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_listen(&sock, "127.0.0.1x", "8004") != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_UNIX);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_listen(&sock, "/", "8004") != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET6);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_listen(&sock, "/", "8004") != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET6);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_listen(&sock, "0.0.0.0", "99999") != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_listen(&sock, "0.0.0.3", "99999") != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET6);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_connect(&sock, "::1", "8006", NULL, NULL) != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_UNIX);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_connect(&sock, "/", "8006", NULL, NULL) != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_connect(&sock, "0.0.0.0", "8006", NULL, NULL) != 0);
|
|
|
|
sc_sock_print(&sock, buf, 128);
|
|
|
|
sc_sock_local_str(&sock, buf, 128);
|
|
|
|
assert(*buf == '\0');
|
|
|
|
sc_sock_remote_str(&sock, buf, 128);
|
|
|
|
assert(*buf == '\0');
|
|
|
|
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_connect(&sock, "0.0.0.0", "8006", "127.0.0.1", "8080") != 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
sc_sock_init(&sock, 0, true, SC_SOCK_INET6);
|
2021-02-01 07:11:12 +03:00
|
|
|
assert(sc_sock_connect(&sock, "0.0.0.0", "8006", "::1", "8080") != 0);
|
|
|
|
assert(sc_sock_term(&sock) == 0);
|
2021-02-04 11:17:43 +03:00
|
|
|
|
|
|
|
struct sc_sock_poll p = {0};
|
|
|
|
assert(sc_sock_poll_wait(&p, 100) != 0);
|
|
|
|
assert(*sc_sock_poll_err(&p) != '\0');
|
2021-02-01 07:11:12 +03:00
|
|
|
}
|
|
|
|
|
2020-11-25 09:25:38 +03:00
|
|
|
int main()
|
|
|
|
{
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
WSADATA data;
|
|
|
|
|
|
|
|
int rc = WSAStartup(MAKEWORD(2, 2), &data);
|
|
|
|
assert(rc == 0);
|
|
|
|
assert(LOBYTE(data.wVersion) == 2 && HIBYTE(data.wVersion) == 2);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
test1();
|
|
|
|
test_ip4();
|
|
|
|
|
|
|
|
#if defined(__x86_64__)
|
|
|
|
test_ip6();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(__APPLE__)
|
|
|
|
test_unix();
|
|
|
|
#endif
|
|
|
|
test_pipe();
|
|
|
|
pipe_fail_test();
|
2021-02-07 20:42:08 +03:00
|
|
|
poll_fail_test();
|
2020-11-25 09:25:38 +03:00
|
|
|
test_poll();
|
2021-02-01 07:11:12 +03:00
|
|
|
test_err();
|
2021-02-07 20:42:08 +03:00
|
|
|
test_poll_mass();
|
2020-11-25 09:25:38 +03:00
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
rc = WSACleanup();
|
|
|
|
assert(rc == 0);
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|