Routino SVN Repository Browser

Check out the latest version of Routino: svn co http://routino.org/svn/trunk routino

ViewVC logotype

Contents of /trunk/src/optimiser.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 123 - (show annotations) (download) (as text)
Sun Feb 15 16:19:06 2009 UTC (16 years, 1 month ago) by amb
File MIME type: text/x-csrc
File size: 29277 byte(s)
Better test for failing to find a route.

1 /***************************************
2 $Header: /home/amb/CVS/routino/src/optimiser.c,v 1.53 2009-02-15 16:19:06 amb Exp $
3
4 Routing optimiser.
5 ******************/ /******************
6 Written by Andrew M. Bishop
7
8 This file Copyright 2008,2009 Andrew M. Bishop
9 It may be distributed under the GNU Public License, version 2, or
10 any higher version. See section COPYING of the GNU Public license
11 for conditions under which this file may be redistributed.
12 ***************************************/
13
14
15 #include <string.h>
16 #include <stdio.h>
17
18 #include "types.h"
19 #include "functions.h"
20 #include "nodes.h"
21 #include "segments.h"
22 #include "ways.h"
23 #include "results.h"
24
25
26 /*+ The option not to print any progress information. +*/
27 extern int option_quiet;
28
29 /*+ The option to calculate the quickest route insted of the shortest. +*/
30 extern int option_quickest;
31
32
33 /*++++++++++++++++++++++++++++++++++++++
34 Find the optimum route between two nodes.
35
36 Results *FindRoute Returns a set of results.
37
38 Nodes *nodes The set of nodes to use.
39
40 Segments *segments The set of segments to use.
41
42 Ways *ways The set of ways to use.
43
44 index_t start The start node.
45
46 index_t finish The finish node.
47
48 Profile *profile The profile containing the transport type, speeds and allowed highways.
49
50 int all A flag to indicate that a big results structure is required.
51 ++++++++++++++++++++++++++++++++++++++*/
52
53 Results *FindRoute(Nodes *nodes,Segments *segments,Ways *ways,index_t start,index_t finish,Profile *profile,int all)
54 {
55 Results *results;
56 index_t node1,node2;
57 distance_t finish_distance;
58 duration_t finish_duration;
59 float finish_lat,finish_lon;
60 speed_t max_speed=0;
61 Result *result1,*result2;
62 Segment *segment;
63 Way *way;
64 int i;
65
66 /* Set up the finish conditions */
67
68 finish_distance=~0;
69 finish_duration=~0;
70
71 GetLatLong(nodes,LookupNode(nodes,finish),&finish_lat,&finish_lon);
72
73 for(i=0;i<sizeof(profile->speed)/sizeof(profile->speed[0]);i++)
74 if(profile->speed[i]>max_speed)
75 max_speed=profile->speed[i];
76
77 /* Insert the first node into the queue */
78
79 if(all)
80 results=NewResultsList(65536);
81 else
82 results=NewResultsList(8);
83
84 result1=InsertResult(results,start);
85
86 result1->node=start;
87 result1->prev=0;
88 result1->next=0;
89 result1->distance=0;
90 result1->duration=0;
91 result1->sortby=0;
92
93 insert_in_queue(result1);
94
95 /* Loop across all nodes in the queue */
96
97 while((result1=pop_from_queue()))
98 {
99 if((option_quickest==0 && result1->sortby>finish_distance) ||
100 (option_quickest==1 && result1->sortby>finish_duration))
101 continue;
102
103 node1=result1->node;
104
105 segment=FirstSegment(segments,LookupNode(nodes,node1));
106
107 while(segment)
108 {
109 distance_t cumulative_distance;
110 duration_t cumulative_duration;
111
112 if(!IsNormalSegment(segment))
113 goto endloop;
114
115 if(profile->oneway && IsOnewayTo(segment,node1))
116 goto endloop;
117
118 node2=OtherNode(segment,node1);
119
120 if(result1->prev==node2)
121 goto endloop;
122
123 way=LookupWay(ways,segment->way);
124
125 if(!(way->allow&profile->allow))
126 goto endloop;
127
128 if(!profile->highways[HIGHWAY(way->type)])
129 goto endloop;
130
131 cumulative_distance=result1->distance+DISTANCE(segment->distance);
132 cumulative_duration=result1->duration+Duration(segment,way,profile);
133
134 if((option_quickest==0 && cumulative_distance>finish_distance) ||
135 (option_quickest==1 && cumulative_duration>finish_duration))
136 goto endloop;
137
138 result2=FindResult(results,node2);
139
140 if(!result2) /* New end node */
141 {
142 result2=InsertResult(results,node2);
143 result2->node=node2;
144 result2->prev=node1;
145 result2->next=0;
146 result2->distance=cumulative_distance;
147 result2->duration=cumulative_duration;
148
149 if(node2==finish)
150 {
151 finish_distance=cumulative_distance;
152 finish_duration=cumulative_duration;
153 }
154 else if(!all)
155 {
156 result2->sortby=result2->distance;
157 insert_in_queue(result2);
158 }
159 else
160 {
161 float lat,lon;
162 GetLatLong(nodes,LookupNode(nodes,node2),&lat,&lon);
163
164 if(option_quickest==0)
165 result2->sortby=result2->distance+Distance(lat,lon,finish_lat,finish_lon);
166 else
167 result2->sortby=result2->duration+distance_speed_to_duration(Distance(lat,lon,finish_lat,finish_lon),max_speed);
168
169 insert_in_queue(result2);
170 }
171 }
172 else if(option_quickest==0) /* shortest */
173 {
174 if(cumulative_distance<result2->distance ||
175 (cumulative_distance==result2->distance &&
176 cumulative_duration<result2->duration)) /* New end node is shorter */
177 {
178 result2->prev=node1;
179 result2->distance=cumulative_distance;
180 result2->duration=cumulative_duration;
181
182 if(node2==finish)
183 {
184 finish_distance=cumulative_distance;
185 finish_duration=cumulative_duration;
186 }
187 else if(!all)
188 {
189 result2->sortby=result2->distance;
190 insert_in_queue(result2);
191 }
192 else
193 {
194 float lat,lon;
195 GetLatLong(nodes,LookupNode(nodes,node2),&lat,&lon);
196
197 result2->sortby=result2->distance+Distance(lat,lon,finish_lat,finish_lon);
198 insert_in_queue(result2);
199 }
200 }
201 }
202 else if(option_quickest==1) /* quickest */
203 {
204 if(cumulative_duration<result2->duration ||
205 (cumulative_duration==result2->duration &&
206 cumulative_distance<result2->distance)) /* New end node is quicker */
207 {
208 result2->prev=node1;
209 result2->distance=cumulative_distance;
210 result2->duration=cumulative_duration;
211
212 if(node2==finish)
213 {
214 finish_distance=cumulative_distance;
215 finish_duration=cumulative_duration;
216 }
217 else if(!all)
218 {
219 result2->sortby=result2->distance;
220 insert_in_queue(result2);
221 }
222 else
223 {
224 float lat,lon;
225 GetLatLong(nodes,LookupNode(nodes,node2),&lat,&lon);
226
227 result2->sortby=result2->duration+distance_speed_to_duration(Distance(lat,lon,finish_lat,finish_lon),max_speed);
228 insert_in_queue(result2);
229 }
230 }
231 }
232
233 endloop:
234
235 if(!option_quiet && !(results->number%10000))
236 {
237 printf("\rRouting: End Nodes=%d %.1fkm %.0fmin ",results->number,
238 distance_to_km(result1->distance),duration_to_minutes(result1->duration));
239 fflush(stdout);
240 }
241
242 segment=NextSegment(segments,segment,node1);
243 }
244 }
245
246 if(!option_quiet)
247 {
248 printf("\rRouted: End Nodes=%d %.1fkm %.0fmin \n",results->number,
249 distance_to_km(finish_distance),duration_to_minutes(finish_duration));
250 fflush(stdout);
251 }
252
253 /* Check it worked */
254
255 result2=FindResult(results,finish);
256
257 if(!result2)
258 {
259 FreeResultsList(results);
260 return(NULL);
261 }
262
263 /* Reverse the results */
264
265 result2=FindResult(results,finish);
266
267 do
268 {
269 if(result2->prev)
270 {
271 index_t node1=result2->prev;
272
273 result1=FindResult(results,node1);
274
275 result1->next=result2->node;
276
277 result2=result1;
278 }
279 else
280 result2=NULL;
281 }
282 while(result2);
283
284 return(results);
285 }
286
287
288 /*++++++++++++++++++++++++++++++++++++++
289 Find the optimum route between two nodes.
290
291 Results *FindRoute3 Returns a set of results.
292
293 Nodes *nodes The set of nodes to use.
294
295 Segments *segments The set of segments to use.
296
297 Ways *ways The set of ways to use.
298
299 index_t start The start node.
300
301 index_t finish The finish node.
302
303 Results *begin The initial portion of the route.
304
305 Results *end The final portion of the route.
306
307 Profile *profile The profile containing the transport type, speeds and allowed highways.
308 ++++++++++++++++++++++++++++++++++++++*/
309
310 Results *FindRoute3(Nodes *nodes,Segments *segments,Ways *ways,index_t start,index_t finish,Results *begin,Results *end,Profile *profile)
311 {
312 Results *results;
313 index_t node1,node2;
314 distance_t finish_distance;
315 duration_t finish_duration;
316 float finish_lat,finish_lon;
317 speed_t max_speed=0;
318 Result *result1,*result2,*result3;
319 Segment *segment;
320 Way *way;
321 int i;
322
323 /* Set up the finish conditions */
324
325 finish_distance=~0;
326 finish_duration=~0;
327
328 GetLatLong(nodes,LookupNode(nodes,finish),&finish_lat,&finish_lon);
329
330 for(i=0;i<sizeof(profile->speed)/sizeof(profile->speed[0]);i++)
331 if(profile->speed[i]>max_speed)
332 max_speed=profile->speed[i];
333
334 /* Insert the start node */
335
336 results=NewResultsList(65536);
337
338 result1=InsertResult(results,start);
339 result3=FindResult(begin,start);
340
341 *result1=*result3;
342
343 /* Insert the finish points of the beginning part of the path into the queue */
344
345 result3=FirstResult(begin);
346
347 while(result3)
348 {
349 if(IsSuperNode(LookupNode(nodes,result3->node)))
350 {
351 if(!(result2=FindResult(results,result3->node)))
352 {
353 result2=InsertResult(results,result3->node);
354
355 *result2=*result3;
356
357 result2->prev=start;
358
359 result2->sortby=result2->distance;
360 }
361
362 insert_in_queue(result2);
363 }
364
365 result3=NextResult(begin,result3);
366 }
367
368 /* Loop across all nodes in the queue */
369
370 while((result1=pop_from_queue()))
371 {
372 if((option_quickest==0 && result1->sortby>finish_distance) ||
373 (option_quickest==1 && result1->sortby>finish_duration))
374 continue;
375
376 node1=result1->node;
377
378 segment=FirstSegment(segments,LookupNode(nodes,node1));
379
380 while(segment)
381 {
382 distance_t cumulative_distance;
383 duration_t cumulative_duration;
384
385 if(!IsSuperSegment(segment))
386 goto endloop;
387
388 if(profile->oneway && IsOnewayTo(segment,node1))
389 goto endloop;
390
391 node2=OtherNode(segment,node1);
392
393 if(result1->prev==node2)
394 goto endloop;
395
396 way=LookupWay(ways,segment->way);
397
398 if(!(way->allow&profile->allow))
399 goto endloop;
400
401 if(!profile->highways[HIGHWAY(way->type)])
402 goto endloop;
403
404 cumulative_distance=result1->distance+DISTANCE(segment->distance);
405 cumulative_duration=result1->duration+Duration(segment,way,profile);
406
407 if((option_quickest==0 && cumulative_distance>finish_distance) ||
408 (option_quickest==1 && cumulative_duration>finish_duration))
409 goto endloop;
410
411 result2=FindResult(results,node2);
412
413 if(!result2) /* New end node */
414 {
415 result2=InsertResult(results,node2);
416 result2->node=node2;
417 result2->prev=node1;
418 result2->next=0;
419 result2->distance=cumulative_distance;
420 result2->duration=cumulative_duration;
421
422 if((result3=FindResult(end,node2)))
423 {
424 finish_distance=cumulative_distance+result3->distance;
425 finish_duration=cumulative_duration+result3->duration;
426 }
427 else
428 {
429 float lat,lon;
430 GetLatLong(nodes,LookupNode(nodes,node2),&lat,&lon);
431
432 if(option_quickest==0)
433 result2->sortby=result2->distance+Distance(lat,lon,finish_lat,finish_lon);
434 else
435 result2->sortby=result2->duration+distance_speed_to_duration(Distance(lat,lon,finish_lat,finish_lon),max_speed);
436
437 insert_in_queue(result2);
438 }
439 }
440 else if(option_quickest==0) /* shortest */
441 {
442 if(cumulative_distance<result2->distance ||
443 (cumulative_distance==result2->distance &&
444 cumulative_duration<result2->duration)) /* New end node is shorter */
445 {
446 result2->prev=node1;
447 result2->distance=cumulative_distance;
448 result2->duration=cumulative_duration;
449
450 if((result3=FindResult(end,node2)))
451 {
452 if((cumulative_distance+result3->distance)<finish_distance ||
453 ((cumulative_distance+result3->distance)==finish_distance &&
454 (cumulative_duration+result3->duration)<finish_duration))
455 {
456 finish_distance=cumulative_distance+result3->distance;
457 finish_duration=cumulative_duration+result3->duration;
458 }
459 }
460 else
461 {
462 float lat,lon;
463 GetLatLong(nodes,LookupNode(nodes,node2),&lat,&lon);
464
465 result2->sortby=result2->distance+Distance(lat,lon,finish_lat,finish_lon);
466 insert_in_queue(result2);
467 }
468 }
469 }
470 else if(option_quickest==1) /* quickest */
471 {
472 if(cumulative_duration<result2->duration ||
473 (cumulative_duration==result2->duration &&
474 cumulative_distance<result2->distance)) /* New end node is quicker */
475 {
476 result2->prev=node1;
477 result2->distance=cumulative_distance;
478 result2->duration=cumulative_duration;
479
480 if((result3=FindResult(end,node2)))
481 {
482 if((cumulative_duration+result3->duration)<finish_duration ||
483 ((cumulative_duration+result3->duration)==finish_duration &&
484 (cumulative_distance+result3->distance)<finish_distance))
485 {
486 finish_distance=cumulative_distance+result3->distance;
487 finish_duration=cumulative_duration+result3->duration;
488 }
489 }
490 else
491 {
492 float lat,lon;
493 GetLatLong(nodes,LookupNode(nodes,node2),&lat,&lon);
494
495 result2->sortby=result2->duration+distance_speed_to_duration(Distance(lat,lon,finish_lat,finish_lon),max_speed);
496 insert_in_queue(result2);
497 }
498 }
499 }
500
501 endloop:
502
503 if(!option_quiet && !(results->number%10000))
504 {
505 printf("\rRouting: End Nodes=%d %.1fkm %.0fmin ",results->number,
506 distance_to_km(result1->distance),duration_to_minutes(result1->duration));
507 fflush(stdout);
508 }
509
510 segment=NextSegment(segments,segment,node1);
511 }
512 }
513
514 if(!option_quiet)
515 {
516 printf("\rRouted: End Super-Nodes=%d %.1fkm %.0fmin \n",results->number,
517 distance_to_km(finish_distance),duration_to_minutes(finish_duration));
518 fflush(stdout);
519 }
520
521 /* Finish off the end part of the route. */
522
523 if(!FindResult(results,finish))
524 {
525 result2=InsertResult(results,finish);
526 result3=FindResult(end,finish);
527
528 *result2=*result3;
529
530 result2->distance=~0;
531 result2->duration=~0;
532
533 result3=FirstResult(end);
534
535 while(result3)
536 {
537 if(IsSuperNode(LookupNode(nodes,result3->node)))
538 if((result1=FindResult(results,result3->node)))
539 {
540 if(option_quickest==0) /* shortest */
541 {
542 if((result1->distance+result3->distance)<result2->distance ||
543 ((result1->distance+result3->distance)==result2->distance &&
544 (result1->duration+result3->duration)<result2->duration))
545 {
546 result2->distance=result1->distance+result3->distance;
547 result2->duration=result1->duration+result3->duration;
548 result2->prev=result1->node;
549 }
550 }
551 else
552 {
553 if((result1->duration+result3->duration)<result2->duration ||
554 ((result1->duration+result3->duration)==result2->duration &&
555 (result1->distance+result3->distance)<result2->distance))
556 {
557 result2->distance=result1->distance+result3->distance;
558 result2->duration=result1->duration+result3->duration;
559 result2->prev=result1->node;
560 }
561 }
562 }
563
564 result3=NextResult(end,result3);
565 }
566 }
567
568 /* Check it worked */
569
570 if(finish_distance == ~0)
571 {
572 FreeResultsList(results);
573 return(NULL);
574 }
575
576 /* Reverse the results */
577
578 result2=FindResult(results,finish);
579
580 do
581 {
582 if(result2->prev)
583 {
584 index_t node1=result2->prev;
585
586 result1=FindResult(results,node1);
587
588 result1->next=result2->node;
589
590 result2=result1;
591 }
592 else
593 result2=NULL;
594 }
595 while(result2);
596
597 return(results);
598 }
599
600
601 /*++++++++++++++++++++++++++++++++++++++
602 Print the optimum route between two nodes.
603
604 Results *Results The set of results to print.
605
606 Nodes *nodes The list of nodes.
607
608 Segments *segments The set of segments to use.
609
610 Ways *ways The list of ways.
611
612 index_t start The start node.
613
614 index_t finish The finish node.
615
616 Profile *profile The profile containing the transport type, speeds and allowed highways.
617 ++++++++++++++++++++++++++++++++++++++*/
618
619 void PrintRoute(Results *results,Nodes *nodes,Segments *segments,Ways *ways,index_t start,index_t finish,Profile *profile)
620 {
621 FILE *textfile,*gpxfile;
622 Result *result;
623
624 if(option_quickest==0)
625 {
626 /* Print the result for the shortest route */
627
628 textfile=fopen("shortest.txt","w");
629 gpxfile=fopen("shortest.gpx","w");
630 }
631 else
632 {
633 /* Print the result for the quickest route */
634
635 textfile=fopen("quickest.txt","w");
636 gpxfile=fopen("quickest.gpx","w");
637 }
638
639 fprintf(gpxfile,"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
640 fprintf(gpxfile,"<gpx version=\"1.0\" creator=\"Router\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns=\"http://www.topografix.com/GPX/1/0\" xsi:schemaLocation=\"http://www.topografix.com/GPX/1/0 http://www.topografix.com/GPX/1/0/gpx.xsd\">\n");
641 fprintf(gpxfile,"<trk>\n");
642 fprintf(gpxfile,"<trkseg>\n");
643
644 result=FindResult(results,start);
645
646 do
647 {
648 float latitude,longitude;
649 Node *node=LookupNode(nodes,result->node);
650
651 GetLatLong(nodes,node,&latitude,&longitude);
652
653 fprintf(gpxfile,"<trkpt lat=\"%.6f\" lon=\"%.6f\"></trkpt>\n",
654 (180/M_PI)*latitude,(180/M_PI)*longitude);
655
656 if(result->prev)
657 {
658 Segment *segment;
659 Way *way;
660
661 segment=FirstSegment(segments,LookupNode(nodes,result->prev));
662 while(OtherNode(segment,result->prev)!=result->node)
663 segment=NextSegment(segments,segment,result->prev);
664
665 way=LookupWay(ways,segment->way);
666
667 fprintf(textfile,"%8.4f %9.4f %8d%c %5.3f %5.2f %7.2f %5.1f %3d %s\n",
668 (180/M_PI)*latitude,(180/M_PI)*longitude,
669 result->node,IsSuperNode(node)?'*':' ',
670 distance_to_km(DISTANCE(segment->distance)),duration_to_minutes(Duration(segment,way,profile)),
671 distance_to_km(result->distance),duration_to_minutes(result->duration),
672 profile->speed[HIGHWAY(way->type)],WayName(ways,way));
673 }
674 else
675 fprintf(textfile,"%8.4f %9.4f %8d%c %5.3f %5.2f %7.2f %5.1f\n",
676 (180/M_PI)*latitude,(180/M_PI)*longitude,
677 result->node,IsSuperNode(node)?'*':' ',
678 0.0,0.0,0.0,0.0);
679
680 if(result->next)
681 result=FindResult(results,result->next);
682 else
683 result=NULL;
684 }
685 while(result);
686
687 fprintf(gpxfile,"</trkseg>\n");
688 fprintf(gpxfile,"</trk>\n");
689 fprintf(gpxfile,"</gpx>\n");
690
691 fclose(textfile);
692 fclose(gpxfile);
693 }
694
695
696 /*++++++++++++++++++++++++++++++++++++++
697 Find all routes from a specified node to any super-node.
698
699 Results *FindRoutes Returns a set of results.
700
701 Nodes *nodes The set of nodes to use.
702
703 Segments *segments The set of segments to use.
704
705 Ways *ways The set of ways to use.
706
707 index_t start The start node.
708
709 Profile *profile The profile containing the transport type, speeds and allowed highways.
710 ++++++++++++++++++++++++++++++++++++++*/
711
712 Results *FindRoutes(Nodes *nodes,Segments *segments,Ways *ways,index_t start,Profile *profile)
713 {
714 Results *results;
715 index_t node1,node2;
716 Result *result1,*result2;
717 Segment *segment;
718 Way *way;
719
720 /* Insert the first node into the queue */
721
722 results=NewResultsList(8);
723
724 result1=InsertResult(results,start);
725
726 result1->node=start;
727 result1->prev=0;
728 result1->next=0;
729 result1->distance=0;
730 result1->duration=0;
731 result1->sortby=0;
732
733 insert_in_queue(result1);
734
735 /* Loop across all nodes in the queue */
736
737 while((result1=pop_from_queue()))
738 {
739 node1=result1->node;
740
741 segment=FirstSegment(segments,LookupNode(nodes,node1));
742
743 while(segment)
744 {
745 distance_t cumulative_distance;
746 duration_t cumulative_duration;
747
748 if(!IsNormalSegment(segment))
749 goto endloop;
750
751 if(profile->oneway && IsOnewayTo(segment,node1))
752 goto endloop;
753
754 node2=OtherNode(segment,node1);
755
756 if(result1->prev==node2)
757 goto endloop;
758
759 way=LookupWay(ways,segment->way);
760
761 if(!(way->allow&profile->allow))
762 goto endloop;
763
764 if(!profile->highways[HIGHWAY(way->type)])
765 goto endloop;
766
767 cumulative_distance=result1->distance+DISTANCE(segment->distance);
768 cumulative_duration=result1->duration+Duration(segment,way,profile);
769
770 result2=FindResult(results,node2);
771
772 if(!result2) /* New end node */
773 {
774 result2=InsertResult(results,node2);
775 result2->node=node2;
776 result2->prev=node1;
777 result2->next=0;
778 result2->distance=cumulative_distance;
779 result2->duration=cumulative_duration;
780
781 if(!IsSuperNode(LookupNode(nodes,node2)))
782 {
783 result2->sortby=result2->distance;
784 insert_in_queue(result2);
785 }
786 }
787 else if(option_quickest==0) /* shortest */
788 {
789 if(cumulative_distance<result2->distance ||
790 (cumulative_distance==result2->distance &&
791 cumulative_duration<result2->duration)) /* New end node is shorter */
792 {
793 result2->prev=node1;
794 result2->distance=cumulative_distance;
795 result2->duration=cumulative_duration;
796
797 if(!IsSuperNode(LookupNode(nodes,node2)))
798 {
799 result2->sortby=result2->distance;
800 insert_in_queue(result2);
801 }
802 }
803 }
804 else if(option_quickest==1) /* quickest */
805 {
806 if(cumulative_duration<result2->duration ||
807 (cumulative_duration==result2->duration &&
808 cumulative_distance<result2->distance)) /* New end node is quicker */
809 {
810 result2->prev=node1;
811 result2->distance=cumulative_distance;
812 result2->duration=cumulative_duration;
813
814 if(!IsSuperNode(LookupNode(nodes,node2)))
815 {
816 result2->sortby=result2->duration;
817 insert_in_queue(result2);
818 }
819 }
820 }
821
822 endloop:
823
824 segment=NextSegment(segments,segment,node1);
825 }
826 }
827
828 /* Check it worked */
829
830 if(results->number==1)
831 {
832 FreeResultsList(results);
833 return(NULL);
834 }
835
836 return(results);
837 }
838
839
840 /*++++++++++++++++++++++++++++++++++++++
841 Find all routes from any super-node to a specific node.
842
843 Results *FindReverseRoute Returns a set of results.
844
845 Nodes *nodes The set of nodes to use.
846
847 Segments *segments The set of segments to use.
848
849 Ways *ways The set of ways to use.
850
851 index_t finish The finishing node.
852
853 Profile *profile The profile containing the transport type, speeds and allowed highways.
854 ++++++++++++++++++++++++++++++++++++++*/
855
856 Results *FindReverseRoutes(Nodes *nodes,Segments *segments,Ways *ways,index_t finish,Profile *profile)
857 {
858 Results *results;
859 index_t node1,node2;
860 Result *result1,*result2;
861 Segment *segment;
862 Way *way;
863
864 /* Insert the first node into the queue */
865
866 results=NewResultsList(8);
867
868 result1=InsertResult(results,finish);
869
870 result1->node=finish;
871 result1->prev=0;
872 result1->next=0;
873 result1->distance=0;
874 result1->duration=0;
875 result1->sortby=0;
876
877 insert_in_queue(result1);
878
879 /* Loop across all nodes in the queue */
880
881 while((result1=pop_from_queue()))
882 {
883 node1=result1->node;
884
885 segment=FirstSegment(segments,LookupNode(nodes,node1));
886
887 while(segment)
888 {
889 distance_t cumulative_distance;
890 duration_t cumulative_duration;
891
892 if(!IsNormalSegment(segment))
893 goto endloop;
894
895 if(profile->oneway && IsOnewayFrom(segment,node1))
896 goto endloop;
897
898 node2=OtherNode(segment,node1);
899
900 if(result1->next==node2)
901 goto endloop;
902
903 way=LookupWay(ways,segment->way);
904
905 if(!(way->allow&profile->allow))
906 goto endloop;
907
908 if(!profile->highways[HIGHWAY(way->type)])
909 goto endloop;
910
911 cumulative_distance=result1->distance+DISTANCE(segment->distance);
912 cumulative_duration=result1->duration+Duration(segment,way,profile);
913
914 result2=FindResult(results,node2);
915
916 if(!result2) /* New end node */
917 {
918 result2=InsertResult(results,node2);
919 result2->node=node2;
920 result2->prev=0;
921 result2->next=node1;
922 result2->distance=cumulative_distance;
923 result2->duration=cumulative_duration;
924
925 if(!IsSuperNode(LookupNode(nodes,node2)))
926 {
927 result2->sortby=result2->distance;
928 insert_in_queue(result2);
929 }
930 }
931 else if(option_quickest==0) /* shortest */
932 {
933 if(cumulative_distance<result2->distance ||
934 (cumulative_distance==result2->distance &&
935 cumulative_duration<result2->duration)) /* New end node is shorter */
936 {
937 result2->next=node1;
938 result2->distance=cumulative_distance;
939 result2->duration=cumulative_duration;
940
941 if(!IsSuperNode(LookupNode(nodes,node2)))
942 {
943 result2->sortby=result2->distance;
944 insert_in_queue(result2);
945 }
946 }
947 }
948 else if(option_quickest==1) /* quickest */
949 {
950 if(cumulative_duration<result2->duration ||
951 (cumulative_duration==result2->duration &&
952 cumulative_distance<result2->distance)) /* New end node is quicker */
953 {
954 result2->next=node1;
955 result2->distance=cumulative_distance;
956 result2->duration=cumulative_duration;
957
958 if(!IsSuperNode(LookupNode(nodes,node2)))
959 {
960 result2->sortby=result2->duration;
961 insert_in_queue(result2);
962 }
963 }
964 }
965
966 endloop:
967
968 segment=NextSegment(segments,segment,node1);
969 }
970 }
971
972 /* Check it worked */
973
974 if(results->number==1)
975 {
976 FreeResultsList(results);
977 return(NULL);
978 }
979
980 return(results);
981 }
982
983
984 /*++++++++++++++++++++++++++++++++++++++
985 Create an optimum route given the set of super-nodes to follow.
986
987 Results *CombineRoutes Returns the results from joining the super-nodes.
988
989 Results *results The set of results from the super-nodes.
990
991 Nodes *nodes The list of nodes.
992
993 Segments *segments The set of segments to use.
994
995 Ways *ways The list of ways.
996
997 index_t start The start node.
998
999 index_t finish The finish node.
1000
1001 Profile *profile The profile containing the transport type, speeds and allowed highways.
1002 ++++++++++++++++++++++++++++++++++++++*/
1003
1004 Results *CombineRoutes(Results *results,Nodes *nodes,Segments *segments,Ways *ways,index_t start,index_t finish,Profile *profile)
1005 {
1006 Result *result1,*result2,*result3,*result4;
1007 Results *combined;
1008 int quiet=option_quiet;
1009
1010 combined=NewResultsList(64);
1011
1012 option_quiet=1;
1013
1014 /* Sort out the combined route */
1015
1016 result1=FindResult(results,start);
1017
1018 result3=InsertResult(combined,start);
1019
1020 result3->node=result1->node;
1021
1022 result3->distance=0;
1023 result3->duration=0;
1024 result3->next=0;
1025 result3->prev=0;
1026
1027 do
1028 {
1029 if(result1->next)
1030 {
1031 Results *results2=FindRoute(nodes,segments,ways,result1->node,result1->next,profile,0);
1032
1033 result2=FindResult(results2,result1->node);
1034
1035 result3->next=result2->next;
1036
1037 result2=FindResult(results2,result2->next);
1038
1039 do
1040 {
1041 result4=InsertResult(combined,result2->node);
1042
1043 result4->node=result2->node;
1044
1045 *result4=*result2;
1046 result4->distance+=result3->distance;
1047 result4->duration+=result3->duration;
1048
1049 if(result2->next)
1050 result2=FindResult(results2,result2->next);
1051 else
1052 result2=NULL;
1053 }
1054 while(result2);
1055
1056 FreeResultsList(results2);
1057
1058 result1=FindResult(results,result1->next);
1059
1060 result3=result4;
1061 }
1062 else
1063 result1=NULL;
1064 }
1065 while(result1);
1066
1067 /* Now print out the result normally */
1068
1069 option_quiet=quiet;
1070
1071 return(combined);
1072 }

Properties

Name Value
cvs:description Route optimiser.