]> git.meshlink.io Git - meshlink/blobdiff - src/meshlink.c
Finished sockaddr_t to sockaddr_storage copy on meshlink_edge_t.
[meshlink] / src / meshlink.c
index d028d4289e0257886bd70035d7253c022d434299..9011879312ea34b27a5bb0833782640ab1372ea8 100644 (file)
@@ -1042,8 +1042,10 @@ void meshlink_send_from_queue(event_loop_t* el,meshlink_handle_t *mesh) {
 
        outpacketqueue_t* p = meshlink_queue_pop(&mesh->outpacketqueue);
        if(!p)
+       {
                pthread_mutex_unlock(&(mesh->mesh_mutex));
                return;
+       }
 
        if (sizeof(meshlink_packethdr_t) + p->len > MAXSIZE) {
                pthread_mutex_unlock(&(mesh->mesh_mutex));
@@ -1812,7 +1814,7 @@ void meshlink_whitelist(meshlink_handle_t *mesh, meshlink_node_t *node) {
 /* Hint that a hostname may be found at an address
  * See header file for detailed comment.
  */
-extern void meshlink_hint_address(meshlink_handle_t *mesh, meshlink_node_t *node, const struct sockaddr *addr) {
+void meshlink_hint_address(meshlink_handle_t *mesh, meshlink_node_t *node, const struct sockaddr *addr) {
        if(!mesh || !node || !addr)
                return;
        
@@ -1834,6 +1836,57 @@ extern void meshlink_hint_address(meshlink_handle_t *mesh, meshlink_node_t *node
        // @TODO do we want to fire off a connection attempt right away?
 }
 
+/* Return an array of edges in the current network graph.
+ * Data captures the current state and will not be updated.
+ * Caller must deallocate data when done.
+ */
+meshlink_edge_t **meshlink_get_all_edges_state(meshlink_handle_t *mesh, size_t *nmemb) {
+       if(!mesh || !nmemb) {
+               meshlink_errno = MESHLINK_EINVAL;
+               return NULL;
+       }
+
+       pthread_mutex_lock(&(mesh->mesh_mutex));
+       
+       meshlink_edge_t **result = NULL;
+       meshlink_edge_t *copy = NULL;
+
+       // mesh->edges->count is the max size
+       *nmemb = mesh->edges->count;
+
+       result = xzalloc(*nmemb * sizeof (meshlink_edge_t*));
+
+       if(result) {
+               meshlink_edge_t **p = result;
+               for splay_each(edge_t, e, mesh->edges) {
+                       // skip edges that do not represent a two-directional connection
+                       if((!e->reverse) || (e->reverse->to != e->from)) {
+                               *nmemb--;
+                               continue;
+                       }
+                       // copy the edge so it can't be mutated
+                       copy = xzalloc(sizeof *copy);
+                       copy->from = (meshlink_node_t*)e->from;
+                       copy->to = (meshlink_node_t*)e->to;
+#ifdef HAVE_STRUCT_SOCKADDR_STORAGE
+                       copy->address = e->address.storage;
+#endif
+                       copy->options = e->options;
+                       copy->weight = e->weight;
+                       *p++ = copy;
+               }
+               // shrink result to the actual amount of memory used
+               result = realloc(result, *nmemb * sizeof (meshlink_edge_t*));
+       } else {
+               *nmemb = 0;
+               meshlink_errno = MESHLINK_ENOMEM;
+       }
+
+       pthread_mutex_unlock(&(mesh->mesh_mutex));
+
+       return result;
+}
+
 static void __attribute__((constructor)) meshlink_init(void) {
        crypto_init();
 }