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 1999 - (show annotations) (download) (as text)
Sat Jul 27 10:33:04 2019 UTC (5 years, 8 months ago) by amb
File MIME type: text/x-csrc
File size: 21550 byte(s)
Add more checking of memory allocation success/failure by combining
the allocation and the assert into one function.

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

Properties

Name Value
cvs:description Extended ways functions.