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

Parent Directory Parent Directory | Revision Log Revision Log


Revision 510 - (show annotations) (download) (as text)
Sat Oct 9 14:14:42 2010 UTC (14 years, 5 months ago) by amb
File MIME type: text/x-csrc
File size: 8396 byte(s)
Fix previous check-in on this set of files.

1 /***************************************
2 $Header: /home/amb/CVS/routino/src/relationsx.c,v 1.7 2010-10-09 14:14:42 amb Exp $
3
4 Extended Relation data type functions.
5
6 Part of the Routino routing software.
7 ******************/ /******************
8 This file Copyright 2010 Andrew M. Bishop
9
10 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 ***************************************/
23
24
25 #include <assert.h>
26 #include <stdlib.h>
27 #include <stdio.h>
28 #include <string.h>
29 #include <sys/stat.h>
30
31 #include "waysx.h"
32 #include "relationsx.h"
33
34 #include "files.h"
35 #include "functions.h"
36
37
38 /* Variables */
39
40 /*+ The command line '--tmpdir' option or its default value. +*/
41 extern char *option_tmpdirname;
42
43
44 /*++++++++++++++++++++++++++++++++++++++
45 Allocate a new relation list (create a new file or open an existing one).
46
47 RelationsX *NewRelationList Returns the relation list.
48
49 int append Set to 1 if the file is to be opened for appending (now or later).
50 ++++++++++++++++++++++++++++++++++++++*/
51
52 RelationsX *NewRelationList(int append)
53 {
54 RelationsX *relationsx;
55
56 relationsx=(RelationsX*)calloc(1,sizeof(RelationsX));
57
58 assert(relationsx); /* Check calloc() worked */
59
60 relationsx->rfilename=(char*)malloc(strlen(option_tmpdirname)+32);
61
62 if(append)
63 sprintf(relationsx->rfilename,"%s/relationsx.route.input.tmp",option_tmpdirname);
64 else
65 sprintf(relationsx->rfilename,"%s/relationsx.route.%p.tmp",option_tmpdirname,relationsx);
66
67 if(append)
68 {
69 off_t size,position=0;
70
71 relationsx->rfd=OpenFileAppend(relationsx->rfilename);
72
73 size=SizeFile(relationsx->rfilename);
74
75 while(position<size)
76 {
77 FILESORT_VARINT relationsize;
78
79 SeekFile(relationsx->rfd,position);
80 ReadFile(relationsx->rfd,&relationsize,FILESORT_VARSIZE);
81
82 relationsx->rxnumber++;
83 position+=relationsize+FILESORT_VARSIZE;
84 }
85
86 SeekFile(relationsx->rfd,size);
87 }
88 else
89 relationsx->rfd=OpenFileNew(relationsx->rfilename);
90
91 return(relationsx);
92 }
93
94
95 /*++++++++++++++++++++++++++++++++++++++
96 Free a relation list.
97
98 RelationsX *relationsx The list to be freed.
99
100 int keep Set to 1 if the file is to be kept.
101 ++++++++++++++++++++++++++++++++++++++*/
102
103 void FreeRelationList(RelationsX *relationsx,int keep)
104 {
105 if(!keep)
106 DeleteFile(relationsx->rfilename);
107
108 free(relationsx->rfilename);
109
110 free(relationsx);
111 }
112
113
114 /*++++++++++++++++++++++++++++++++++++++
115 Append a single relation to an unsorted route relation list.
116
117 RelationsX* relationsx The set of relations to process.
118
119 relation_t id The ID of the relation.
120
121 allow_t routes The types of routes that this relation is for.
122
123 way_t *ways The array of ways that are members of the relation.
124
125 int nways The number of ways that are members of the relation.
126
127 relation_t *relations The array of relations that are members of the relation.
128
129 int nrelations The number of relations that are members of the relation.
130 ++++++++++++++++++++++++++++++++++++++*/
131
132 void AppendRouteRelation(RelationsX* relationsx,relation_t id,allow_t routes,
133 way_t *ways,int nways,
134 relation_t *relations,int nrelations)
135 {
136 RouteRelX relationx;
137 FILESORT_VARINT size;
138 way_t zeroway=0;
139 relation_t zerorelation=0;
140
141 relationx.id=id;
142 relationx.routes=routes;
143
144 size=sizeof(RouteRelX)+(nways+1)*sizeof(way_t)+(nrelations+1)*sizeof(relation_t);
145
146 WriteFile(relationsx->rfd,&size,FILESORT_VARSIZE);
147 WriteFile(relationsx->rfd,&relationx,sizeof(RouteRelX));
148
149 WriteFile(relationsx->rfd,ways ,nways*sizeof(way_t));
150 WriteFile(relationsx->rfd,&zeroway, sizeof(way_t));
151
152 WriteFile(relationsx->rfd,relations ,nrelations*sizeof(relation_t));
153 WriteFile(relationsx->rfd,&zerorelation, sizeof(relation_t));
154
155 relationsx->rxnumber++;
156
157 assert(!(relationsx->rxnumber==0)); /* Zero marks the high-water mark for relations. */
158 }
159
160
161 /*++++++++++++++++++++++++++++++++++++++
162 Sort the list of relations.
163
164 RelationsX* relationsx The set of relations to process.
165 ++++++++++++++++++++++++++++++++++++++*/
166
167 void SortRelationList(RelationsX* relationsx)
168 {
169 /* Don't need to sort route relations */
170 }
171
172
173 /*++++++++++++++++++++++++++++++++++++++
174 Process the route relations and apply the information to the ways.
175
176 RelationsX *relationsx The set of relations to process.
177
178 WaysX *waysx The set of ways to update.
179 ++++++++++++++++++++++++++++++++++++++*/
180
181 void ProcessRouteRelations(RelationsX *relationsx,WaysX *waysx)
182 {
183 RouteRelX *unmatched=NULL,*lastunmatched=NULL;
184 int nunmatched=0,lastnunmatched=0,iteration=0;
185 int i,j;
186
187 if(waysx->number==0)
188 return;
189
190 /* Map into memory */
191
192 #if !SLIM
193 waysx->xdata=MapFileWriteable(waysx->filename);
194 #endif
195
196 /* Open the file and read through it */
197
198 relationsx->rfd=ReOpenFile(relationsx->rfilename);
199
200 do
201 {
202 SeekFile(relationsx->rfd,0);
203
204 /* Print the start message */
205
206 printf("Processing Route Relations: Iteration=%d Relations=0",iteration);
207 fflush(stdout);
208
209 for(i=0;i<relationsx->rxnumber;i++)
210 {
211 FILESORT_VARINT size;
212 RouteRelX relationx;
213 way_t wayid;
214 relation_t relationid;
215 allow_t routes=Allow_None;
216
217 /* Read each route relation */
218
219 ReadFile(relationsx->rfd,&size,FILESORT_VARSIZE);
220 ReadFile(relationsx->rfd,&relationx,sizeof(RouteRelX));
221
222 /* Decide what type of route it is */
223
224 if(iteration==0)
225 routes=relationx.routes;
226 else
227 {
228 if((lastunmatched[j].routes|relationx.routes)==relationx.routes)
229 routes=0; /* Nothing new to add */
230 else
231 for(j=0;j<lastnunmatched;j++)
232 if(lastunmatched[j].id==relationx.id)
233 {
234 routes=lastunmatched[j].routes;
235 break;
236 }
237 }
238
239 /* Loop through the ways */
240
241 do
242 {
243 ReadFile(relationsx->rfd,&wayid,sizeof(way_t));
244
245 /* Update the ways that are listed for the relation */
246
247 if(wayid && routes)
248 {
249 index_t way=IndexWayX(waysx,wayid);
250
251 if(way!=NO_WAY)
252 {
253 WayX *wayx=LookupWayX(waysx,way,1);
254
255 if(routes&Allow_Foot)
256 wayx->way.props|=Properties_FootRoute;
257
258 if(routes&Allow_Bicycle)
259 wayx->way.props|=Properties_BicycleRoute;
260
261 #if SLIM
262 PutBackWayX(waysx,way,1);
263 #endif
264 }
265 }
266 }
267 while(wayid);
268
269 /* Loop through the relations */
270
271 do
272 {
273 ReadFile(relationsx->rfd,&relationid,sizeof(relation_t));
274
275 /* Add the relations that are listed for this relation to the list for next time */
276
277 if(relationid && routes && relationid!=relationx.id)
278 {
279 if(nunmatched%256==0)
280 unmatched=(RouteRelX*)realloc((void*)unmatched,(nunmatched+256)*sizeof(RouteRelX));
281
282 unmatched[nunmatched].id=relationid;
283 unmatched[nunmatched].routes=routes;
284
285 nunmatched++;
286 }
287 }
288 while(relationid);
289
290 if(!((i+1)%10000))
291 {
292 printf("\rProcessing Route Relations: Iteration=%d Relations=%d",iteration,i+1);
293 fflush(stdout);
294 }
295 }
296
297 if(lastunmatched)
298 free(lastunmatched);
299
300 lastunmatched=unmatched;
301 lastnunmatched=nunmatched;
302
303 unmatched=NULL;
304 nunmatched=0;
305
306 /* Print the final message */
307
308 printf("\rProcessed Route Relations: Iteration=%d Relations=%d \n",iteration,relationsx->rxnumber);
309 fflush(stdout);
310 }
311 while(lastnunmatched && ++iteration<5);
312
313 if(lastunmatched)
314 free(lastunmatched);
315
316 CloseFile(relationsx->rfd);
317
318 /* Unmap from memory */
319
320 #if !SLIM
321 waysx->xdata=UnmapFile(waysx->filename);
322 #endif
323 }