src/lib/tile.c
author Tero Marttila <terom@fixme.fi>
Sun, 24 Jan 2010 23:20:39 +0200
changeset 55 a3542e78ecd8
parent 21 47f15166e25a
child 56 d5e3089906da
permissions -rw-r--r--
fixfix
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     1
#include "tile.h"
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     2
#include "error.h"
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     3
#include "shared/log.h" // only FATAL
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     4
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     5
#include <stdlib.h>
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     6
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
     7
int pt_tile_new (struct pt_tile **tile_ptr)
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     8
{
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
     9
    struct pt_tile *tile;
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    10
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    11
    if ((tile = calloc(1, sizeof(*tile))) == NULL)
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    12
        return -PT_ERR_MEM;
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    13
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    14
    *tile_ptr = tile;
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    15
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    16
    return 0;
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    17
}
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    18
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    19
static void pt_tile_init (struct pt_tile *tile, struct pt_cache *cache, const struct pt_tile_info *info, enum pt_tile_output out_type)
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    20
{
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    21
    // init
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    22
    tile->cache = cache;
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    23
    tile->info = *info;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    24
    tile->out_type = out_type;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    25
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    26
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    27
int pt_tile_init_file (struct pt_tile *tile, struct pt_cache *cache, const struct pt_tile_info *info, FILE *out)
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    28
{
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    29
    pt_tile_init(tile, cache, info, PT_TILE_OUT_FILE);
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    30
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    31
    tile->out.file = out;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    32
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    33
    return 0;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    34
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    35
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    36
int pt_tile_init_mem (struct pt_tile *tile, struct pt_cache *cache, const struct pt_tile_info *info)
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    37
{
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    38
    pt_tile_init(tile, cache, info, PT_TILE_OUT_MEM);
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    39
    
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    40
    // init buffer
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    41
    if ((tile->out.mem.base = malloc(PT_TILE_BUF_SIZE)) == NULL)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    42
        RETURN_ERROR(PT_ERR_MEM);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    43
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    44
    tile->out.mem.len = PT_TILE_BUF_SIZE;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    45
    tile->out.mem.off = 0;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    46
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    47
    return 0;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    48
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    49
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    50
static void pt_tile_mem_write (png_structp png, png_bytep data, png_size_t length)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    51
{
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    52
    struct pt_tile_mem *buf = png_get_io_ptr(png);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    53
    size_t buf_len = buf->len;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    54
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    55
    // grow?
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    56
    while (buf->off + length > buf_len)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    57
        buf_len *= 2;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    58
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    59
    if (buf_len != buf->len) {
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    60
        char *tmp;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    61
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    62
        if ((tmp = realloc(buf->base, buf_len)) == NULL)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    63
            png_error(png, "pt_tile_buf_write - realloc failed");
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    64
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    65
        buf->base = tmp;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    66
        buf->len = buf_len;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    67
    }
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    68
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    69
    // copy
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    70
    memcpy(buf->base + buf->off, data, length);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    71
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    72
    buf->off += length;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    73
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    74
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    75
static void pt_tile_mem_flush (png_structp png_ptr)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    76
{
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    77
    // no-op
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    78
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    79
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    80
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
    81
int pt_tile_render (struct pt_tile *tile)
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    82
{
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    83
    png_structp png = NULL;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    84
    png_infop info = NULL;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    85
    int err = 0;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    86
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    87
    // open PNG writer
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    88
    if ((png = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL)) == NULL)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    89
        JUMP_SET_ERROR(err, PT_ERR_PNG_CREATE);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    90
    
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    91
    if ((info = png_create_info_struct(png)) == NULL)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    92
        JUMP_SET_ERROR(err, PT_ERR_PNG_CREATE);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    93
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    94
    // libpng error trap
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    95
    if (setjmp(png_jmpbuf(png)))
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    96
        JUMP_SET_ERROR(err, PT_ERR_PNG);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    97
 
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    98
    // setup output I/O
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    99
    switch (tile->out_type) {
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   100
        case PT_TILE_OUT_FILE:
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   101
            // use default FILE* operation
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   102
            png_init_io(png, tile->out.file);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   103
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   104
            break;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   105
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   106
        case PT_TILE_OUT_MEM:
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   107
            // use pt_tile_mem struct via pt_tile_mem_* callbacks
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   108
            png_set_write_fn(png, &tile->out.mem, pt_tile_mem_write, pt_tile_mem_flush);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   109
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   110
            break;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   111
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   112
        default:
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   113
            FATAL("tile->out_type: %d", tile->out_type);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   114
    }
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   115
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   116
    // render tile
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   117
    if ((err = pt_cache_tile_png(tile->cache, png, info, &tile->info)))
18
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   118
        JUMP_ERROR(err);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   119
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   120
    // done
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   121
    png_write_end(png, info);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   122
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   123
error:
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   124
    // cleanup
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   125
    png_destroy_write_struct(&png, &info);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   126
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   127
    return err;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   128
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   129
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   130
void pt_tile_abort (struct pt_tile *tile)
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   131
{
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   132
    // cleanup
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   133
    switch (tile->out_type) {
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   134
        case PT_TILE_OUT_FILE:
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   135
            // no-op
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   136
            break;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   137
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   138
        case PT_TILE_OUT_MEM:
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   139
            // drop buffer
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   140
            free(tile->out.mem.base);
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   141
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   142
            break;
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   143
    }
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   144
}
f92a24ab046e add missing lib/tile.*
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   145
19
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   146
void pt_tile_destroy (struct pt_tile *tile)
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   147
{
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   148
    pt_tile_abort(tile);
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   149
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   150
    free(tile);
ebcc49de97d0 implement pt_ctx threadpool and pt_image_tile_async
Tero Marttila <terom@fixme.fi>
parents: 18
diff changeset
   151
}
21
Tero Marttila <terom@fixme.fi>
parents: 19
diff changeset
   152