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

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1347 - (show annotations) (download) (as text)
Mon May 27 17:36:11 2013 UTC (11 years, 10 months ago) by amb
File MIME type: text/x-csrc
File size: 21018 byte(s)
Redistributed error log messages from osmparser way handling.

1 /***************************************
2 Extended Way data type functions.
3
4 Part of the Routino routing software.
5 ******************/ /******************
6 This file Copyright 2008-2013 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
26 #include "types.h"
27 #include "ways.h"
28
29 #include "typesx.h"
30 #include "nodesx.h"
31 #include "segmentsx.h"
32 #include "waysx.h"
33
34 #include "files.h"
35 #include "logging.h"
36 #include "sorting.h"
37
38
39 /* Global variables */
40
41 /*+ The command line '--tmpdir' option or its default value. +*/
42 extern char *option_tmpdirname;
43
44 /* Local variables */
45
46 /*+ Temporary file-local variables for use by the sort functions. +*/
47 static WaysX *sortwaysx;
48 static SegmentsX *sortsegmentsx;
49
50 /* Local functions */
51
52 static int sort_by_id(WayX *a,WayX *b);
53 static int deduplicate_by_id(WayX *wayx,index_t index);
54
55 static int sort_by_name(WayX *a,WayX *b);
56 static int index_by_id(WayX *wayx,index_t index);
57
58 static int delete_unused(WayX *wayx,index_t index);
59 static int sort_by_name_and_prop_and_id(WayX *a,WayX *b);
60 static int deduplicate_and_index_by_compact_id(WayX *wayx,index_t index);
61
62
63 /*++++++++++++++++++++++++++++++++++++++
64 Allocate a new way list (create a new file or open an existing one).
65
66 WaysX *NewWayList Returns the way list.
67
68 int append Set to 1 if the file is to be opened for appending.
69
70 int readonly Set to 1 if the file is to be opened for reading.
71 ++++++++++++++++++++++++++++++++++++++*/
72
73 WaysX *NewWayList(int append,int readonly)
74 {
75 WaysX *waysx;
76
77 waysx=(WaysX*)calloc(1,sizeof(WaysX));
78
79 logassert(waysx,"Failed to allocate memory (try using slim mode?)"); /* Check calloc() worked */
80
81 waysx->filename =(char*)malloc(strlen(option_tmpdirname)+32);
82 waysx->filename_tmp=(char*)malloc(strlen(option_tmpdirname)+32);
83
84 sprintf(waysx->filename ,"%s/waysx.parsed.mem",option_tmpdirname);
85 sprintf(waysx->filename_tmp,"%s/waysx.%p.tmp" ,option_tmpdirname,(void*)waysx);
86
87 if(append || readonly)
88 if(ExistsFile(waysx->filename))
89 {
90 off_t size,position=0;
91 int fd;
92
93 size=SizeFile(waysx->filename);
94
95 fd=ReOpenFile(waysx->filename);
96
97 while(position<size)
98 {
99 FILESORT_VARINT waysize;
100
101 SeekReadFile(fd,&waysize,FILESORT_VARSIZE,position);
102
103 waysx->number++;
104
105 position+=waysize+FILESORT_VARSIZE;
106 }
107
108 CloseFile(fd);
109
110 RenameFile(waysx->filename,waysx->filename_tmp);
111 }
112
113 if(append)
114 waysx->fd=OpenFileAppend(waysx->filename_tmp);
115 else if(!readonly)
116 waysx->fd=OpenFileNew(waysx->filename_tmp);
117 else
118 waysx->fd=-1;
119
120 #if SLIM
121 waysx->cache=NewWayXCache();
122 #endif
123
124
125 waysx->nfilename_tmp=(char*)malloc(strlen(option_tmpdirname)+32);
126
127 sprintf(waysx->nfilename_tmp,"%s/waynames.%p.tmp",option_tmpdirname,(void*)waysx);
128
129 return(waysx);
130 }
131
132
133 /*++++++++++++++++++++++++++++++++++++++
134 Free a way list.
135
136 WaysX *waysx The set of ways to be freed.
137
138 int keep If set then the results file is to be kept.
139 ++++++++++++++++++++++++++++++++++++++*/
140
141 void FreeWayList(WaysX *waysx,int keep)
142 {
143 if(keep)
144 RenameFile(waysx->filename_tmp,waysx->filename);
145 else
146 DeleteFile(waysx->filename_tmp);
147
148 free(waysx->filename);
149 free(waysx->filename_tmp);
150
151 if(waysx->idata)
152 free(waysx->idata);
153
154 if(waysx->odata)
155 free(waysx->odata);
156
157 if(waysx->cdata)
158 free(waysx->cdata);
159
160 DeleteFile(waysx->nfilename_tmp);
161
162 free(waysx->nfilename_tmp);
163
164 #if SLIM
165 DeleteWayXCache(waysx->cache);
166 #endif
167
168 free(waysx);
169 }
170
171
172 /*++++++++++++++++++++++++++++++++++++++
173 Append a single way to an unsorted way list.
174
175 WaysX *waysx The set of ways to process.
176
177 way_t id The ID of the way.
178
179 Way *way The way data itself.
180
181 node_t *nodes The list of nodes for this way.
182
183 int nnodes The number of nodes for this way.
184
185 const char *name The name or reference of the way.
186 ++++++++++++++++++++++++++++++++++++++*/
187
188 void AppendWayList(WaysX *waysx,way_t id,Way *way,node_t *nodes,int nnodes,const char *name)
189 {
190 WayX wayx;
191 FILESORT_VARINT size;
192 node_t nonode=NO_NODE_ID;
193
194 wayx.id=id;
195 wayx.way=*way;
196
197 size=sizeof(WayX)+(nnodes+1)*sizeof(node_t)+strlen(name)+1;
198
199 WriteFile(waysx->fd,&size,FILESORT_VARSIZE);
200 WriteFile(waysx->fd,&wayx,sizeof(WayX));
201
202 WriteFile(waysx->fd,nodes ,nnodes*sizeof(node_t));
203 WriteFile(waysx->fd,&nonode, sizeof(node_t));
204
205 WriteFile(waysx->fd,name,strlen(name)+1);
206
207 waysx->number++;
208
209 logassert(waysx->number!=0,"Too many ways (change index_t to 64-bits?)"); /* Zero marks the high-water mark for ways. */
210 }
211
212
213 /*++++++++++++++++++++++++++++++++++++++
214 Finish appending ways and change the filename over.
215
216 WaysX *waysx The ways that have been appended.
217 ++++++++++++++++++++++++++++++++++++++*/
218
219 void FinishWayList(WaysX *waysx)
220 {
221 if(waysx->fd!=-1)
222 waysx->fd=CloseFile(waysx->fd);
223 }
224
225
226 /*++++++++++++++++++++++++++++++++++++++
227 Find a particular way index.
228
229 index_t IndexWayX Returns the index of the extended way with the specified id.
230
231 WaysX *waysx The set of ways to process.
232
233 way_t id The way id to look for.
234 ++++++++++++++++++++++++++++++++++++++*/
235
236 index_t IndexWayX(WaysX *waysx,way_t id)
237 {
238 index_t start=0;
239 index_t end=waysx->number-1;
240 index_t mid;
241
242 if(waysx->number==0) /* There are no ways */
243 return(NO_WAY);
244
245 if(id<waysx->idata[start]) /* Key is before start */
246 return(NO_WAY);
247
248 if(id>waysx->idata[end]) /* Key is after end */
249 return(NO_WAY);
250
251 /* Binary search - search key exact match only is required.
252 *
253 * # <- start | Check mid and move start or end if it doesn't match
254 * # |
255 * # | Since an exact match is wanted we can set end=mid-1
256 * # <- mid | or start=mid+1 because we know that mid doesn't match.
257 * # |
258 * # | Eventually either end=start or end=start+1 and one of
259 * # <- end | start or end is the wanted one.
260 */
261
262 do
263 {
264 mid=(start+end)/2; /* Choose mid point */
265
266 if(waysx->idata[mid]<id) /* Mid point is too low */
267 start=mid+1;
268 else if(waysx->idata[mid]>id) /* Mid point is too high */
269 end=mid?(mid-1):mid;
270 else /* Mid point is correct */
271 return(mid);
272 }
273 while((end-start)>1);
274
275 if(waysx->idata[start]==id) /* Start is correct */
276 return(start);
277
278 if(waysx->idata[end]==id) /* End is correct */
279 return(end);
280
281 return(NO_WAY);
282 }
283
284
285 /*++++++++++++++++++++++++++++++++++++++
286 Sort the list of ways.
287
288 WaysX *waysx The set of ways to process.
289 ++++++++++++++++++++++++++++++++++++++*/
290
291 void SortWayList(WaysX *waysx)
292 {
293 index_t xnumber;
294 int fd;
295
296 /* Print the start message */
297
298 printf_first("Sorting Ways");
299
300 /* Re-open the file read-only and a new file writeable */
301
302 waysx->fd=ReOpenFile(waysx->filename_tmp);
303
304 DeleteFile(waysx->filename_tmp);
305
306 fd=OpenFileNew(waysx->filename_tmp);
307
308 /* Sort the ways by ID and index them */
309
310 xnumber=waysx->number;
311
312 waysx->number=filesort_vary(waysx->fd,fd,NULL,
313 (int (*)(const void*,const void*))sort_by_id,
314 (int (*)(void*,index_t))deduplicate_by_id);
315
316 /* Close the files */
317
318 waysx->fd=CloseFile(waysx->fd);
319 CloseFile(fd);
320
321 /* Print the final message */
322
323 printf_last("Sorted Ways: Ways=%"Pindex_t" Duplicates=%"Pindex_t,xnumber,xnumber-waysx->number);
324 }
325
326
327 /*++++++++++++++++++++++++++++++++++++++
328 Sort the ways into id order.
329
330 int sort_by_id Returns the comparison of the id fields.
331
332 WayX *a The first extended way.
333
334 WayX *b The second extended way.
335 ++++++++++++++++++++++++++++++++++++++*/
336
337 static int sort_by_id(WayX *a,WayX *b)
338 {
339 way_t a_id=a->id;
340 way_t b_id=b->id;
341
342 if(a_id<b_id)
343 return(-1);
344 else if(a_id>b_id)
345 return(1);
346 else
347 return(-FILESORT_PRESERVE_ORDER(a,b)); /* latest version first */
348 }
349
350
351 /*++++++++++++++++++++++++++++++++++++++
352 Discard duplicate ways.
353
354 int deduplicate_by_id Return 1 if the value is to be kept, otherwise 0.
355
356 WayX *wayx The extended way.
357
358 index_t index The number of sorted ways that have already been written to the output file.
359 ++++++++++++++++++++++++++++++++++++++*/
360
361 static int deduplicate_by_id(WayX *wayx,index_t index)
362 {
363 static way_t previd=NO_WAY_ID;
364
365 if(wayx->id!=previd)
366 {
367 previd=wayx->id;
368
369 if(wayx->way.type==WAY_DELETED)
370 return(0);
371 else
372 return(1);
373 }
374 else
375 return(0);
376 }
377
378
379 /*++++++++++++++++++++++++++++++++++++++
380 Generate the segments from the ways.
381
382 WaysX *waysx The set of ways to process.
383
384 NodesX *nodesx The set of nodes to use for creating the segments.
385
386 int keep If set to 1 then keep the old data file otherwise delete it.
387 ++++++++++++++++++++++++++++++++++++++*/
388
389 SegmentsX *GenerateSegments(WaysX *waysx,NodesX *nodesx,int keep)
390 {
391 SegmentsX *segmentsx;
392 index_t i;
393 int fd;
394 char *name=NULL;
395 int namelen=0;
396
397 /* Print the start message */
398
399 printf_first("Generating Segments: Ways=0 Segments=0");
400
401 segmentsx=NewSegmentList();
402
403 /* Re-open the file read-only and a new file writeable */
404
405 waysx->fd=ReOpenFile(waysx->filename_tmp);
406
407 if(keep)
408 RenameFile(waysx->filename_tmp,waysx->filename);
409 else
410 DeleteFile(waysx->filename_tmp);
411
412 waysx->knumber=waysx->number;
413
414 fd=OpenFileNew(waysx->filename_tmp);
415
416 /* Loop through the ways and create the segments */
417
418 for(i=0;i<waysx->number;i++)
419 {
420 WayX wayx;
421 FILESORT_VARINT size;
422 node_t node,prevnode=NO_NODE_ID;
423
424 ReadFile(waysx->fd,&size,FILESORT_VARSIZE);
425
426 ReadFile(waysx->fd,&wayx,sizeof(WayX));
427
428 while(!ReadFile(waysx->fd,&node,sizeof(node_t)) && node!=NO_NODE_ID)
429 {
430 if(prevnode==NO_NODE_ID)
431 ;
432 else if(prevnode==node)
433 logerror("Node %"Pnode_t" in way %"Pway_t" is connected to itself.\n",logerror_node(node),logerror_way(wayx.id));
434 else
435 {
436 distance_t segment_flags=0;
437
438 if(wayx.way.type&Highway_OneWay)
439 segment_flags|=ONEWAY_1TO2;
440
441 if(wayx.way.type&Highway_Area)
442 segment_flags|=SEGMENT_AREA;
443
444 AppendSegmentList(segmentsx,wayx.id,prevnode,node,segment_flags);
445 }
446
447 prevnode=node;
448
449 size-=sizeof(node_t);
450 }
451
452 size-=sizeof(node_t);
453
454 if(namelen<size)
455 name=(char*)realloc((void*)name,namelen=size);
456
457 ReadFile(waysx->fd,name,size-sizeof(WayX));
458
459 WriteFile(fd,&size,FILESORT_VARSIZE);
460 WriteFile(fd,&wayx,sizeof(WayX));
461 WriteFile(fd,name,size-sizeof(WayX));
462
463 if(!((i+1)%1000))
464 printf_middle("Generating Segments: Ways=%"Pindex_t" Segments=%"Pindex_t,i+1,segmentsx->number);
465 }
466
467 FinishSegmentList(segmentsx);
468
469 if(name) free(name);
470
471 /* Close the files */
472
473 waysx->fd=CloseFile(waysx->fd);
474 CloseFile(fd);
475
476 /* Print the final message */
477
478 printf_last("Generated Segments: Ways=%"Pindex_t" Segments=%"Pindex_t,waysx->number,segmentsx->number);
479
480 return(segmentsx);
481 }
482
483
484
485 /*++++++++++++++++++++++++++++++++++++++
486 Extract the way names from the ways and reference the list of names from the ways.
487
488 WaysX *waysx The set of ways to process.
489 ++++++++++++++++++++++++++++++++++++++*/
490
491 void ExtractWayNames(WaysX *waysx)
492 {
493 index_t i;
494 int fd;
495 char *names[2]={NULL,NULL};
496 int namelen[2]={0,0};
497 int nnames=0;
498 uint32_t lastlength=0;
499
500 /* Print the start message */
501
502 printf_first("Sorting Ways by Name");
503
504 /* Re-open the file read-only and new files writeable */
505
506 waysx->fd=ReOpenFile(waysx->filename_tmp);
507
508 DeleteFile(waysx->filename_tmp);
509
510 fd=OpenFileNew(waysx->filename_tmp);
511
512 /* Sort the ways to allow separating the names */
513
514 filesort_vary(waysx->fd,fd,NULL,
515 (int (*)(const void*,const void*))sort_by_name,
516 NULL);
517
518 /* Close the files */
519
520 waysx->fd=CloseFile(waysx->fd);
521 CloseFile(fd);
522
523 /* Print the final message */
524
525 printf_last("Sorted Ways by Name: Ways=%"Pindex_t,waysx->number);
526
527
528 /* Print the start message */
529
530 printf_first("Separating Way Names: Ways=0 Names=0");
531
532 /* Re-open the file read-only and new files writeable */
533
534 waysx->fd=ReOpenFile(waysx->filename_tmp);
535
536 DeleteFile(waysx->filename_tmp);
537
538 fd=OpenFileNew(waysx->filename_tmp);
539
540 waysx->nfd=OpenFileNew(waysx->nfilename_tmp);
541
542 /* Copy from the single file into two files */
543
544 for(i=0;i<waysx->number;i++)
545 {
546 WayX wayx;
547 FILESORT_VARINT size;
548
549 ReadFile(waysx->fd,&size,FILESORT_VARSIZE);
550
551 if(namelen[nnames%2]<size)
552 names[nnames%2]=(char*)realloc((void*)names[nnames%2],namelen[nnames%2]=size);
553
554 ReadFile(waysx->fd,&wayx,sizeof(WayX));
555 ReadFile(waysx->fd,names[nnames%2],size-sizeof(WayX));
556
557 if(nnames==0 || strcmp(names[0],names[1]))
558 {
559 WriteFile(waysx->nfd,names[nnames%2],size-sizeof(WayX));
560
561 lastlength=waysx->nlength;
562 waysx->nlength+=size-sizeof(WayX);
563
564 nnames++;
565 }
566
567 wayx.way.name=lastlength;
568
569 WriteFile(fd,&wayx,sizeof(WayX));
570
571 if(!((i+1)%1000))
572 printf_middle("Separating Way Names: Ways=%"Pindex_t" Names=%"Pindex_t,i+1,nnames);
573 }
574
575 if(names[0]) free(names[0]);
576 if(names[1]) free(names[1]);
577
578 /* Close the files */
579
580 waysx->fd=CloseFile(waysx->fd);
581 CloseFile(fd);
582
583 waysx->nfd=CloseFile(waysx->nfd);
584
585 /* Print the final message */
586
587 printf_last("Separated Way Names: Ways=%"Pindex_t" Names=%"Pindex_t,waysx->number,nnames);
588
589
590 /* Print the start message */
591
592 printf_first("Sorting Ways");
593
594 /* Re-open the file read-only and a new file writeable */
595
596 waysx->fd=ReOpenFile(waysx->filename_tmp);
597
598 DeleteFile(waysx->filename_tmp);
599
600 fd=OpenFileNew(waysx->filename_tmp);
601
602 /* Allocate the array of indexes */
603
604 waysx->idata=(way_t*)malloc(waysx->number*sizeof(way_t));
605
606 logassert(waysx->idata,"Failed to allocate memory (try using slim mode?)"); /* Check malloc() worked */
607
608 /* Sort the ways by ID */
609
610 sortwaysx=waysx;
611
612 filesort_fixed(waysx->fd,fd,sizeof(WayX),NULL,
613 (int (*)(const void*,const void*))sort_by_id,
614 (int (*)(void*,index_t))index_by_id);
615
616 /* Close the files */
617
618 waysx->fd=CloseFile(waysx->fd);
619 CloseFile(fd);
620
621 /* Print the final message */
622
623 printf_last("Sorted Ways: Ways=%"Pindex_t,waysx->number);
624 }
625
626
627 /*++++++++++++++++++++++++++++++++++++++
628 Sort the ways into name order and then id order.
629
630 int sort_by_name Returns the comparison of the name fields.
631
632 WayX *a The first extended Way.
633
634 WayX *b The second extended Way.
635 ++++++++++++++++++++++++++++++++++++++*/
636
637 static int sort_by_name(WayX *a,WayX *b)
638 {
639 int compare;
640 char *a_name=(char*)a+sizeof(WayX);
641 char *b_name=(char*)b+sizeof(WayX);
642
643 compare=strcmp(a_name,b_name);
644
645 if(compare)
646 return(compare);
647 else
648 return(FILESORT_PRESERVE_ORDER(a,b));
649 }
650
651
652 /*++++++++++++++++++++++++++++++++++++++
653 Create the index of identifiers.
654
655 int index_by_id Return 1 if the value is to be kept, otherwise 0.
656
657 WayX *wayx The extended way.
658
659 index_t index The number of sorted ways that have already been written to the output file.
660 ++++++++++++++++++++++++++++++++++++++*/
661
662 static int index_by_id(WayX *wayx,index_t index)
663 {
664 sortwaysx->idata[index]=wayx->id;
665
666 return(1);
667 }
668
669
670 /*++++++++++++++++++++++++++++++++++++++
671 Compact the way list, removing duplicated ways and unused ways.
672
673 WaysX *waysx The set of ways to process.
674
675 SegmentsX *segmentsx The set of segments to check.
676 ++++++++++++++++++++++++++++++++++++++*/
677
678 void CompactWayList(WaysX *waysx,SegmentsX *segmentsx)
679 {
680 int fd;
681 index_t cnumber;
682
683 if(waysx->number==0)
684 return;
685
686 /* Print the start message */
687
688 printf_first("Sorting Ways and Compacting");
689
690 /* Allocate the array of indexes */
691
692 waysx->cdata=(index_t*)malloc(waysx->number*sizeof(index_t));
693
694 logassert(waysx->cdata,"Failed to allocate memory (try using slim mode?)"); /* Check malloc() worked */
695
696 /* Re-open the file read-only and a new file writeable */
697
698 waysx->fd=ReOpenFile(waysx->filename_tmp);
699
700 DeleteFile(waysx->filename_tmp);
701
702 fd=OpenFileNew(waysx->filename_tmp);
703
704 /* Sort the ways to allow compacting according to the properties */
705
706 sortwaysx=waysx;
707 sortsegmentsx=segmentsx;
708
709 cnumber=filesort_fixed(waysx->fd,fd,sizeof(WayX),(int (*)(void*,index_t))delete_unused,
710 (int (*)(const void*,const void*))sort_by_name_and_prop_and_id,
711 (int (*)(void*,index_t))deduplicate_and_index_by_compact_id);
712
713 /* Close the files */
714
715 waysx->fd=CloseFile(waysx->fd);
716 CloseFile(fd);
717
718 /* Print the final message */
719
720 printf_last("Sorted and Compacted Ways: Ways=%"Pindex_t" Unique=%"Pindex_t,waysx->number,cnumber);
721 waysx->number=cnumber;
722
723 free(segmentsx->usedway);
724 segmentsx->usedway=NULL;
725 }
726
727
728 /*++++++++++++++++++++++++++++++++++++++
729 Delete the ways that are no longer being used.
730
731 int delete_unused Return 1 if the value is to be kept, otherwise 0.
732
733 WayX *wayx The extended way.
734
735 index_t index The number of unsorted ways that have been read from the input file.
736 ++++++++++++++++++++++++++++++++++++++*/
737
738 static int delete_unused(WayX *wayx,index_t index)
739 {
740 if(sortsegmentsx && !IsBitSet(sortsegmentsx->usedway,index))
741 {
742 sortwaysx->cdata[index]=NO_WAY;
743
744 return(0);
745 }
746 else
747 {
748 wayx->id=index;
749
750 return(1);
751 }
752 }
753
754
755 /*++++++++++++++++++++++++++++++++++++++
756 Sort the ways into name, properties and id order.
757
758 int sort_by_name_and_prop_and_id Returns the comparison of the name, properties and id fields.
759
760 WayX *a The first extended Way.
761
762 WayX *b The second extended Way.
763 ++++++++++++++++++++++++++++++++++++++*/
764
765 static int sort_by_name_and_prop_and_id(WayX *a,WayX *b)
766 {
767 int compare;
768 index_t a_name=a->way.name;
769 index_t b_name=b->way.name;
770
771 if(a_name<b_name)
772 return(-1);
773 else if(a_name>b_name)
774 return(1);
775
776 compare=WaysCompare(&a->way,&b->way);
777
778 if(compare)
779 return(compare);
780
781 return(sort_by_id(a,b));
782 }
783
784
785 /*++++++++++++++++++++++++++++++++++++++
786 Create the index of compacted Way identifiers and ignore Ways with duplicated properties.
787
788 int deduplicate_and_index_by_compact_id Return 1 if the value is to be kept, otherwise 0.
789
790 WayX *wayx The extended way.
791
792 index_t index The number of sorted ways that have already been written to the output file.
793 ++++++++++++++++++++++++++++++++++++++*/
794
795 static int deduplicate_and_index_by_compact_id(WayX *wayx,index_t index)
796 {
797 static Way lastway;
798
799 if(index==0 || wayx->way.name!=lastway.name || WaysCompare(&lastway,&wayx->way))
800 {
801 lastway=wayx->way;
802
803 sortwaysx->cdata[wayx->id]=index;
804
805 wayx->id=index;
806
807 return(1);
808 }
809 else
810 {
811 sortwaysx->cdata[wayx->id]=index-1;
812
813 return(0);
814 }
815 }
816
817
818 /*++++++++++++++++++++++++++++++++++++++
819 Save the way list to a file.
820
821 WaysX *waysx The set of ways to save.
822
823 const char *filename The name of the file to save.
824 ++++++++++++++++++++++++++++++++++++++*/
825
826 void SaveWayList(WaysX *waysx,const char *filename)
827 {
828 index_t i;
829 int fd;
830 index_t position=0;
831 WayX wayx;
832 WaysFile waysfile={0};
833 highways_t highways=0;
834 transports_t allow=0;
835 properties_t props=0;
836
837 /* Print the start message */
838
839 printf_first("Writing Ways: Ways=0");
840
841 /* Re-open the files */
842
843 waysx->fd=ReOpenFile(waysx->filename_tmp);
844 waysx->nfd=ReOpenFile(waysx->nfilename_tmp);
845
846 /* Write out the ways data */
847
848 fd=OpenFileNew(filename);
849
850 SeekFile(fd,sizeof(WaysFile));
851
852 for(i=0;i<waysx->number;i++)
853 {
854 ReadFile(waysx->fd,&wayx,sizeof(WayX));
855
856 highways|=HIGHWAYS(wayx.way.type);
857 allow |=wayx.way.allow;
858 props |=wayx.way.props;
859
860 WriteFile(fd,&wayx.way,sizeof(Way));
861
862 if(!((i+1)%1000))
863 printf_middle("Writing Ways: Ways=%"Pindex_t,i+1);
864 }
865
866 /* Write out the ways names */
867
868 SeekFile(fd,sizeof(WaysFile)+(off_t)waysx->number*sizeof(Way));
869
870 while(position<waysx->nlength)
871 {
872 size_t len=1024;
873 char temp[1024];
874
875 if((waysx->nlength-position)<1024)
876 len=waysx->nlength-position;
877
878 ReadFile(waysx->nfd,temp,len);
879
880 WriteFile(fd,temp,len);
881
882 position+=len;
883 }
884
885 /* Close the files */
886
887 waysx->fd=CloseFile(waysx->fd);
888 waysx->nfd=CloseFile(waysx->nfd);
889
890 /* Write out the header structure */
891
892 waysfile.number =waysx->number;
893
894 waysfile.highways=highways;
895 waysfile.allow =allow;
896 waysfile.props =props;
897
898 SeekFile(fd,0);
899 WriteFile(fd,&waysfile,sizeof(WaysFile));
900
901 CloseFile(fd);
902
903 /* Print the final message */
904
905 printf_last("Wrote Ways: Ways=%"Pindex_t,waysx->number);
906 }

Properties

Name Value
cvs:description Extended ways functions.