]> git.meshlink.io Git - meshlink/blob - test/utils.c
f1f1c7700334e20ba1080b5a2e850c779203bbba
[meshlink] / test / utils.c
1 #define _GNU_SOURCE 1
2
3 #ifdef NDEBUG
4 #undef NDEBUG
5 #endif
6
7 #include <stdio.h>
8 #include <stdlib.h>
9 #include <pthread.h>
10 #include <assert.h>
11 #include <errno.h>
12 #include <time.h>
13
14 #include "utils.h"
15
16 void init_sync_flag(struct sync_flag *s) {
17         assert(pthread_mutex_init(&s->mutex, NULL) == 0);
18         assert(pthread_cond_init(&s->cond, NULL) == 0);
19 }
20
21 void set_sync_flag(struct sync_flag *s, bool value) {
22         assert(pthread_mutex_lock(&s->mutex) == 0);
23         s->flag = value;
24         assert(pthread_cond_broadcast(&s->cond) == 0);
25         assert(pthread_mutex_unlock(&s->mutex) == 0);
26 }
27
28 void reset_sync_flag(struct sync_flag *s) {
29         assert(pthread_mutex_lock(&s->mutex) == 0);
30         s->flag = false;
31         assert(pthread_mutex_unlock(&s->mutex) == 0);
32 }
33
34 bool check_sync_flag(struct sync_flag *s) {
35         bool flag;
36         assert(pthread_mutex_lock(&s->mutex) == 0);
37         flag = s->flag;
38         assert(pthread_mutex_unlock(&s->mutex) == 0);
39         return flag;
40 }
41
42 bool wait_sync_flag(struct sync_flag *s, int seconds) {
43         struct timespec timeout;
44         clock_gettime(CLOCK_REALTIME, &timeout);
45         timeout.tv_sec += seconds;
46
47         assert(pthread_mutex_lock(&s->mutex) == 0);
48
49         while(!s->flag) {
50                 if(!pthread_cond_timedwait(&s->cond, &s->mutex, &timeout) || errno != EINTR) {
51                         break;
52                 }
53         }
54
55         assert(pthread_mutex_unlock(&s->mutex) == 0);
56
57         return s->flag;
58 }
59
60 void link_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
61         // Import and export both side's data
62
63         assert(meshlink_set_canonical_address(a, meshlink_get_self(a), "localhost", NULL));
64         assert(meshlink_set_canonical_address(b, meshlink_get_self(b), "localhost", NULL));
65
66         char *data = meshlink_export(a);
67         assert(data);
68         assert(meshlink_import(b, data));
69         free(data);
70
71         data = meshlink_export(b);
72         assert(data);
73         assert(meshlink_import(a, data));
74         free(data);
75 }
76
77 void open_meshlink_pair(meshlink_handle_t **pa, meshlink_handle_t **pb, const char *prefix) {
78         // Create two new MeshLink instances
79
80         *pa = *pb = NULL;
81
82         char *a_name, *b_name;
83
84         assert(asprintf(&a_name, "%s_conf.1", prefix) > 0);
85         assert(a_name);
86
87         assert(asprintf(&b_name, "%s_conf.2", prefix) > 0);
88         assert(b_name);
89
90         assert(meshlink_destroy(a_name));
91         assert(meshlink_destroy(b_name));
92
93         meshlink_handle_t *a = meshlink_open(a_name, "a", prefix, DEV_CLASS_BACKBONE);
94         assert(a);
95
96         meshlink_handle_t *b = meshlink_open(b_name, "b", prefix, DEV_CLASS_BACKBONE);
97         assert(b);
98
99         free(a_name);
100         free(b_name);
101
102         meshlink_enable_discovery(a, false);
103         meshlink_enable_discovery(b, false);
104
105         link_meshlink_pair(a, b);
106
107         *pa = a;
108         *pb = b;
109 }
110
111 void open_meshlink_pair_ephemeral(meshlink_handle_t **pa, meshlink_handle_t **pb, const char *prefix) {
112         // Create two new MeshLink instances
113
114         *pa = *pb = NULL;
115
116         meshlink_handle_t *a = meshlink_open_ephemeral("a", prefix, DEV_CLASS_BACKBONE);
117         meshlink_handle_t *b = meshlink_open_ephemeral("b", prefix, DEV_CLASS_BACKBONE);
118
119         assert(a);
120         assert(b);
121
122         meshlink_enable_discovery(a, false);
123         meshlink_enable_discovery(b, false);
124
125         link_meshlink_pair(a, b);
126
127         *pa = a;
128         *pb = b;
129 }
130
131 // Don't poll in the application thread, use a condition variable to signal when the peer is online.
132 static void pair_status_cb(meshlink_handle_t *mesh, meshlink_node_t *node, bool reachable) {
133         (void)node;
134
135         if(reachable) {
136                 set_sync_flag(mesh->priv, true);
137         }
138 }
139
140 void start_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
141         struct sync_flag pair_status = {.flag = false};
142         init_sync_flag(&pair_status);
143
144         a->priv = &pair_status;
145         meshlink_set_node_status_cb(a, pair_status_cb);
146
147         assert(meshlink_start(a));
148         assert(meshlink_start(b));
149
150         assert(wait_sync_flag(&pair_status, 5));
151
152         meshlink_set_node_status_cb(a, NULL);
153         a->priv = NULL;
154 }
155
156 void stop_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
157         meshlink_stop(a);
158         meshlink_stop(b);
159 }
160
161 void close_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
162         meshlink_close(a);
163         meshlink_close(b);
164 }
165
166 void log_cb(meshlink_handle_t *mesh, meshlink_log_level_t level, const char *text) {
167         static const char *levelstr[] = {
168                 [MESHLINK_DEBUG] = "DEBUG",
169                 [MESHLINK_INFO] = "INFO",
170                 [MESHLINK_WARNING] = "WARNING",
171                 [MESHLINK_ERROR] = "ERROR",
172                 [MESHLINK_CRITICAL] = "CRITICAL",
173         };
174
175         static struct timespec ts0;
176         struct timespec ts;
177
178         clock_gettime(CLOCK_MONOTONIC, &ts);
179
180         if(ts0.tv_sec == 0) {
181                 ts0 = ts;
182         }
183
184         float diff = (ts.tv_sec - ts0.tv_sec) + (ts.tv_nsec - ts0.tv_nsec) * 1e-9;
185
186         fprintf(stderr, "%7.3f (%s) [%s] %s\n",
187                 diff,
188                 mesh ? mesh->name : "",
189                 levelstr[level],
190                 text);
191 }