]> git.meshlink.io Git - meshlink/blob - test/utils.c
Never automatically try to bind to ports >= 32768.
[meshlink] / test / utils.c
1 #define _GNU_SOURCE 1
2
3 #include <stdio.h>
4 #include <stdlib.h>
5 #include <pthread.h>
6 #include <assert.h>
7 #include <errno.h>
8
9 #include "utils.h"
10
11 void set_sync_flag(struct sync_flag *s, bool value) {
12         pthread_mutex_lock(&s->mutex);
13         s->flag = value;
14         pthread_cond_broadcast(&s->cond);
15         pthread_mutex_unlock(&s->mutex);
16 }
17
18 bool check_sync_flag(struct sync_flag *s) {
19         bool flag;
20         pthread_mutex_lock(&s->mutex);
21         flag = s->flag;
22         pthread_mutex_unlock(&s->mutex);
23         return flag;
24 }
25
26 bool wait_sync_flag(struct sync_flag *s, int seconds) {
27         struct timespec timeout;
28         clock_gettime(CLOCK_REALTIME, &timeout);
29         timeout.tv_sec += seconds;
30
31         pthread_mutex_lock(&s->mutex);
32
33         while(!s->flag) {
34                 if(!pthread_cond_timedwait(&s->cond, &s->mutex, &timeout) || errno != EINTR) {
35                         break;
36                 }
37         }
38
39         pthread_mutex_unlock(&s->mutex);
40
41         return s->flag;
42 }
43
44 void open_meshlink_pair(meshlink_handle_t **pa, meshlink_handle_t **pb, const char *prefix) {
45         // Create two new MeshLink instances
46
47         *pa = *pb = NULL;
48
49         char *a_name, *b_name;
50         int ret_val;
51         (void)ret_val;
52
53         ret_val = asprintf(&a_name, "%s_conf.1", prefix);
54         assert(a_name);
55
56         ret_val = asprintf(&b_name, "%s_conf.2", prefix);
57         assert(b_name);
58
59         meshlink_handle_t *a = meshlink_open(a_name, "a", prefix, DEV_CLASS_BACKBONE);
60         assert(a);
61
62         meshlink_handle_t *b = meshlink_open(b_name, "b", prefix, DEV_CLASS_BACKBONE);
63         assert(b);
64
65         meshlink_enable_discovery(a, false);
66         meshlink_enable_discovery(b, false);
67
68         // Import and export both side's data
69
70         meshlink_add_address(a, "localhost");
71
72         char *data = meshlink_export(a);
73         assert(data);
74         assert(meshlink_import(b, data));
75         free(data);
76
77         data = meshlink_export(b);
78         assert(data);
79         assert(meshlink_import(a, data));
80         free(data);
81
82         *pa = a;
83         *pb = b;
84 }
85
86 // Don't poll in the application thread, use a condition variable to signal when the peer is online.
87 static void pair_status_cb(meshlink_handle_t *mesh, meshlink_node_t *node, bool reachable) {
88         (void)node;
89
90         set_sync_flag(mesh->priv, reachable);
91 }
92
93 void start_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
94         struct sync_flag pair_status = {.flag = false};
95
96         a->priv = &pair_status;
97         meshlink_set_node_status_cb(a, pair_status_cb);
98
99         meshlink_start(a);
100         meshlink_start(b);
101
102         assert(wait_sync_flag(&pair_status, 5));
103
104         meshlink_set_node_status_cb(a, NULL);
105         a->priv = NULL;
106 }
107
108 void stop_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
109         meshlink_stop(a);
110         meshlink_stop(b);
111 }
112
113 void close_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b, const char *prefix) {
114         int ret_val;
115         (void)ret_val;
116         meshlink_close(a);
117         meshlink_close(b);
118
119         if(prefix) {
120                 char *a_name, *b_name;
121
122                 ret_val = asprintf(&a_name, "%s_conf.1", prefix);
123                 assert(a_name);
124                 assert(meshlink_destroy(a_name));
125
126                 ret_val = asprintf(&b_name, "%s_conf.2", prefix);
127                 assert(b_name);
128                 assert(meshlink_destroy(b_name));
129         }
130 }