Routino SVN Repository Browser

Check out the latest version of Routino: svn co http://routino.org/svn/trunk routino

ViewVC logotype

Annotation of /trunk/src/waysx.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1699 - (hide annotations) (download) (as text)
Tue Jun 9 18:05:06 2015 UTC (9 years, 9 months ago) by amb
File MIME type: text/x-csrc
File size: 21333 byte(s)
Merge the MS-Windows branch back into the trunk.
Code changes and documentation for Cygwin and MinGW compilers.

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

Properties

Name Value
cvs:description Extended ways functions.