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

Parent Directory Parent Directory | Revision Log Revision Log


Revision 448 - (hide annotations) (download) (as text)
Sun Jul 11 10:56:51 2010 UTC (14 years, 8 months ago) by amb
File MIME type: text/x-csrc
File size: 22687 byte(s)
Made the planetsplitter slim mode handle the output node and segment data in a
slim way as well as in the input data.

1 amb 110 /***************************************
2 amb 448 $Header: /home/amb/CVS/routino/src/nodesx.c,v 1.58 2010-07-11 10:56:50 amb Exp $
3 amb 110
4     Extented Node data type functions.
5 amb 151
6     Part of the Routino routing software.
7 amb 110 ******************/ /******************
8 amb 326 This file Copyright 2008-2010 Andrew M. Bishop
9 amb 110
10 amb 151 This program is free software: you can redistribute it and/or modify
11     it under the terms of the GNU Affero General Public License as published by
12     the Free Software Foundation, either version 3 of the License, or
13     (at your option) any later version.
14    
15     This program is distributed in the hope that it will be useful,
16     but WITHOUT ANY WARRANTY; without even the implied warranty of
17     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18     GNU Affero General Public License for more details.
19    
20     You should have received a copy of the GNU Affero General Public License
21     along with this program. If not, see <http://www.gnu.org/licenses/>.
22 amb 110 ***************************************/
23    
24    
25     #include <assert.h>
26     #include <stdlib.h>
27     #include <stdio.h>
28 amb 249 #include <string.h>
29 amb 326 #include <sys/stat.h>
30 amb 110
31     #include "types.h"
32     #include "functions.h"
33     #include "nodesx.h"
34     #include "segmentsx.h"
35 amb 204 #include "waysx.h"
36 amb 228 #include "segments.h"
37     #include "nodes.h"
38 amb 110
39    
40 amb 252 /* Variables */
41 amb 110
42 amb 289 /*+ The command line '--slim' option. +*/
43 amb 252 extern int option_slim;
44 amb 289
45     /*+ The command line '--tmpdir' option or its default value. +*/
46 amb 284 extern char *option_tmpdirname;
47 amb 110
48 amb 284 /*+ A temporary file-local variable for use by the sort functions. +*/
49     static NodesX *sortnodesx;
50    
51 amb 110 /* Functions */
52    
53 amb 271 static int sort_by_id(NodeX *a,NodeX *b);
54 amb 273 static int index_by_id(NodeX *nodex,index_t index);
55 amb 271
56 amb 281 static int sort_by_lat_long(NodeX *a,NodeX *b);
57     static int index_by_lat_long(NodeX *nodex,index_t index);
58 amb 110
59    
60     /*++++++++++++++++++++++++++++++++++++++
61 amb 326 Allocate a new node list (create a new file or open an existing one).
62 amb 110
63     NodesX *NewNodeList Returns the node list.
64 amb 326
65     int append Set to 1 if the file is to be opened for appending (now or later).
66 amb 110 ++++++++++++++++++++++++++++++++++++++*/
67    
68 amb 326 NodesX *NewNodeList(int append)
69 amb 110 {
70     NodesX *nodesx;
71    
72 amb 213 nodesx=(NodesX*)calloc(1,sizeof(NodesX));
73 amb 110
74 amb 243 assert(nodesx); /* Check calloc() worked */
75    
76 amb 284 nodesx->filename=(char*)malloc(strlen(option_tmpdirname)+32);
77 amb 216
78 amb 326 if(append)
79 amb 447 sprintf(nodesx->filename,"%s/nodesx.input.tmp",option_tmpdirname);
80 amb 326 else
81 amb 447 sprintf(nodesx->filename,"%s/nodesx.%p.tmp",option_tmpdirname,nodesx);
82 amb 249
83 amb 448 nodesx->nfilename=(char*)malloc(strlen(option_tmpdirname)+32);
84    
85     sprintf(nodesx->nfilename,"%s/nodes.%p.tmp",option_tmpdirname,nodesx);
86    
87 amb 326 if(append)
88     {
89 amb 331 off_t size;
90 amb 326
91     nodesx->fd=AppendFile(nodesx->filename);
92    
93 amb 331 size=SizeFile(nodesx->filename);
94 amb 326
95 amb 331 nodesx->xnumber=size/sizeof(NodeX);
96 amb 326 }
97     else
98     nodesx->fd=OpenFile(nodesx->filename);
99    
100 amb 110 return(nodesx);
101     }
102    
103    
104     /*++++++++++++++++++++++++++++++++++++++
105 amb 226 Free a node list.
106    
107     NodesX *nodesx The list to be freed.
108 amb 326
109     int keep Set to 1 if the file is to be kept.
110 amb 226 ++++++++++++++++++++++++++++++++++++++*/
111    
112 amb 326 void FreeNodeList(NodesX *nodesx,int keep)
113 amb 226 {
114 amb 326 if(!keep)
115     DeleteFile(nodesx->filename);
116    
117 amb 283 free(nodesx->filename);
118 amb 226
119     if(nodesx->idata)
120     free(nodesx->idata);
121    
122     if(nodesx->ndata)
123     free(nodesx->ndata);
124    
125 amb 448 DeleteFile(nodesx->nfilename);
126    
127     free(nodesx->nfilename);
128    
129 amb 283 if(nodesx->super)
130     free(nodesx->super);
131    
132 amb 257 if(nodesx->offsets)
133     free(nodesx->offsets);
134    
135 amb 226 free(nodesx);
136     }
137    
138    
139     /*++++++++++++++++++++++++++++++++++++++
140 amb 252 Append a node to a newly created node list (unsorted).
141 amb 243
142 amb 252 NodesX* nodesx The set of nodes to process.
143 amb 243
144 amb 252 node_t id The node identification.
145 amb 110
146 amb 252 double latitude The latitude of the node.
147 amb 110
148 amb 252 double longitude The longitude of the node.
149     ++++++++++++++++++++++++++++++++++++++*/
150 amb 110
151 amb 252 void AppendNode(NodesX* nodesx,node_t id,double latitude,double longitude)
152     {
153     NodeX nodex;
154 amb 249
155 amb 252 assert(!nodesx->idata); /* Must not have idata filled in => unsorted */
156    
157     nodex.id=id;
158     nodex.latitude =radians_to_latlong(latitude);
159     nodex.longitude=radians_to_latlong(longitude);
160    
161     WriteFile(nodesx->fd,&nodex,sizeof(NodeX));
162    
163     nodesx->xnumber++;
164 amb 110 }
165    
166    
167     /*++++++++++++++++++++++++++++++++++++++
168 amb 263 Sort the node list (i.e. create the sortable indexes).
169 amb 110
170     NodesX* nodesx The set of nodes to process.
171     ++++++++++++++++++++++++++++++++++++++*/
172    
173 amb 263 void SortNodeList(NodesX* nodesx)
174 amb 110 {
175 amb 263 int fd;
176 amb 110
177 amb 263 /* Check the start conditions */
178    
179 amb 252 assert(!nodesx->idata); /* Must not have idata filled in => unsorted */
180    
181 amb 263 /* Print the start message */
182    
183     printf("Sorting Nodes");
184 amb 227 fflush(stdout);
185 amb 132
186 amb 263 /* Close the files and re-open them (finished appending) */
187 amb 260
188     CloseFile(nodesx->fd);
189     nodesx->fd=ReOpenFile(nodesx->filename);
190    
191 amb 271 DeleteFile(nodesx->filename);
192    
193     fd=OpenFile(nodesx->filename);
194    
195 amb 252 /* Allocate the array of indexes */
196 amb 249
197 amb 252 nodesx->idata=(node_t*)malloc(nodesx->xnumber*sizeof(node_t));
198 amb 249
199 amb 252 assert(nodesx->idata); /* Check malloc() worked */
200 amb 249
201 amb 271 /* Sort by node indexes */
202 amb 249
203 amb 271 sortnodesx=nodesx;
204 amb 110
205 amb 311 filesort_fixed(nodesx->fd,fd,sizeof(NodeX),(int (*)(const void*,const void*))sort_by_id,(int (*)(void*,index_t))index_by_id);
206 amb 252
207 amb 260 /* Close the files and re-open them */
208    
209 amb 257 CloseFile(nodesx->fd);
210     CloseFile(fd);
211 amb 252
212 amb 257 nodesx->fd=ReOpenFile(nodesx->filename);
213 amb 252
214 amb 263 /* Print the final message */
215    
216 amb 273 printf("\rSorted Nodes: Nodes=%d Duplicates=%d\n",nodesx->xnumber,nodesx->xnumber-nodesx->number);
217 amb 263 fflush(stdout);
218 amb 110 }
219    
220    
221     /*++++++++++++++++++++++++++++++++++++++
222     Sort the nodes into id order.
223    
224     int sort_by_id Returns the comparison of the id fields.
225    
226 amb 271 NodeX *a The first extended node.
227 amb 110
228 amb 271 NodeX *b The second extended node.
229 amb 110 ++++++++++++++++++++++++++++++++++++++*/
230    
231 amb 271 static int sort_by_id(NodeX *a,NodeX *b)
232 amb 110 {
233 amb 271 node_t a_id=a->id;
234     node_t b_id=b->id;
235 amb 252
236     if(a_id<b_id)
237     return(-1);
238     else if(a_id>b_id)
239 amb 249 return(1);
240 amb 110 else
241 amb 252 return(0);
242 amb 110 }
243    
244    
245 amb 271 /*++++++++++++++++++++++++++++++++++++++
246     Index the nodes after sorting.
247 amb 252
248 amb 289 int index_by_id Return 1 if the value is to be kept, otherwise zero.
249 amb 273
250 amb 271 NodeX *nodex The extended node.
251 amb 252
252 amb 271 index_t index The index of this node in the total.
253     ++++++++++++++++++++++++++++++++++++++*/
254    
255 amb 273 static int index_by_id(NodeX *nodex,index_t index)
256 amb 271 {
257 amb 273 if(index==0 || sortnodesx->idata[index-1]!=nodex->id)
258     {
259     sortnodesx->idata[index]=nodex->id;
260 amb 271
261 amb 273 sortnodesx->number++;
262    
263     return(1);
264     }
265    
266     return(0);
267 amb 271 }
268    
269    
270 amb 110 /*++++++++++++++++++++++++++++++++++++++
271 amb 212 Sort the node list geographically.
272    
273     NodesX* nodesx The set of nodes to process.
274     ++++++++++++++++++++++++++++++++++++++*/
275    
276     void SortNodeListGeographically(NodesX* nodesx)
277     {
278 amb 281 int fd;
279 amb 212
280 amb 263 /* Print the start message */
281    
282 amb 227 printf("Sorting Nodes Geographically");
283     fflush(stdout);
284 amb 212
285 amb 281 /* Allocate the memory for the geographical offsets array */
286 amb 275
287 amb 281 nodesx->offsets=(index_t*)malloc((nodesx->latbins*nodesx->lonbins+1)*sizeof(index_t));
288 amb 275
289 amb 281 nodesx->latlonbin=0;
290 amb 212
291 amb 281 /* Close the files and re-open them */
292 amb 212
293 amb 281 CloseFile(nodesx->fd);
294     nodesx->fd=ReOpenFile(nodesx->filename);
295 amb 243
296 amb 281 DeleteFile(nodesx->filename);
297 amb 212
298 amb 281 fd=OpenFile(nodesx->filename);
299 amb 212
300 amb 281 /* Sort geographically */
301 amb 252
302 amb 281 sortnodesx=nodesx;
303 amb 257
304 amb 311 filesort_fixed(nodesx->fd,fd,sizeof(NodeX),(int (*)(const void*,const void*))sort_by_lat_long,(int (*)(void*,index_t))index_by_lat_long);
305 amb 257
306 amb 281 /* Close the files and re-open them */
307 amb 257
308 amb 281 CloseFile(nodesx->fd);
309     CloseFile(fd);
310 amb 257
311 amb 281 nodesx->fd=ReOpenFile(nodesx->filename);
312 amb 257
313 amb 281 /* Finish off the indexing */
314 amb 257
315 amb 281 for(;nodesx->latlonbin<=(nodesx->latbins*nodesx->lonbins);nodesx->latlonbin++)
316     nodesx->offsets[nodesx->latlonbin]=nodesx->number;
317 amb 257
318 amb 263 /* Print the final message */
319 amb 257
320 amb 227 printf("\rSorted Nodes Geographically \n");
321     fflush(stdout);
322 amb 212 }
323    
324    
325     /*++++++++++++++++++++++++++++++++++++++
326 amb 110 Sort the nodes into latitude and longitude order.
327    
328     int sort_by_lat_long Returns the comparison of the latitude and longitude fields.
329    
330 amb 281 NodeX *a The first extended node.
331 amb 110
332 amb 281 NodeX *b The second extended node.
333 amb 110 ++++++++++++++++++++++++++++++++++++++*/
334    
335 amb 281 static int sort_by_lat_long(NodeX *a,NodeX *b)
336 amb 110 {
337 amb 281 ll_bin_t a_lon=latlong_to_bin(a->longitude);
338     ll_bin_t b_lon=latlong_to_bin(b->longitude);
339 amb 110
340     if(a_lon<b_lon)
341     return(-1);
342     else if(a_lon>b_lon)
343     return(1);
344     else
345     {
346 amb 281 ll_bin_t a_lat=latlong_to_bin(a->latitude);
347     ll_bin_t b_lat=latlong_to_bin(b->latitude);
348 amb 110
349     if(a_lat<b_lat)
350     return(-1);
351     else if(a_lat>b_lat)
352     return(1);
353     else
354 amb 281 {
355     #ifdef REGRESSION_TESTING
356     // Need this for regression testing because heapsort() is not order
357     // preserving like qsort() is (or was when tested).
358    
359     index_t a_id=a->id;
360     index_t b_id=b->id;
361    
362     if(a_id<b_id)
363     return(-1);
364     else if(a_id>b_id)
365     return(1);
366     else
367     #endif
368     return(0);
369     }
370 amb 110 }
371     }
372    
373    
374     /*++++++++++++++++++++++++++++++++++++++
375 amb 281 Index the nodes after sorting.
376    
377 amb 289 int index_by_lat_long Return 1 if the value is to be kept, otherwise zero.
378 amb 281
379     NodeX *nodex The extended node.
380    
381     index_t index The index of this node in the total.
382     ++++++++++++++++++++++++++++++++++++++*/
383    
384     static int index_by_lat_long(NodeX *nodex,index_t index)
385     {
386     /* Work out the offsets */
387    
388     ll_bin_t latbin=latlong_to_bin(nodex->latitude )-sortnodesx->latzero;
389     ll_bin_t lonbin=latlong_to_bin(nodex->longitude)-sortnodesx->lonzero;
390     int llbin=lonbin*sortnodesx->latbins+latbin;
391    
392     for(;sortnodesx->latlonbin<=llbin;sortnodesx->latlonbin++)
393     sortnodesx->offsets[sortnodesx->latlonbin]=index;
394    
395     return(1);
396     }
397    
398    
399     /*++++++++++++++++++++++++++++++++++++++
400 amb 285 Find a particular node index.
401    
402     index_t IndexNodeX Returns the index of the extended node with the specified id.
403    
404     NodesX* nodesx The set of nodes to process.
405    
406     node_t id The node id to look for.
407     ++++++++++++++++++++++++++++++++++++++*/
408    
409     index_t IndexNodeX(NodesX* nodesx,node_t id)
410     {
411     int start=0;
412     int end=nodesx->number-1;
413     int mid;
414    
415     assert(nodesx->idata); /* Must have idata filled in => sorted by id */
416    
417     /* Binary search - search key exact match only is required.
418     *
419     * # <- start | Check mid and move start or end if it doesn't match
420     * # |
421     * # | Since an exact match is wanted we can set end=mid-1
422     * # <- mid | or start=mid+1 because we know that mid doesn't match.
423     * # |
424     * # | Eventually either end=start or end=start+1 and one of
425     * # <- end | start or end is the wanted one.
426     */
427    
428     if(end<start) /* There are no nodes */
429     return(NO_NODE);
430     else if(id<nodesx->idata[start]) /* Check key is not before start */
431     return(NO_NODE);
432     else if(id>nodesx->idata[end]) /* Check key is not after end */
433     return(NO_NODE);
434     else
435     {
436     do
437     {
438     mid=(start+end)/2; /* Choose mid point */
439    
440     if(nodesx->idata[mid]<id) /* Mid point is too low */
441     start=mid+1;
442     else if(nodesx->idata[mid]>id) /* Mid point is too high */
443     end=mid-1;
444     else /* Mid point is correct */
445     return(mid);
446     }
447     while((end-start)>1);
448    
449     if(nodesx->idata[start]==id) /* Start is correct */
450     return(start);
451    
452     if(nodesx->idata[end]==id) /* End is correct */
453     return(end);
454     }
455    
456     return(NO_NODE);
457     }
458    
459    
460     /*++++++++++++++++++++++++++++++++++++++
461 amb 448 Lookup a particular extended node.
462 amb 285
463     NodeX *LookupNodeX Returns a pointer to the extended node with the specified id.
464    
465     NodesX* nodesx The set of nodes to process.
466    
467     index_t index The node index to look for.
468    
469     int position The position in the cache to use.
470     ++++++++++++++++++++++++++++++++++++++*/
471    
472     NodeX *LookupNodeX(NodesX* nodesx,index_t index,int position)
473     {
474     assert(index!=NO_NODE); /* Must be a valid node */
475    
476     if(option_slim)
477     {
478     SeekFile(nodesx->fd,index*sizeof(NodeX));
479    
480     ReadFile(nodesx->fd,&nodesx->cached[position-1],sizeof(NodeX));
481    
482     return(&nodesx->cached[position-1]);
483     }
484     else
485     {
486     return(&nodesx->xdata[index]);
487     }
488     }
489    
490    
491     /*++++++++++++++++++++++++++++++++++++++
492 amb 448 Lookup a particular extended node's normal node.
493    
494     Node *LookupNodeXNode Returns a pointer to the node with the specified id.
495    
496     NodesX* nodesx The set of nodes to process.
497    
498     index_t index The node index to look for.
499    
500     int position The position in the cache to use.
501     ++++++++++++++++++++++++++++++++++++++*/
502    
503     Node *LookupNodeXNode(NodesX* nodesx,index_t index,int position)
504     {
505     assert(index!=NO_NODE); /* Must be a valid node */
506    
507     if(option_slim)
508     {
509     SeekFile(nodesx->nfd,index*sizeof(Node));
510    
511     ReadFile(nodesx->nfd,&nodesx->ncached[position-1],sizeof(Node));
512    
513     return(&nodesx->ncached[position-1]);
514     }
515     else
516     {
517     return(&nodesx->ndata[index]);
518     }
519     }
520    
521    
522     /*++++++++++++++++++++++++++++++++++++++
523     Put back an extended node's normal node.
524    
525     NodesX* nodesx The set of nodes to process.
526    
527     index_t index The node index to look for.
528    
529     int position The position in the cache to use.
530     ++++++++++++++++++++++++++++++++++++++*/
531    
532     void PutBackNodeXNode(NodesX* nodesx,index_t index,int position)
533     {
534     assert(index!=NO_NODE); /* Must be a valid node */
535    
536     if(option_slim)
537     {
538     SeekFile(nodesx->nfd,index*sizeof(Node));
539    
540     WriteFile(nodesx->nfd,&nodesx->ncached[position-1],sizeof(Node));
541     }
542     }
543    
544    
545     /*++++++++++++++++++++++++++++++++++++++
546 amb 110 Remove any nodes that are not part of a highway.
547    
548     NodesX *nodesx The complete node list.
549    
550     SegmentsX *segmentsx The list of segments.
551     ++++++++++++++++++++++++++++++++++++++*/
552    
553     void RemoveNonHighwayNodes(NodesX *nodesx,SegmentsX *segmentsx)
554     {
555 amb 263 NodeX nodex;
556 amb 273 int total=0,highway=0,nothighway=0;
557 amb 281 ll_bin_t lat_min_bin,lat_max_bin,lon_min_bin,lon_max_bin;
558     latlong_t lat_min,lat_max,lon_min,lon_max;
559 amb 263 int fd;
560 amb 110
561 amb 263 /* Check the start conditions */
562    
563 amb 249 assert(nodesx->idata); /* Must have idata filled in => data sorted */
564 amb 213
565 amb 263 /* Print the start message */
566    
567 amb 227 printf("Checking: Nodes=0");
568     fflush(stdout);
569    
570 amb 281 /* While we are here we can work out the range of data */
571    
572     lat_min=radians_to_latlong( 2);
573     lat_max=radians_to_latlong(-2);
574     lon_min=radians_to_latlong( 4);
575     lon_max=radians_to_latlong(-4);
576    
577 amb 263 /* Modify the on-disk image */
578    
579     DeleteFile(nodesx->filename);
580    
581     fd=OpenFile(nodesx->filename);
582     SeekFile(nodesx->fd,0);
583    
584     while(!ReadFile(nodesx->fd,&nodex,sizeof(NodeX)))
585 amb 110 {
586 amb 275 if(IndexFirstSegmentX(segmentsx,nodex.id)==NO_SEGMENT)
587 amb 271 nothighway++;
588     else
589 amb 263 {
590 amb 280 nodex.id=highway;
591    
592 amb 263 WriteFile(fd,&nodex,sizeof(NodeX));
593    
594     nodesx->idata[highway]=nodesx->idata[total];
595 amb 110 highway++;
596 amb 281
597     if(nodex.latitude<lat_min)
598     lat_min=nodex.latitude;
599     if(nodex.latitude>lat_max)
600     lat_max=nodex.latitude;
601     if(nodex.longitude<lon_min)
602     lon_min=nodex.longitude;
603     if(nodex.longitude>lon_max)
604     lon_max=nodex.longitude;
605 amb 263 }
606 amb 110
607 amb 263 total++;
608    
609     if(!(total%10000))
610 amb 110 {
611 amb 273 printf("\rChecking: Nodes=%d Highway=%d not-Highway=%d",total,highway,nothighway);
612 amb 110 fflush(stdout);
613     }
614     }
615    
616 amb 263 /* Close the files and re-open them */
617    
618     CloseFile(nodesx->fd);
619     CloseFile(fd);
620    
621     nodesx->fd=ReOpenFile(nodesx->filename);
622    
623     nodesx->number=highway;
624    
625 amb 281 /* Work out the number of bins */
626    
627     lat_min_bin=latlong_to_bin(lat_min);
628     lon_min_bin=latlong_to_bin(lon_min);
629     lat_max_bin=latlong_to_bin(lat_max);
630     lon_max_bin=latlong_to_bin(lon_max);
631    
632     nodesx->latzero=lat_min_bin;
633     nodesx->lonzero=lon_min_bin;
634    
635     nodesx->latbins=(lat_max_bin-lat_min_bin)+1;
636     nodesx->lonbins=(lon_max_bin-lon_min_bin)+1;
637    
638 amb 263 /* Allocate and clear the super-node markers */
639    
640     nodesx->super=(uint8_t*)calloc(nodesx->number,sizeof(uint8_t));
641    
642     assert(nodesx->super); /* Check calloc() worked */
643    
644     /* Print the final message */
645    
646 amb 273 printf("\rChecked: Nodes=%d Highway=%d not-Highway=%d \n",total,highway,nothighway);
647 amb 110 fflush(stdout);
648     }
649    
650    
651     /*++++++++++++++++++++++++++++++++++++++
652 amb 212 Create the real node data.
653 amb 110
654 amb 212 NodesX *nodesx The set of nodes to use.
655 amb 110
656 amb 212 int iteration The final super-node iteration.
657 amb 110 ++++++++++++++++++++++++++++++++++++++*/
658    
659 amb 212 void CreateRealNodes(NodesX *nodesx,int iteration)
660 amb 110 {
661 amb 214 index_t i;
662 amb 110
663 amb 275 /* Check the start conditions */
664    
665 amb 243 assert(!nodesx->ndata); /* Must not have ndata filled in => no real nodes */
666 amb 110
667 amb 275 /* Print the start message */
668    
669 amb 227 printf("Creating Real Nodes: Nodes=0");
670     fflush(stdout);
671    
672 amb 275 /* Map into memory */
673    
674     if(!option_slim)
675     nodesx->xdata=MapFile(nodesx->filename);
676    
677 amb 448 /* Allocate the memory (or open the file) */
678 amb 212
679 amb 448 if(!option_slim)
680     {
681     nodesx->ndata=(Node*)malloc(nodesx->number*sizeof(Node));
682 amb 212
683 amb 448 assert(nodesx->ndata); /* Check malloc() worked */
684     }
685     else
686     nodesx->nfd=OpenFile(nodesx->nfilename);
687 amb 243
688 amb 212 /* Loop through and allocate. */
689    
690 amb 110 for(i=0;i<nodesx->number;i++)
691     {
692 amb 275 NodeX *nodex=LookupNodeX(nodesx,i,1);
693 amb 448 Node *node =LookupNodeXNode(nodesx,nodex->id,1);
694 amb 208
695 amb 448 node->latoffset=latlong_to_off(nodex->latitude);
696     node->lonoffset=latlong_to_off(nodex->longitude);
697     node->firstseg=SEGMENT(NO_SEGMENT);
698 amb 212
699 amb 281 if(nodesx->super[nodex->id]==iteration)
700 amb 448 node->firstseg|=NODE_SUPER;
701 amb 212
702 amb 448 if(option_slim)
703     PutBackNodeXNode(nodesx,nodex->id,1);
704    
705 amb 110 if(!((i+1)%10000))
706     {
707 amb 212 printf("\rCreating Real Nodes: Nodes=%d",i+1);
708 amb 110 fflush(stdout);
709     }
710     }
711    
712 amb 280 /* Free the unneeded memory */
713    
714     free(nodesx->super);
715     nodesx->super=NULL;
716    
717 amb 275 /* Unmap from memory */
718    
719     if(!option_slim)
720     nodesx->xdata=UnmapFile(nodesx->filename);
721    
722     /* Print the final message */
723    
724 amb 212 printf("\rCreating Real Nodes: Nodes=%d \n",nodesx->number);
725 amb 110 fflush(stdout);
726     }
727    
728    
729     /*++++++++++++++++++++++++++++++++++++++
730     Assign the segment indexes to the nodes.
731    
732     NodesX *nodesx The list of nodes to process.
733    
734     SegmentsX* segmentsx The set of segments to use.
735     ++++++++++++++++++++++++++++++++++++++*/
736    
737 amb 209 void IndexNodes(NodesX *nodesx,SegmentsX *segmentsx)
738 amb 110 {
739 amb 214 index_t i;
740 amb 110
741 amb 275 /* Check the start conditions */
742    
743 amb 448 if(!option_slim)
744     assert(nodesx->ndata); /* Must have ndata filled in => real nodes exist */
745     if(!option_slim)
746     assert(segmentsx->sdata); /* Must have sdata filled in => real segments exist */
747 amb 110
748 amb 275 /* Print the start message */
749    
750 amb 227 printf("Indexing Segments: Segments=0");
751     fflush(stdout);
752    
753 amb 275 /* Map into memory */
754    
755     if(!option_slim)
756 amb 281 {
757     nodesx->xdata=MapFile(nodesx->filename);
758 amb 275 segmentsx->xdata=MapFile(segmentsx->filename);
759 amb 281 }
760 amb 275
761 amb 110 /* Index the nodes */
762    
763     for(i=0;i<segmentsx->number;i++)
764     {
765 amb 275 SegmentX *segmentx=LookupSegmentX(segmentsx,i,1);
766 amb 257 node_t id1=segmentx->node1;
767     node_t id2=segmentx->node2;
768 amb 448 Node *node1=LookupNodeXNode(nodesx,id1,1);
769     Node *node2=LookupNodeXNode(nodesx,id2,2);
770 amb 110
771     /* Check node1 */
772    
773 amb 212 if(SEGMENT(node1->firstseg)==SEGMENT(NO_SEGMENT))
774 amb 110 {
775 amb 212 node1->firstseg^=SEGMENT(NO_SEGMENT);
776     node1->firstseg|=i;
777 amb 448
778     if(option_slim)
779     PutBackNodeXNode(nodesx,id1,1);
780 amb 110 }
781     else
782     {
783 amb 212 index_t index=SEGMENT(node1->firstseg);
784 amb 110
785     do
786     {
787 amb 275 segmentx=LookupSegmentX(segmentsx,index,1);
788 amb 256
789 amb 257 if(segmentx->node1==id1)
790 amb 110 {
791 amb 209 index++;
792 amb 128
793 amb 256 if(index>=segmentsx->number)
794 amb 209 break;
795 amb 256
796 amb 275 segmentx=LookupSegmentX(segmentsx,index,1);
797 amb 256
798 amb 257 if(segmentx->node1!=id1)
799 amb 256 break;
800 amb 110 }
801     else
802     {
803 amb 448 Segment *segment=LookupSegmentXSegment(segmentsx,index,1);
804    
805     if(segment->next2==NO_NODE)
806 amb 110 {
807 amb 448 segment->next2=i;
808    
809     PutBackSegmentXSegment(segmentsx,index,1);
810    
811 amb 209 break;
812 amb 110 }
813     else
814 amb 448 index=segment->next2;
815 amb 110 }
816     }
817 amb 209 while(1);
818 amb 110 }
819    
820     /* Check node2 */
821    
822 amb 212 if(SEGMENT(node2->firstseg)==SEGMENT(NO_SEGMENT))
823 amb 110 {
824 amb 212 node2->firstseg^=SEGMENT(NO_SEGMENT);
825     node2->firstseg|=i;
826 amb 448
827     if(option_slim)
828     PutBackNodeXNode(nodesx,id2,2);
829 amb 110 }
830     else
831     {
832 amb 212 index_t index=SEGMENT(node2->firstseg);
833 amb 110
834     do
835     {
836 amb 275 segmentx=LookupSegmentX(segmentsx,index,1);
837 amb 256
838 amb 257 if(segmentx->node1==id2)
839 amb 110 {
840 amb 209 index++;
841 amb 128
842 amb 256 if(index>=segmentsx->number)
843 amb 209 break;
844 amb 256
845 amb 275 segmentx=LookupSegmentX(segmentsx,index,1);
846 amb 256
847 amb 257 if(segmentx->node1!=id2)
848 amb 256 break;
849 amb 110 }
850     else
851     {
852 amb 448 Segment *segment=LookupSegmentXSegment(segmentsx,index,1);
853    
854     if(segment->next2==NO_NODE)
855 amb 110 {
856 amb 448 segment->next2=i;
857    
858     PutBackSegmentXSegment(segmentsx,index,1);
859    
860 amb 209 break;
861 amb 110 }
862     else
863 amb 448 index=segment->next2;
864 amb 110 }
865     }
866 amb 209 while(1);
867 amb 110 }
868    
869     if(!((i+1)%10000))
870     {
871     printf("\rIndexing Segments: Segments=%d",i+1);
872     fflush(stdout);
873     }
874     }
875    
876 amb 275 /* Unmap from memory */
877    
878     if(!option_slim)
879 amb 281 {
880     nodesx->xdata=UnmapFile(nodesx->filename);
881 amb 275 segmentsx->xdata=UnmapFile(segmentsx->filename);
882 amb 281 }
883 amb 275
884     /* Print the final message */
885    
886 amb 110 printf("\rIndexed Segments: Segments=%d \n",segmentsx->number);
887     fflush(stdout);
888     }
889 amb 285
890    
891     /*++++++++++++++++++++++++++++++++++++++
892     Save the node list to a file.
893    
894     NodesX* nodesx The set of nodes to save.
895    
896     const char *filename The name of the file to save.
897     ++++++++++++++++++++++++++++++++++++++*/
898    
899     void SaveNodeList(NodesX* nodesx,const char *filename)
900     {
901     index_t i;
902     int fd;
903     Nodes *nodes;
904     int super_number=0;
905    
906     /* Check the start conditions */
907    
908 amb 448 if(!option_slim)
909     assert(nodesx->ndata); /* Must have ndata filled in => real nodes exist */
910 amb 285
911     /* Print the start message */
912    
913     printf("Writing Nodes: Nodes=0");
914     fflush(stdout);
915    
916     /* Map into memory */
917    
918     if(!option_slim)
919     nodesx->xdata=MapFile(nodesx->filename);
920    
921     /* Count the number of super-nodes */
922    
923     for(i=0;i<nodesx->number;i++)
924 amb 448 {
925     Node *node=LookupNodeXNode(nodesx,i,1);
926    
927     if(node->firstseg&NODE_SUPER)
928 amb 285 super_number++;
929 amb 448 }
930 amb 285
931     /* Fill in a Nodes structure with the offset of the real data in the file after
932     the Node structure itself. */
933    
934     nodes=calloc(1,sizeof(Nodes));
935    
936     assert(nodes); /* Check calloc() worked */
937    
938     nodes->number=nodesx->number;
939     nodes->snumber=super_number;
940    
941     nodes->latbins=nodesx->latbins;
942     nodes->lonbins=nodesx->lonbins;
943    
944     nodes->latzero=nodesx->latzero;
945     nodes->lonzero=nodesx->lonzero;
946    
947     nodes->data=NULL;
948 amb 385 nodes->offsets=NULL;
949     nodes->nodes=NULL;
950 amb 285
951     /* Write out the Nodes structure and then the real data. */
952    
953     fd=OpenFile(filename);
954    
955     WriteFile(fd,nodes,sizeof(Nodes));
956    
957     WriteFile(fd,nodesx->offsets,(nodesx->latbins*nodesx->lonbins+1)*sizeof(index_t));
958    
959     for(i=0;i<nodes->number;i++)
960     {
961     NodeX *nodex=LookupNodeX(nodesx,i,1);
962 amb 448 Node *node=LookupNodeXNode(nodesx,nodex->id,1);
963 amb 285
964     WriteFile(fd,node,sizeof(Node));
965    
966     if(!((i+1)%10000))
967     {
968     printf("\rWriting Nodes: Nodes=%d",i+1);
969     fflush(stdout);
970     }
971     }
972    
973     CloseFile(fd);
974    
975     /* Unmap from memory */
976    
977     if(!option_slim)
978     nodesx->xdata=UnmapFile(nodesx->filename);
979    
980     /* Print the final message */
981    
982     printf("\rWrote Nodes: Nodes=%d \n",nodes->number);
983     fflush(stdout);
984    
985     /* Free the fake Nodes */
986    
987     free(nodes);
988     }

Properties

Name Value
cvs:description Extended nodes functions.