Check out the latest version of Routino: svn co http://routino.org/svn/trunk routino
Contents of /trunk/src/optimiser.c
Parent Directory
|
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)
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. |