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

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1234 - (show annotations) (download) (as text)
Sat Dec 29 11:11:11 2012 UTC (12 years, 3 months ago) by amb
File MIME type: text/x-csrc
File size: 35983 byte(s)
Re-factor parsing code to remove duplicated parts from three parsers
(osmo5mparse.c, osmpbfparse.c and osmxmlparse.c) into a common place
(osmparser.c), also removes lots of global variables.
Change the node, way and relation count to uint64_t instead of index_t to avoid
wrap-around (although it would have been a cosmetic problem only), also removes
dependency on types.h.
Make the node, way and relation counters be 'int64_t' instead of 'long long' in
the XML parsers for consistency with the non-XML parsers.

1 /***************************************
2 OSM file parser (either JOSM or planet)
3
4 Part of the Routino routing software.
5 ******************/ /******************
6 This file Copyright 2008-2012 Andrew M. Bishop
7
8 This program is free software: you can redistribute it and/or modify
9 it under the terms of the GNU Affero General Public License as published by
10 the Free Software Foundation, either version 3 of the License, or
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU Affero General Public License for more details.
17
18 You should have received a copy of the GNU Affero General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>.
20 ***************************************/
21
22
23 #include <stdlib.h>
24 #include <string.h>
25 #include <stdint.h>
26 #include <ctype.h>
27
28 #include "types.h"
29 #include "typesx.h"
30
31 #include "nodesx.h"
32 #include "segmentsx.h"
33 #include "waysx.h"
34 #include "relationsx.h"
35
36 #include "osmparser.h"
37 #include "tagging.h"
38 #include "logging.h"
39
40
41 /* Macros */
42
43 /*+ Checks if a value in the XML is one of the allowed values for true. +*/
44 #define ISTRUE(xx) (!strcmp(xx,"true") || !strcmp(xx,"yes") || !strcmp(xx,"1"))
45
46 /*+ Checks if a value in the XML is one of the allowed values for false. +*/
47 #define ISFALSE(xx) (!strcmp(xx,"false") || !strcmp(xx,"no") || !strcmp(xx,"0"))
48
49 /* Local variables */
50
51 static NodesX *nodes;
52 static SegmentsX *segments;
53 static WaysX *ways;
54 static RelationsX *relations;
55
56 static node_t *way_nodes=NULL;
57 static int way_nnodes=0;
58
59 static node_t *relation_nodes=NULL;
60 static int relation_nnodes=0;
61 static way_t *relation_ways=NULL;
62 static int relation_nways=0;
63 static relation_t *relation_relations=NULL;
64 static int relation_nrelations=0;
65 static way_t relation_from=NO_WAY_ID;
66 static way_t relation_to=NO_WAY_ID;
67 static node_t relation_via=NO_NODE_ID;
68
69 /* Local functions */
70
71 static double parse_speed(way_t id,const char *k,const char *v);
72 static double parse_weight(way_t id,const char *k,const char *v);
73 static double parse_length(way_t id,const char *k,const char *v);
74
75
76 /*++++++++++++++++++++++++++++++++++++++
77 Parse an OSM XML file (from JOSM or planet download).
78
79 int ParseOSMFile Returns 0 if OK or something else in case of an error.
80
81 int fd The file descriptor of the file to read from.
82
83 NodesX *OSMNodes The data structure of nodes to fill in.
84
85 SegmentsX *OSMSegments The data structure of segments to fill in.
86
87 WaysX *OSMWays The data structure of ways to fill in.
88
89 RelationsX *OSMRelations The data structure of relations to fill in.
90 ++++++++++++++++++++++++++++++++++++++*/
91
92 int ParseOSMFile(int fd,NodesX *OSMNodes,SegmentsX *OSMSegments,WaysX *OSMWays,RelationsX *OSMRelations)
93 {
94 int retval;
95
96 /* Copy the function parameters and initialise the variables */
97
98 nodes=OSMNodes;
99 segments=OSMSegments;
100 ways=OSMWays;
101 relations=OSMRelations;
102
103 way_nodes=(node_t*)malloc(256*sizeof(node_t));
104
105 relation_nodes =(node_t *)malloc(256*sizeof(node_t));
106 relation_ways =(way_t *)malloc(256*sizeof(way_t));
107 relation_relations=(relation_t*)malloc(256*sizeof(relation_t));
108
109 /* Parse the file */
110
111 retval=ParseXML(fd,xml_osm_toplevel_tags,XMLPARSE_UNKNOWN_ATTR_IGNORE);
112
113 /* Free the variables */
114
115 free(way_nodes);
116
117 free(relation_nodes);
118 free(relation_ways);
119 free(relation_relations);
120
121 return(retval);
122 }
123
124
125 /*++++++++++++++++++++++++++++++++++++++
126 Parse an OSC XML file (from planet download).
127
128 int ParseOSCFile Returns 0 if OK or something else in case of an error.
129
130 int fd The file descriptor of the file to read from.
131
132 NodesX *OSMNodes The data structure of nodes to fill in.
133
134 SegmentsX *OSMSegments The data structure of segments to fill in.
135
136 WaysX *OSMWays The data structure of ways to fill in.
137
138 RelationsX *OSMRelations The data structure of relations to fill in.
139 ++++++++++++++++++++++++++++++++++++++*/
140
141 int ParseOSCFile(int fd,NodesX *OSMNodes,SegmentsX *OSMSegments,WaysX *OSMWays,RelationsX *OSMRelations)
142 {
143 int retval;
144
145 /* Copy the function parameters and initialise the variables */
146
147 nodes=OSMNodes;
148 segments=OSMSegments;
149 ways=OSMWays;
150 relations=OSMRelations;
151
152 way_nodes=(node_t*)malloc(256*sizeof(node_t));
153
154 relation_nodes =(node_t *)malloc(256*sizeof(node_t));
155 relation_ways =(way_t *)malloc(256*sizeof(way_t));
156 relation_relations=(relation_t*)malloc(256*sizeof(relation_t));
157
158 /* Parse the file */
159
160 retval=ParseXML(fd,xml_osc_toplevel_tags,XMLPARSE_UNKNOWN_ATTR_IGNORE);
161
162 /* Free the variables */
163
164 free(way_nodes);
165
166 free(relation_nodes);
167 free(relation_ways);
168 free(relation_relations);
169
170 return(retval);
171 }
172
173
174 /*++++++++++++++++++++++++++++++++++++++
175 Parse a PBF format OSM file (from planet download).
176
177 int ParsePBFFile Returns 0 if OK or something else in case of an error.
178
179 int fd The file descriptor of the file to read from.
180
181 NodesX *OSMNodes The data structure of nodes to fill in.
182
183 SegmentsX *OSMSegments The data structure of segments to fill in.
184
185 WaysX *OSMWays The data structure of ways to fill in.
186
187 RelationsX *OSMRelations The data structure of relations to fill in.
188 ++++++++++++++++++++++++++++++++++++++*/
189
190 int ParsePBFFile(int fd,NodesX *OSMNodes,SegmentsX *OSMSegments,WaysX *OSMWays,RelationsX *OSMRelations)
191 {
192 int retval;
193
194 /* Copy the function parameters and initialise the variables */
195
196 nodes=OSMNodes;
197 segments=OSMSegments;
198 ways=OSMWays;
199 relations=OSMRelations;
200
201 way_nodes=(node_t*)malloc(256*sizeof(node_t));
202
203 relation_nodes =(node_t *)malloc(256*sizeof(node_t));
204 relation_ways =(way_t *)malloc(256*sizeof(way_t));
205 relation_relations=(relation_t*)malloc(256*sizeof(relation_t));
206
207 /* Parse the file */
208
209 retval=ParsePBF(fd);
210
211 /* Free the variables */
212
213 free(way_nodes);
214
215 free(relation_nodes);
216 free(relation_ways);
217 free(relation_relations);
218
219 return(retval);
220 }
221
222
223 /*++++++++++++++++++++++++++++++++++++++
224 Parse an O5M format OSM file (from planet download).
225
226 int ParseO5MFile Returns 0 if OK or something else in case of an error.
227
228 int fd The file descriptor of the file to read from.
229
230 NodesX *OSMNodes The data structure of nodes to fill in.
231
232 SegmentsX *OSMSegments The data structure of segments to fill in.
233
234 WaysX *OSMWays The data structure of ways to fill in.
235
236 RelationsX *OSMRelations The data structure of relations to fill in.
237 ++++++++++++++++++++++++++++++++++++++*/
238
239 int ParseO5MFile(int fd,NodesX *OSMNodes,SegmentsX *OSMSegments,WaysX *OSMWays,RelationsX *OSMRelations)
240 {
241 int retval;
242
243 /* Copy the function parameters and initialise the variables */
244
245 nodes=OSMNodes;
246 segments=OSMSegments;
247 ways=OSMWays;
248 relations=OSMRelations;
249
250 way_nodes=(node_t*)malloc(256*sizeof(node_t));
251
252 relation_nodes =(node_t *)malloc(256*sizeof(node_t));
253 relation_ways =(way_t *)malloc(256*sizeof(way_t));
254 relation_relations=(relation_t*)malloc(256*sizeof(relation_t));
255
256 /* Parse the file */
257
258 retval=ParseO5M(fd,0);
259
260 /* Free the variables */
261
262 free(way_nodes);
263
264 free(relation_nodes);
265 free(relation_ways);
266 free(relation_relations);
267
268 return(retval);
269 }
270
271
272 /*++++++++++++++++++++++++++++++++++++++
273 Parse an O5C format OSM file (from planet download).
274
275 int ParseO5CFile Returns 0 if OK or something else in case of an error.
276
277 int fd The file descriptor of the file to read from.
278
279 NodesX *OSMNodes The data structure of nodes to fill in.
280
281 SegmentsX *OSMSegments The data structure of segments to fill in.
282
283 WaysX *OSMWays The data structure of ways to fill in.
284
285 RelationsX *OSMRelations The data structure of relations to fill in.
286 ++++++++++++++++++++++++++++++++++++++*/
287
288 int ParseO5CFile(int fd,NodesX *OSMNodes,SegmentsX *OSMSegments,WaysX *OSMWays,RelationsX *OSMRelations)
289 {
290 int retval;
291
292 /* Copy the function parameters and initialise the variables */
293
294 nodes=OSMNodes;
295 segments=OSMSegments;
296 ways=OSMWays;
297 relations=OSMRelations;
298
299 way_nodes=(node_t*)malloc(256*sizeof(node_t));
300
301 relation_nodes =(node_t *)malloc(256*sizeof(node_t));
302 relation_ways =(way_t *)malloc(256*sizeof(way_t));
303 relation_relations=(relation_t*)malloc(256*sizeof(relation_t));
304
305 /* Parse the file */
306
307 retval=ParseO5M(fd,1);
308
309 /* Free the variables */
310
311 free(way_nodes);
312
313 free(relation_nodes);
314 free(relation_ways);
315 free(relation_relations);
316
317 return(retval);
318 }
319
320
321 /*++++++++++++++++++++++++++++++++++++++
322 Add node references to a way.
323
324 int64_t node_id The node ID to add or zero to clear the list.
325 ++++++++++++++++++++++++++++++++++++++*/
326
327 void AddWayRefs(int64_t node_id)
328 {
329 if(node_id==0)
330 way_nnodes=0;
331 else
332 {
333 node_t id;
334
335 if(way_nnodes && (way_nnodes%256)==0)
336 way_nodes=(node_t*)realloc((void*)way_nodes,(way_nnodes+256)*sizeof(node_t));
337
338 id=(node_t)node_id;
339 logassert((int64_t)id==node_id,"Node ID too large (change node_t to 64-bits?)"); /* check node id can be stored in node_t data type. */
340
341 way_nodes[way_nnodes++]=id;
342 }
343 }
344
345
346 /*++++++++++++++++++++++++++++++++++++++
347 Add node, way or relation references to a relation.
348
349 int64_t node_id The node ID to add or zero if it is not a node.
350
351 int64_t way_id The way ID to add or zero if it is not a way.
352
353 int64_t relation_id The relation ID to add or zero if it is not a relation.
354
355 const char *role The role played by this referenced item or NULL.
356
357 If all of node_id, way_id and relation_id are zero then the list is cleared.
358 ++++++++++++++++++++++++++++++++++++++*/
359
360 void AddRelationRefs(int64_t node_id,int64_t way_id,int64_t relation_id,const char *role)
361 {
362 if(node_id==0 && way_id==0 && relation_id==0)
363 {
364 relation_nnodes=0;
365 relation_nways=0;
366 relation_nrelations=0;
367 }
368 else if(node_id!=0)
369 {
370 node_t id;
371
372 id=(node_t)node_id;
373 logassert((int64_t)id==node_id,"Node ID too large (change node_t to 64-bits?)"); /* check node id can be stored in node_t data type. */
374
375 if(relation_nnodes && (relation_nnodes%256)==0)
376 relation_nodes=(node_t*)realloc((void*)relation_nodes,(relation_nnodes+256)*sizeof(node_t));
377
378 relation_nodes[relation_nnodes++]=id;
379
380 if(role)
381 {
382 if(!strcmp(role,"via"))
383 relation_via=id;
384 }
385 }
386 else if(way_id!=0)
387 {
388 way_t id;
389
390 id=(way_t)way_id;
391 logassert((int64_t)id==way_id,"Way ID too large (change way_t to 64-bits?)"); /* check way id can be stored in way_t data type. */
392
393 if(relation_nways && (relation_nways%256)==0)
394 relation_ways=(way_t*)realloc((void*)relation_ways,(relation_nways+256)*sizeof(way_t));
395
396 relation_ways[relation_nways++]=id;
397
398 if(role)
399 {
400 if(!strcmp(role,"from"))
401 relation_from=id;
402 else if(!strcmp(role,"to"))
403 relation_to=id;
404 }
405 }
406 else /* if(relation_id!=0) */
407 {
408 relation_t id;
409
410 id=(relation_t)relation_id;
411 logassert((int64_t)id==relation_id,"Relation ID too large (change relation_t to 64-bits?)"); /* check relation id can be stored in relation_t data type. */
412
413 if(relation_nrelations && (relation_nrelations%256)==0)
414 relation_relations=(relation_t*)realloc((void*)relation_relations,(relation_nrelations+256)*sizeof(relation_t));
415
416 relation_relations[relation_nrelations++]=relation_id;
417 }
418 }
419
420
421 /*++++++++++++++++++++++++++++++++++++++
422 Process the tags associated with a node.
423
424 TagList *tags The list of node tags.
425
426 int64_t node_id The id of the node.
427
428 double latitude The latitude of the node.
429
430 double longitude The longitude of the node.
431
432 int mode The mode of operation to take (create, modify, delete).
433 ++++++++++++++++++++++++++++++++++++++*/
434
435 void ProcessNodeTags(TagList *tags,int64_t node_id,double latitude,double longitude,int mode)
436 {
437 transports_t allow=Transports_ALL;
438 nodeflags_t flags=0;
439 node_t id;
440 int i;
441
442 /* Convert id */
443
444 id=(node_t)node_id;
445 logassert((int64_t)id==node_id,"Node ID too large (change node_t to 64-bits?)"); /* check node id can be stored in node_t data type. */
446
447 /* Delete */
448
449 if(mode==MODE_DELETE)
450 {
451 AppendNodeList(nodes,id,degrees_to_radians(latitude),degrees_to_radians(longitude),allow,NODE_DELETED);
452
453 return;
454 }
455
456 /* Parse the tags */
457
458 for(i=0;i<tags->ntags;i++)
459 {
460 int recognised=0;
461 char *k=tags->k[i];
462 char *v=tags->v[i];
463
464 switch(*k)
465 {
466 case 'b':
467 if(!strcmp(k,"bicycle"))
468 {
469 if(ISFALSE(v))
470 allow&=~Transports_Bicycle;
471 else if(!ISTRUE(v))
472 logerror("Node %"Pnode_t" has an unrecognised tag value 'bicycle' = '%s' (after tagging rules); using 'yes'.\n",id,v);
473 recognised=1; break;
474 }
475
476 break;
477
478 case 'f':
479 if(!strcmp(k,"foot"))
480 {
481 if(ISFALSE(v))
482 allow&=~Transports_Foot;
483 else if(!ISTRUE(v))
484 logerror("Node %"Pnode_t" has an unrecognised tag value 'foot' = '%s' (after tagging rules); using 'yes'.\n",id,v);
485 recognised=1; break;
486 }
487
488 break;
489
490 case 'g':
491 if(!strcmp(k,"goods"))
492 {
493 if(ISFALSE(v))
494 allow&=~Transports_Goods;
495 else if(!ISTRUE(v))
496 logerror("Node %"Pnode_t" has an unrecognised tag value 'goods' = '%s' (after tagging rules); using 'yes'.\n",id,v);
497 recognised=1; break;
498 }
499
500 break;
501
502 case 'h':
503 if(!strcmp(k,"highway"))
504 if(!strcmp(v,"mini_roundabout"))
505 {
506 flags|=NODE_MINIRNDBT;
507 recognised=1; break;
508 }
509
510 if(!strcmp(k,"horse"))
511 {
512 if(ISFALSE(v))
513 allow&=~Transports_Horse;
514 else if(!ISTRUE(v))
515 logerror("Node %"Pnode_t" has an unrecognised tag value 'horse' = '%s' (after tagging rules); using 'yes'.\n",id,v);
516 recognised=1; break;
517 }
518
519 if(!strcmp(k,"hgv"))
520 {
521 if(ISFALSE(v))
522 allow&=~Transports_HGV;
523 else if(!ISTRUE(v))
524 logerror("Node %"Pnode_t" has an unrecognised tag value 'hgv' = '%s' (after tagging rules); using 'yes'.\n",id,v);
525 recognised=1; break;
526 }
527
528 break;
529
530 case 'm':
531 if(!strcmp(k,"moped"))
532 {
533 if(ISFALSE(v))
534 allow&=~Transports_Moped;
535 else if(!ISTRUE(v))
536 logerror("Node %"Pnode_t" has an unrecognised tag value 'moped' = '%s' (after tagging rules); using 'yes'.\n",id,v);
537 recognised=1; break;
538 }
539
540 if(!strcmp(k,"motorbike"))
541 {
542 if(ISFALSE(v))
543 allow&=~Transports_Motorbike;
544 else if(!ISTRUE(v))
545 logerror("Node %"Pnode_t" has an unrecognised tag value 'motorbike' = '%s' (after tagging rules); using 'yes'.\n",id,v);
546 recognised=1; break;
547 }
548
549 if(!strcmp(k,"motorcar"))
550 {
551 if(ISFALSE(v))
552 allow&=~Transports_Motorcar;
553 else if(!ISTRUE(v))
554 logerror("Node %"Pnode_t" has an unrecognised tag value 'motorcar' = '%s' (after tagging rules); using 'yes'.\n",id,v);
555 recognised=1; break;
556 }
557
558 break;
559
560 case 'p':
561 if(!strcmp(k,"psv"))
562 {
563 if(ISFALSE(v))
564 allow&=~Transports_PSV;
565 else if(!ISTRUE(v))
566 logerror("Node %"Pnode_t" has an unrecognised tag value 'psv' = '%s' (after tagging rules); using 'yes'.\n",id,v);
567 recognised=1; break;
568 }
569
570 break;
571
572 case 'w':
573 if(!strcmp(k,"wheelchair"))
574 {
575 if(ISFALSE(v))
576 allow&=~Transports_Wheelchair;
577 else if(!ISTRUE(v))
578 logerror("Node %"Pnode_t" has an unrecognised tag value 'wheelchair' = '%s' (after tagging rules); using 'yes'.\n",id,v);
579 recognised=1; break;
580 }
581
582 break;
583
584 default:
585 break;
586 }
587
588 if(!recognised)
589 logerror("Node %"Pnode_t" has an unrecognised tag '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
590 }
591
592 /* Create the node */
593
594 AppendNodeList(nodes,id,degrees_to_radians(latitude),degrees_to_radians(longitude),allow,flags);
595 }
596
597
598 /*++++++++++++++++++++++++++++++++++++++
599 Process the tags associated with a way.
600
601 TagList *tags The list of way tags.
602
603 int64_t way_id The id of the way.
604
605 int mode The mode of operation to take (create, modify, delete).
606 ++++++++++++++++++++++++++++++++++++++*/
607
608 void ProcessWayTags(TagList *tags,int64_t way_id,int mode)
609 {
610 Way way={0};
611 distance_t oneway=0,area=0;
612 int roundabout=0;
613 char *name=NULL,*ref=NULL,*refname=NULL;
614 way_t id;
615 int i,j;
616
617 /* Convert id */
618
619 id=(way_t)way_id;
620 logassert((int64_t)id==way_id,"Way ID too large (change way_t to 64-bits?)"); /* check way id can be stored in way_t data type. */
621
622 /* Delete */
623
624 if(mode==MODE_DELETE || mode==MODE_MODIFY)
625 {
626 way.type=WAY_DELETED;
627
628 AppendWayList(ways,id,&way,"");
629
630 way.type=Highway_None;
631
632 AppendSegmentList(segments,id,NO_NODE_ID,NO_NODE_ID,0);
633 }
634
635 if(mode==MODE_DELETE)
636 return;
637
638 /* Sanity check */
639
640 if(way_nnodes==0)
641 {
642 logerror("Way %"Pway_t" has no nodes.\n",id);
643 return;
644 }
645
646 if(way_nnodes==1)
647 {
648 logerror("Way %"Pway_t" has only one node.\n",id);
649 return;
650 }
651
652 /* Parse the tags - just look for highway */
653
654 for(i=0;i<tags->ntags;i++)
655 {
656 char *k=tags->k[i];
657 char *v=tags->v[i];
658
659 if(!strcmp(k,"highway"))
660 {
661 way.type=HighwayType(v);
662
663 if(way.type==Highway_None)
664 logerror("Way %"Pway_t" has an unrecognised highway type '%s' (after tagging rules); ignoring it.\n",id,v);
665
666 break;
667 }
668 }
669
670 /* Don't continue if this is not a highway (bypass error logging) */
671
672 if(way.type==Highway_None)
673 return;
674
675 /* Parse the tags - look for the others */
676
677 for(i=0;i<tags->ntags;i++)
678 {
679 int recognised=0;
680 char *k=tags->k[i];
681 char *v=tags->v[i];
682
683 switch(*k)
684 {
685 case 'a':
686 if(!strcmp(k,"area"))
687 {
688 if(ISTRUE(v))
689 area=SEGMENT_AREA;
690 else if(!ISFALSE(v))
691 logerror("Way %"Pway_t" has an unrecognised tag value 'area' = '%s' (after tagging rules); using 'no'.\n",id,v);
692 recognised=1; break;
693 }
694
695 break;
696
697 case 'b':
698 if(!strcmp(k,"bicycle"))
699 {
700 if(ISTRUE(v))
701 way.allow|=Transports_Bicycle;
702 else if(!ISFALSE(v))
703 logerror("Way %"Pway_t" has an unrecognised tag value 'bicycle' = '%s' (after tagging rules); using 'no'.\n",id,v);
704 recognised=1; break;
705 }
706
707 if(!strcmp(k,"bicycleroute"))
708 {
709 if(ISTRUE(v))
710 way.props|=Properties_BicycleRoute;
711 else if(!ISFALSE(v))
712 logerror("Way %"Pway_t" has an unrecognised tag value 'bicycleroute' = '%s' (after tagging rules); using 'no'.\n",id,v);
713 recognised=1; break;
714 }
715
716 if(!strcmp(k,"bridge"))
717 {
718 if(ISTRUE(v))
719 way.props|=Properties_Bridge;
720 else if(!ISFALSE(v))
721 logerror("Way %"Pway_t" has an unrecognised tag value 'bridge' = '%s' (after tagging rules); using 'no'.\n",id,v);
722 recognised=1; break;
723 }
724
725 break;
726
727 case 'f':
728 if(!strcmp(k,"foot"))
729 {
730 if(ISTRUE(v))
731 way.allow|=Transports_Foot;
732 else if(!ISFALSE(v))
733 logerror("Way %"Pway_t" has an unrecognised tag value 'foot' = '%s' (after tagging rules); using 'no'.\n",id,v);
734 recognised=1; break;
735 }
736
737 if(!strcmp(k,"footroute"))
738 {
739 if(ISTRUE(v))
740 way.props|=Properties_FootRoute;
741 else if(!ISFALSE(v))
742 logerror("Way %"Pway_t" has an unrecognised tag value 'footroute' = '%s' (after tagging rules); using 'no'.\n",id,v);
743 recognised=1; break;
744 }
745
746 break;
747
748 case 'g':
749 if(!strcmp(k,"goods"))
750 {
751 if(ISTRUE(v))
752 way.allow|=Transports_Goods;
753 else if(!ISFALSE(v))
754 logerror("Way %"Pway_t" has an unrecognised tag value 'goods' = '%s' (after tagging rules); using 'no'.\n",id,v);
755 recognised=1; break;
756 }
757
758 break;
759
760 case 'h':
761 if(!strcmp(k,"highway"))
762 {recognised=1; break;}
763
764 if(!strcmp(k,"horse"))
765 {
766 if(ISTRUE(v))
767 way.allow|=Transports_Horse;
768 else if(!ISFALSE(v))
769 logerror("Way %"Pway_t" has an unrecognised tag value 'horse' = '%s' (after tagging rules); using 'no'.\n",id,v);
770 recognised=1; break;
771 }
772
773 if(!strcmp(k,"hgv"))
774 {
775 if(ISTRUE(v))
776 way.allow|=Transports_HGV;
777 else if(!ISFALSE(v))
778 logerror("Way %"Pway_t" has an unrecognised tag value 'hgv' = '%s' (after tagging rules); using 'no'.\n",id,v);
779 recognised=1; break;
780 }
781
782 break;
783
784 case 'm':
785 if(!strncmp(k,"max",3))
786 {
787 if(!strcmp(k+3,"speed"))
788 {
789 way.speed=kph_to_speed(parse_speed(id,k,v));
790 recognised=1; break;
791 }
792
793 if(!strcmp(k+3,"weight"))
794 {
795 way.weight=tonnes_to_weight(parse_weight(id,k,v));
796 recognised=1; break;
797 }
798
799 if(!strcmp(k+3,"height"))
800 {
801 way.height=metres_to_height(parse_length(id,k,v));
802 recognised=1; break;
803 }
804
805 if(!strcmp(k+3,"width"))
806 {
807 way.width=metres_to_height(parse_length(id,k,v));
808 recognised=1; break;
809 }
810
811 if(!strcmp(k+3,"length"))
812 {
813 way.length=metres_to_height(parse_length(id,k,v));
814 recognised=1; break;
815 }
816 }
817
818 if(!strcmp(k,"moped"))
819 {
820 if(ISTRUE(v))
821 way.allow|=Transports_Moped;
822 else if(!ISFALSE(v))
823 logerror("Way %"Pway_t" has an unrecognised tag value 'moped' = '%s' (after tagging rules); using 'no'.\n",id,v);
824 recognised=1; break;
825 }
826
827 if(!strcmp(k,"motorbike"))
828 {
829 if(ISTRUE(v))
830 way.allow|=Transports_Motorbike;
831 else if(!ISFALSE(v))
832 logerror("Way %"Pway_t" has an unrecognised tag value 'motorbike' = '%s' (after tagging rules); using 'no'.\n",id,v);
833 recognised=1; break;
834 }
835
836 if(!strcmp(k,"motorcar"))
837 {
838 if(ISTRUE(v))
839 way.allow|=Transports_Motorcar;
840 else if(!ISFALSE(v))
841 logerror("Way %"Pway_t" has an unrecognised tag value 'motorcar' = '%s' (after tagging rules); using 'no'.\n",id,v);
842 recognised=1; break;
843 }
844
845 if(!strcmp(k,"multilane"))
846 {
847 if(ISTRUE(v))
848 way.props|=Properties_Multilane;
849 else if(!ISFALSE(v))
850 logerror("Way %"Pway_t" has an unrecognised tag value 'multilane' = '%s' (after tagging rules); using 'no'.\n",id,v);
851 recognised=1; break;
852 }
853
854 break;
855
856 case 'n':
857 if(!strcmp(k,"name"))
858 {
859 name=v;
860 recognised=1; break;
861 }
862
863 break;
864
865 case 'o':
866 if(!strcmp(k,"oneway"))
867 {
868 if(ISTRUE(v))
869 oneway=ONEWAY_1TO2;
870 else if(!strcmp(v,"-1"))
871 oneway=ONEWAY_2TO1;
872 else if(!ISFALSE(v))
873 logerror("Way %"Pway_t" has an unrecognised tag value 'oneway' = '%s' (after tagging rules); using 'no'.\n",id,v);
874 recognised=1; break;
875 }
876
877 break;
878
879 case 'p':
880 if(!strcmp(k,"paved"))
881 {
882 if(ISTRUE(v))
883 way.props|=Properties_Paved;
884 else if(!ISFALSE(v))
885 logerror("Way %"Pway_t" has an unrecognised tag value 'paved' = '%s' (after tagging rules); using 'no'.\n",id,v);
886 recognised=1; break;
887 }
888
889 if(!strcmp(k,"psv"))
890 {
891 if(ISTRUE(v))
892 way.allow|=Transports_PSV;
893 else if(!ISFALSE(v))
894 logerror("Way %"Pway_t" has an unrecognised tag value 'psv' = '%s' (after tagging rules); using 'no'.\n",id,v);
895 recognised=1; break;
896 }
897
898 break;
899
900 case 'r':
901 if(!strcmp(k,"ref"))
902 {
903 ref=v;
904 recognised=1; break;
905 }
906
907 if(!strcmp(k,"roundabout"))
908 {
909 if(ISTRUE(v))
910 roundabout=1;
911 else if(!ISFALSE(v))
912 logerror("Way %"Pway_t" has an unrecognised tag value 'roundabout' = '%s' (after tagging rules); using 'no'.\n",id,v);
913 recognised=1; break;
914 }
915
916 break;
917
918 case 't':
919 if(!strcmp(k,"tunnel"))
920 {
921 if(ISTRUE(v))
922 way.props|=Properties_Tunnel;
923 else if(!ISFALSE(v))
924 logerror("Way %"Pway_t" has an unrecognised tag value 'tunnel' = '%s' (after tagging rules); using 'no'.\n",id,v);
925 recognised=1; break;
926 }
927
928 break;
929
930 case 'w':
931 if(!strcmp(k,"wheelchair"))
932 {
933 if(ISTRUE(v))
934 way.allow|=Transports_Wheelchair;
935 else if(!ISFALSE(v))
936 logerror("Way %"Pway_t" has an unrecognised tag value 'wheelchair' = '%s' (after tagging rules); using 'no'.\n",id,v);
937 recognised=1; break;
938 }
939
940 break;
941
942 default:
943 break;
944 }
945
946 if(!recognised)
947 logerror("Way %"Pway_t" has an unrecognised tag '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
948 }
949
950 /* Create the way */
951
952 if(area && oneway)
953 {
954 logerror("Way %"Pway_t" is an area and oneway; ignoring area tagging.\n",id);
955 area=0;
956 }
957
958 if(!way.allow)
959 return;
960
961 if(oneway)
962 way.type|=Highway_OneWay;
963
964 if(roundabout)
965 way.type|=Highway_Roundabout;
966
967 if(ref && name)
968 {
969 refname=(char*)malloc(strlen(ref)+strlen(name)+4);
970 sprintf(refname,"%s (%s)",name,ref);
971 }
972 else if(ref && !name)
973 refname=ref;
974 else if(!ref && name)
975 refname=name;
976 else /* if(!ref && !name) */
977 refname="";
978
979 AppendWayList(ways,id,&way,refname);
980
981 if(ref && name)
982 free(refname);
983
984 if(area && way_nodes[0]!=way_nodes[way_nnodes-1])
985 logerror("Way %"Pway_t" is an area but not closed.\n",id);
986
987 for(i=1;i<way_nnodes;i++)
988 {
989 node_t from=way_nodes[i-1];
990 node_t to =way_nodes[i];
991
992 if(from==to)
993 logerror("Node %"Pnode_t" in way %"Pway_t" is connected to itself.\n",from,id);
994 else
995 {
996 int nto=1,duplicated=0;
997
998 for(j=1;j<i;j++)
999 {
1000 node_t n1=way_nodes[j-1];
1001 node_t n2=way_nodes[j];
1002
1003 if(n1==to && (i!=way_nnodes-1 || j!=1))
1004 nto++;
1005
1006 if((n1==from && n2==to) || (n2==from && n1==to))
1007 {
1008 duplicated=1;
1009 logerror("Segment connecting nodes %"Pnode_t" and %"Pnode_t" in way %"Pway_t" is duplicated.\n",n1,n2,id);
1010 }
1011 }
1012
1013 if(nto>=2 && !duplicated)
1014 logerror("Node %"Pnode_t" in way %"Pway_t" appears more than once.\n",to,id);
1015
1016 if(!duplicated)
1017 AppendSegmentList(segments,id,from,to,area+oneway);
1018 }
1019 }
1020 }
1021
1022
1023 /*++++++++++++++++++++++++++++++++++++++
1024 Process the tags associated with a relation.
1025
1026 TagList *tags The list of relation tags.
1027
1028 int64_t relation_id The id of the relation.
1029
1030 int mode The mode of operation to take (create, modify, delete).
1031 ++++++++++++++++++++++++++++++++++++++*/
1032
1033 void ProcessRelationTags(TagList *tags,int64_t relation_id,int mode)
1034 {
1035 transports_t routes=Transports_None;
1036 transports_t except=Transports_None;
1037 int relation_turn_restriction=0;
1038 TurnRestriction restriction=TurnRestrict_None;
1039 relation_t id;
1040 int i;
1041
1042 /* Convert id */
1043
1044 id=(relation_t)relation_id;
1045 logassert((int64_t)id==relation_id,"Relation ID too large (change relation_t to 64-bits?)"); /* check relation id can be stored in relation_t data type. */
1046
1047 /* Delete */
1048
1049 if(mode==MODE_DELETE || mode==MODE_MODIFY)
1050 {
1051 AppendRouteRelationList(relations,id,RELATION_DELETED,
1052 relation_ways,relation_nways,
1053 relation_relations,relation_nrelations);
1054
1055 AppendTurnRelationList(relations,id,
1056 relation_from,relation_to,relation_via,
1057 restriction,RELATION_DELETED);
1058 }
1059
1060 if(mode==MODE_DELETE)
1061 return;
1062
1063 /* Sanity check */
1064
1065 if(relation_nnodes==0 && relation_nways==0 && relation_nrelations==0)
1066 {
1067 logerror("Relation %"Prelation_t" has no nodes, ways or relations.\n",id);
1068 return;
1069 }
1070
1071 /* Parse the tags */
1072
1073 for(i=0;i<tags->ntags;i++)
1074 {
1075 int recognised=0;
1076 char *k=tags->k[i];
1077 char *v=tags->v[i];
1078
1079 switch(*k)
1080 {
1081 case 'b':
1082 if(!strcmp(k,"bicycleroute"))
1083 {
1084 if(ISTRUE(v))
1085 routes|=Transports_Bicycle;
1086 else if(!ISFALSE(v))
1087 logerror("Relation %"Prelation_t" has an unrecognised tag value 'bicycleroute' = '%s' (after tagging rules); using 'no'.\n",id,v);
1088 recognised=1; break;
1089 }
1090
1091 break;
1092
1093 case 'e':
1094 if(!strcmp(k,"except"))
1095 {
1096 for(i=1;i<Transport_Count;i++)
1097 if(strstr(v,TransportName(i)))
1098 except|=TRANSPORTS(i);
1099
1100 if(except==Transports_None)
1101 logerror("Relation %"Prelation_t" has an unrecognised tag value 'except' = '%s' (after tagging rules); ignoring it.\n",id,v);
1102
1103 recognised=1; break;
1104 }
1105
1106 break;
1107
1108 case 'f':
1109 if(!strcmp(k,"footroute"))
1110 {
1111 if(ISTRUE(v))
1112 routes|=Transports_Foot;
1113 else if(!ISFALSE(v))
1114 logerror("Relation %"Prelation_t" has an unrecognised tag value 'footroute' = '%s' (after tagging rules); using 'no'.\n",id,v);
1115 recognised=1; break;
1116 }
1117
1118 break;
1119
1120 case 'r':
1121 if(!strcmp(k,"restriction"))
1122 {
1123 if(!strcmp(v,"no_right_turn" )) restriction=TurnRestrict_no_right_turn;
1124 if(!strcmp(v,"no_left_turn" )) restriction=TurnRestrict_no_left_turn;
1125 if(!strcmp(v,"no_u_turn" )) restriction=TurnRestrict_no_u_turn;
1126 if(!strcmp(v,"no_straight_on" )) restriction=TurnRestrict_no_straight_on;
1127 if(!strcmp(v,"only_right_turn" )) restriction=TurnRestrict_only_right_turn;
1128 if(!strcmp(v,"only_left_turn" )) restriction=TurnRestrict_only_left_turn;
1129 if(!strcmp(v,"only_straight_on")) restriction=TurnRestrict_only_straight_on;
1130
1131 if(restriction==TurnRestrict_None)
1132 logerror("Relation %"Prelation_t" has an unrecognised tag value 'restriction' = '%s' (after tagging rules); ignoring it.\n",id,v);
1133
1134 recognised=1; break;
1135 }
1136
1137 break;
1138
1139 case 't':
1140 if(!strcmp(k,"type"))
1141 {
1142 if(!strcmp(v,"restriction"))
1143 relation_turn_restriction=1;
1144
1145 /* Don't log an error for relations of types that we don't handle - there are so many */
1146 recognised=1; break;
1147 }
1148
1149 break;
1150
1151 default:
1152 break;
1153 }
1154
1155 if(!recognised)
1156 logerror("Relation %"Prelation_t" has an unrecognised tag '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1157 }
1158
1159 /* Create the route relation (must store all relations that have ways or
1160 relations even if they are not routes because they might be referenced by
1161 other relations that are routes) */
1162
1163 if((relation_nways || relation_nrelations) && !relation_turn_restriction)
1164 AppendRouteRelationList(relations,id,routes,
1165 relation_ways,relation_nways,
1166 relation_relations,relation_nrelations);
1167
1168 /* Create the turn restriction relation. */
1169
1170 if(relation_turn_restriction && restriction!=TurnRestrict_None)
1171 {
1172 if(relation_from==NO_WAY_ID)
1173 logerror("Relation %"Prelation_t" is a turn restriction but has no 'from' way.\n",id);
1174 else if(relation_to==NO_WAY_ID)
1175 logerror("Relation %"Prelation_t" is a turn restriction but has no 'to' way.\n",id);
1176 else if(relation_via==NO_NODE_ID)
1177 logerror("Relation %"Prelation_t" is a turn restriction but has no 'via' node.\n",id);
1178 else
1179 AppendTurnRelationList(relations,id,
1180 relation_from,relation_to,relation_via,
1181 restriction,except);
1182 }
1183 }
1184
1185
1186 /*++++++++++++++++++++++++++++++++++++++
1187 Convert a string containing a speed into a double precision.
1188
1189 double parse_speed Returns the speed in km/h if it can be parsed.
1190
1191 way_t id The way being processed.
1192
1193 const char *k The tag key.
1194
1195 const char *v The tag value.
1196 ++++++++++++++++++++++++++++++++++++++*/
1197
1198 static double parse_speed(way_t id,const char *k,const char *v)
1199 {
1200 char *ev;
1201 double value=strtod(v,&ev);
1202
1203 if(v==ev)
1204 logerror("Way %"Pway_t" has an unrecognised tag value '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1205 else
1206 {
1207 while(isspace(*ev)) ev++;
1208
1209 if(!strcmp(ev,"mph"))
1210 return(1.609*value);
1211
1212 if(*ev==0 || !strcmp(ev,"kph"))
1213 return(value);
1214
1215 logerror("Way %"Pway_t" has an un-parseable tag value '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1216 }
1217
1218 return(0);
1219 }
1220
1221
1222 /*++++++++++++++++++++++++++++++++++++++
1223 Convert a string containing a weight into a double precision.
1224
1225 double parse_weight Returns the weight in tonnes if it can be parsed.
1226
1227 way_t id The way being processed.
1228
1229 const char *k The tag key.
1230
1231 const char *v The tag value.
1232 ++++++++++++++++++++++++++++++++++++++*/
1233
1234 static double parse_weight(way_t id,const char *k,const char *v)
1235 {
1236 char *ev;
1237 double value=strtod(v,&ev);
1238
1239 if(v==ev)
1240 logerror("Way %"Pway_t" has an unrecognised tag value '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1241 else
1242 {
1243 while(isspace(*ev)) ev++;
1244
1245 if(!strcmp(ev,"kg"))
1246 return(value/1000.0);
1247
1248 if(*ev==0 || !strcmp(ev,"T") || !strcmp(ev,"t")
1249 || !strcmp(ev,"ton") || !strcmp(ev,"tons")
1250 || !strcmp(ev,"tonne") || !strcmp(ev,"tonnes"))
1251 return(value);
1252
1253 logerror("Way %"Pway_t" has an un-parseable tag value '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1254 }
1255
1256 return(0);
1257 }
1258
1259
1260 /*++++++++++++++++++++++++++++++++++++++
1261 Convert a string containing a length into a double precision.
1262
1263 double parse_length Returns the length in metres if it can be parsed.
1264
1265 way_t id The way being processed.
1266
1267 const char *k The tag key.
1268
1269 const char *v The tag value.
1270 ++++++++++++++++++++++++++++++++++++++*/
1271
1272 static double parse_length(way_t id,const char *k,const char *v)
1273 {
1274 char *ev;
1275 double value=strtod(v,&ev);
1276
1277 if(v==ev)
1278 logerror("Way %"Pway_t" has an unrecognised tag value '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1279 else
1280 {
1281 int en=0;
1282 int feet=0,inches=0;
1283
1284 if(sscanf(v,"%d' %d\"%n",&feet,&inches,&en)==2 && en && !v[en])
1285 return((feet+(double)inches/12.0)*0.254);
1286
1287 if(sscanf(v,"%d'%d\"%n",&feet,&inches,&en)==2 && en && !v[en])
1288 return((feet+(double)inches/12.0)*0.254);
1289
1290 if(sscanf(v,"%d'-%d\"%n",&feet,&inches,&en)==2 && en && !v[en])
1291 return((feet+(double)inches/12.0)*0.254);
1292
1293 if(sscanf(v,"%d - %d%n",&feet,&inches,&en)==2 && en && !v[en])
1294 return((feet+(double)inches/12.0)*0.254);
1295
1296 if(sscanf(v,"%d ft %d in%n",&feet,&inches,&en)==2 && en && !v[en])
1297 return((feet+(double)inches/12.0)*0.254);
1298
1299 if(sscanf(v,"%d feet %d inches%n",&feet,&inches,&en)==2 && en && !v[en])
1300 return((feet+(double)inches/12.0)*0.254);
1301
1302 if(!strcmp(ev,"'"))
1303 return(feet*0.254);
1304
1305 while(isspace(*ev)) ev++;
1306
1307 if(!strcmp(ev,"ft") || !strcmp(ev,"feet"))
1308 return(value*0.254);
1309
1310 if(*ev==0 || !strcmp(ev,"m") || !strcmp(ev,"metre") || !strcmp(ev,"metres"))
1311 return(value);
1312
1313 logerror("Way %"Pway_t" has an un-parseable tag value '%s' = '%s' (after tagging rules); ignoring it.\n",id,k,v);
1314 }
1315
1316 return(0);
1317 }

Properties

Name Value
cvs:description OSM XML file parser.