You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

578 lines
18 KiB

  1. /**CFile***********************************************************************
  2. FileName [ntrShort.c]
  3. PackageName [ntr]
  4. Synopsis [Symbolic shortest paths algorithms.]
  5. Description [This file contains the functions that implement the
  6. symbolic version of several shortest path algorithms described in the
  7. JFM paper on ADDs.]
  8. Author [Fabio Somenzi, Iris Bahar]
  9. Copyright [Copyright (c) 1995-2012, Regents of the University of Colorado
  10. All rights reserved.
  11. Redistribution and use in source and binary forms, with or without
  12. modification, are permitted provided that the following conditions
  13. are met:
  14. Redistributions of source code must retain the above copyright
  15. notice, this list of conditions and the following disclaimer.
  16. Redistributions in binary form must reproduce the above copyright
  17. notice, this list of conditions and the following disclaimer in the
  18. documentation and/or other materials provided with the distribution.
  19. Neither the name of the University of Colorado nor the names of its
  20. contributors may be used to endorse or promote products derived from
  21. this software without specific prior written permission.
  22. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  23. "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  24. LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
  25. FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
  26. COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
  27. INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
  28. BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  29. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  30. CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  31. LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
  32. ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  33. POSSIBILITY OF SUCH DAMAGE.]
  34. ******************************************************************************/
  35. #include "ntr.h"
  36. #include "cuddInt.h"
  37. /*---------------------------------------------------------------------------*/
  38. /* Constant declarations */
  39. /*---------------------------------------------------------------------------*/
  40. /*---------------------------------------------------------------------------*/
  41. /* Stucture declarations */
  42. /*---------------------------------------------------------------------------*/
  43. /*---------------------------------------------------------------------------*/
  44. /* Type declarations */
  45. /*---------------------------------------------------------------------------*/
  46. /*---------------------------------------------------------------------------*/
  47. /* Variable declarations */
  48. /*---------------------------------------------------------------------------*/
  49. #ifndef lint
  50. static char rcsid[] UTIL_UNUSED = "$Id: ntrShort.c,v 1.5 2012/02/05 01:53:01 fabio Exp fabio $";
  51. #endif
  52. /*---------------------------------------------------------------------------*/
  53. /* Macro declarations */
  54. /*---------------------------------------------------------------------------*/
  55. /**AutomaticStart*************************************************************/
  56. /*---------------------------------------------------------------------------*/
  57. /* Static function prototypes */
  58. /*---------------------------------------------------------------------------*/
  59. static DdNode * ntrBellman (DdManager *dd, DdNode *D, DdNode *source, DdNode **x, DdNode **y, int vars, int pr);
  60. static DdNode * ntrWarshall (DdManager *dd, DdNode *D, DdNode **x, DdNode **y, int vars, int pr);
  61. static DdNode * ntrSquare (DdManager *dd, DdNode *D, DdNode **x, DdNode **y, DdNode **z, int vars, int pr, int st);
  62. /**AutomaticEnd***************************************************************/
  63. /*---------------------------------------------------------------------------*/
  64. /* Definition of exported functions */
  65. /*---------------------------------------------------------------------------*/
  66. /**Function********************************************************************
  67. Synopsis [Computes shortest paths in a state graph.]
  68. Description [Computes shortest paths in the state transition graph of
  69. a network. Three methods are availabe:
  70. <ul>
  71. <li> Bellman-Ford algorithm for single-source shortest paths; the
  72. algorithm computes the distance (number of transitions) from the initial
  73. states to all states.
  74. <li> Floyd-Warshall algorithm for all-pair shortest paths.
  75. <li> Repeated squaring algorithm for all-pair shortest paths.
  76. </ul>
  77. The function returns 1 in case of success; 0 otherwise.
  78. ]
  79. SideEffects [ADD variables are created in the manager.]
  80. SeeAlso []
  81. ******************************************************************************/
  82. int
  83. Ntr_ShortestPaths(
  84. DdManager * dd,
  85. BnetNetwork * net,
  86. NtrOptions * option)
  87. {
  88. NtrPartTR *TR;
  89. DdNode *edges, *source, *res, *r, *q, *bddSource;
  90. DdNode **xadd, **yadd, **zadd;
  91. int i;
  92. int pr = option->verb;
  93. int algorithm = option->shortPath;
  94. int selectiveTrace = option->selectiveTrace;
  95. int nvars = net->nlatches;
  96. /* Set background to infinity for shortest paths. */
  97. Cudd_SetBackground(dd,Cudd_ReadPlusInfinity(dd));
  98. /* Build the monolithic TR. */
  99. TR = Ntr_buildTR(dd,net,option,NTR_IMAGE_MONO);
  100. /* Build the ADD variable vectors for x and y. */
  101. xadd = ALLOC(DdNode *, nvars);
  102. yadd = ALLOC(DdNode *, nvars);
  103. for(i = 0; i < nvars; i++) {
  104. q = Cudd_addIthVar(dd,TR->x[i]->index);
  105. Cudd_Ref(q);
  106. xadd[i] = q;
  107. q = Cudd_addIthVar(dd,TR->y[i]->index);
  108. Cudd_Ref(q);
  109. yadd[i] = q;
  110. }
  111. /* Convert the transition relation BDD into an ADD... */
  112. q = Cudd_BddToAdd(dd,TR->part[0]);
  113. Cudd_Ref(q);
  114. /* ...replacing zeroes with infinities... */
  115. r = Cudd_addIte(dd,q,Cudd_ReadOne(dd),Cudd_ReadPlusInfinity(dd));
  116. Cudd_Ref(r);
  117. Cudd_RecursiveDeref(dd,q);
  118. /* ...and zeroing the diagonal. */
  119. q = Cudd_addXeqy(dd,nvars,xadd,yadd);
  120. Cudd_Ref(q);
  121. edges = Cudd_addIte(dd,q,Cudd_ReadZero(dd),r);
  122. Cudd_Ref(edges);
  123. Cudd_RecursiveDeref(dd,r);
  124. Cudd_RecursiveDeref(dd,q);
  125. switch(algorithm) {
  126. case NTR_SHORT_BELLMAN:
  127. bddSource = Ntr_initState(dd,net,option);
  128. source = Cudd_BddToAdd(dd,bddSource);
  129. Cudd_Ref(source);
  130. res = ntrBellman(dd,edges,source,xadd,yadd,nvars,pr);
  131. if (res == NULL) return(0);
  132. Cudd_Ref(res);
  133. Cudd_RecursiveDeref(dd,source);
  134. Cudd_RecursiveDeref(dd,bddSource);
  135. if (pr >= 0) {
  136. (void) fprintf(stdout,"Distance Matrix");
  137. Cudd_PrintDebug(dd,res,nvars,pr);
  138. }
  139. break;
  140. case NTR_SHORT_FLOYD:
  141. res = ntrWarshall(dd,edges,xadd,yadd,nvars,pr);
  142. if (res == NULL) return(0);
  143. Cudd_Ref(res);
  144. if (pr >= 0) {
  145. (void) fprintf(stdout,"Distance Matrix");
  146. Cudd_PrintDebug(dd,res,2*nvars,pr);
  147. }
  148. break;
  149. case NTR_SHORT_SQUARE:
  150. /* Create a third set of ADD variables. */
  151. zadd = ALLOC(DdNode *, nvars);
  152. for(i = 0; i < nvars; i++) {
  153. int level;
  154. level = Cudd_ReadIndex(dd,TR->x[i]->index);
  155. q = Cudd_addNewVarAtLevel(dd,level);
  156. Cudd_Ref(q);
  157. zadd[i] = q;
  158. }
  159. /* Compute the shortest paths. */
  160. res = ntrSquare(dd,edges,zadd,yadd,xadd,nvars,pr,selectiveTrace);
  161. if (res == NULL) return(0);
  162. Cudd_Ref(res);
  163. /* Dispose of the extra variables. */
  164. for(i = 0; i < nvars; i++) {
  165. Cudd_RecursiveDeref(dd,zadd[i]);
  166. }
  167. FREE(zadd);
  168. if (pr >= 0) {
  169. (void) fprintf(stdout,"Distance Matrix");
  170. Cudd_PrintDebug(dd,res,2*nvars,pr);
  171. }
  172. break;
  173. default:
  174. (void) printf("Unrecognized method. Try again.\n");
  175. return(0);
  176. }
  177. /* Here we should compute the paths. */
  178. /* Clean up. */
  179. Ntr_freeTR(dd,TR);
  180. Cudd_RecursiveDeref(dd,edges);
  181. Cudd_RecursiveDeref(dd,res);
  182. for(i = 0; i < nvars; i++) {
  183. Cudd_RecursiveDeref(dd,xadd[i]);
  184. Cudd_RecursiveDeref(dd,yadd[i]);
  185. }
  186. FREE(xadd);
  187. FREE(yadd);
  188. if (option->autoDyn & 1) {
  189. (void) printf("Order after short path computation\n");
  190. if (!Bnet_PrintOrder(net,dd)) return(0);
  191. }
  192. return(1);
  193. } /* end of Ntr_ShortestPaths */
  194. /*---------------------------------------------------------------------------*/
  195. /* Definition of internal functions */
  196. /*---------------------------------------------------------------------------*/
  197. /*---------------------------------------------------------------------------*/
  198. /* Definition of static functions */
  199. /*---------------------------------------------------------------------------*/
  200. /**Function********************************************************************
  201. Synopsis [Bellman-Ford algorithm for single-source shortest paths.]
  202. Description [Bellman-Ford algorithm for single-source shortest
  203. paths. Returns the vector of the distances of all states from the
  204. initial states. In case of multiple initial states the distance for
  205. each state is from the nearest initial state. Negative-weight
  206. cycles are detected, though only in the naive way. (Lack of
  207. convergence after nodes-1 iterations.) In such a case, a constant
  208. ADD with value minus infinity is returned. Bellman-Ford is based on
  209. matrix-vector multiplication. The matrix is the distance matrix
  210. D(x,y), such that D(a,b) is the length of the arc connecting state a
  211. to state b. The vector V(x) stores the distances of all states from
  212. the initial states. The actual vector used in the matrix-vector
  213. multiplication is diff(x), that holds those distances that have
  214. changed during the last update.]
  215. SideEffects []
  216. SeeAlso [ntrWarshall ntrSquare]
  217. ******************************************************************************/
  218. static DdNode *
  219. ntrBellman(
  220. DdManager *dd,
  221. DdNode *D,
  222. DdNode *source,
  223. DdNode **x,
  224. DdNode **y,
  225. int vars,
  226. int pr)
  227. {
  228. DdNode *u, *w, *V, *min, *diff;
  229. DdApaNumber i, nodes, one;
  230. int digits = vars + 1;
  231. /* To avoid overflow when there are many variables, use APA. */
  232. nodes = Cudd_NewApaNumber(digits);
  233. Cudd_ApaPowerOfTwo(digits,nodes,vars);
  234. i = Cudd_NewApaNumber(digits);
  235. one = Cudd_NewApaNumber(digits);
  236. Cudd_ApaSetToLiteral(digits,one,1);
  237. #if 0
  238. /* Find the distances from the initial state along paths using one
  239. ** arc. */
  240. w = Cudd_Cofactor(dd,D,source); /* works only if source is a cube */
  241. Cudd_Ref(w);
  242. V = Cudd_addSwapVariables(dd,w,x,y,vars);
  243. Cudd_Ref(V);
  244. Cudd_RecursiveDeref(dd,w);
  245. #endif
  246. /* The initial states are at distance 0. The other states are
  247. ** initially at infinite distance. */
  248. V = Cudd_addIte(dd,source,Cudd_ReadZero(dd),Cudd_ReadPlusInfinity(dd));
  249. Cudd_Ref(V);
  250. /* Selective trace algorithm. For the next update, only consider the
  251. ** nodes whose distance has changed in the last update. */
  252. diff = V;
  253. Cudd_Ref(diff);
  254. for (Cudd_ApaSetToLiteral(digits,i,1);
  255. Cudd_ApaCompare(digits,i,digits,nodes) < 0;
  256. Cudd_ApaAdd(digits,i,one,i)) {
  257. if (pr>2) {(void) printf("V"); Cudd_PrintDebug(dd,V,vars,pr);}
  258. /* Compute the distances via triangulation as a function of x. */
  259. w = Cudd_addTriangle(dd,diff,D,x,vars);
  260. Cudd_Ref(w);
  261. Cudd_RecursiveDeref(dd,diff);
  262. u = Cudd_addSwapVariables(dd,w,x,y,vars);
  263. Cudd_Ref(u);
  264. Cudd_RecursiveDeref(dd,w);
  265. if (pr>2) {(void) printf("u"); Cudd_PrintDebug(dd,u,vars,pr);}
  266. /* Take the minimum of the previous distances and those just
  267. ** computed. */
  268. min = Cudd_addApply(dd,Cudd_addMinimum,V,u);
  269. Cudd_Ref(min);
  270. Cudd_RecursiveDeref(dd,u);
  271. if (pr>2) {(void) printf("min"); Cudd_PrintDebug(dd,min,vars,pr);}
  272. if (V == min) { /* convergence */
  273. Cudd_RecursiveDeref(dd,min);
  274. if (pr>0) {
  275. (void) printf("Terminating after ");
  276. Cudd_ApaPrintDecimal(stdout,digits,i);
  277. (void) printf(" iterations\n");
  278. }
  279. break;
  280. }
  281. /* Find the distances that decreased. */
  282. diff = Cudd_addApply(dd,Cudd_addDiff,V,min);
  283. Cudd_Ref(diff);
  284. if (pr>2) {(void) printf("diff"); Cudd_PrintDebug(dd,diff,vars,pr);}
  285. Cudd_RecursiveDeref(dd,V);
  286. V = min;
  287. }
  288. /* Negative cycle detection. */
  289. if (Cudd_ApaCompare(digits,i,digits,nodes) == 0 &&
  290. diff != Cudd_ReadPlusInfinity(dd)) {
  291. (void) printf("Negative cycle\n");
  292. Cudd_RecursiveDeref(dd,diff);
  293. Cudd_RecursiveDeref(dd,V);
  294. V = Cudd_ReadMinusInfinity(dd);
  295. Cudd_Ref(V);
  296. }
  297. Cudd_Deref(V);
  298. FREE(i);
  299. FREE(nodes);
  300. FREE(one);
  301. return(V);
  302. } /* end of ntrBellman */
  303. /**Function********************************************************************
  304. Synopsis [Floyd-Warshall algorithm for all-pair shortest paths.]
  305. Description []
  306. SideEffects []
  307. SeeAlso []
  308. ******************************************************************************/
  309. static DdNode *
  310. ntrWarshall(
  311. DdManager *dd,
  312. DdNode *D,
  313. DdNode **x,
  314. DdNode **y,
  315. int vars,
  316. int pr)
  317. {
  318. DdNode *one, *zero;
  319. DdNode *xminterm, *w, *V, *V2;
  320. DdNode *P, *R;
  321. int i;
  322. int nodes;
  323. int k,u;
  324. long start_time;
  325. if (vars > 30)
  326. nodes = 1000000000;
  327. else
  328. nodes = 1 << vars;
  329. one = DD_ONE(dd);
  330. zero = DD_ZERO(dd);
  331. Cudd_Ref(R = D); /* make copy of original matrix */
  332. /* Extract pivot row and column from D */
  333. start_time = util_cpu_time();
  334. for (k = 0; k < nodes; k++) {
  335. if (k % 10000 == 0) {
  336. (void) printf("Starting iteration %d at time %s\n",
  337. k,util_print_time(util_cpu_time() - start_time));
  338. }
  339. Cudd_Ref(xminterm = one);
  340. u = k;
  341. for (i = vars-1; i >= 0; i--) {
  342. if (u&1) {
  343. Cudd_Ref(w = Cudd_addIte(dd,x[i],xminterm,zero));
  344. } else {
  345. Cudd_Ref(w = Cudd_addIte(dd,x[i],zero,xminterm));
  346. }
  347. Cudd_RecursiveDeref(dd,xminterm);
  348. xminterm = w;
  349. u >>= 1;
  350. }
  351. Cudd_Ref(V = Cudd_Cofactor(dd,R,xminterm));
  352. Cudd_RecursiveDeref(dd,xminterm);
  353. if (pr>2) {(void) printf("V"); Cudd_PrintDebug(dd,V,vars,pr);}
  354. Cudd_Ref(xminterm = one);
  355. u = k;
  356. for (i = vars-1; i >= 0; i--) {
  357. if (u&1) {
  358. Cudd_Ref(w = Cudd_addIte(dd,y[i],xminterm,zero));
  359. } else {
  360. Cudd_Ref(w = Cudd_addIte(dd,y[i],zero,xminterm));
  361. }
  362. Cudd_RecursiveDeref(dd,xminterm);
  363. xminterm = w;
  364. u >>= 1;
  365. }
  366. Cudd_Ref(V2 = Cudd_Cofactor(dd,R,xminterm));
  367. Cudd_RecursiveDeref(dd,xminterm);
  368. if (pr>2) {(void) printf("V2"); Cudd_PrintDebug(dd,V2,vars,pr);}
  369. Cudd_Ref(P = Cudd_addOuterSum(dd,R,V,V2));
  370. Cudd_RecursiveDeref(dd,V);
  371. Cudd_RecursiveDeref(dd,V2);
  372. Cudd_RecursiveDeref(dd,R);
  373. R = P;
  374. if (pr>2) {(void) printf("R"); Cudd_PrintDebug(dd,R,vars,pr);}
  375. }
  376. Cudd_Deref(R);
  377. return(R);
  378. } /* end of ntrWarshall */
  379. /**Function********************************************************************
  380. Synopsis [Repeated squaring algorithm for all-pairs shortest paths.]
  381. Description []
  382. SideEffects []
  383. SeeAlso []
  384. ******************************************************************************/
  385. static DdNode *
  386. ntrSquare(
  387. DdManager *dd /* manager */,
  388. DdNode *D /* D(z,y): distance matrix */,
  389. DdNode **x /* array of x variables */,
  390. DdNode **y /* array of y variables */,
  391. DdNode **z /* array of z variables */,
  392. int vars /* number of variables in each of the three arrays */,
  393. int pr /* verbosity level */,
  394. int st /* use the selective trace algorithm */)
  395. {
  396. DdNode *zero;
  397. DdNode *I; /* identity matirix */
  398. DdNode *w, *V, *P, *M, *R, *RT;
  399. DdNode *diff, *min, *minDiag;
  400. int n;
  401. int neg;
  402. long start_time;
  403. zero = Cudd_ReadZero(dd);
  404. /* Make a working copy of the original matrix. */
  405. R = D;
  406. Cudd_Ref(R);
  407. I = Cudd_addXeqy(dd,vars,z,y); /* identity matrix */
  408. Cudd_Ref(I);
  409. /* Make a copy of the matrix for the selective trace algorithm. */
  410. diff = R;
  411. Cudd_Ref(diff);
  412. start_time = util_cpu_time();
  413. for (n = vars; n >= 0; n--) {
  414. printf("Starting iteration %d at time %s\n",vars-n,
  415. util_print_time(util_cpu_time() - start_time));
  416. /* Check for negative cycles: They are identified by negative
  417. ** elements on the diagonal.
  418. */
  419. /* Extract values from the diagonal. */
  420. Cudd_Ref(w = Cudd_addIte(dd,I,R,zero));
  421. minDiag = Cudd_addFindMin(dd,w); /* no need to ref */
  422. neg = Cudd_V(minDiag) < 0;
  423. Cudd_RecursiveDeref(dd,w);
  424. if (neg) {
  425. Cudd_RecursiveDeref(dd,diff);
  426. (void) printf("Negative cycle after %d iterations!\n",vars-n);
  427. break;
  428. }
  429. /* Prepare the first operand of matrix multiplication:
  430. ** diff(z,y) -> RT(x,y) -> V(x,z)
  431. */
  432. /* RT(x,y) */
  433. Cudd_Ref(RT = Cudd_addSwapVariables(dd,diff,x,z,vars));
  434. Cudd_RecursiveDeref(dd,diff);
  435. /* V(x,z) */
  436. Cudd_Ref(V = Cudd_addSwapVariables(dd,RT,y,z,vars));
  437. Cudd_RecursiveDeref(dd,RT);
  438. if (pr > 0) {
  439. double pathcount;
  440. (void) printf("V"); Cudd_PrintDebug(dd,V,2*vars,pr);
  441. pathcount = Cudd_CountPath(V);
  442. (void) printf("Path count = %g\n", pathcount);
  443. }
  444. /* V(x,z) * R(z,y) -> P(x,y) */
  445. Cudd_Ref(P = Cudd_addTriangle(dd,V,R,z,vars));
  446. Cudd_RecursiveDeref(dd,V);
  447. /* P(x,y) => M(z,y) */
  448. Cudd_Ref(M = Cudd_addSwapVariables(dd,P,x,z,vars));
  449. Cudd_RecursiveDeref(dd,P);
  450. if (pr>0) {(void) printf("M"); Cudd_PrintDebug(dd,M,2*vars,pr);}
  451. /* min(z,y) */
  452. Cudd_Ref(min = Cudd_addApply(dd,Cudd_addMinimum,R,M));
  453. Cudd_RecursiveDeref(dd,M);
  454. if (R == min) {
  455. Cudd_RecursiveDeref(dd,min);
  456. if (pr>0) {printf("Done after %d iterations\n",vars-n+1); }
  457. break;
  458. }
  459. /* diff(z,y) */
  460. if (st) {
  461. Cudd_Ref(diff = Cudd_addApply(dd,Cudd_addDiff,min,R));
  462. } else {
  463. Cudd_Ref(diff = min);
  464. }
  465. Cudd_RecursiveDeref(dd,R);
  466. R = min; /* keep a copy of matrix at current iter. */
  467. if (pr > 0) {
  468. double pathcount;
  469. (void) printf("R"); Cudd_PrintDebug(dd,R,2*vars,pr);
  470. pathcount = Cudd_CountPath(R);
  471. (void) printf("Path count = %g\n", pathcount);
  472. }
  473. if (n == 0) {
  474. (void) printf("Negative cycle!\n");
  475. break;
  476. }
  477. }
  478. Cudd_RecursiveDeref(dd,I);
  479. Cudd_Deref(R);
  480. return(R);
  481. } /* end of ntrSquare */