Graphviz 13.0.0~dev.20250121.0651
Loading...
Searching...
No Matches
stuff.c
Go to the documentation of this file.
1/*************************************************************************
2 * Copyright (c) 2011 AT&T Intellectual Property
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * https://www.eclipse.org/legal/epl-v10.html
7 *
8 * Contributors: Details at https://graphviz.org
9 *************************************************************************/
10
11
12#include "config.h"
13#include <math.h>
14#include <neatogen/neato.h>
15#include <neatogen/stress.h>
16#include <stdatomic.h>
17#include <stdlib.h>
18#include <time.h>
19#include <util/alloc.h>
20#ifndef _WIN32
21#include <unistd.h>
22#endif
23
24static double Epsilon2;
25static Agnode_t *choose_node(graph_t *, int);
26static void make_spring(graph_t *, Agnode_t *, Agnode_t *, double);
27static void move_node(graph_t *, int, Agnode_t *);
28
29static double fpow32(double x)
30{
31 x = sqrt(x);
32 return x * x * x;
33}
34
35static double distvec(double *p0, double *p1, double *vec)
36{
37 int k;
38 double dist = 0.0;
39
40 for (k = 0; k < Ndim; k++) {
41 vec[k] = p0[k] - p1[k];
42 dist += vec[k] * vec[k];
43 }
44 dist = sqrt(dist);
45 return dist;
46}
47
48double **new_array(int m, int n, double ival)
49{
50 int i, j;
51
52 double **rv = gv_calloc(m, sizeof(double*));
53 double *mem = gv_calloc(m * n, sizeof(double));
54 for (i = 0; i < m; i++) {
55 rv[i] = mem;
56 mem += n;
57 for (j = 0; j < n; j++)
58 rv[i][j] = ival;
59 }
60 return rv;
61}
62
63void free_array(double **rv)
64{
65 if (rv) {
66 free(rv[0]);
67 free(rv);
68 }
69}
70
71
72static double ***new_3array(int m, int n, int p, double ival)
73{
74 int i, j, k;
75
76 double ***rv = gv_calloc(m + 1, sizeof(double**));
77 for (i = 0; i < m; i++) {
78 rv[i] = gv_calloc(n + 1, sizeof(double*));
79 for (j = 0; j < n; j++) {
80 rv[i][j] = gv_calloc(p, sizeof(double));
81 for (k = 0; k < p; k++)
82 rv[i][j][k] = ival;
83 }
84 rv[i][j] = NULL; /* NULL terminate so we can clean up */
85 }
86 rv[i] = NULL;
87 return rv;
88}
89
90static void free_3array(double ***rv)
91{
92 int i, j;
93
94 if (rv) {
95 for (i = 0; rv[i]; i++) {
96 for (j = 0; rv[i][j]; j++)
97 free(rv[i][j]);
98 free(rv[i]);
99 }
100 free(rv);
101 }
102}
103
104
105/* lenattr:
106 * Return 1 if attribute not defined
107 * Return 2 if attribute string bad
108 */
109static int lenattr(edge_t* e, Agsym_t* index, double* val)
110{
111 char* s;
112
113 if (index == NULL)
114 return 1;
115
116 s = agxget(e, index);
117 if (*s == '\0') return 1;
118
119 if (sscanf(s, "%lf", val) < 1 || *val < 0 || (*val == 0 && !Nop)) {
120 agwarningf("bad edge len \"%s\"", s);
121 return 2;
122 }
123 else
124 return 0;
125}
126
127
128/* degreeKind;
129 * Returns degree of n ignoring loops and multiedges.
130 * Returns 0, 1 or many (2)
131 * For case of 1, returns other endpoint of edge.
132 */
133static int degreeKind(graph_t * g, node_t * n, node_t ** op)
134{
135 edge_t *ep;
136 int deg = 0;
137 node_t *other = NULL;
138
139 for (ep = agfstedge(g, n); ep; ep = agnxtedge(g, ep, n)) {
140 if (aghead(ep) == agtail(ep))
141 continue; /* ignore loops */
142 if (deg == 1) {
143 if ((agtail(ep) == n && aghead(ep) == other) || /* ignore multiedge */
144 (agtail(ep) == other && aghead(ep) == n))
145 continue;
146 return 2;
147 } else { /* deg == 0 */
148 if (agtail(ep) == n)
149 other = aghead(ep);
150 else
151 other = agtail(ep);
152 *op = other;
153 deg++;
154 }
155 }
156 return deg;
157}
158
159
160/* prune:
161 * np is at end of a chain. If its degree is 0, remove it.
162 * If its degree is 1, remove it and recurse.
163 * If its degree > 1, stop.
164 * The node next is the next node to be visited during iteration.
165 * If it is equal to a node being deleted, set it to next one.
166 * Delete from root graph, since G may be a connected component subgraph.
167 * Return next.
168 */
169static node_t *prune(graph_t * G, node_t * np, node_t * next)
170{
171 node_t *other;
172 int deg;
173
174 while (np) {
175 deg = degreeKind(G, np, &other);
176 if (deg == 0) {
177 if (next == np)
178 next = agnxtnode(G, np);
179 agdelete(G->root, np);
180 np = 0;
181 } else if (deg == 1) {
182 if (next == np)
183 next = agnxtnode(G, np);
184 agdelete(G->root, np);
185 np = other;
186 } else
187 np = 0;
188
189 }
190 return next;
191}
192
193static double setEdgeLen(graph_t * G, node_t * np, Agsym_t* lenx, double dfltlen)
194{
195 edge_t *ep;
196 double total_len = 0.0;
197 double len;
198 int err;
199
200 for (ep = agfstout(G, np); ep; ep = agnxtout(G, ep)) {
201 if ((err = lenattr(ep, lenx, &len))) {
202 if (err == 2) agerr(AGPREV, " in %s - setting to %.02f\n", agnameof(G), dfltlen);
203 len = dfltlen;
204 }
205 ED_dist(ep) = len;
206 total_len += len;
207 }
208 return total_len;
209}
210
211/* scan_graph_mode:
212 * Prepare the graph and data structures depending on the layout mode.
213 * If Reduce is true, eliminate singletons and trees. Since G may be a
214 * subgraph, we remove the nodes from the root graph.
215 * Return the number of nodes in the reduced graph.
216 */
218{
219 int i, nV, nE, deg;
220 char *str;
221 node_t *np, *xp, *other;
222 double total_len = 0.0;
223 double dfltlen = 1.0;
224 Agsym_t* lenx;
225
226 if (Verbose)
227 fprintf(stderr, "Scanning graph %s, %d nodes\n", agnameof(G),
228 agnnodes(G));
229
230
231 /* Eliminate singletons and trees */
232 if (Reduce) {
233 for (np = agfstnode(G); np; np = xp) {
234 xp = agnxtnode(G, np);
235 deg = degreeKind(G, np, &other);
236 if (deg == 0) { /* singleton node */
237 agdelete(G->root, np);
238 } else if (deg == 1) {
239 agdelete(G->root, np);
240 xp = prune(G, other, xp);
241 }
242 }
243 }
244
245 nV = agnnodes(G);
246 nE = agnedges(G);
247
248 lenx = agattr(G, AGEDGE, "len", 0);
249 if (mode == MODE_KK) {
250 Epsilon = .0001 * nV;
251 getdouble(G, "epsilon", &Epsilon);
252 if ((str = agget(G->root, "Damping")))
253 Damping = atof(str);
254 else
255 Damping = .99;
256 GD_neato_nlist(G) = gv_calloc(nV + 1, sizeof(node_t*));
257 for (i = 0, np = agfstnode(G); np; np = agnxtnode(G, np)) {
258 GD_neato_nlist(G)[i] = np;
259 ND_id(np) = i++;
260 ND_heapindex(np) = -1;
261 total_len += setEdgeLen(G, np, lenx, dfltlen);
262 }
263 } else if (mode == MODE_SGD) {
264 Epsilon = .01;
265 getdouble(G, "epsilon", &Epsilon);
266 GD_neato_nlist(G) = gv_calloc(nV + 1, sizeof(node_t*)); // not sure why but sometimes needs the + 1
267 for (i = 0, np = agfstnode(G); np; np = agnxtnode(G, np)) {
268 GD_neato_nlist(G)[i] = np;
269 ND_id(np) = i++;
270 total_len += setEdgeLen(G, np, lenx, dfltlen);
271 }
272 } else {
274 getdouble(G, "epsilon", &Epsilon);
275 for (i = 0, np = agfstnode(G); np; np = agnxtnode(G, np)) {
276 ND_id(np) = i++;
277 total_len += setEdgeLen(G, np, lenx, dfltlen);
278 }
279 }
280
281 str = agget(G, "defaultdist");
282 if (str && str[0])
283 Initial_dist = fmax(Epsilon, atof(str));
284 else
285 Initial_dist = total_len / (nE > 0 ? nE : 1) * sqrt(nV) + 1;
286
287 if (!Nop && mode == MODE_KK) {
288 GD_dist(G) = new_array(nV, nV, Initial_dist);
289 GD_spring(G) = new_array(nV, nV, 1.0);
290 GD_sum_t(G) = new_array(nV, Ndim, 1.0);
291 GD_t(G) = new_3array(nV, nV, Ndim, 0.0);
292 }
293
294 return nV;
295}
296
298{
299 return scan_graph_mode(g, MODE_KK);
300}
301
303{
305 if (!Nop) {
309 free_3array(GD_t(g));
310 GD_t(g) = NULL;
311 }
312}
313
314void jitter_d(node_t * np, int nG, int n)
315{
316 int k;
317 for (k = n; k < Ndim; k++)
318 ND_pos(np)[k] = nG * drand48();
319}
320
321void jitter3d(node_t * np, int nG)
322{
323 jitter_d(np, nG, 2);
324}
325
326void randompos(node_t * np, int nG)
327{
328 ND_pos(np)[0] = nG * drand48();
329 ND_pos(np)[1] = nG * drand48();
330 if (Ndim > 2)
331 jitter3d(np, nG);
332}
333
335{
336 int init, i;
337 node_t *np;
338 static atomic_flag once;
339
340 if (Verbose)
341 fprintf(stderr, "Setting initial positions\n");
342
344 if (init == INIT_REGULAR)
345 return;
346 if (init == INIT_SELF && !atomic_flag_test_and_set(&once)) {
347 agwarningf("start=0 not supported with mode=self - ignored\n");
348 }
349
350 for (i = 0; (np = GD_neato_nlist(G)[i]); i++) {
351 if (hasPos(np))
352 continue;
353 randompos(np, 1);
354 }
355}
356
357void diffeq_model(graph_t * G, int nG)
358{
359 int i, j, k;
360 double dist, **D, **K, del[MAXDIM], f;
361 node_t *vi, *vj;
362 edge_t *e;
363
364 if (Verbose) {
365 fprintf(stderr, "Setting up spring model: ");
366 start_timer();
367 }
368 /* init springs */
369 K = GD_spring(G);
370 D = GD_dist(G);
371 for (i = 0; i < nG; i++) {
372 for (j = 0; j < i; j++) {
373 f = Spring_coeff / (D[i][j] * D[i][j]);
374 if ((e = agfindedge(G, GD_neato_nlist(G)[i], GD_neato_nlist(G)[j])))
375 f = f * ED_factor(e);
376 K[i][j] = K[j][i] = f;
377 }
378 }
379
380 /* init differential equation solver */
381 for (i = 0; i < nG; i++)
382 for (k = 0; k < Ndim; k++)
383 GD_sum_t(G)[i][k] = 0.0;
384
385 for (i = 0; (vi = GD_neato_nlist(G)[i]); i++) {
386 for (j = 0; j < nG; j++) {
387 if (i == j)
388 continue;
389 vj = GD_neato_nlist(G)[j];
390 dist = distvec(ND_pos(vi), ND_pos(vj), del);
391 for (k = 0; k < Ndim; k++) {
392 GD_t(G)[i][j][k] =
393 GD_spring(G)[i][j] * (del[k] -
394 GD_dist(G)[i][j] * del[k] /
395 dist);
396 GD_sum_t(G)[i][k] += GD_t(G)[i][j][k];
397 }
398 }
399 }
400 if (Verbose) {
401 fprintf(stderr, "%.2f sec\n", elapsed_sec());
402 }
403}
404
405/* total_e:
406 * Return 2*energy of system.
407 */
408static double total_e(graph_t * G, int nG)
409{
410 int i, j, d;
411 double e = 0.0; /* 2*energy */
412 double t0; /* distance squared */
413 double t1;
414 node_t *ip, *jp;
415
416 for (i = 0; i < nG - 1; i++) {
417 ip = GD_neato_nlist(G)[i];
418 for (j = i + 1; j < nG; j++) {
419 jp = GD_neato_nlist(G)[j];
420 for (t0 = 0.0, d = 0; d < Ndim; d++) {
421 t1 = ND_pos(ip)[d] - ND_pos(jp)[d];
422 t0 += t1 * t1;
423 }
424 e = e + GD_spring(G)[i][j] *
425 (t0 + GD_dist(G)[i][j] * GD_dist(G)[i][j]
426 - 2.0 * GD_dist(G)[i][j] * sqrt(t0));
427 }
428 }
429 return e;
430}
431
432void solve_model(graph_t * G, int nG)
433{
434 node_t *np;
435
437
438 while ((np = choose_node(G, nG))) {
439 move_node(G, nG, np);
440 }
441 if (Verbose) {
442 fprintf(stderr, "\nfinal e = %f", total_e(G, nG));
443 fprintf(stderr, " %d%s iterations %.2f sec\n",
444 GD_move(G), GD_move(G) == MaxIter ? "!" : "",
445 elapsed_sec());
446 }
447 if (GD_move(G) == MaxIter)
448 agwarningf("Max. iterations (%d) reached on graph %s\n",
449 MaxIter, agnameof(G));
450}
451
452static void update_arrays(graph_t * G, int nG, int i)
453{
454 int j, k;
455 double del[MAXDIM], dist, old;
456 node_t *vi, *vj;
457
458 vi = GD_neato_nlist(G)[i];
459 for (k = 0; k < Ndim; k++)
460 GD_sum_t(G)[i][k] = 0.0;
461 for (j = 0; j < nG; j++) {
462 if (i == j)
463 continue;
464 vj = GD_neato_nlist(G)[j];
465 dist = distvec(ND_pos(vi), ND_pos(vj), del);
466 for (k = 0; k < Ndim; k++) {
467 old = GD_t(G)[i][j][k];
468 GD_t(G)[i][j][k] =
469 GD_spring(G)[i][j] * (del[k] -
470 GD_dist(G)[i][j] * del[k] / dist);
471 GD_sum_t(G)[i][k] += GD_t(G)[i][j][k];
472 old = GD_t(G)[j][i][k];
473 GD_t(G)[j][i][k] = -GD_t(G)[i][j][k];
474 GD_sum_t(G)[j][k] += GD_t(G)[j][i][k] - old;
475 }
476 }
477}
478
479#define Msub(i,j) M[(i)*Ndim+(j)]
480static void D2E(graph_t * G, int nG, int n, double *M)
481{
482 int i, l, k;
483 node_t *vi, *vn;
484 double scale, sq, t[MAXDIM];
485 double **K = GD_spring(G);
486 double **D = GD_dist(G);
487
488 vn = GD_neato_nlist(G)[n];
489 for (l = 0; l < Ndim; l++)
490 for (k = 0; k < Ndim; k++)
491 Msub(l, k) = 0.0;
492 for (i = 0; i < nG; i++) {
493 if (n == i)
494 continue;
495 vi = GD_neato_nlist(G)[i];
496 sq = 0.0;
497 for (k = 0; k < Ndim; k++) {
498 t[k] = ND_pos(vn)[k] - ND_pos(vi)[k];
499 sq += (t[k] * t[k]);
500 }
501 scale = 1 / fpow32(sq);
502 for (k = 0; k < Ndim; k++) {
503 for (l = 0; l < k; l++)
504 Msub(l, k) += K[n][i] * D[n][i] * t[k] * t[l] * scale;
505 Msub(k, k) +=
506 K[n][i] * (1.0 - D[n][i] * (sq - t[k] * t[k]) * scale);
507 }
508 }
509 for (k = 1; k < Ndim; k++)
510 for (l = 0; l < k; l++)
511 Msub(k, l) = Msub(l, k);
512}
513
515{
516 int i, k;
517 double m, max;
518 node_t *choice, *np;
519 static int cnt = 0;
520
521 cnt++;
522 if (GD_move(G) >= MaxIter)
523 return NULL;
524 max = 0.0;
525 choice = NULL;
526 for (i = 0; i < nG; i++) {
527 np = GD_neato_nlist(G)[i];
528 if (ND_pinned(np) > P_SET)
529 continue;
530 for (m = 0.0, k = 0; k < Ndim; k++)
531 m += GD_sum_t(G)[i][k] * GD_sum_t(G)[i][k];
532 /* could set the color=energy of the node here */
533 if (m > max) {
534 choice = np;
535 max = m;
536 }
537 }
538 if (max < Epsilon2)
539 choice = NULL;
540 else {
541 if (Verbose && cnt % 100 == 0) {
542 fprintf(stderr, "%.3f ", sqrt(max));
543 if (cnt % 1000 == 0)
544 fprintf(stderr, "\n");
545 }
546 }
547 return choice;
548}
549
550void move_node(graph_t * G, int nG, node_t * n)
551{
552 int i, m;
553 double b[MAXDIM] = {0};
554 double c[MAXDIM] = {0};
555
556 m = ND_id(n);
557 double *a = gv_calloc((size_t)Ndim * Ndim, sizeof(double));
558 D2E(G, nG, m, a);
559 for (i = 0; i < Ndim; i++)
560 c[i] = -GD_sum_t(G)[m][i];
561 solve(a, b, c, Ndim);
562 for (i = 0; i < Ndim; i++) {
563 b[i] = (Damping + 2 * (1 - Damping) * drand48()) * b[i];
564 ND_pos(n)[i] += b[i];
565 }
566 GD_move(G)++;
567 update_arrays(G, nG, m);
568 if (test_toggle()) {
569 double sum = 0;
570 for (i = 0; i < Ndim; i++) {
571 sum += fabs(b[i]);
572 } /* Why not squared? */
573 sum = sqrt(sum);
574 fprintf(stderr, "%s %.3f\n", agnameof(n), sum);
575 }
576 free(a);
577}
578
579static node_t **Heap;
580static int Heapsize;
581static node_t *Src;
582
583static void heapup(node_t * v)
584{
585 int i, par;
586 node_t *u;
587
588 for (i = ND_heapindex(v); i > 0; i = par) {
589 par = (i - 1) / 2;
590 u = Heap[par];
591 if (ND_dist(u) <= ND_dist(v))
592 break;
593 Heap[par] = v;
594 ND_heapindex(v) = par;
595 Heap[i] = u;
596 ND_heapindex(u) = i;
597 }
598}
599
600static void heapdown(node_t * v)
601{
602 int i, left, right, c;
603 node_t *u;
604
605 i = ND_heapindex(v);
606 while ((left = 2 * i + 1) < Heapsize) {
607 right = left + 1;
608 if ((right < Heapsize)
609 && (ND_dist(Heap[right]) < ND_dist(Heap[left])))
610 c = right;
611 else
612 c = left;
613 u = Heap[c];
614 if (ND_dist(v) <= ND_dist(u))
615 break;
616 Heap[c] = v;
617 ND_heapindex(v) = c;
618 Heap[i] = u;
619 ND_heapindex(u) = i;
620 i = c;
621 }
622}
623
625{
626 int i;
627
628 assert(ND_heapindex(v) < 0);
629 i = Heapsize++;
630 ND_heapindex(v) = i;
631 Heap[i] = v;
632 if (i > 0)
633 heapup(v);
634}
635
637{
638 int i;
639 node_t *rv, *v;
640
641 if (Heapsize == 0)
642 return NULL;
643 rv = Heap[0];
644 i = --Heapsize;
645 v = Heap[i];
646 Heap[0] = v;
647 ND_heapindex(v) = 0;
648 if (i > 1)
649 heapdown(v);
650 ND_heapindex(rv) = -1;
651 return rv;
652}
653
654void shortest_path(graph_t * G, int nG)
655{
656 node_t *v;
657
658 Heap = gv_calloc(nG + 1, sizeof(node_t*));
659 if (Verbose) {
660 fprintf(stderr, "Calculating shortest paths: ");
661 start_timer();
662 }
663 for (v = agfstnode(G); v; v = agnxtnode(G, v))
664 s1(G, v);
665 if (Verbose) {
666 fprintf(stderr, "%.2f sec\n", elapsed_sec());
667 }
668 free(Heap);
669}
670
672{
673 node_t *v, *u;
674 edge_t *e;
675 int t;
676 double f;
677
678 for (t = 0; (v = GD_neato_nlist(G)[t]); t++)
680 Src = node;
681 ND_dist(Src) = 0;
682 ND_hops(Src) = 0;
684
685 while ((v = neato_dequeue())) {
686 if (v != Src)
687 make_spring(G, Src, v, ND_dist(v));
688 for (e = agfstedge(G, v); e; e = agnxtedge(G, e, v)) {
689 if ((u = agtail(e)) == v)
690 u = aghead(e);
691 f = ND_dist(v) + ED_dist(e);
692 if (ND_dist(u) > f) {
693 ND_dist(u) = f;
694 if (ND_heapindex(u) >= 0)
695 heapup(u);
696 else {
697 ND_hops(u) = ND_hops(v) + 1;
698 neato_enqueue(u);
699 }
700 }
701 }
702 }
703}
704
705void make_spring(graph_t * G, node_t * u, node_t * v, double f)
706{
707 int i, j;
708
709 i = ND_id(u);
710 j = ND_id(v);
711 GD_dist(G)[i][j] = GD_dist(G)[j][i] = f;
712}
Memory allocation wrappers that exit on failure.
static void * gv_calloc(size_t nmemb, size_t size)
Definition alloc.h:26
#define Epsilon
Definition arcball.h:210
#define right(i)
Definition closest.c:79
int test_toggle(void)
Definition utils.c:412
double drand48(void)
Definition utils.c:1561
#define P_SET
Definition const.h:261
#define MAXDIM
Definition const.h:169
#define Spring_coeff
Definition const.h:167
mode
Definition cvtgxl.c:33
static char * err
Definition delaunay.c:696
#define left
Definition dthdr.h:12
static void del(Dict_t *d, Dtlink_t **set, Agedge_t *e)
Definition edge.c:157
static void init(int argc, char *argv[], double *angle, double *accuracy, int *check_edges_with_same_endpoint, int *seed, const char **color_scheme, int *lightness)
static double dist(int dim, double *x, double *y)
#define G
Definition gdefs.h:7
static pointf scale(double c, pointf p)
Definition geomprocs.h:155
double Initial_dist
Definition globals.h:64
int MaxIter
Definition globals.h:60
int Nop
Definition globals.h:54
bool Reduce
Definition globals.h:51
unsigned short Ndim
Definition globals.h:61
double Damping
Definition globals.h:65
static double len(glCompPoint p)
Definition glutils.c:150
static bool Verbose
Definition gml2gv.c:23
void free(void *)
node NULL
Definition grammar.y:163
static int cnt(Dict_t *d, Dtlink_t **set)
Definition graph.c:206
int agnedges(Agraph_t *g)
Definition graph.c:171
int agnnodes(Agraph_t *g)
Definition graph.c:165
Agsym_t * agattr(Agraph_t *g, int kind, char *name, const char *value)
creates or looks up attributes of a graph
Definition attr.c:371
char * agget(void *obj, char *name)
Definition attr.c:465
char * agxget(void *obj, Agsym_t *sym)
Definition attr.c:481
#define ED_dist(e)
Definition types.h:602
Agedge_t * agfstout(Agraph_t *g, Agnode_t *n)
Definition edge.c:24
#define agtail(e)
Definition cgraph.h:888
#define agfindedge(g, t, h)
Definition types.h:609
Agedge_t * agnxtedge(Agraph_t *g, Agedge_t *e, Agnode_t *n)
Definition edge.c:94
#define ED_factor(e)
Definition types.h:585
#define aghead(e)
Definition cgraph.h:889
Agedge_t * agnxtout(Agraph_t *g, Agedge_t *e)
Definition edge.c:39
Agedge_t * agfstedge(Agraph_t *g, Agnode_t *n)
Definition edge.c:85
void agwarningf(const char *fmt,...)
Definition agerror.c:173
int agerr(agerrlevel_t level, const char *fmt,...)
Definition agerror.c:155
@ AGPREV
Definition cgraph.h:857
#define GD_sum_t(g)
Definition types.h:404
#define GD_move(g)
Definition types.h:388
#define GD_t(g)
Definition types.h:405
#define GD_dist(g)
Definition types.h:357
#define GD_spring(g)
Definition types.h:403
#define GD_neato_nlist(g)
Definition types.h:392
Agnode_t * agnxtnode(Agraph_t *g, Agnode_t *n)
Definition node.c:47
Agnode_t * agfstnode(Agraph_t *g)
Definition node.c:40
#define ND_pinned(n)
Definition types.h:519
#define ND_heapindex(n)
Definition types.h:497
#define ND_hops(n)
Definition types.h:499
#define ND_pos(n)
Definition types.h:520
#define ND_dist(n)
Definition types.h:491
char * agnameof(void *)
returns a string descriptor for the object.
Definition id.c:143
int agdelete(Agraph_t *g, void *obj)
deletes object. Equivalent to agclose, agdelnode, and agdeledge for obj being a graph,...
Definition obj.c:20
@ AGEDGE
Definition cgraph.h:207
#define D
Definition hierarchy.c:119
textitem scanner parser str
Definition htmlparse.y:224
void getdouble(graph_t *g, char *name, double *result)
Definition input.c:490
#define hasPos(n)
Definition macros.h:18
#define ND_id(n)
Definition mm2gv.c:39
#define INIT_SELF
Definition neato.h:27
#define MODE_SGD
Definition neato.h:24
#define MODE_KK
Definition neato.h:20
#define INIT_RANDOM
Definition neato.h:29
#define INIT_REGULAR
Definition neato.h:28
int checkStart(graph_t *G, int nG, int dflt)
Definition neatoinit.c:1021
NEATOPROCS_API void solve(double *, double *, double *, size_t)
Definition solve.c:26
#define M
Definition randomkit.c:90
#define DFLT_TOLERANCE
Definition stress.h:23
graph or subgraph
Definition cgraph.h:424
string attribute descriptor symbol in Agattr_s.dict
Definition cgraph.h:641
static void heapup(node_t *v)
Definition stuff.c:583
static void update_arrays(graph_t *G, int nG, int i)
Definition stuff.c:452
static void make_spring(graph_t *, Agnode_t *, Agnode_t *, double)
Definition stuff.c:705
static double Epsilon2
Definition stuff.c:24
void solve_model(graph_t *G, int nG)
Definition stuff.c:432
void randompos(node_t *np, int nG)
Definition stuff.c:326
int scan_graph(graph_t *g)
Definition stuff.c:297
void s1(graph_t *G, node_t *node)
Definition stuff.c:671
#define Msub(i, j)
Definition stuff.c:479
void jitter3d(node_t *np, int nG)
Definition stuff.c:321
void shortest_path(graph_t *G, int nG)
Definition stuff.c:654
static node_t * prune(graph_t *G, node_t *np, node_t *next)
Definition stuff.c:169
static void D2E(graph_t *G, int nG, int n, double *M)
Definition stuff.c:480
static double setEdgeLen(graph_t *G, node_t *np, Agsym_t *lenx, double dfltlen)
Definition stuff.c:193
node_t * neato_dequeue(void)
Definition stuff.c:636
void jitter_d(node_t *np, int nG, int n)
Definition stuff.c:314
static node_t ** Heap
Definition stuff.c:579
int scan_graph_mode(graph_t *G, int mode)
Definition stuff.c:217
void free_array(double **rv)
Definition stuff.c:63
void free_scan_graph(graph_t *g)
Definition stuff.c:302
void diffeq_model(graph_t *G, int nG)
Definition stuff.c:357
static void move_node(graph_t *, int, Agnode_t *)
Definition stuff.c:550
void initial_positions(graph_t *G, int nG)
Definition stuff.c:334
static Agnode_t * choose_node(graph_t *, int)
Definition stuff.c:514
static int degreeKind(graph_t *g, node_t *n, node_t **op)
Definition stuff.c:133
static void heapdown(node_t *v)
Definition stuff.c:600
static void free_3array(double ***rv)
Definition stuff.c:90
static double total_e(graph_t *G, int nG)
Definition stuff.c:408
static double fpow32(double x)
Definition stuff.c:29
void neato_enqueue(node_t *v)
Definition stuff.c:624
static double distvec(double *p0, double *p1, double *vec)
Definition stuff.c:35
static double *** new_3array(int m, int n, int p, double ival)
Definition stuff.c:72
static node_t * Src
Definition stuff.c:581
static int Heapsize
Definition stuff.c:580
static int lenattr(edge_t *e, Agsym_t *index, double *val)
Definition stuff.c:109
double ** new_array(int m, int n, double ival)
Definition stuff.c:48
double elapsed_sec(void)
Definition timing.c:48
void start_timer(void)
Definition timing.c:43
Definition grammar.c:93