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 149 - (show annotations) (download) (as text)
Sat Mar 28 14:31:06 2009 UTC (16 years ago) by amb
File MIME type: text/x-csrc
File size: 31657 byte(s)
Fix file headers (again) and fix segment distance/duration for abbreviated text
output.

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

Properties

Name Value
cvs:description Route optimiser.