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