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

Parent Directory Parent Directory | Revision Log Revision Log


Revision 17 - (show annotations) (download) (as text)
Wed Jan 7 19:21:06 2009 UTC (16 years, 3 months ago) by amb
File MIME type: text/x-csrc
File size: 5627 byte(s)
Lots of modifications:
Two data structures - in memory (pointers) and in file (array).
Data is hashed into multiple bins.
Each function takes a nodes structure as an argument.

1 /***************************************
2 $Header: /home/amb/CVS/routino/src/nodes.c,v 1.4 2009-01-07 19:21:06 amb Exp $
3
4 Node data type functions.
5 ******************/ /******************
6 Written by Andrew M. Bishop
7
8 This file Copyright 2008,2009 Andrew M. Bishop
9 It may be distributed under the GNU Public License, version 2, or
10 any higher version. See section COPYING of the GNU Public license
11 for conditions under which this file may be redistributed.
12 ***************************************/
13
14
15 #include <assert.h>
16 #include <stdlib.h>
17
18 #include "functions.h"
19 #include "nodes.h"
20
21
22 /* Functions */
23
24 static int sort_by_id(Node *a,Node *b);
25
26
27 /*++++++++++++++++++++++++++++++++++++++
28 Allocate a new node list.
29
30 NodesMem *NewNodeList Returns the node list.
31 ++++++++++++++++++++++++++++++++++++++*/
32
33 NodesMem *NewNodeList(void)
34 {
35 NodesMem *nodes;
36 int i;
37
38 nodes=(NodesMem*)calloc(sizeof(NodesMem),1);
39
40 nodes->alloced=INCREMENT_NODES;
41
42 for(i=0;i<NBINS_NODES;i++)
43 nodes->bins[i]=(Node*)malloc(nodes->alloced*sizeof(Node));
44
45 return(nodes);
46 }
47
48
49 /*++++++++++++++++++++++++++++++++++++++
50 Load in a node list from a file.
51
52 NodesFile* LoadNodeList Returns the node list.
53
54 const char *filename The name of the file to load.
55 ++++++++++++++++++++++++++++++++++++++*/
56
57 NodesFile *LoadNodeList(const char *filename)
58 {
59 return((NodesFile*)MapFile(filename));
60 }
61
62
63 /*++++++++++++++++++++++++++++++++++++++
64 Save the node list to a file.
65
66 NodesFile* SaveNodeList Returns the node list that has just been saved.
67
68 NodesMem* nodesm The set of nodes to save.
69
70 const char *filename The name of the file to save.
71 ++++++++++++++++++++++++++++++++++++++*/
72
73 NodesFile *SaveNodeList(NodesMem* nodesm,const char *filename)
74 {
75 NodesFile nodesf;
76 int i;
77
78 assert(nodesm->sorted); /* Must be sorted */
79
80 nodesf.offset[0]=0;
81 for(i=1;i<=NBINS_NODES;i++)
82 nodesf.offset[i]=nodesf.offset[i-1]+nodesm->number[i-1];
83
84 if(WriteFile(filename,(void*)&nodesf,sizeof(nodesf.offset),0))
85 assert(0);
86
87 for(i=0;i<NBINS_NODES;i++)
88 if(WriteFile(filename,(void*)nodesm->bins[i],nodesm->number[i]*sizeof(Node),1))
89 assert(0);
90
91 for(i=0;i<NBINS_NODES;i++)
92 free(nodesm->bins[i]);
93
94 free(nodesm);
95
96 return(LoadNodeList(filename));
97 }
98
99
100 /*++++++++++++++++++++++++++++++++++++++
101 Find a particular node.
102
103 Node *FindNode Returns a pointer to the node with the specified id.
104
105 NodesFile* nodes The set of nodes to process.
106
107 node_t id The node id to look for.
108 ++++++++++++++++++++++++++++++++++++++*/
109
110 Node *FindNode(NodesFile* nodes,node_t id)
111 {
112 int bin=id%NBINS_NODES;
113 int start=nodes->offset[bin];
114 int end=nodes->offset[bin+1]-1;
115 int mid;
116
117 /* Binary search - search key exact match only is required.
118 *
119 * # <- start | Check mid and move start or end if it doesn't match
120 * # |
121 * # | Since an exact match is wanted we can set end=mid-1
122 * # <- mid | or start=mid+1 because we know that mid doesn't match.
123 * # |
124 * # | Eventually either end=start or end=start+1 and one of
125 * # <- end | start or end is the wanted one.
126 */
127
128 if(end<start) /* There are no nodes */
129 return(NULL);
130 else if(id<nodes->nodes[start].id) /* Check key is not before start */
131 return(NULL);
132 else if(id>nodes->nodes[end].id) /* Check key is not after end */
133 return(NULL);
134 else
135 {
136 do
137 {
138 mid=(start+end)/2; /* Choose mid point */
139
140 if(nodes->nodes[mid].id<id) /* Mid point is too low */
141 start=mid+1;
142 else if(nodes->nodes[mid].id>id) /* Mid point is too high */
143 end=mid-1;
144 else /* Mid point is correct */
145 return(&nodes->nodes[mid]);
146 }
147 while((end-start)>1);
148
149 if(nodes->nodes[start].id==id) /* Start is correct */
150 return(&nodes->nodes[start]);
151
152 if(nodes->nodes[end].id==id) /* End is correct */
153 return(&nodes->nodes[end]);
154 }
155
156 return(NULL);
157 }
158
159
160 /*++++++++++++++++++++++++++++++++++++++
161 Append a node to a newly created node list (unsorted).
162
163 NodesMem* nodes The set of nodes to process.
164
165 node_t id The node identification.
166
167 latlong_t latitude The latitude of the node.
168
169 latlong_t longitude The longitude of the node.
170 ++++++++++++++++++++++++++++++++++++++*/
171
172 void AppendNode(NodesMem* nodes,node_t id,latlong_t latitude,latlong_t longitude)
173 {
174 int i;
175 int bin=id%NBINS_NODES;
176
177 /* Check that the arrays have enough space. */
178
179 if(nodes->number[bin]==nodes->alloced)
180 {
181 nodes->alloced+=INCREMENT_NODES;
182
183 for(i=0;i<NBINS_NODES;i++)
184 nodes->bins[i]=(Node*)realloc((void*)nodes->bins[i],nodes->alloced*sizeof(Node));
185 }
186
187 /* Insert the node */
188
189 nodes->bins[bin][nodes->number[bin]].id=id;
190 nodes->bins[bin][nodes->number[bin]].latitude=latitude;
191 nodes->bins[bin][nodes->number[bin]].longitude=longitude;
192
193 nodes->number[bin]++;
194
195 nodes->sorted=0;
196 }
197
198
199 /*++++++++++++++++++++++++++++++++++++++
200 Sort the node list.
201
202 NodesMem* nodes The set of nodes to process.
203 ++++++++++++++++++++++++++++++++++++++*/
204
205 void SortNodeList(NodesMem* nodes)
206 {
207 int i;
208
209 for(i=0;i<NBINS_NODES;i++)
210 qsort(nodes->bins[i],nodes->number[i],sizeof(Node),(int (*)(const void*,const void*))sort_by_id);
211
212 nodes->sorted=1;
213 }
214
215
216 /*++++++++++++++++++++++++++++++++++++++
217 Sort the nodes into id order.
218
219 int sort_by_id Returns the comparison of the id fields.
220
221 Node *a The first Node.
222
223 Node *b The second Node.
224 ++++++++++++++++++++++++++++++++++++++*/
225
226 static int sort_by_id(Node *a,Node *b)
227 {
228 node_t a_id=a->id;
229 node_t b_id=b->id;
230
231 return(a_id-b_id);
232 }

Properties

Name Value
cvs:description Node data type.