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 1497 - (show annotations) (download) (as text)
Mon Jan 27 19:48:26 2014 UTC (11 years, 2 months ago) by amb
File MIME type: text/x-csrc
File size: 33580 byte(s)
Detect the "cycleway=opposite_lane" tag and allow bicycles to travel in both
directions along those highways.  Based on parts of
https://github.com/cgravier/routino-2.6-bikeopposite-elevation/commit/47d68b37f1ea0d2f967ea6aa1555991747491200
https://github.com/cgravier/routino-2.6-bikeopposite-elevation/commit/764832f2671e6f6d8176be65cea3992bd1a488d3
https://github.com/cgravier/routino-2.6-bikeopposite-elevation/commit/37af908880c045309fba1125c4d683f6925f7d25
by Christophe Collard.

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

Properties

Name Value
cvs:description OSM XML file parser.