Routino SVN Repository Browser

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

ViewVC logotype

Annotation of /trunk/src/optimiser.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 307 - (hide annotations) (download) (as text)
Wed Nov 25 15:00:37 2009 UTC (15 years, 4 months ago) by amb
File MIME type: text/x-csrc
File size: 23517 byte(s)
Store the selected options when parsing (planetsplitter) and display them in the
statistics (filedumper) and check them when routing (router).

1 amb 2 /***************************************
2 amb 307 $Header: /home/amb/CVS/routino/src/optimiser.c,v 1.82 2009-11-25 15:00:37 amb Exp $
3 amb 2
4     Routing optimiser.
5 amb 151
6     Part of the Routino routing software.
7 amb 2 ******************/ /******************
8 amb 151 This file Copyright 2008,2009 Andrew M. Bishop
9 amb 2
10 amb 151 This program is free software: you can redistribute it and/or modify
11     it under the terms of the GNU Affero General Public License as published by
12     the Free Software Foundation, either version 3 of the License, or
13     (at your option) any later version.
14    
15     This program is distributed in the hope that it will be useful,
16     but WITHOUT ANY WARRANTY; without even the implied warranty of
17     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18     GNU Affero General Public License for more details.
19    
20     You should have received a copy of the GNU Affero General Public License
21     along with this program. If not, see <http://www.gnu.org/licenses/>.
22 amb 2 ***************************************/
23    
24    
25 amb 109 #include <stdio.h>
26 amb 2
27 amb 96 #include "types.h"
28 amb 109 #include "functions.h"
29     #include "nodes.h"
30     #include "segments.h"
31     #include "ways.h"
32 amb 28 #include "results.h"
33 amb 2
34 amb 26
35 amb 113 /*+ The option not to print any progress information. +*/
36 amb 107 extern int option_quiet;
37 amb 2
38 amb 113 /*+ The option to calculate the quickest route insted of the shortest. +*/
39     extern int option_quickest;
40 amb 31
41 amb 113
42 amb 2 /*++++++++++++++++++++++++++++++++++++++
43 amb 238 Find the optimum route between two nodes not passing through a super-node.
44 amb 2
45 amb 238 Results *FindNormalRoute Returns a set of results.
46 amb 26
47     Nodes *nodes The set of nodes to use.
48    
49     Segments *segments The set of segments to use.
50    
51 amb 54 Ways *ways The set of ways to use.
52    
53 amb 96 index_t start The start node.
54 amb 2
55 amb 96 index_t finish The finish node.
56 amb 54
57 amb 82 Profile *profile The profile containing the transport type, speeds and allowed highways.
58 amb 2 ++++++++++++++++++++++++++++++++++++++*/
59    
60 amb 238 Results *FindNormalRoute(Nodes *nodes,Segments *segments,Ways *ways,index_t start,index_t finish,Profile *profile)
61 amb 2 {
62 amb 31 Results *results;
63 amb 236 Queue *queue;
64 amb 96 index_t node1,node2;
65 amb 166 score_t finish_score;
66 amb 219 double finish_lat,finish_lon;
67 amb 2 Result *result1,*result2;
68     Segment *segment;
69 amb 54 Way *way;
70 amb 2
71 amb 10 /* Set up the finish conditions */
72    
73 amb 176 finish_score=INF_SCORE;
74 amb 10
75 amb 303 if(IsFakeNode(finish))
76     GetFakeLatLong(finish,&finish_lat,&finish_lon);
77     else
78     GetLatLong(nodes,finish,&finish_lat,&finish_lon);
79 amb 119
80 amb 165 /* Create the list of results and insert the first node into the queue */
81 amb 2
82 amb 238 results=NewResultsList(8);
83 amb 2
84 amb 165 results->start=start;
85     results->finish=finish;
86    
87 amb 31 result1=InsertResult(results,start);
88 amb 28
89 amb 166 ZeroResult(result1);
90 amb 2
91 amb 236 queue=NewQueueList();
92 amb 2
93 amb 236 InsertInQueue(queue,result1);
94    
95 amb 2 /* Loop across all nodes in the queue */
96    
97 amb 236 while((result1=PopFromQueue(queue)))
98 amb 2 {
99 amb 238 if(result1->score>finish_score)
100 amb 119 continue;
101    
102 amb 43 node1=result1->node;
103 amb 2
104 amb 303 if(IsFakeNode(node1))
105     segment=FirstFakeSegment(node1);
106     else
107     segment=FirstSegment(segments,nodes,node1);
108 amb 2
109     while(segment)
110     {
111 amb 298 score_t segment_pref,segment_score,cumulative_score;
112     int i;
113 amb 59
114 amb 303 node2=OtherNode(segment,node1); /* need this here because we use node2 later */
115    
116 amb 95 if(!IsNormalSegment(segment))
117 amb 90 goto endloop;
118    
119 amb 105 if(profile->oneway && IsOnewayTo(segment,node1))
120 amb 64 goto endloop;
121    
122 amb 113 if(result1->prev==node2)
123 amb 64 goto endloop;
124    
125 amb 303 if(node2!=finish && !IsFakeNode(node2) && IsSuperNode(nodes,node2))
126 amb 238 goto endloop;
127    
128 amb 96 way=LookupWay(ways,segment->way);
129 amb 54
130 amb 82 if(!(way->allow&profile->allow))
131 amb 54 goto endloop;
132    
133 amb 168 if(!profile->highway[HIGHWAY(way->type)])
134 amb 75 goto endloop;
135    
136 amb 197 if(way->weight && way->weight<profile->weight)
137 amb 135 goto endloop;
138    
139 amb 197 if((way->height && way->height<profile->height) ||
140     (way->width && way->width <profile->width ) ||
141     (way->length && way->length<profile->length))
142 amb 135 goto endloop;
143    
144 amb 298 segment_pref=profile->highway[HIGHWAY(way->type)];
145    
146     for(i=1;i<Property_Count;i++)
147 amb 307 if(ways->props && PROPERTIES(i))
148     {
149     if(way->props && PROPERTIES(i))
150     segment_pref*=profile->props_yes[i];
151     else
152     segment_pref*=profile->props_no[i];
153     }
154 amb 298
155     if(segment_pref==0)
156     goto endloop;
157    
158 amb 166 if(option_quickest==0)
159 amb 298 segment_score=(score_t)DISTANCE(segment->distance)/segment_pref;
160 amb 166 else
161 amb 298 segment_score=(score_t)Duration(segment,way,profile)/segment_pref;
162 amb 59
163 amb 170 cumulative_score=result1->score+segment_score;
164 amb 166
165     if(cumulative_score>finish_score)
166 amb 2 goto endloop;
167    
168 amb 42 result2=FindResult(results,node2);
169    
170 amb 2 if(!result2) /* New end node */
171     {
172 amb 31 result2=InsertResult(results,node2);
173 amb 113 result2->prev=node1;
174 amb 176 result2->next=NO_NODE;
175 amb 170 result2->score=cumulative_score;
176     result2->segment=segment;
177 amb 2
178     if(node2==finish)
179     {
180 amb 170 finish_score=cumulative_score;
181 amb 2 }
182     else
183 amb 119 {
184 amb 238 result2->sortby=result2->score;
185 amb 166
186 amb 236 InsertInQueue(queue,result2);
187 amb 119 }
188 amb 2 }
189 amb 166 else if(cumulative_score<result2->score) /* New end node is better */
190 amb 2 {
191 amb 166 result2->prev=node1;
192 amb 170 result2->score=cumulative_score;
193     result2->segment=segment;
194 amb 166
195     if(node2==finish)
196 amb 2 {
197 amb 166 finish_score=cumulative_score;
198 amb 2 }
199 amb 166 else
200     {
201 amb 238 result2->sortby=result2->score;
202 amb 2
203 amb 238 if(result2->score<finish_score)
204 amb 236 InsertInQueue(queue,result2);
205 amb 2 }
206     }
207    
208     endloop:
209    
210 amb 303 if(IsFakeNode(node1))
211     segment=NextFakeSegment(segment,node1);
212     else if(IsFakeNode(node2))
213     segment=NULL; /* cannot call NextSegment() with a fake segment */
214     else
215     {
216     segment=NextSegment(segments,segment,node1);
217    
218     if(!segment && IsFakeNode(finish))
219     segment=ExtraFakeSegment(node1,finish);
220     }
221 amb 2 }
222     }
223    
224 amb 236 FreeQueueList(queue);
225    
226 amb 77 /* Check it worked */
227    
228 amb 290 if(!FindResult(results,finish))
229 amb 77 {
230     FreeResultsList(results);
231     return(NULL);
232     }
233    
234 amb 290 FixForwardRoute(results,finish);
235 amb 31
236     return(results);
237 amb 2 }
238    
239    
240     /*++++++++++++++++++++++++++++++++++++++
241 amb 238 Find the optimum route between two nodes where the start and end are a set of pre-routed super-nodes.
242 amb 34
243 amb 238 Results *FindMiddleRoute Returns a set of results.
244 amb 34
245 amb 95 Nodes *nodes The set of nodes to use.
246 amb 34
247 amb 95 Segments *segments The set of segments to use.
248 amb 34
249 amb 95 Ways *ways The set of ways to use.
250 amb 54
251 amb 34 Results *begin The initial portion of the route.
252    
253     Results *end The final portion of the route.
254 amb 54
255 amb 82 Profile *profile The profile containing the transport type, speeds and allowed highways.
256 amb 34 ++++++++++++++++++++++++++++++++++++++*/
257    
258 amb 238 Results *FindMiddleRoute(Nodes *nodes,Segments *segments,Ways *ways,Results *begin,Results *end,Profile *profile)
259 amb 34 {
260     Results *results;
261 amb 236 Queue *queue;
262 amb 96 index_t node1,node2;
263 amb 166 score_t finish_score;
264 amb 219 double finish_lat,finish_lon;
265 amb 34 Result *result1,*result2,*result3;
266     Segment *segment;
267 amb 54 Way *way;
268 amb 34
269 amb 227 if(!option_quiet)
270     {
271 amb 305 printf("Routing: Super-Nodes checked = 0");
272 amb 227 fflush(stdout);
273     }
274    
275 amb 34 /* Set up the finish conditions */
276    
277 amb 302 finish_score=INF_DISTANCE;
278 amb 34
279 amb 303 if(IsFakeNode(end->finish))
280     GetFakeLatLong(end->finish,&finish_lat,&finish_lon);
281     else
282     GetLatLong(nodes,end->finish,&finish_lat,&finish_lon);
283 amb 119
284 amb 165 /* Create the list of results and insert the first node into the queue */
285 amb 34
286 amb 71 results=NewResultsList(65536);
287 amb 34
288 amb 165 results->start=begin->start;
289     results->finish=end->finish;
290 amb 34
291 amb 165 result1=InsertResult(results,begin->start);
292     result3=FindResult(begin,begin->start);
293    
294 amb 81 *result1=*result3;
295 amb 34
296 amb 236 queue=NewQueueList();
297    
298 amb 34 /* Insert the finish points of the beginning part of the path into the queue */
299    
300 amb 79 result3=FirstResult(begin);
301    
302     while(result3)
303     {
304 amb 303 if(result3->node!=begin->start && !IsFakeNode(result3->node) && IsSuperNode(nodes,result3->node))
305 amb 45 {
306 amb 238 result2=InsertResult(results,result3->node);
307 amb 34
308 amb 238 *result2=*result3;
309 amb 34
310 amb 238 result2->prev=begin->start;
311 amb 119
312 amb 238 result2->sortby=result2->score;
313 amb 34
314 amb 236 InsertInQueue(queue,result2);
315 amb 45 }
316 amb 34
317 amb 79 result3=NextResult(begin,result3);
318     }
319    
320 amb 238 if(begin->number==1)
321     InsertInQueue(queue,result1);
322    
323 amb 95 /* Loop across all nodes in the queue */
324 amb 34
325 amb 236 while((result1=PopFromQueue(queue)))
326 amb 34 {
327 amb 238 if(result1->score>finish_score)
328 amb 119 continue;
329    
330 amb 43 node1=result1->node;
331 amb 34
332 amb 303 if(IsFakeNode(node1))
333     segment=FirstFakeSegment(node1);
334     else
335     segment=FirstSegment(segments,nodes,node1);
336 amb 34
337     while(segment)
338     {
339 amb 298 score_t segment_pref,segment_score,cumulative_score;
340     int i;
341 amb 59
342 amb 95 if(!IsSuperSegment(segment))
343     goto endloop;
344    
345 amb 105 if(profile->oneway && IsOnewayTo(segment,node1))
346 amb 64 goto endloop;
347    
348 amb 105 node2=OtherNode(segment,node1);
349 amb 34
350 amb 113 if(result1->prev==node2)
351 amb 64 goto endloop;
352    
353 amb 96 way=LookupWay(ways,segment->way);
354 amb 54
355 amb 82 if(!(way->allow&profile->allow))
356 amb 54 goto endloop;
357    
358 amb 168 if(!profile->highway[HIGHWAY(way->type)])
359 amb 75 goto endloop;
360    
361 amb 197 if(way->weight && way->weight<profile->weight)
362 amb 135 goto endloop;
363    
364 amb 197 if((way->height && way->height<profile->height) ||
365     (way->width && way->width <profile->width ) ||
366     (way->length && way->length<profile->length))
367 amb 135 goto endloop;
368    
369 amb 298 segment_pref=profile->highway[HIGHWAY(way->type)];
370    
371     for(i=1;i<Property_Count;i++)
372 amb 307 if(ways->props && PROPERTIES(i))
373     {
374     if(way->props && PROPERTIES(i))
375     segment_pref*=profile->props_yes[i];
376     else
377     segment_pref*=profile->props_no[i];
378     }
379 amb 298
380     if(segment_pref==0)
381     goto endloop;
382    
383 amb 166 if(option_quickest==0)
384 amb 298 segment_score=(score_t)DISTANCE(segment->distance)/segment_pref;
385 amb 166 else
386 amb 298 segment_score=(score_t)Duration(segment,way,profile)/segment_pref;
387 amb 34
388 amb 170 cumulative_score=result1->score+segment_score;
389 amb 166
390     if(cumulative_score>finish_score)
391 amb 34 goto endloop;
392    
393 amb 42 result2=FindResult(results,node2);
394    
395 amb 34 if(!result2) /* New end node */
396     {
397     result2=InsertResult(results,node2);
398 amb 113 result2->prev=node1;
399 amb 176 result2->next=NO_NODE;
400 amb 170 result2->score=cumulative_score;
401     result2->segment=segment;
402 amb 34
403 amb 113 if((result3=FindResult(end,node2)))
404     {
405 amb 170 finish_score=result2->score+result3->score;
406 amb 113 }
407     else
408 amb 119 {
409 amb 219 double lat,lon;
410 amb 166 distance_t direct;
411    
412 amb 303 if(IsFakeNode(node2))
413     GetFakeLatLong(node2,&lat,&lon);
414     else
415     GetLatLong(nodes,node2,&lat,&lon);
416 amb 166 direct=Distance(lat,lon,finish_lat,finish_lon);
417 amb 119
418     if(option_quickest==0)
419 amb 173 result2->sortby=result2->score+(score_t)direct/profile->max_pref;
420 amb 119 else
421 amb 173 result2->sortby=result2->score+(score_t)distance_speed_to_duration(direct,profile->max_speed)/profile->max_pref;
422 amb 119
423 amb 236 InsertInQueue(queue,result2);
424 amb 119 }
425 amb 34 }
426 amb 166 else if(cumulative_score<result2->score) /* New end node is better */
427 amb 34 {
428 amb 166 result2->prev=node1;
429 amb 170 result2->score=cumulative_score;
430     result2->segment=segment;
431 amb 166
432     if((result3=FindResult(end,node2)))
433 amb 34 {
434 amb 166 if((result2->score+result3->score)<finish_score)
435 amb 113 {
436 amb 166 finish_score=result2->score+result3->score;
437 amb 113 }
438 amb 34 }
439 amb 238 else if(result2->score<finish_score)
440 amb 34 {
441 amb 219 double lat,lon;
442 amb 166 distance_t direct;
443 amb 34
444 amb 303 if(IsFakeNode(node2))
445     GetFakeLatLong(node2,&lat,&lon);
446     else
447     GetLatLong(nodes,node2,&lat,&lon);
448 amb 166 direct=Distance(lat,lon,finish_lat,finish_lon);
449    
450     if(option_quickest==0)
451 amb 173 result2->sortby=result2->score+(score_t)direct/profile->max_pref;
452 amb 113 else
453 amb 173 result2->sortby=result2->score+(score_t)distance_speed_to_duration(direct,profile->max_speed)/profile->max_pref;
454 amb 119
455 amb 238 InsertInQueue(queue,result2);
456 amb 34 }
457     }
458    
459     endloop:
460    
461 amb 107 if(!option_quiet && !(results->number%10000))
462 amb 95 {
463 amb 305 printf("\rRouting: Super-Nodes checked = %d",results->number);
464 amb 95 fflush(stdout);
465     }
466 amb 34
467 amb 303 if(IsFakeNode(node1))
468     segment=NextFakeSegment(segment,node1);
469     else
470     segment=NextSegment(segments,segment,node1);
471 amb 34 }
472     }
473    
474 amb 107 if(!option_quiet)
475 amb 34 {
476 amb 305 printf("\rRouted: Super-Nodes checked = %d\n",results->number);
477 amb 34 fflush(stdout);
478     }
479    
480     /* Finish off the end part of the route. */
481    
482 amb 165 if(!FindResult(results,end->finish))
483 amb 47 {
484 amb 165 result2=InsertResult(results,end->finish);
485     result3=FindResult(end,end->finish);
486 amb 34
487 amb 81 *result2=*result3;
488 amb 47
489 amb 302 result2->score=INF_DISTANCE;
490 amb 54
491 amb 79 result3=FirstResult(end);
492    
493     while(result3)
494     {
495 amb 303 if(!IsFakeNode(result3->node) && IsSuperNode(nodes,result3->node))
496 amb 79 if((result1=FindResult(results,result3->node)))
497 amb 170 if((result1->score+result3->score)<result2->score)
498 amb 47 {
499 amb 170 result2->score=result1->score+result3->score;
500     result2->prev=result1->node;
501 amb 47 }
502 amb 79
503     result3=NextResult(end,result3);
504     }
505 amb 47 }
506 amb 34
507 amb 236 FreeQueueList(queue);
508    
509 amb 77 /* Check it worked */
510    
511 amb 290 if(!FindResult(results,end->finish))
512 amb 77 {
513     FreeResultsList(results);
514     return(NULL);
515     }
516    
517 amb 290 FixForwardRoute(results,end->finish);
518 amb 34
519     return(results);
520     }
521    
522    
523     /*++++++++++++++++++++++++++++++++++++++
524 amb 95 Find all routes from a specified node to any super-node.
525 amb 3
526 amb 126 Results *FindStartRoutes Returns a set of results.
527 amb 31
528     Nodes *nodes The set of nodes to use.
529    
530     Segments *segments The set of segments to use.
531    
532 amb 54 Ways *ways The set of ways to use.
533    
534 amb 96 index_t start The start node.
535 amb 31
536 amb 82 Profile *profile The profile containing the transport type, speeds and allowed highways.
537 amb 31 ++++++++++++++++++++++++++++++++++++++*/
538    
539 amb 126 Results *FindStartRoutes(Nodes *nodes,Segments *segments,Ways *ways,index_t start,Profile *profile)
540 amb 31 {
541     Results *results;
542 amb 236 Queue *queue;
543 amb 96 index_t node1,node2;
544 amb 31 Result *result1,*result2;
545     Segment *segment;
546 amb 54 Way *way;
547 amb 31
548     /* Insert the first node into the queue */
549    
550 amb 71 results=NewResultsList(8);
551 amb 31
552 amb 165 results->start=start;
553    
554 amb 31 result1=InsertResult(results,start);
555    
556 amb 166 ZeroResult(result1);
557 amb 31
558 amb 236 queue=NewQueueList();
559 amb 31
560 amb 236 InsertInQueue(queue,result1);
561    
562 amb 31 /* Loop across all nodes in the queue */
563    
564 amb 236 while((result1=PopFromQueue(queue)))
565 amb 31 {
566 amb 43 node1=result1->node;
567 amb 31
568 amb 303 if(IsFakeNode(node1))
569     segment=FirstFakeSegment(node1);
570     else
571     segment=FirstSegment(segments,nodes,node1);
572 amb 31
573     while(segment)
574     {
575 amb 298 score_t segment_pref,segment_score,cumulative_score;
576     int i;
577 amb 59
578 amb 95 if(!IsNormalSegment(segment))
579     goto endloop;
580    
581 amb 105 if(profile->oneway && IsOnewayTo(segment,node1))
582 amb 64 goto endloop;
583    
584 amb 105 node2=OtherNode(segment,node1);
585 amb 31
586 amb 113 if(result1->prev==node2)
587 amb 64 goto endloop;
588    
589 amb 96 way=LookupWay(ways,segment->way);
590 amb 54
591 amb 82 if(!(way->allow&profile->allow))
592 amb 54 goto endloop;
593    
594 amb 168 if(!profile->highway[HIGHWAY(way->type)])
595 amb 75 goto endloop;
596    
597 amb 197 if(way->weight && way->weight<profile->weight)
598 amb 135 goto endloop;
599    
600 amb 197 if((way->height && way->height<profile->height) ||
601     (way->width && way->width <profile->width ) ||
602     (way->length && way->length<profile->length))
603 amb 135 goto endloop;
604    
605 amb 298 segment_pref=profile->highway[HIGHWAY(way->type)];
606    
607     for(i=1;i<Property_Count;i++)
608 amb 307 if(ways->props && PROPERTIES(i))
609     {
610     if(way->props && PROPERTIES(i))
611     segment_pref*=profile->props_yes[i];
612     else
613     segment_pref*=profile->props_no[i];
614     }
615 amb 298
616     if(segment_pref==0)
617     goto endloop;
618    
619 amb 166 if(option_quickest==0)
620 amb 298 segment_score=(score_t)DISTANCE(segment->distance)/segment_pref;
621 amb 166 else
622 amb 298 segment_score=(score_t)Duration(segment,way,profile)/segment_pref;
623 amb 59
624 amb 170 cumulative_score=result1->score+segment_score;
625 amb 166
626 amb 31 result2=FindResult(results,node2);
627    
628     if(!result2) /* New end node */
629     {
630     result2=InsertResult(results,node2);
631 amb 113 result2->prev=node1;
632 amb 176 result2->next=NO_NODE;
633 amb 170 result2->score=cumulative_score;
634     result2->segment=segment;
635 amb 31
636 amb 303 if(!IsFakeNode(node2) && !IsSuperNode(nodes,node2))
637 amb 119 {
638 amb 166 result2->sortby=result2->score;
639 amb 236 InsertInQueue(queue,result2);
640 amb 119 }
641 amb 31 }
642 amb 166 else if(cumulative_score<result2->score) /* New end node is better */
643 amb 31 {
644 amb 166 result2->prev=node1;
645 amb 170 result2->score=cumulative_score;
646     result2->segment=segment;
647 amb 31
648 amb 303 if(!IsFakeNode(node2) && !IsSuperNode(nodes,node2))
649 amb 31 {
650 amb 166 result2->sortby=result2->score;
651 amb 236 InsertInQueue(queue,result2);
652 amb 31 }
653     }
654    
655     endloop:
656    
657 amb 303 if(IsFakeNode(node1))
658     segment=NextFakeSegment(segment,node1);
659     else
660     segment=NextSegment(segments,segment,node1);
661 amb 31 }
662     }
663    
664 amb 236 FreeQueueList(queue);
665    
666 amb 112 /* Check it worked */
667    
668     if(results->number==1)
669     {
670     FreeResultsList(results);
671     return(NULL);
672     }
673    
674 amb 31 return(results);
675 amb 2 }
676    
677    
678     /*++++++++++++++++++++++++++++++++++++++
679 amb 95 Find all routes from any super-node to a specific node.
680 amb 34
681 amb 126 Results *FindFinishRoutes Returns a set of results.
682 amb 34
683     Nodes *nodes The set of nodes to use.
684    
685     Segments *segments The set of segments to use.
686    
687 amb 54 Ways *ways The set of ways to use.
688    
689 amb 96 index_t finish The finishing node.
690 amb 54
691 amb 82 Profile *profile The profile containing the transport type, speeds and allowed highways.
692 amb 34 ++++++++++++++++++++++++++++++++++++++*/
693    
694 amb 126 Results *FindFinishRoutes(Nodes *nodes,Segments *segments,Ways *ways,index_t finish,Profile *profile)
695 amb 34 {
696     Results *results;
697 amb 236 Queue *queue;
698 amb 96 index_t node1,node2;
699 amb 34 Result *result1,*result2;
700     Segment *segment;
701 amb 54 Way *way;
702 amb 34
703     /* Insert the first node into the queue */
704    
705 amb 71 results=NewResultsList(8);
706 amb 34
707 amb 165 results->finish=finish;
708    
709 amb 34 result1=InsertResult(results,finish);
710    
711 amb 166 ZeroResult(result1);
712 amb 34
713 amb 236 queue=NewQueueList();
714 amb 34
715 amb 236 InsertInQueue(queue,result1);
716    
717 amb 34 /* Loop across all nodes in the queue */
718    
719 amb 236 while((result1=PopFromQueue(queue)))
720 amb 34 {
721 amb 43 node1=result1->node;
722 amb 34
723 amb 303 if(IsFakeNode(node1))
724     segment=FirstFakeSegment(node1);
725     else
726     segment=FirstSegment(segments,nodes,node1);
727 amb 34
728     while(segment)
729     {
730 amb 298 score_t segment_pref,segment_score,cumulative_score;
731     int i;
732 amb 59
733 amb 104 if(!IsNormalSegment(segment))
734 amb 34 goto endloop;
735    
736 amb 105 if(profile->oneway && IsOnewayFrom(segment,node1))
737 amb 95 goto endloop;
738    
739 amb 105 node2=OtherNode(segment,node1);
740 amb 64
741 amb 113 if(result1->next==node2)
742 amb 64 goto endloop;
743    
744 amb 104 way=LookupWay(ways,segment->way);
745 amb 54
746 amb 82 if(!(way->allow&profile->allow))
747 amb 54 goto endloop;
748    
749 amb 168 if(!profile->highway[HIGHWAY(way->type)])
750 amb 75 goto endloop;
751    
752 amb 197 if(way->weight && way->weight<profile->weight)
753 amb 135 goto endloop;
754    
755 amb 197 if((way->height && way->height<profile->height) ||
756     (way->width && way->width <profile->width ) ||
757     (way->length && way->length<profile->length))
758 amb 135 goto endloop;
759    
760 amb 298 segment_pref=profile->highway[HIGHWAY(way->type)];
761    
762     for(i=1;i<Property_Count;i++)
763 amb 307 if(ways->props && PROPERTIES(i))
764     {
765     if(way->props && PROPERTIES(i))
766     segment_pref*=profile->props_yes[i];
767     else
768     segment_pref*=profile->props_no[i];
769     }
770 amb 298
771     if(segment_pref==0)
772     goto endloop;
773    
774 amb 166 if(option_quickest==0)
775 amb 298 segment_score=(score_t)DISTANCE(segment->distance)/segment_pref;
776 amb 166 else
777 amb 298 segment_score=(score_t)Duration(segment,way,profile)/segment_pref;
778 amb 59
779 amb 170 cumulative_score=result1->score+segment_score;
780 amb 166
781 amb 34 result2=FindResult(results,node2);
782    
783     if(!result2) /* New end node */
784     {
785     result2=InsertResult(results,node2);
786 amb 176 result2->prev=NO_NODE;
787 amb 113 result2->next=node1;
788 amb 170 result2->score=cumulative_score;
789     result2->segment=segment;
790 amb 34
791 amb 303 if(!IsFakeNode(node2) && !IsSuperNode(nodes,node2))
792 amb 119 {
793 amb 166 result2->sortby=result2->score;
794 amb 236 InsertInQueue(queue,result2);
795 amb 119 }
796 amb 34 }
797 amb 166 else if(cumulative_score<result2->score) /* New end node is better */
798 amb 34 {
799 amb 166 result2->next=node1;
800 amb 170 result2->score=cumulative_score;
801     result2->segment=segment;
802 amb 34
803 amb 303 if(!IsFakeNode(node2) && !IsSuperNode(nodes,node2))
804 amb 34 {
805 amb 166 result2->sortby=result2->score;
806 amb 236 InsertInQueue(queue,result2);
807 amb 34 }
808     }
809    
810     endloop:
811    
812 amb 303 if(IsFakeNode(node1))
813     segment=NextFakeSegment(segment,node1);
814     else
815     segment=NextSegment(segments,segment,node1);
816 amb 34 }
817     }
818    
819 amb 236 FreeQueueList(queue);
820    
821 amb 112 /* Check it worked */
822    
823     if(results->number==1)
824     {
825     FreeResultsList(results);
826     return(NULL);
827     }
828    
829 amb 34 return(results);
830     }
831    
832    
833     /*++++++++++++++++++++++++++++++++++++++
834 amb 95 Create an optimum route given the set of super-nodes to follow.
835 amb 31
836 amb 58 Results *CombineRoutes Returns the results from joining the super-nodes.
837 amb 55
838 amb 58 Results *results The set of results from the super-nodes.
839 amb 31
840     Nodes *nodes The list of nodes.
841    
842     Segments *segments The set of segments to use.
843    
844     Ways *ways The list of ways.
845    
846 amb 82 Profile *profile The profile containing the transport type, speeds and allowed highways.
847 amb 31 ++++++++++++++++++++++++++++++++++++++*/
848    
849 amb 165 Results *CombineRoutes(Results *results,Nodes *nodes,Segments *segments,Ways *ways,Profile *profile)
850 amb 31 {
851     Result *result1,*result2,*result3,*result4;
852     Results *combined;
853    
854 amb 71 combined=NewResultsList(64);
855 amb 31
856 amb 165 combined->start=results->start;
857     combined->finish=results->finish;
858    
859 amb 113 /* Sort out the combined route */
860 amb 31
861 amb 165 result1=FindResult(results,results->start);
862 amb 31
863 amb 165 result3=InsertResult(combined,results->start);
864 amb 31
865 amb 166 ZeroResult(result3);
866 amb 36
867 amb 58 do
868     {
869 amb 176 if(result1->next!=NO_NODE)
870 amb 31 {
871 amb 238 Results *results2=FindNormalRoute(nodes,segments,ways,result1->node,result1->next,profile);
872 amb 31
873     result2=FindResult(results2,result1->node);
874    
875 amb 113 result3->next=result2->next;
876 amb 36
877 amb 113 result2=FindResult(results2,result2->next);
878 amb 36
879 amb 31 do
880     {
881 amb 58 result4=InsertResult(combined,result2->node);
882 amb 31
883 amb 113 *result4=*result2;
884 amb 170 result4->score+=result3->score;
885 amb 31
886 amb 176 if(result2->next!=NO_NODE)
887 amb 113 result2=FindResult(results2,result2->next);
888 amb 31 else
889     result2=NULL;
890     }
891     while(result2);
892    
893     FreeResultsList(results2);
894    
895 amb 113 result1=FindResult(results,result1->next);
896 amb 58
897     result3=result4;
898 amb 31 }
899     else
900     result1=NULL;
901     }
902     while(result1);
903    
904 amb 55 return(combined);
905 amb 31 }
906 amb 290
907    
908     /*++++++++++++++++++++++++++++++++++++++
909     Fx the forward route (i.e. setup next nodes for forward path from prev nodes on reverse path).
910    
911     Results *results The set of results to update.
912    
913     index_t finish The finish point.
914     ++++++++++++++++++++++++++++++++++++++*/
915    
916     void FixForwardRoute(Results *results,index_t finish)
917     {
918     Result *result2=FindResult(results,finish);
919     Result *result1;
920    
921     /* Create the forward links for the optimum path */
922    
923     do
924     {
925     if(result2->prev!=NO_NODE)
926     {
927     index_t node1=result2->prev;
928    
929     result1=FindResult(results,node1);
930    
931     result1->next=result2->node;
932    
933     result2=result1;
934     }
935     else
936     result2=NULL;
937     }
938     while(result2);
939    
940     results->finish=finish;
941     }

Properties

Name Value
cvs:description Route optimiser.