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/output.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 306 - (show annotations) (download) (as text)
Mon Nov 23 18:42:40 2009 UTC (15 years, 4 months ago) by amb
File MIME type: text/x-csrc
File size: 15634 byte(s)
Add in "steps" as a new highway type.

1 /***************************************
2 $Header: /home/amb/CVS/routino/src/output.c,v 1.17 2009-11-23 18:42:40 amb Exp $
3
4 Routing output generator.
5
6 Part of the Routino routing software.
7 ******************/ /******************
8 This file Copyright 2008,2009 Andrew M. Bishop
9
10 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 ***************************************/
23
24
25 #include <stdlib.h>
26 #include <string.h>
27 #include <ctype.h>
28 #include <stdio.h>
29 #include <sys/types.h>
30 #include <sys/stat.h>
31 #include <unistd.h>
32
33 #include "types.h"
34 #include "functions.h"
35 #include "nodes.h"
36 #include "segments.h"
37 #include "ways.h"
38 #include "results.h"
39
40
41 /*+ The option to calculate the quickest route insted of the shortest. +*/
42 extern int option_quickest;
43
44 /*+ The files to write to. +*/
45 static FILE *gpxtrackfile=NULL,*gpxroutefile=NULL,*textfile=NULL,*textallfile=NULL;
46
47 /*+ The final latitude, longitude point. +*/
48 static double finish_latitude,finish_longitude;
49
50 /*+ Heuristics for determining if a junction is important. +*/
51 static char junction_other_way[Way_Count][Way_Count]=
52 { /* M, T, P, S, T, U, R, S, T, C, P, S = Way type of route not taken */
53 { 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0 }, /* Motorway */
54 { 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, /* Trunk */
55 { 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0 }, /* Primary */
56 { 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0 }, /* Secondary */
57 { 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0 }, /* Tertiary */
58 { 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0 }, /* Unclassified */
59 { 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0 }, /* Residential */
60 { 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0 }, /* Service */
61 { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* Track */
62 { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* Cycleway */
63 { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* Path */
64 { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 }, /* Steps */
65 };
66
67
68 /*++++++++++++++++++++++++++++++++++++++
69 Open the files and print the head.
70
71 const char *copyright The name of a file that might exist and contain copyright information.
72 ++++++++++++++++++++++++++++++++++++++*/
73
74 void PrintRouteHead(const char *copyright)
75 {
76 char *source=NULL,*license=NULL;
77
78 if(copyright)
79 {
80 struct stat buf;
81
82 if(!stat(copyright,&buf))
83 {
84 FILE *file=fopen(copyright,"r");
85 char *string=(char*)malloc(buf.st_size+1);
86 char *p;
87
88 fread(string,buf.st_size,1,file);
89 string[buf.st_size]=0;
90
91 p=string;
92 while(*p)
93 {
94 if(!strncmp(p,"Source:",7))
95 {
96 p+=7;
97 while(*p==' ' || *p=='t')
98 p++;
99 source=p;
100 while(*p && *p!='\r' && *p!='\n')
101 p++;
102 while(*p=='\r' || *p=='\n')
103 *p++=0;
104 }
105 else if(!strncmp(p,"License:",8) || !strncmp(p,"Licence:",8))
106 {
107 p+=8;
108 while(*p==' ' || *p=='t')
109 p++;
110 license=p;
111 while(*p && *p!='\r' && *p!='\n')
112 p++;
113 while(*p=='\r' || *p=='\n')
114 *p++=0;
115 }
116 else
117 {
118 while(*p && *p!='\r' && *p!='\n')
119 p++;
120 while(*p=='\r' || *p=='\n')
121 *p++=0;
122 }
123 }
124
125 fclose(file);
126 }
127 }
128
129 /* Open the files */
130
131 if(option_quickest==0)
132 {
133 /* Print the result for the shortest route */
134
135 gpxtrackfile=fopen("shortest-track.gpx","w");
136 gpxroutefile=fopen("shortest-route.gpx","w");
137 textfile =fopen("shortest.txt","w");
138 textallfile =fopen("shortest-all.txt","w");
139
140 if(!gpxtrackfile)
141 fprintf(stderr,"Warning: Cannot open file 'shortest-track.gpx' to write.\n");
142 if(!gpxroutefile)
143 fprintf(stderr,"Warning: Cannot open file 'shortest-route.gpx' to write.\n");
144 if(!textfile)
145 fprintf(stderr,"Warning: Cannot open file 'shortest.txt' to write.\n");
146 if(!textallfile)
147 fprintf(stderr,"Warning: Cannot open file 'shortest-all.txt' to write.\n");
148 }
149 else
150 {
151 /* Print the result for the quickest route */
152
153 gpxtrackfile=fopen("quickest-track.gpx","w");
154 gpxroutefile=fopen("quickest-route.gpx","w");
155 textfile =fopen("quickest.txt","w");
156 textallfile =fopen("quickest-all.txt","w");
157
158 if(!gpxtrackfile)
159 fprintf(stderr,"Warning: Cannot open file 'quickest-track.gpx' to write.\n");
160 if(!gpxroutefile)
161 fprintf(stderr,"Warning: Cannot open file 'quickest-route.gpx' to write.\n");
162 if(!textfile)
163 fprintf(stderr,"Warning: Cannot open file 'quickest.txt' to write.\n");
164 if(!textallfile)
165 fprintf(stderr,"Warning: Cannot open file 'quickest-all.txt' to write.\n");
166 }
167
168 /* Print the head of the files */
169
170 if(gpxtrackfile)
171 {
172 fprintf(gpxtrackfile,"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
173 fprintf(gpxtrackfile,"<gpx version=\"1.1\" creator=\"Routino\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns=\"http://www.topografix.com/GPX/1/1\" xsi:schemaLocation=\"http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd\">\n");
174
175 fprintf(gpxtrackfile,"<metadata>\n");
176 fprintf(gpxtrackfile,"<desc><![CDATA[%s route between 'start' and 'finish' waypoints]]></desc>\n",option_quickest?"Quickest":"Shortest");
177 if(source)
178 fprintf(gpxtrackfile,"<copyright author=\"%s\">\n",source);
179 if(license)
180 fprintf(gpxtrackfile,"<license>%s</license>\n",license);
181 if(source)
182 fprintf(gpxtrackfile,"</copyright>\n");
183 fprintf(gpxtrackfile,"</metadata>\n");
184
185 fprintf(gpxtrackfile,"<trk>\n");
186 }
187
188 if(gpxroutefile)
189 {
190 fprintf(gpxroutefile,"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
191 fprintf(gpxroutefile,"<gpx version=\"1.1\" creator=\"Routino\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns=\"http://www.topografix.com/GPX/1/1\" xsi:schemaLocation=\"http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd\">\n");
192
193 fprintf(gpxroutefile,"<metadata>\n");
194 fprintf(gpxroutefile,"<desc><![CDATA[%s route between 'start' and 'finish' waypoints]]></desc>\n",option_quickest?"Quickest":"Shortest");
195 if(source)
196 fprintf(gpxroutefile,"<copyright author=\"%s\">\n",source);
197 if(license)
198 fprintf(gpxroutefile,"<license>%s</license>\n",license);
199 if(source)
200 fprintf(gpxroutefile,"</copyright>\n");
201 fprintf(gpxroutefile,"</metadata>\n");
202
203 fprintf(gpxroutefile,"<rte>\n");
204 fprintf(gpxroutefile,"<name>%s route</name>\n",option_quickest?"Quickest":"Shortest");
205 }
206
207 if(textfile)
208 {
209 if(source)
210 fprintf(textfile,"# Source: %s\n",source);
211 if(license)
212 fprintf(textfile,"# License: %s\n",license);
213 if(source || license)
214 fprintf(textfile,"#\n");
215 fprintf(textfile,"#Latitude\tLongitude\tSegment \tSegment \tTotal \tTotal \tHighway\n");
216 fprintf(textfile,"# \t \tDistance\tDuration\tDistance\tDurat'n\t \n");
217 /* "%10.6f\t%11.6f\t%6.3f km\t%4.1f min\t%5.1f km\t%3.0f min\t%s\n" */
218 }
219
220 if(textallfile)
221 {
222 if(source)
223 fprintf(textallfile,"# Source: %s\n",source);
224 if(license)
225 fprintf(textallfile,"# License: %s\n",license);
226 if(source || license)
227 fprintf(textallfile,"#\n");
228 fprintf(textallfile,"#Latitude\tLongitude\t Node\tSegment\tSegment\tTotal\tTotal \tSpeed\tHighway\n");
229 fprintf(textallfile,"# \t \t \tDist \tDurat'n\tDist \tDurat'n\t \t \n");
230 /* "%10.6f\t%11.6f\t%8d%c\t%5.3f\t%5.2f\t%5.2f\t%5.1f\t%3d\t%s\n" */
231 }
232 }
233
234
235 /*++++++++++++++++++++++++++++++++++++++
236 Print the optimum route between two nodes.
237
238 Results *results The set of results to print.
239
240 Nodes *nodes The list of nodes.
241
242 Segments *segments The set of segments to use.
243
244 Ways *ways The list of ways.
245
246 Profile *profile The profile containing the transport type, speeds and allowed highways.
247 ++++++++++++++++++++++++++++++++++++++*/
248
249 void PrintRoute(Results *results,Nodes *nodes,Segments *segments,Ways *ways,Profile *profile)
250 {
251 static distance_t cum_distance=0;
252 static duration_t cum_duration=0;
253 static int segment_count=0;
254 static int route_count=0;
255 double finish_lat,finish_lon;
256 double start_lat,start_lon;
257 distance_t junc_distance=0;
258 duration_t junc_duration=0;
259 Result *result;
260
261 fprintf(gpxtrackfile,"<trkseg>\n");
262
263 if(IsFakeNode(results->start))
264 GetFakeLatLong(results->start,&start_lat,&start_lon);
265 else
266 GetLatLong(nodes,results->start,&start_lat,&start_lon);
267
268 if(IsFakeNode(results->finish))
269 GetFakeLatLong(results->finish,&finish_lat,&finish_lon);
270 else
271 GetLatLong(nodes,results->finish,&finish_lat,&finish_lon);
272
273 result=FindResult(results,results->start);
274
275 do
276 {
277 double latitude,longitude;
278
279 if(result->node==results->start)
280 {latitude=start_lat; longitude=start_lon;}
281 else if(result->node==results->finish)
282 {latitude=finish_lat; longitude=finish_lon;}
283 else
284 GetLatLong(nodes,result->node,&latitude,&longitude);
285
286 if(gpxtrackfile)
287 fprintf(gpxtrackfile,"<trkpt lat=\"%.6f\" lon=\"%.6f\"/>\n",
288 radians_to_degrees(latitude),radians_to_degrees(longitude));
289
290 if(result->node!=results->start)
291 {
292 distance_t seg_distance=0;
293 duration_t seg_duration=0;
294 Segment *segment;
295 Way *resultway;
296 int important=0;
297
298 /* Get the properties of this segment */
299
300 resultway=LookupWay(ways,result->segment->way);
301
302 seg_distance+=DISTANCE(result->segment->distance);
303 seg_duration+=Duration(result->segment,resultway,profile);
304 junc_distance+=seg_distance;
305 junc_duration+=seg_duration;
306 cum_distance+=seg_distance;
307 cum_duration+=seg_duration;
308
309 /* Decide if this is an important junction */
310
311 if(result->node==results->finish)
312 important=1;
313 else
314 {
315 segment=FirstSegment(segments,nodes,result->node);
316
317 do
318 {
319 index_t othernode=OtherNode(segment,result->node);
320
321 if(othernode!=result->prev)
322 if(IsNormalSegment(segment) && (!profile->oneway || !IsOnewayTo(segment,result->node)))
323 {
324 Way *way=LookupWay(ways,segment->way);
325
326 if(othernode==result->next) /* the next segment that we follow */
327 {
328 if(HIGHWAY(way->type)!=HIGHWAY(resultway->type))
329 {important=1; break;}
330 }
331 else /* a segment that we don't follow */
332 if(junction_other_way[HIGHWAY(resultway->type)-1][HIGHWAY(way->type)-1])
333 {important=1; break;}
334 }
335
336 segment=NextSegment(segments,segment,result->node);
337 }
338 while(segment);
339 }
340
341 /* Print out the junctions */
342
343 if(important)
344 {
345 if(result->node!=results->finish)
346 {
347 /* Don't print the intermediate finish points (because they are also intermediate start points) */
348
349 if(gpxroutefile)
350 fprintf(gpxroutefile,"<rtept lat=\"%.6f\" lon=\"%.6f\"><name>TRIP%03d</name></rtept>\n",
351 radians_to_degrees(latitude),radians_to_degrees(longitude),
352 ++route_count);
353 }
354 else
355 {
356 finish_latitude=latitude;
357 finish_longitude=longitude;
358 }
359
360 /* Do print the intermediate finish points (because they have correct junction distances) */
361
362 if(textfile)
363 fprintf(textfile,"%10.6f\t%11.6f\t%6.3f km\t%4.1f min\t%5.1f km\t%3.0f min\t%s\n",
364 radians_to_degrees(latitude),radians_to_degrees(longitude),
365 distance_to_km(junc_distance),duration_to_minutes(junc_duration),
366 distance_to_km(cum_distance),duration_to_minutes(cum_duration),
367 WayName(ways,resultway));
368
369 junc_distance=0;
370 junc_duration=0;
371 }
372
373 /* Print out all of the results */
374
375 if(textallfile)
376 fprintf(textallfile,"%10.6f\t%11.6f\t%8d%c\t%5.3f\t%5.2f\t%5.2f\t%5.1f\t%3d\t%s\n",
377 radians_to_degrees(latitude),radians_to_degrees(longitude),
378 IsFakeNode(result->node)?-(result->node&(~NODE_SUPER)):result->node,
379 (!IsFakeNode(result->node) && IsSuperNode(nodes,result->node))?'*':' ',
380 distance_to_km(seg_distance),duration_to_minutes(seg_duration),
381 distance_to_km(cum_distance),duration_to_minutes(cum_duration),
382 profile->speed[HIGHWAY(resultway->type)],WayName(ways,resultway));
383 }
384 else if(!cum_distance)
385 {
386 /* Print out the very first start point */
387
388 if(gpxroutefile)
389 fprintf(gpxroutefile,"<rtept lat=\"%.6f\" lon=\"%.6f\"><name>START</name></rtept>\n",
390 radians_to_degrees(latitude),radians_to_degrees(longitude));
391
392 if(textfile)
393 fprintf(textfile,"%10.6f\t%11.6f\t%6.3f km\t%4.1f min\t%5.1f km\t%3.0f min\t\n",
394 radians_to_degrees(latitude),radians_to_degrees(longitude),
395 0.0,0.0,0.0,0.0);
396
397 if(textallfile)
398 fprintf(textallfile,"%10.6f\t%11.6f\t%8d%c\t%5.3f\t%5.2f\t%5.2f\t%5.1f\n",
399 radians_to_degrees(latitude),radians_to_degrees(longitude),
400 IsFakeNode(result->node)?-(result->node&(~NODE_SUPER)):result->node,
401 (!IsFakeNode(result->node) && IsSuperNode(nodes,result->node))?'*':' ',
402 0.0,0.0,0.0,0.0);
403 }
404 else
405 {
406 /* Print out the intermediate start points */
407
408 if(gpxroutefile)
409 fprintf(gpxroutefile,"<rtept lat=\"%.6f\" lon=\"%.6f\"><name>INTER%d</name></rtept>\n",
410 radians_to_degrees(latitude),radians_to_degrees(longitude),
411 ++segment_count);
412 }
413
414 result=FindResult(results,result->next);
415 }
416 while(result);
417
418 if(gpxtrackfile)
419 fprintf(gpxtrackfile,"</trkseg>\n");
420 }
421
422
423 /*++++++++++++++++++++++++++++++++++++++
424 Print the tail and close the files.
425 ++++++++++++++++++++++++++++++++++++++*/
426
427 void PrintRouteTail(void)
428 {
429 /* Print the very final point in the route */
430
431 if(gpxroutefile)
432 fprintf(gpxroutefile,"<rtept lat=\"%.6f\" lon=\"%.6f\"><name>FINISH</name></rtept>\n",
433 radians_to_degrees(finish_latitude),radians_to_degrees(finish_longitude));
434
435 /* Print the tail of the files */
436
437 if(gpxtrackfile)
438 {
439 fprintf(gpxtrackfile,"</trk>\n");
440 fprintf(gpxtrackfile,"</gpx>\n");
441 }
442
443 if(gpxroutefile)
444 {
445 fprintf(gpxroutefile,"</rte>\n");
446 fprintf(gpxroutefile,"</gpx>\n");
447 }
448
449 /* Close the files */
450
451 if(gpxtrackfile)
452 fclose(gpxtrackfile);
453 if(gpxroutefile)
454 fclose(gpxroutefile);
455 if(textfile)
456 fclose(textfile);
457 if(textallfile)
458 fclose(textallfile);
459 }

Properties

Name Value
cvs:description New file to contain the function to print the output.