Mercurial > geeqie.yaz
changeset 1014:4d3c98219246
prepared image loader code for threads
author | nadvornik |
---|---|
date | Sat, 30 Aug 2008 13:04:06 +0000 |
parents | 88ebc61e33ae |
children | 2cdcf67e9300 |
files | src/image-load.c src/image-load.h |
diffstat | 2 files changed, 228 insertions(+), 80 deletions(-) [+] |
line wrap: on
line diff
--- a/src/image-load.c Sat Aug 30 10:39:35 2008 +0000 +++ b/src/image-load.c Sat Aug 30 13:04:06 2008 +0000 @@ -22,6 +22,11 @@ #include <fcntl.h> #include <sys/mman.h> + +/**************************************************************************************/ +/* image looader class */ + + enum { SIGNAL_AREA_READY = 0, SIGNAL_ERROR, @@ -142,10 +147,26 @@ ImageLoader *il = (ImageLoader *)object; image_loader_stop(il); + + DEBUG_1("freeing image loader %p bytes_read=%d", il, il->bytes_read); + if (il->idle_done_id != -1) g_source_remove(il->idle_done_id); + + while (g_source_remove_by_user_data(il)) + { + DEBUG_1("pending signals detected"); + } + + while (il->area_param_list) + { + DEBUG_1("pending area_ready signals detected"); + while (g_source_remove_by_user_data(il->area_param_list->data)) {} + g_free(il->area_param_list->data); + il->area_param_list = g_list_delete_link(il->area_param_list, il->area_param_list); + } + if (il->pixbuf) gdk_pixbuf_unref(il->pixbuf); file_data_unref(il->fd); - DEBUG_1("freeing image loader %p bytes_read=%d", il, il->bytes_read); } void image_loader_free(ImageLoader *il) @@ -168,6 +189,86 @@ return il; } +/**************************************************************************************/ +/* send signals via idle calbacks - the callback are executed in the main thread */ + +typedef struct _ImageLoaderAreaParam ImageLoaderAreaParam; +struct _ImageLoaderAreaParam { + ImageLoader *il; + guint x; + guint y; + guint w; + guint h; +}; + + +static gint image_loader_emit_area_ready_cb(gpointer data) +{ + ImageLoaderAreaParam *par = data; + ImageLoader *il = par->il; + g_signal_emit(il, signals[SIGNAL_AREA_READY], 0, par->x, par->y, par->w, par->h); + il->area_param_list = g_list_remove(il->area_param_list, par); + g_free(par); + + return FALSE; +} + +static gint image_loader_emit_done_cb(gpointer data) +{ + ImageLoader *il = data; + g_signal_emit(il, signals[SIGNAL_DONE], 0); + return FALSE; +} + +static gint image_loader_emit_error_cb(gpointer data) +{ + ImageLoader *il = data; + g_signal_emit(il, signals[SIGNAL_ERROR], 0); + return FALSE; +} + +static gint image_loader_emit_percent_cb(gpointer data) +{ + ImageLoader *il = data; + g_signal_emit(il, signals[SIGNAL_PERCENT], 0, (gdouble)il->bytes_read / il->bytes_total); + return FALSE; +} + +/* DONE and ERROR are emited only once, thus they can have normal priority + PERCENT and AREA_READY should be processed ASAP +*/ + +static void image_loader_emit_done(ImageLoader *il) +{ + g_idle_add_full(il->idle_priority, image_loader_emit_done_cb, il, NULL); +} + +static void image_loader_emit_error(ImageLoader *il) +{ + g_idle_add_full(il->idle_priority, image_loader_emit_error_cb, il, NULL); +} + +static void image_loader_emit_percent(ImageLoader *il) +{ + g_idle_add_full(G_PRIORITY_HIGH, image_loader_emit_percent_cb, il, NULL); +} + +static void image_loader_emit_area_ready(ImageLoader *il, guint x, guint y, guint w, guint h) +{ + ImageLoaderAreaParam *par = g_new0(ImageLoaderAreaParam, 1); + par->il = il; + par->x = x; + par->y = y; + par->w = w; + par->h = h; + + il->area_param_list = g_list_prepend(il->area_param_list, par); + g_idle_add_full(G_PRIORITY_HIGH, image_loader_emit_area_ready_cb, par, NULL); +} + +/**************************************************************************************/ +/* the following functions may be executed in separate thread */ + static void image_loader_sync_pixbuf(ImageLoader *il) { GdkPixbuf *pb; @@ -197,7 +298,7 @@ log_printf("critical: area_ready signal with NULL pixbuf (out of mem?)\n"); } } - g_signal_emit(il, signals[SIGNAL_AREA_READY], 0, x, y, w, h); + image_loader_emit_area_ready(il, x, y, w, h); } static void image_loader_area_prepared_cb(GdkPixbufLoader *loader, gpointer data) @@ -273,16 +374,10 @@ } } -static void image_loader_stop(ImageLoader *il) +static void image_loader_stop_loader(ImageLoader *il) { if (!il) return; - if (il->idle_id != -1) - { - g_source_remove(il->idle_id); - il->idle_id = -1; - } - if (il->loader) { /* some loaders do not have a pixbuf till close, order is important here */ @@ -292,63 +387,44 @@ il->loader = NULL; } - if (il->mapped_file) - { - if (il->preview) - { - exif_free_preview(il->mapped_file); - } - else - { - munmap(il->mapped_file, il->bytes_total); - } - il->mapped_file = NULL; - } - il->done = TRUE; } +static void image_loader_setup_loader(ImageLoader *il) +{ + il->loader = gdk_pixbuf_loader_new(); + g_signal_connect(G_OBJECT(il->loader), "area_updated", + G_CALLBACK(image_loader_area_updated_cb), il); + g_signal_connect(G_OBJECT(il->loader), "size_prepared", + G_CALLBACK(image_loader_size_cb), il); + g_signal_connect(G_OBJECT(il->loader), "area_prepared", + G_CALLBACK(image_loader_area_prepared_cb), il); +} + + static void image_loader_done(ImageLoader *il) { - image_loader_stop(il); - - g_signal_emit(il, signals[SIGNAL_DONE], 0); -} - -static gint image_loader_done_delay_cb(gpointer data) -{ - ImageLoader *il = data; + image_loader_stop_loader(il); - il->idle_done_id = -1; - image_loader_done(il); - return FALSE; -} - -static void image_loader_done_delay(ImageLoader *il) -{ - if (il->idle_done_id == -1) il->idle_done_id = g_idle_add_full(il->idle_priority, - image_loader_done_delay_cb, il, NULL); + image_loader_emit_done(il); } static void image_loader_error(ImageLoader *il) { - image_loader_stop(il); + image_loader_stop_loader(il); DEBUG_1("pixbuf_loader reported load error for: %s", il->fd->path); - g_signal_emit(il, signals[SIGNAL_ERROR], 0); + image_loader_emit_error(il); } -static gint image_loader_idle_cb(gpointer data) +static gint image_loader_continue(ImageLoader *il) { - ImageLoader *il = data; gint b; gint c; if (!il) return FALSE; - if (il->idle_id == -1) return FALSE; - c = il->idle_read_loop_count ? il->idle_read_loop_count : 1; while (c > 0) { @@ -373,7 +449,7 @@ if (il->bytes_total > 0) { - g_signal_emit(il, signals[SIGNAL_PERCENT], 0, (gdouble)il->bytes_read / il->bytes_total); + image_loader_emit_percent(il); } return TRUE; @@ -382,20 +458,17 @@ static gint image_loader_begin(ImageLoader *il) { gint b; - - if (!il->loader || il->pixbuf) return FALSE; + + if (il->pixbuf) return FALSE; b = MIN(il->read_buffer_size, il->bytes_total - il->bytes_read); + if (b < 1) return FALSE; - if (b < 1) - { - image_loader_stop(il); - return FALSE; - } + image_loader_setup_loader(il); if (!gdk_pixbuf_loader_write(il->loader, il->mapped_file + il->bytes_read, b, NULL)) { - image_loader_stop(il); + image_loader_stop_loader(il); return FALSE; } @@ -407,7 +480,7 @@ b = MIN(il->read_buffer_size, il->bytes_total - il->bytes_read); if (b < 0 || (b > 0 && !gdk_pixbuf_loader_write(il->loader, il->mapped_file + il->bytes_read, b, NULL))) { - image_loader_stop(il); + image_loader_stop_loader(il); return FALSE; } il->bytes_read += b; @@ -417,27 +490,28 @@ if (il->bytes_read == il->bytes_total || b < 1) { /* done, handle (broken) loaders that do not have pixbuf till close */ - image_loader_stop(il); + image_loader_stop_loader(il); if (!il->pixbuf) return FALSE; - image_loader_done_delay(il); + image_loader_done(il); return TRUE; } if (!il->pixbuf) { - image_loader_stop(il); + image_loader_stop_loader(il); return FALSE; } - /* finally, progressive loading :) */ - il->idle_id = g_idle_add_full(il->idle_priority, image_loader_idle_cb, il, NULL); - return TRUE; } -static gint image_loader_setup(ImageLoader *il) +/**************************************************************************************/ +/* the following functions are always executed in the main thread */ + + +static gint image_loader_setup_source(ImageLoader *il) { struct stat st; gchar *pathl; @@ -490,19 +564,99 @@ } il->preview = FALSE; } + + return TRUE; +} + +static void image_loader_stop_source(ImageLoader *il) +{ + if (!il) return; + + if (il->mapped_file) + { + if (il->preview) + { + exif_free_preview(il->mapped_file); + } + else + { + munmap(il->mapped_file, il->bytes_total); + } + il->mapped_file = NULL; + } +} - il->loader = gdk_pixbuf_loader_new(); - g_signal_connect(G_OBJECT(il->loader), "area_updated", - G_CALLBACK(image_loader_area_updated_cb), il); - g_signal_connect(G_OBJECT(il->loader), "size_prepared", - G_CALLBACK(image_loader_size_cb), il); - g_signal_connect(G_OBJECT(il->loader), "area_prepared", - G_CALLBACK(image_loader_area_prepared_cb), il); +/* +static gint image_loader_setup(ImageLoader *il) +{ + if (!image_loader_setup_source(il)) return FALSE; + + return image_loader_begin(il); +} +*/ + +static void image_loader_stop(ImageLoader *il) +{ + if (!il) return; + + if (il->idle_id != -1) + { + g_source_remove(il->idle_id); + il->idle_id = -1; + } + + + image_loader_stop_loader(il); + image_loader_stop_source(il); + +} + +/**************************************************************************************/ +/* execution via idle calls */ - il->shrunk = FALSE; +static gint image_loader_idle_cb(gpointer data) +{ + gint ret = FALSE; + ImageLoader *il = data; + if (il->idle_id != -1) + { + ret = image_loader_continue(il); + } + if (!ret) + { + image_loader_stop_source(il); + } + return ret; +} + + +gint image_loader_start_idle(ImageLoader *il) +{ + gint ret; + if (!il) return FALSE; - return image_loader_begin(il); + if (!il->fd) return FALSE; + + if (!image_loader_setup_source(il)) return FALSE; + + ret = image_loader_begin(il); + + if (ret && !il->done) il->idle_id = g_idle_add_full(il->idle_priority, image_loader_idle_cb, il, NULL); + return ret; +} + +/**************************************************************************************/ +/* public interface */ + + +gint image_loader_start(ImageLoader *il) +{ + if (!il) return FALSE; + + if (!il->fd) return FALSE; + + return image_loader_start_idle(il); } @@ -557,14 +711,6 @@ il->idle_priority = priority; } -gint image_loader_start(ImageLoader *il) -{ - if (!il) return FALSE; - - if (!il->fd) return FALSE; - - return image_loader_setup(il); -} gdouble image_loader_get_percent(ImageLoader *il) { @@ -597,6 +743,7 @@ } +/* FIXME - this can be rather slow and blocks until the size is known */ gint image_load_dimensions(FileData *fd, gint *width, gint *height) { ImageLoader *il; @@ -604,7 +751,7 @@ il = image_loader_new(fd); - success = image_loader_start(il); + success = image_loader_start_idle(il); if (success && il->pixbuf) {