Graphviz 13.0.0~dev.20241220.2304
Loading...
Searching...
No Matches
circle.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#include <assert.h>
12#include <cgraph/list.h>
13#include <twopigen/circle.h>
14#include <inttypes.h>
15#include <limits.h>
16#include <math.h>
17#include <stdbool.h>
18#include <stdint.h>
19#include <stdlib.h>
20#include <string.h>
21#include <util/alloc.h>
22#include <util/gv_ctype.h>
23#include <util/gv_math.h>
24#include <util/streq.h>
25#define DEF_RANKSEP 1.00
26#define UNSET 10.00
27
28/* dfs to set distance from a particular leaf.
29 * Note that termination is implicit in the test
30 * for reduced number of steps. Proof?
31 */
33{
34 Agnode_t *next;
35
36 uint64_t nsteps = SLEAF(n) + 1;
37
38 for (Agedge_t *ep = agfstedge(g, n); ep; ep = agnxtedge(g, ep, n)) {
39 if ((next = agtail(ep)) == n)
40 next = aghead(ep);
41
42 if (prev == next)
43 continue;
44
45 if (nsteps < SLEAF(next)) { /* handles loops and multiedges */
46 SLEAF(next) = nsteps;
47 setNStepsToLeaf(g, next, n);
48 }
49 }
50}
51
52/* isLeaf:
53 * Return true if n is a leaf node.
54 */
55static bool isLeaf(Agraph_t * g, Agnode_t * n)
56{
57 Agnode_t *neighp = 0;
58 Agnode_t *np;
59
60 for (Agedge_t *ep = agfstedge(g, n); ep; ep = agnxtedge(g, ep, n)) {
61 if ((np = agtail(ep)) == n)
62 np = aghead(ep);
63 if (n == np)
64 continue; /* loop */
65 if (neighp) {
66 if (neighp != np)
67 return false; /* two different neighbors */
68 } else
69 neighp = np;
70 }
71 return true;
72}
73
74static void initLayout(Agraph_t * g)
75{
76 int nnodes = agnnodes(g);
77 assert(nnodes >= 0);
78 uint64_t INF = (uint64_t)nnodes * (uint64_t)nnodes;
79
80 for (Agnode_t *n = agfstnode(g); n; n = agnxtnode(g, n)) {
81 SCENTER(n) = INF;
82 THETA(n) = UNSET; /* marks theta as unset, since 0 <= theta <= 2PI */
83 if (isLeaf(g, n))
84 SLEAF(n) = 0;
85 else
86 SLEAF(n) = INF;
87 }
88}
89
90/*
91 * Working recursively in from each leaf node (ie, each node
92 * with nStepsToLeaf == 0; see initLayout), set the
93 * minimum value of nStepsToLeaf for each node. Using
94 * that information, assign some node to be the centerNode.
95*/
97{
99 uint64_t maxNStepsToLeaf = 0;
100
101 /* dfs from each leaf node */
102 for (Agnode_t *n = agfstnode(g); n; n = agnxtnode(g, n)) {
103 if (SLEAF(n) == 0)
104 setNStepsToLeaf(g, n, 0);
105 }
106
107 for (Agnode_t *n = agfstnode(g); n; n = agnxtnode(g, n)) {
108 if (center == NULL || SLEAF(n) > maxNStepsToLeaf) {
109 maxNStepsToLeaf = SLEAF(n);
110 center = n;
111 }
112 }
113 return center;
114}
115
116DEFINE_LIST(node_queue, Agnode_t *)
117
118/* bfs to create tree structure */
120{
121 Agnode_t *next;
122 Agsym_t* wt = agfindedgeattr(g,"weight");
123 node_queue_t q = {0};
124
125 node_queue_push_back(&q, n);
126 while (!node_queue_is_empty(&q)) {
127 n = node_queue_pop_front(&q);
128 uint64_t nsteps = SCENTER(n) + 1;
129 for (Agedge_t *ep = agfstedge(g, n); ep; ep = agnxtedge(g, ep, n)) {
130 if (wt && streq(ag_xget(ep,wt),"0")) continue;
131 if ((next = agtail(ep)) == n)
132 next = aghead(ep);
133 if (nsteps < SCENTER(next)) {
134 SCENTER(next) = nsteps;
135 SPARENT(next) = n;
136 NCHILD(n)++;
137 node_queue_push_back(&q, next);
138 }
139 }
140 }
141 node_queue_free(&q);
142}
143
144/*
145 * Work out from the center and determine the value of
146 * nStepsToCenter and parent node for each node.
147 * Return UINT64_MAX if some node was not reached.
148 */
149static uint64_t setParentNodes(Agraph_t * sg, Agnode_t * center)
150{
151 uint64_t maxn = 0;
152 uint64_t unset = SCENTER(center);
153
154 SCENTER(center) = 0;
155 SPARENT(center) = 0;
157
158 /* find the maximum number of steps from the center */
159 for (Agnode_t *n = agfstnode(sg); n; n = agnxtnode(sg, n)) {
160 if (SCENTER(n) == unset) {
161 return UINT64_MAX;
162 }
163 else if (SCENTER(n) > maxn) {
164 maxn = SCENTER(n);
165 }
166 }
167 return maxn;
168}
169
170/* Sets each node's subtreeSize, which counts the number of
171 * leaves in subtree rooted at the node.
172 * At present, this is done bottom-up.
173 */
174static void setSubtreeSize(Agraph_t * g)
175{
176 for (Agnode_t *n = agfstnode(g); n; n = agnxtnode(g, n)) {
177 if (NCHILD(n) > 0)
178 continue;
179 STSIZE(n)++;
181 STSIZE(parent)++;
182 }
183 }
184}
185
187{
188 Agnode_t *next;
189
190 double ratio = SPAN(n) / STSIZE(n);
191 for (Agedge_t *ep = agfstedge(g, n); ep; ep = agnxtedge(g, ep, n)) {
192 if ((next = agtail(ep)) == n)
193 next = aghead(ep);
194 if (SPARENT(next) != n)
195 continue; /* handles loops */
196
197 if (SPAN(next) != 0.0)
198 continue; /* multiedges */
199 SPAN(next) = ratio * STSIZE(next);
200
201 if (NCHILD(next) > 0) {
202 setChildSubtreeSpans(g, next);
203 }
204 }
205}
206
208{
209 SPAN(center) = 2 * M_PI;
211}
212
214static bool is_set(double a) { return !is_exactly_equal(a, UNSET); }
215
216 /* Set the node positions for the 2nd and later rings. */
217static void setChildPositions(Agraph_t * sg, Agnode_t * n)
218{
219 Agnode_t *next;
220 double theta; /* theta is the lower boundary radius of the fan */
221
222 if (SPARENT(n) == 0) /* center */
223 theta = 0;
224 else
225 theta = THETA(n) - SPAN(n) / 2;
226
227 for (Agedge_t *ep = agfstedge(sg, n); ep; ep = agnxtedge(sg, ep, n)) {
228 if ((next = agtail(ep)) == n)
229 next = aghead(ep);
230 if (SPARENT(next) != n)
231 continue; /* handles loops */
232 if (is_set(THETA(next)))
233 continue; /* handles multiedges */
234
235 THETA(next) = theta + SPAN(next) / 2.0;
236 theta += SPAN(next);
237
238 if (NCHILD(next) > 0)
239 setChildPositions(sg, next);
240 }
241}
242
244{
245 THETA(center) = 0;
247}
248
249/* getRankseps:
250 * Return array of doubles of size maxrank+1 containing the radius of each
251 * rank. Position 0 always contains 0. Use the colon-separated list of
252 * doubles provided by ranksep to get the deltas for each additional rank.
253 * If not enough values are provided, the last value is repeated.
254 * If the ranksep attribute is not provided, use DEF_RANKSEP for all values.
255 */
256static double*
257getRankseps (Agraph_t* g, uint64_t maxrank)
258{
259 char *p;
260 char *endp;
261 char c;
262 uint64_t rk = 1;
263 double* ranks = gv_calloc(maxrank + 1, sizeof(double));
264 double xf = 0.0, delx = 0.0, d;
265
266 if ((p = late_string(g, agfindgraphattr(g->root, "ranksep"), NULL))) {
267 while (rk <= maxrank && (d = strtod (p, &endp)) > 0) {
268 delx = fmax(d, MIN_RANKSEP);
269 xf += delx;
270 ranks[rk++] = xf;
271 p = endp;
272 while ((c = *p) && (gv_isspace(c) || c == ':'))
273 p++;
274 }
275 }
276 else {
277 delx = DEF_RANKSEP;
278 }
279
280 for (uint64_t i = rk; i <= maxrank; i++) {
281 xf += delx;
282 ranks[i] = xf;
283 }
284
285 return ranks;
286}
287
288static void setAbsolutePos(Agraph_t * g, uint64_t maxrank)
289{
290 double* ranksep = getRankseps (g, maxrank);
291 if (Verbose) {
292 fputs ("Rank separation = ", stderr);
293 for (uint64_t i = 0; i <= maxrank; i++)
294 fprintf (stderr, "%.03lf ", ranksep[i]);
295 fputs ("\n", stderr);
296 }
297
298 /* Convert circular to cartesian coordinates */
299 for (Agnode_t *n = agfstnode(g); n; n = agnxtnode(g, n)) {
300 double hyp = ranksep[SCENTER(n)];
301 ND_pos(n)[0] = hyp * cos(THETA(n));
302 ND_pos(n)[1] = hyp * sin(THETA(n));
303 }
304 free (ranksep);
305}
306
307/* circleLayout:
308 * We assume sg is is connected and non-empty.
309 * Also, if center != 0, we are guaranteed that center is
310 * in the graph.
311 */
313{
314 if (agnnodes(sg) == 1) {
315 Agnode_t *n = agfstnode(sg);
316 ND_pos(n)[0] = 0;
317 ND_pos(n)[1] = 0;
318 return center;
319 }
320
321 initLayout(sg);
322
323 if (!center)
325
326 uint64_t maxNStepsToCenter = setParentNodes(sg,center);
327 if (Verbose)
328 fprintf(stderr, "root = %s max steps to root = %" PRIu64 "\n",
329 agnameof(center), maxNStepsToCenter);
330 if (maxNStepsToCenter == UINT64_MAX) {
331 agerrorf("twopi: use of weight=0 creates disconnected component.\n");
332 return center;
333 }
334
335 setSubtreeSize(sg);
336
338
339 setPositions(sg, center);
340
341 setAbsolutePos(sg, maxNStepsToCenter);
342 return center;
343}
Memory allocation wrappers that exit on failure.
static void * gv_calloc(size_t nmemb, size_t size)
Definition alloc.h:26
#define M_PI
Definition arith.h:41
static void setNStepsToCenter(Agraph_t *g, Agnode_t *n)
Definition circle.c:119
static double * getRankseps(Agraph_t *g, uint64_t maxrank)
Definition circle.c:257
static void setPositions(Agraph_t *sg, Agnode_t *center)
Definition circle.c:243
static void setSubtreeSize(Agraph_t *g)
Definition circle.c:174
static void setChildPositions(Agraph_t *sg, Agnode_t *n)
Definition circle.c:217
static void setChildSubtreeSpans(Agraph_t *g, Agnode_t *n)
Definition circle.c:186
static bool is_set(double a)
has the given value been assigned?
Definition circle.c:214
static void setNStepsToLeaf(Agraph_t *g, Agnode_t *n, Agnode_t *prev)
Definition circle.c:32
static void setAbsolutePos(Agraph_t *g, uint64_t maxrank)
Definition circle.c:288
static void initLayout(Agraph_t *g)
Definition circle.c:74
static Agnode_t * findCenterNode(Agraph_t *g)
Definition circle.c:96
Agnode_t * circleLayout(Agraph_t *sg, Agnode_t *center)
Definition circle.c:312
#define UNSET
Definition circle.c:26
static void setSubtreeSpans(Agraph_t *sg, Agnode_t *center)
Definition circle.c:207
#define DEF_RANKSEP
Definition circle.c:25
static uint64_t setParentNodes(Agraph_t *sg, Agnode_t *center)
Definition circle.c:149
static bool isLeaf(Agraph_t *g, Agnode_t *n)
Definition circle.c:55
#define SLEAF(n)
Definition circle.h:30
#define THETA(n)
Definition circle.h:36
#define NCHILD(n)
Definition circle.h:32
#define SCENTER(n)
Definition circle.h:33
#define SPAN(n)
Definition circle.h:35
#define STSIZE(n)
Definition circle.h:31
#define parent(i)
Definition closest.c:80
char * late_string(void *obj, attrsym_t *attr, char *defaultValue)
Definition utils.c:78
#define MIN_RANKSEP
Definition const.h:88
static bool Verbose
Definition gml2gv.c:23
void free(void *)
node NULL
Definition grammar.y:163
int agnnodes(Agraph_t *g)
Definition graph.c:169
#define agfindedgeattr(g, a)
Definition types.h:617
#define agtail(e)
Definition cgraph.h:880
Agedge_t * agnxtedge(Agraph_t *g, Agedge_t *e, Agnode_t *n)
Definition edge.c:94
#define aghead(e)
Definition cgraph.h:881
Agedge_t * agfstedge(Agraph_t *g, Agnode_t *n)
Definition edge.c:85
void agerrorf(const char *fmt,...)
Definition agerror.c:165
#define agfindgraphattr(g, a)
Definition types.h:613
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_pos(n)
Definition types.h:520
char * agnameof(void *)
returns a string descriptor for the object.
Definition id.c:158
replacements for ctype.h functions
static bool gv_isspace(int c)
Definition gv_ctype.h:55
Arithmetic helper functions.
static bool is_exactly_equal(double a, double b)
are two values precisely the same?
Definition gv_math.h:43
$2 u p prev
Definition htmlparse.y:297
#define DEFINE_LIST(name, type)
Definition list.h:26
#define SPARENT(n)
Definition patchwork.h:25
#define INF
Definition sccmap.c:41
static bool streq(const char *a, const char *b)
are a and b equal?
Definition streq.h:11
graph or subgraph
Definition cgraph.h:425
Agraph_t * root
subgraphs - ancestors
Definition cgraph.h:434
string attribute descriptor symbol in Agattr_s.dict
Definition cgraph.h:637
static point center(point vertex[], size_t n)
#define ag_xget(x, a)
Definition types.h:606