dijkstra.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. /**
  2. * @file
  3. * @brief single-source distance filter
  4. */
  5. /*************************************************************************
  6. * Copyright (c) 2011 AT&T Intellectual Property
  7. * All rights reserved. This program and the accompanying materials
  8. * are made available under the terms of the Eclipse Public License v1.0
  9. * which accompanies this distribution, and is available at
  10. * https://www.eclipse.org/legal/epl-v10.html
  11. *
  12. * Contributors: Details at https://graphviz.org
  13. *************************************************************************/
  14. #include "config.h"
  15. #include <stdbool.h>
  16. #include <stdio.h>
  17. #include <stdlib.h>
  18. #include <math.h>
  19. #include <cgraph/cgraph.h>
  20. #include <cgraph/ingraphs.h>
  21. #include <getopt.h>
  22. #include <util/alloc.h>
  23. #include <util/exit.h>
  24. #include <util/unreachable.h>
  25. static char *CmdName;
  26. static char **Files;
  27. static char **Nodes;
  28. static bool setall; /* if false, don't set dist attribute for
  29. * nodes in different components.
  30. */
  31. static bool doPath; /* if true, record shortest paths */
  32. static bool doDirected; /* if true, use directed paths */
  33. static Agsym_t *len_sym;
  34. typedef struct {
  35. Agrec_t hdr;
  36. double dist; /* always positive for scanned nodes */
  37. Agnode_t* prev;
  38. bool done; ///< true if finished
  39. } nodedata_t;
  40. static double getlength(Agedge_t * e)
  41. {
  42. double len;
  43. char* lens;
  44. char* p;
  45. if (len_sym && (*(lens = agxget(e, len_sym)))) {
  46. len = strtod (lens, &p);
  47. if (len < 0 || p == lens)
  48. len = 1;
  49. }
  50. else
  51. len = 1;
  52. return len;
  53. }
  54. static double getdist(Agnode_t * n)
  55. {
  56. nodedata_t *data;
  57. data = (nodedata_t *) (n->base.data);
  58. return data->dist;
  59. }
  60. static void setdist(Agnode_t * n, double dist)
  61. {
  62. nodedata_t *data;
  63. data = (nodedata_t *) (n->base.data);
  64. data->dist = dist;
  65. }
  66. #define getprev(n) (((nodedata_t*)((n)->base.data))->prev)
  67. #define setprev(n,p) (((nodedata_t*)((n)->base.data))->prev = (p))
  68. #define isDone(n) (((nodedata_t*)((n)->base.data))->done)
  69. #define setDone(n) (((nodedata_t*)((n)->base.data))->done = true)
  70. static int cmpf(void *key1, void *key2) {
  71. const double dist1 = getdist(key1);
  72. const double dist2 = getdist(key2);
  73. if (dist1 < dist2)
  74. return -1;
  75. if (dist1 > dist2)
  76. return 1;
  77. if (key1 < key2)
  78. return -1;
  79. if (key1 > key2)
  80. return 1;
  81. return 0;
  82. }
  83. static Dtdisc_t MyDisc = {
  84. 0, /* int key */
  85. 0, /* int size */
  86. -1, /* int link */
  87. 0, /* Dtmake_f makef */
  88. 0, /* Dtfree_f freef */
  89. cmpf, /* Dtcompar_f comparf */
  90. };
  91. static Agnode_t *extract_min(Dict_t * Q)
  92. {
  93. Agnode_t *rv;
  94. rv = dtfirst(Q);
  95. dtdelete(Q, rv);
  96. return rv;
  97. }
  98. static void update(Dict_t * Q, Agnode_t * dest, Agnode_t * src, double len)
  99. {
  100. double newlen = getdist(src) + len;
  101. double oldlen = getdist(dest);
  102. if (oldlen == 0) { /* first time to see dest */
  103. setdist(dest, newlen);
  104. if (doPath) setprev(dest, src);
  105. dtinsert(Q, dest);
  106. } else if (newlen < oldlen) {
  107. dtdelete(Q, dest);
  108. setdist(dest, newlen);
  109. if (doPath) setprev(dest, src);
  110. dtinsert(Q, dest);
  111. }
  112. }
  113. static void pre(Agraph_t * g)
  114. {
  115. len_sym = agattr(g, AGEDGE, "len", NULL);
  116. aginit(g, AGNODE, "dijkstra", sizeof(nodedata_t), true);
  117. }
  118. static void post(Agraph_t * g)
  119. {
  120. Agnode_t *v;
  121. Agnode_t *prev;
  122. char buf[256];
  123. char dflt[256];
  124. Agsym_t *sym;
  125. Agsym_t *psym = NULL;
  126. double dist, oldmax;
  127. double maxdist = 0.0; /* maximum "finite" distance */
  128. sym = agattr(g, AGNODE, "dist", "");
  129. if (doPath)
  130. psym = agattr(g, AGNODE, "prev", "");
  131. if (setall)
  132. snprintf(dflt, sizeof(dflt), "%.3lf", HUGE_VAL);
  133. for (v = agfstnode(g); v; v = agnxtnode(g, v)) {
  134. dist = getdist(v);
  135. if (dist) {
  136. dist--;
  137. snprintf(buf, sizeof(buf), "%.3lf", dist);
  138. agxset(v, sym, buf);
  139. if (doPath && (prev = getprev(v)))
  140. agxset(v, psym, agnameof(prev));
  141. if (maxdist < dist)
  142. maxdist = dist;
  143. } else if (setall)
  144. agxset(v, sym, dflt);
  145. }
  146. sym = agattrsym(g, "maxdist");
  147. if (sym) {
  148. if (!setall) {
  149. /* if we are preserving distances in other components,
  150. * check previous value of maxdist.
  151. */
  152. oldmax = atof(agxget(g, sym));
  153. if (oldmax > maxdist)
  154. maxdist = oldmax;
  155. }
  156. snprintf(buf, sizeof(buf), "%.3lf", maxdist);
  157. agxset(g, sym, buf);
  158. } else {
  159. snprintf(buf, sizeof(buf), "%.3lf", maxdist);
  160. agattr(g, AGRAPH, "maxdist", buf);
  161. }
  162. agclean(g, AGNODE, "dijkstra");
  163. agclean(g, AGEDGE, "dijkstra");
  164. }
  165. static void dijkstra(Dict_t * Q, Agraph_t * G, Agnode_t * n)
  166. {
  167. Agnode_t *u;
  168. Agedge_t *e;
  169. pre(G);
  170. setdist(n, 1);
  171. dtinsert(Q, n);
  172. if (doDirected) {
  173. while ((u = extract_min(Q))) {
  174. setDone (u);
  175. for (e = agfstout(G, u); e; e = agnxtout(G, e)) {
  176. if (!isDone(e->node)) update(Q, e->node, u, getlength(e));
  177. }
  178. }
  179. } else {
  180. while ((u = extract_min(Q))) {
  181. setDone (u);
  182. for (e = agfstedge(G, u); e; e = agnxtedge(G, e, u)) {
  183. if (!isDone(e->node)) update(Q, e->node, u, getlength(e));
  184. }
  185. }
  186. }
  187. post(G);
  188. }
  189. static char *useString =
  190. "Usage: dijkstra [-ap?] <node> [<file> <node> <file>]\n\
  191. -a - for nodes in a different component, set dist very large\n\
  192. -d - use forward directed edges\n\
  193. -p - attach shortest path info\n\
  194. -? - print usage\n\
  195. If no files are specified, stdin is used\n";
  196. static void usage(int v)
  197. {
  198. printf("%s",useString);
  199. graphviz_exit(v);
  200. }
  201. static void init(int argc, char *argv[])
  202. {
  203. int i, j, c;
  204. CmdName = argv[0];
  205. opterr = 0;
  206. while ((c = getopt(argc, argv, "adp?")) != -1) {
  207. switch (c) {
  208. case 'a':
  209. setall = true;
  210. break;
  211. case 'd':
  212. doDirected = true;
  213. break;
  214. case 'p':
  215. doPath = true;
  216. break;
  217. case '?':
  218. if (optopt == '\0' || optopt == '?')
  219. usage(0);
  220. else {
  221. fprintf(stderr, "%s: option -%c unrecognized\n",
  222. CmdName, optopt);
  223. usage(1);
  224. }
  225. break;
  226. default:
  227. UNREACHABLE();
  228. }
  229. }
  230. argv += optind;
  231. argc -= optind;
  232. if (argc == 0) {
  233. fprintf(stderr, "%s: no node specified\n", CmdName);
  234. usage(1);
  235. }
  236. Files = gv_calloc((size_t)argc / 2 + 2, sizeof(char *));
  237. Nodes = gv_calloc((size_t)argc / 2 + 2, sizeof(char *));
  238. for (j = i = 0; i < argc; i++) {
  239. Nodes[j] = argv[i++];
  240. Files[j] = argv[i] ? argv[i] : "-";
  241. j++;
  242. }
  243. Nodes[j] = Files[j] = NULL;
  244. }
  245. int main(int argc, char **argv)
  246. {
  247. Agraph_t *g;
  248. Agnode_t *n;
  249. ingraph_state ig;
  250. size_t i = 0;
  251. int code = 0;
  252. Dict_t *Q;
  253. init(argc, argv);
  254. newIngraph(&ig, Files);
  255. Q = dtopen(&MyDisc, Dtoset);
  256. while ((g = nextGraph(&ig)) != 0) {
  257. dtclear(Q);
  258. if ((n = agnode(g, Nodes[i], 0)))
  259. dijkstra(Q, g, n);
  260. else {
  261. fprintf(stderr, "%s: no node %s in graph %s in %s\n",
  262. CmdName, Nodes[i], agnameof(g), fileName(&ig));
  263. code = 1;
  264. }
  265. agwrite(g, stdout);
  266. fflush(stdout);
  267. agclose(g);
  268. i++;
  269. }
  270. free(Nodes);
  271. free(Files);
  272. graphviz_exit(code);
  273. }