source: subversion/applications/utils/mod_tile/render_expired.c @ 28718

Last change on this file since 28718 was 28680, checked in by apmon, 7 years ago

[mod_tile] bzero is defined in <strings.h> but was not included

On Linux strings.h must have been included through some other header,
but on solaris it isn't. So it needs to explicitly be included.

File size: 18.8 KB
Line 
1/**
2 * A modified version of render_list.c that does the following:
3 *
4 * - read list of expired tiles from stdin
5 * - calculate a list of all meta tiles between minZoom and maxZoom
6 *   affected by that expiry
7 * - for all expired meta tiles that are actually present on disk,
8 *   issue a re-render request to renderd
9 *
10 * If you run a tile server that servers z0-z17, it makes sense to run
11 * osm2pgsql with "-e14-14" (i.e. three zoom levels lower than your max)
12 * and then run this script with maxZoom=17. Due to z17 metatiles being
13 * exactly the area of a z14 tile, your expiry list just becomes unnecessarily
14 * large if you use -e17-17 (although this would lead to the same tiles
15 * being re-rendered).
16 *
17 * Be careful about minZoom; if you set it to 0, the tile x=0,y=0,z=0 will
18 * be expired every time the script is run. Having minZoom somewhere between
19 * z8 and z12 probably makes sense, and then use another, time-based mechanism
20 * to expire tiles.
21 *
22 * NOTE: format on stdin is one tile per line, formatted "z/x/y". This is different
23 * from render_list which wants "x y z". "z/x/y" is the format written by osm2pgsql.
24 *
25 * See also
26 * https://subversion.nexusuk.org/trac/browser/openpistemap/trunk/scripts/expire_tiles.py
27 * for a database-backed expiry solution, or
28 * http://trac.openstreetmap.org/browser/applications/utils/export/tile_expiry
29 * for a solution that works outside of osm2pgsql.
30 *
31 * This program made by Frederik Ramm <frederik@remote.org>. My ideas and
32 * contributions are in the public domain but being based on GPL'ed code
33 * this program inherits the GPL.
34 */
35
36#include <stdio.h>
37#include <stdlib.h>
38#include <unistd.h>
39#include <sys/types.h>
40#include <sys/socket.h>
41#include <sys/stat.h>
42#include <sys/time.h>
43#include <sys/un.h>
44#include <getopt.h>
45#include <time.h>
46#include <utime.h>
47#include <string.h>
48#include <strings.h>
49#include <limits.h>
50#include <utime.h>
51
52#include <pthread.h>
53
54#include "protocol.h"
55#include "render_config.h"
56#include "dir_utils.h"
57
58char *tile_dir = HASH_PATH;
59
60// macros handling our tile marking arrays (these are essentially bit arrays
61// that have one bit for each tile on the repsective zoom level; since we only
62// need them for meta tile levels, even if someone were to render level 20,
63// we'd still only use 4^17 bits = 2 GB RAM (plus a little for the lower zoom
64// levels) - this saves us the hassle of working with a tree structure.
65
66#define TILE_REQUESTED(z,x,y) \
67   (tile_requested[z][((x)*twopow[z]+(y))/(8*sizeof(int))]>>(((x)*twopow[z]+(y))%(8*sizeof(int))))&0x01
68#define SET_TILE_REQUESTED(z,x,y) \
69   tile_requested[z][((x)*twopow[z]+(y))/(8*sizeof(int))] |= (0x01 << (((x)*twopow[z]+(y))%(8*sizeof(int))));
70
71
72#ifndef METATILE
73#warning("render_expired not implemented for non-metatile mode. Feel free to submit fix")
74int main(int argc, char **argv)
75{
76    fprintf(stderr, "render_expired not implemented for non-metatile mode. Feel free to submit fix!\n");
77    return -1;
78}
79#else
80
81// tile marking arrays
82unsigned int **tile_requested;
83
84// "two raised to the power of [...]" - don't trust pow() to be efficient
85// for base 2
86unsigned long long twopow[18];
87
88static int minZoom = 0;
89static int maxZoom = MAX_ZOOM;
90static int verbose = 0;
91int work_complete;
92
93void display_rate(struct timeval start, struct timeval end, int num) 
94{
95    int d_s, d_us;
96    float sec;
97
98    d_s  = end.tv_sec  - start.tv_sec;
99    d_us = end.tv_usec - start.tv_usec;
100
101    sec = d_s + d_us / 1000000.0;
102
103    printf("Rendered %d tiles in %.2f seconds (%.2f tiles/s)\n", num, sec, num / sec);
104    fflush(NULL);
105}
106
107int process_loop(int fd, const char *mapname, int x, int y, int z)
108{
109    struct protocol cmd, rsp;
110    //struct pollfd fds[1];
111    int ret = 0;
112
113    bzero(&cmd, sizeof(cmd));
114
115    cmd.ver = 2;
116    cmd.cmd = cmdRenderBulk;
117    cmd.z = z;
118    cmd.x = x;
119    cmd.y = y;
120    strcpy(cmd.xmlname, mapname);
121
122    //printf("Sending request\n");
123    ret = send(fd, &cmd, sizeof(cmd), 0);
124    if (ret != sizeof(cmd)) {
125        perror("send error");
126    }
127
128    //printf("Waiting for response\n");
129    bzero(&rsp, sizeof(rsp));
130    ret = recv(fd, &rsp, sizeof(rsp), 0);
131    if (ret != sizeof(rsp)) 
132    {
133        perror("recv error");
134        return 0;
135    }
136    //printf("Got response\n");
137
138    if (rsp.cmd != cmdDone) 
139    {
140        printf("rendering failed, pausing\n");
141        sleep(10);
142    }
143
144    if (!ret)
145        perror("Socket send error");
146    return ret;
147}
148
149void process(const char *tilepath, int fd, const char *name)
150{
151    char xmlconfig[XMLCONFIG_MAX];
152    int x, y, z;
153
154    if (path_to_xyz(tilepath, name, xmlconfig, &x, &y, &z))
155        return;
156
157    printf("Requesting xml(%s) x(%d) y(%d) z(%d)\n", xmlconfig, x, y, z);
158    process_loop(fd, xmlconfig, x, y, z);
159}
160
161#define QMAX 32
162
163pthread_mutex_t qLock;
164pthread_cond_t qCondNotEmpty;
165pthread_cond_t qCondNotFull;
166
167unsigned int qLen;
168struct qItem {
169    char *path;
170    struct qItem *next;
171};
172
173struct qItem *qHead, *qTail;
174
175char *fetch(void)
176{
177    // Fetch path to render from queue or NULL on work completion
178    // Must free() pointer after use
179    char *path;
180
181    pthread_mutex_lock(&qLock);
182
183    while (qLen == 0) {
184        if (work_complete) {
185            pthread_mutex_unlock(&qLock);
186            return NULL;
187        }
188        pthread_cond_wait(&qCondNotEmpty, &qLock);
189    }
190
191    // Fetch item from queue
192    if (!qHead) {
193        fprintf(stderr, "Queue failure, null qHead with %d items in list\n", qLen);
194        exit(1);
195    }
196    path = qHead->path;
197
198    if (qHead == qTail) {
199        free(qHead);
200        qHead = NULL;
201        qTail = NULL;
202        qLen = 0;
203    } else {
204        struct qItem *e = qHead;
205        qHead = qHead->next;
206        free(e);
207        qLen--;
208        pthread_cond_signal(&qCondNotFull);
209    }
210
211    pthread_mutex_unlock(&qLock);
212    return path;
213}
214
215void enqueue(const char *path)
216{
217    // Add this path in the local render queue
218    struct qItem *e = malloc(sizeof(struct qItem));
219
220    e->path = strdup(path);
221    e->next = NULL;
222
223    if (!e->path) {
224        fprintf(stderr, "Malloc failure\n");
225        exit(1);
226    }
227
228    pthread_mutex_lock(&qLock);
229
230    while (qLen == QMAX) 
231    {
232        pthread_cond_wait(&qCondNotFull, &qLock);
233    }
234
235    // Append item to end of queue
236    if (qTail)
237        qTail->next = e;
238    else
239        qHead = e;
240    qTail = e;
241    qLen++;
242    pthread_cond_signal(&qCondNotEmpty);
243
244    pthread_mutex_unlock(&qLock);
245}
246
247
248void *thread_main(void *arg)
249{
250    const char *spath = (const char *)arg;
251    int fd;
252    struct sockaddr_un addr;
253    char *tile;
254
255    fd = socket(PF_UNIX, SOCK_STREAM, 0);
256    if (fd < 0) {
257        fprintf(stderr, "failed to create unix socket\n");
258        exit(2);
259    }
260
261    bzero(&addr, sizeof(addr));
262    addr.sun_family = AF_UNIX;
263    strncpy(addr.sun_path, spath, sizeof(addr.sun_path));
264
265    if (connect(fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
266        fprintf(stderr, "socket connect failed for: %s\n", spath);
267        close(fd);
268        return NULL;
269    }
270
271    while(1) {
272        if (!(tile = fetch())) break;
273        process(tile_dir, fd, tile);
274        free(tile);
275    }
276
277    close(fd);
278
279    return NULL;
280}
281
282
283pthread_t *workers;
284
285void spawn_workers(int num, const char *spath)
286{
287    int i;
288
289    // Setup request queue
290    pthread_mutex_init(&qLock, NULL);
291    pthread_cond_init(&qCondNotEmpty, NULL);
292    pthread_cond_init(&qCondNotFull, NULL);
293
294    printf("Starting %d rendering threads\n", num);
295    workers = calloc(sizeof(pthread_t), num);
296    if (!workers) {
297        perror("Error allocating worker memory");
298        exit(1);
299    }
300    for(i=0; i<num; i++) {
301        if (pthread_create(&workers[i], NULL, thread_main, (void *)spath)) {
302            perror("Thread creation failed");
303            exit(1);
304        }
305    }
306}
307
308void finish_workers(int num)
309{
310    int i;
311
312    printf("Waiting for rendering threads to finish\n");
313    pthread_mutex_lock(&qLock);
314    work_complete = 1;
315    pthread_mutex_unlock(&qLock);
316    pthread_cond_broadcast(&qCondNotEmpty);
317
318    for(i=0; i<num; i++)
319        pthread_join(workers[i], NULL);
320    free(workers);
321    workers = NULL;
322}
323
324
325int main(int argc, char **argv)
326{
327    char *spath = RENDER_SOCKET;
328    char *mapname = XMLCONFIG_DEFAULT;
329    int x, y, z;
330    char name[PATH_MAX];
331    struct timeval start, end;
332    int num_render = 0, num_all = 0, num_read = 0, num_ignore = 0, num_unlink = 0, num_touch = 0;
333    int c;
334    int all=0;
335    int numThreads = 1;
336    int deleteFrom = -1;
337    int touchFrom = -1;
338    int doRender = 0;
339    int i;
340
341    // Initialize touchFrom timestamp
342    struct utimbuf touchTime;
343    touchTime.actime = 946681200;
344    touchTime.modtime = 946681200; // Jan 1 00:00 2000
345
346    // excess_zoomlevels is how many zoom levels at the large end
347    // we can ignore because their tiles will share one meta tile.
348    // with the default METATILE==8 this is 3.
349    int excess_zoomlevels = 0;
350    int mt = METATILE;
351    while (mt > 1)
352    {
353        excess_zoomlevels++;
354        mt >>= 1; 
355    }
356
357    // initialise arrays for tile markings
358
359    tile_requested = (unsigned int **) malloc((maxZoom - excess_zoomlevels + 1)*sizeof(unsigned int *));
360
361    for (i=0; i<=maxZoom - excess_zoomlevels; i++)
362    {
363        // initialize twopow array
364        twopow[i] = (i==0) ? 1 : twopow[i-1]*2;
365        unsigned long long fourpow=twopow[i]*twopow[i];
366        tile_requested[i] = (unsigned int *) malloc((fourpow / METATILE) + 1);
367        if (NULL == tile_requested[i])
368        {
369            fprintf(stderr, "not enough memory available.\n");
370            return 1;
371        }
372        memset(tile_requested[i], 0, (fourpow / METATILE) + 1);
373    }
374
375    while (1) 
376    {
377        int option_index = 0;
378        static struct option long_options[] = 
379        {
380            {"min-zoom", 1, 0, 'z'},
381            {"max-zoom", 1, 0, 'Z'},
382            {"socket", 1, 0, 's'},
383            {"num-threads", 1, 0, 'n'},
384            {"delete-from", 1, 0, 'd'},
385            {"touch-from", 1, 0, 'T'},
386            {"tile-dir", 1, 0, 't'},
387            {"map", 1, 0, 'm'},
388            {"verbose", 0, 0, 'v'},
389            {"help", 0, 0, 'h'},
390            {0, 0, 0, 0}
391        };
392
393        c = getopt_long(argc, argv, "hvz:Z:s:m:t:n:", long_options, &option_index);
394
395        if (c == -1)
396            break;
397
398        switch (c) {
399            case 'a':   /* -a, --all */
400                all=1;
401                break;
402            case 's':   /* -s, --socket */
403                spath = strdup(optarg);
404                break;
405            case 't':   /* -t, --tile-dir */
406                tile_dir=strdup(optarg);
407                break;
408            case 'm':   /* -m, --map */
409                mapname=strdup(optarg);
410                break;
411            case 'n':   /* -n, --num-threads */
412                numThreads=atoi(optarg);
413                if (numThreads <= 0) {
414                    fprintf(stderr, "Invalid number of threads, must be at least 1\n");
415                    return 1;
416                }
417                break;
418            case 'd':   /* -d, --delete-from */
419                deleteFrom=atoi(optarg);
420                if (deleteFrom < 0 || deleteFrom > MAX_ZOOM) 
421                {
422                    fprintf(stderr, "Invalid 'delete-from' zoom, must be between 0 and %d\n", MAX_ZOOM);
423                    return 1;
424                }
425                break;
426            case 'T':   /* -T, --touch-from */
427                touchFrom=atoi(optarg);
428                if (touchFrom < 0 || touchFrom > MAX_ZOOM)
429                {
430                    fprintf(stderr, "Invalid 'touch-from' zoom, must be between 0 and %d\n", MAX_ZOOM);
431                    return 1;
432                }
433                break;
434            case 'z':   /* -z, --min-zoom */
435                minZoom=atoi(optarg);
436                if (minZoom < 0 || minZoom > MAX_ZOOM) {
437                    fprintf(stderr, "Invalid minimum zoom selected, must be between 0 and %d\n", MAX_ZOOM);
438                    return 1;
439                }
440                break;
441            case 'Z':   /* -Z, --max-zoom */
442                maxZoom=atoi(optarg);
443                if (maxZoom < 0 || maxZoom > MAX_ZOOM) {
444                    fprintf(stderr, "Invalid maximum zoom selected, must be between 0 and %d\n", MAX_ZOOM);
445                    return 1;
446                }
447                break;
448            case 'v':   /* -v, --verbose */
449                verbose=1;
450                break;
451            case 'h':   /* -h, --help */
452                fprintf(stderr, "Usage: render_expired [OPTION] ...\n");
453                fprintf(stderr, "  -m, --map=MAP        render tiles in this map (defaults to '" XMLCONFIG_DEFAULT "')\n");
454                fprintf(stderr, "  -s, --socket=SOCKET  unix domain socket name for contacting renderd\n");
455                fprintf(stderr, "  -n, --num-threads=N the number of parallel request threads (default 1)\n");
456                fprintf(stderr, "  -t, --tile-dir       tile cache directory (defaults to '" HASH_PATH "')\n");
457                fprintf(stderr, "  -z, --min-zoom=ZOOM  filter input to only render tiles greater or equal to this zoom level (default is 0)\n");
458                fprintf(stderr, "  -Z, --max-zoom=ZOOM  filter input to only render tiles less than or equal to this zoom level (default is %d)\n", MAX_ZOOM);
459                fprintf(stderr, "  -d, --delete-from=ZOOM  when expiring tiles of ZOOM or higher, delete them instead of re-rendering (default is off)\n");
460                fprintf(stderr, "  -T, --touch-from=ZOOM   when expiring tiles of ZOOM or higher, touch them instead of re-rendering (default is off)\n");
461                fprintf(stderr, "Send a list of tiles to be rendered from STDIN in the format:\n");
462                fprintf(stderr, "  z/x/y\n");
463                fprintf(stderr, "e.g.\n");
464                fprintf(stderr, "  1/0/1\n");
465                fprintf(stderr, "  1/1/1\n");
466                fprintf(stderr, "  1/0/0\n");
467                fprintf(stderr, "  1/1/0\n");
468                fprintf(stderr, "The above would cause all 4 tiles at zoom 1 to be rendered\n");
469                return -1;
470            default:
471                fprintf(stderr, "unhandled char '%c'\n", c);
472                break;
473        }
474    }
475
476    if (maxZoom < minZoom) {
477        fprintf(stderr, "Invalid zoom range, max zoom must be greater or equal to minimum zoom\n");
478        return 1;
479    }
480
481    if (minZoom < excess_zoomlevels) minZoom = excess_zoomlevels;
482
483    fprintf(stderr, "Rendering client\n");
484
485    gettimeofday(&start, NULL);
486
487    if (   ( touchFrom != -1 && minZoom < touchFrom )
488        || (deleteFrom != -1 && minZoom < deleteFrom)
489        || ( touchFrom == -1 && deleteFrom == -1) ) {
490        // No need to spawn render threads, when we're not actually going to rerender tiles
491        spawn_workers(numThreads, spath);
492        doRender = 1;
493    }
494
495    while(!feof(stdin)) 
496    {
497        struct stat s;
498        int n = fscanf(stdin, "%d/%d/%d", &z, &x, &y);
499
500        if (verbose)
501            printf("read: x=%d y=%d z=%d\n", x, y, z);
502        if (n != 3) {
503            // Discard input line
504            char tmp[1024];
505            char *r = fgets(tmp, sizeof(tmp), stdin);
506            if (!r)
507                continue;
508            // fprintf(stderr, "bad line %d: %s", num_all, tmp);
509            continue;
510        }
511
512        while (z > maxZoom)
513        {
514            x>>=1; y>>=1; z--;
515        }
516        while (z < maxZoom)
517        {
518            x<<=1; y<<=1; z++;
519        }
520        //printf("loop: x=%d y=%d z=%d up to z=%d\n", x, y, z, minZoom);
521        num_read++;
522        if (num_read % 100 == 0)
523            printf("Read and expanded %i tiles from list.\n", num_read);
524
525        for (; z>= minZoom; z--, x>>=1, y>>=1)
526        {
527            if (verbose)
528                printf("process: x=%d y=%d z=%d\n", x, y, z);
529
530            // don't do anything if this tile was already requested.
531            // renderd does keep a list internally to avoid enqueing the same tile
532            // twice but in case it has already rendered the tile we don't want to
533            // cause extra work.
534            if (TILE_REQUESTED(z - excess_zoomlevels,x>>excess_zoomlevels,y>>excess_zoomlevels)) 
535            { 
536                if (verbose)
537                    printf("already requested\n"); 
538                break; 
539            }
540
541            // mark tile as requested. (do this even if, below, the tile is not
542            // actually requested due to not being present on disk, to avoid
543            // unnecessary later stat'ing).
544            SET_TILE_REQUESTED(z - excess_zoomlevels,x>>excess_zoomlevels,y>>excess_zoomlevels); 
545
546            // commented out - seems to cause problems in MT environment,
547            // trying to write to already-closed file
548            //check_load();
549
550            num_all++;
551            xyz_to_meta(name, sizeof(name), tile_dir, mapname, x, y, z);
552
553            if (stat(name, &s) == 0) // 0 is success
554            {
555                // tile exists on disk; render it
556                if (deleteFrom != -1 && z >= deleteFrom)
557                {
558                    if (verbose)
559                        printf("unlink: %s\n", name);
560                    unlink(name);
561                    num_unlink++;
562                }
563                else if (touchFrom != -1 && z >= touchFrom)
564                {
565                    if (verbose)
566                        printf("touch: %s\n", name);
567                    if (-1 == utime(name, &touchTime))
568                    {
569                        perror("modifying timestamp failed");
570                    }
571                    num_touch++;
572                }
573                else if (doRender)
574                {
575                    printf("render: %s\n", name);
576                    enqueue(name);
577                    num_render++;
578                }
579                /*
580                if (!(num_render % 10))
581                {
582                    gettimeofday(&end, NULL);
583                    printf("\n");
584                    printf("Meta tiles rendered: ");
585                    display_rate(start, end, num_render);
586                    printf("Total tiles rendered: ");
587                    display_rate(start, end, num_render * METATILE * METATILE);
588                    printf("Total tiles in input: %d\n", num_read);
589                    printf("Total tiles expanded from input: %d\n", num_all);
590                    printf("Total tiles ignored (not on disk): %d\n", num_ignore);
591                }
592                */
593            }
594            else
595            {
596                if (verbose)
597                    printf("not on disk: %s\n", name);
598                num_ignore++;
599            }
600        }
601    }
602
603    if (doRender) {
604        finish_workers(numThreads);
605    }
606
607    gettimeofday(&end, NULL);
608    printf("\nTotal for all tiles rendered\n");
609    printf("Meta tiles rendered: ");
610    display_rate(start, end, num_render);
611    printf("Total tiles rendered: ");
612    display_rate(start, end, num_render * METATILE * METATILE);
613    printf("Total tiles in input: %d\n", num_read);
614    printf("Total tiles expanded from input: %d\n", num_all);
615    printf("Total meta tiles deleted: %d\n", num_unlink);
616    printf("Total meta tiles touched: %d\n", num_touch);
617    printf("Total tiles ignored (not on disk): %d\n", num_ignore);
618
619    return 0;
620}
621#endif
Note: See TracBrowser for help on using the repository browser.