17 void init_sync_flag(struct sync_flag *s) {
18 assert(pthread_mutex_init(&s->mutex, NULL) == 0);
19 assert(pthread_cond_init(&s->cond, NULL) == 0);
23 void set_sync_flag(struct sync_flag *s, bool value) {
24 assert(pthread_mutex_lock(&s->mutex) == 0);
26 assert(pthread_cond_broadcast(&s->cond) == 0);
27 assert(pthread_mutex_unlock(&s->mutex) == 0);
30 void reset_sync_flag(struct sync_flag *s) {
31 assert(pthread_mutex_lock(&s->mutex) == 0);
33 assert(pthread_mutex_unlock(&s->mutex) == 0);
36 bool check_sync_flag(struct sync_flag *s) {
38 assert(pthread_mutex_lock(&s->mutex) == 0);
40 assert(pthread_mutex_unlock(&s->mutex) == 0);
44 bool wait_sync_flag(struct sync_flag *s, int seconds) {
45 struct timespec timeout;
46 clock_gettime(CLOCK_REALTIME, &timeout);
47 timeout.tv_sec += seconds;
49 assert(pthread_mutex_lock(&s->mutex) == 0);
52 if(!pthread_cond_timedwait(&s->cond, &s->mutex, &timeout) || errno != EINTR) {
57 assert(pthread_mutex_unlock(&s->mutex) == 0);
62 void link_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
63 // Import and export both side's data
65 assert(meshlink_set_canonical_address(a, meshlink_get_self(a), "localhost", NULL));
66 assert(meshlink_set_canonical_address(b, meshlink_get_self(b), "localhost", NULL));
68 char *data = meshlink_export(a);
70 assert(meshlink_import(b, data));
73 data = meshlink_export(b);
75 assert(meshlink_import(a, data));
79 void open_meshlink_pair(meshlink_handle_t **pa, meshlink_handle_t **pb, const char *prefix) {
80 // Create two new MeshLink instances
84 char *a_name, *b_name;
86 assert(asprintf(&a_name, "%s_conf.1", prefix) > 0);
89 assert(asprintf(&b_name, "%s_conf.2", prefix) > 0);
92 assert(meshlink_destroy(a_name));
93 assert(meshlink_destroy(b_name));
95 meshlink_handle_t *a = meshlink_open(a_name, "a", prefix, DEV_CLASS_BACKBONE);
98 meshlink_handle_t *b = meshlink_open(b_name, "b", prefix, DEV_CLASS_BACKBONE);
104 link_meshlink_pair(a, b);
110 void open_meshlink_pair_ephemeral(meshlink_handle_t **pa, meshlink_handle_t **pb, const char *prefix) {
111 // Create two new MeshLink instances
115 meshlink_handle_t *a = meshlink_open_ephemeral("a", prefix, DEV_CLASS_BACKBONE);
116 meshlink_handle_t *b = meshlink_open_ephemeral("b", prefix, DEV_CLASS_BACKBONE);
121 link_meshlink_pair(a, b);
127 // Don't poll in the application thread, use a condition variable to signal when the peer is online.
128 static void pair_status_cb(meshlink_handle_t *mesh, meshlink_node_t *node, bool reachable) {
131 if(reachable && meshlink_get_self(mesh) != node) {
132 set_sync_flag(mesh->priv, true);
136 void start_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
137 struct sync_flag pair_status = {.flag = false};
138 init_sync_flag(&pair_status);
140 a->priv = &pair_status;
141 meshlink_set_node_status_cb(a, pair_status_cb);
143 assert(meshlink_start(a));
144 assert(meshlink_start(b));
146 assert(wait_sync_flag(&pair_status, 5));
148 meshlink_set_node_status_cb(a, NULL);
152 void start_full_tiny_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
153 struct sync_flag pair_status = {.flag = false};
154 init_sync_flag(&pair_status);
156 a->priv = &pair_status;
157 full_meshlink_set_node_status_cb(a, pair_status_cb);
159 assert(full_meshlink_start(a));
160 assert(meshlink_start(b));
162 assert(wait_sync_flag(&pair_status, 5));
164 full_meshlink_set_node_status_cb(a, NULL);
168 void stop_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
173 void stop_full_tiny_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
174 full_meshlink_stop(a);
178 void close_meshlink_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
183 void close_full_tiny_pair(meshlink_handle_t *a, meshlink_handle_t *b) {
184 full_meshlink_close(a);
188 void log_cb(meshlink_handle_t *mesh, meshlink_log_level_t level, const char *text) {
189 static const char *levelstr[] = {
190 [MESHLINK_DEBUG] = "DEBUG",
191 [MESHLINK_INFO] = "INFO",
192 [MESHLINK_WARNING] = "WARNING",
193 [MESHLINK_ERROR] = "ERROR",
194 [MESHLINK_CRITICAL] = "CRITICAL",
197 static struct timespec ts0;
200 clock_gettime(CLOCK_MONOTONIC, &ts);
202 if(ts0.tv_sec == 0) {
206 float diff = (ts.tv_sec - ts0.tv_sec) + (ts.tv_nsec - ts0.tv_nsec) * 1e-9;
208 fprintf(stderr, "%7.3f (%s) [%s] %s\n",
210 mesh ? mesh->name : "",