Mercurial > pidgin
view libpurple/buddyicon.c @ 15976:a6a79b8616bf
I commonly see a crash in which socket_ready_cb(), shortly after a laptop wakes from sleep, is passed invalid (previously freed) connect_data. It looks like this:
Thread 0 Crashed:
0 libobjc.A.dylib 0x90a59380 objc_msgSend 16
1 Libgaim 0x0fe23bcd gaim_proxy_connect_data_disconnect 172
2 Libgaim 0x0fe23d63 socket_ready_cb 199
3 com.apple.CoreFoundation 0x90843ffd __CFSocketDoCallback 551
(objc_msgSend is how ObjC routes messages... it's being called because connect_data->cconnect_cb is invalid).
It appears that when this crash happens, the socket is marked as ready just before the computer sleeps; on the next run loop, the callback will be called [socket_ready_cb()]. The computer sleeps and every account is disconnected first, which calls gaim_proxy_connect_cancel_with_handle(), destroying the connect_data. When it awakens, it calls socket_ready_cb() and the crash occurs.
I've added PURPLE_PROXY_CONNECT_DATA_IS_VALID, which takes advantage of the fact that all valid connect_data objects are stored in the handles GSList, just as PURPLE_GAIM_CONNECTION_IS_VALID works.
author | Evan Schoenberg <evan.s@dreskin.net> |
---|---|
date | Sun, 01 Apr 2007 02:17:06 +0000 |
parents | 32c366eeeb99 |
children | c9b4ff420140 |
line wrap: on
line source
/** * @file icon.c Buddy Icon API * @ingroup core * * purple * * Purple is the legal property of its developers, whose names are too numerous * to list here. Please refer to the COPYRIGHT file distributed with this * source distribution. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "internal.h" #include "buddyicon.h" #include "conversation.h" #include "dbus-maybe.h" #include "debug.h" #include "util.h" static GHashTable *account_cache = NULL; static char *cache_dir = NULL; static gboolean icon_caching = TRUE; static PurpleBuddyIcon * purple_buddy_icon_create(PurpleAccount *account, const char *username) { PurpleBuddyIcon *icon; GHashTable *icon_cache; icon = g_new0(PurpleBuddyIcon, 1); PURPLE_DBUS_REGISTER_POINTER(icon, PurpleBuddyIcon); purple_buddy_icon_set_account(icon, account); purple_buddy_icon_set_username(icon, username); icon_cache = g_hash_table_lookup(account_cache, account); if (icon_cache == NULL) { icon_cache = g_hash_table_new(g_str_hash, g_str_equal); g_hash_table_insert(account_cache, account, icon_cache); } g_hash_table_insert(icon_cache, (char *)purple_buddy_icon_get_username(icon), icon); return icon; } PurpleBuddyIcon * purple_buddy_icon_new(PurpleAccount *account, const char *username, void *icon_data, size_t icon_len) { PurpleBuddyIcon *icon; g_return_val_if_fail(account != NULL, NULL); g_return_val_if_fail(username != NULL, NULL); g_return_val_if_fail(icon_data != NULL, NULL); g_return_val_if_fail(icon_len > 0, NULL); icon = purple_buddy_icons_find(account, username); if (icon == NULL) icon = purple_buddy_icon_create(account, username); purple_buddy_icon_ref(icon); purple_buddy_icon_set_data(icon, icon_data, icon_len); purple_buddy_icon_set_path(icon, NULL); /* purple_buddy_icon_set_data() makes blist.c or * conversation.c, or both, take a reference. * * Plus, we leave one for the caller of this function. */ return icon; } void purple_buddy_icon_destroy(PurpleBuddyIcon *icon) { PurpleConversation *conv; PurpleAccount *account; GHashTable *icon_cache; const char *username; GSList *sl, *list; g_return_if_fail(icon != NULL); if (icon->ref_count > 0) { /* If the ref count is greater than 0, then we weren't called from * purple_buddy_icon_unref(). So we go through and ask everyone to * unref us. Then we return, since we know somewhere along the * line we got called recursively by one of the unrefs, and the * icon is already destroyed. */ account = purple_buddy_icon_get_account(icon); username = purple_buddy_icon_get_username(icon); conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, username, account); if (conv != NULL) purple_conv_im_set_icon(PURPLE_CONV_IM(conv), NULL); for (list = sl = purple_find_buddies(account, username); sl != NULL; sl = sl->next) { PurpleBuddy *buddy = (PurpleBuddy *)sl->data; purple_buddy_set_icon(buddy, NULL); } g_slist_free(list); return; } icon_cache = g_hash_table_lookup(account_cache, purple_buddy_icon_get_account(icon)); if (icon_cache != NULL) g_hash_table_remove(icon_cache, purple_buddy_icon_get_username(icon)); g_free(icon->username); g_free(icon->data); g_free(icon->path); PURPLE_DBUS_UNREGISTER_POINTER(icon); g_free(icon); } PurpleBuddyIcon * purple_buddy_icon_ref(PurpleBuddyIcon *icon) { g_return_val_if_fail(icon != NULL, NULL); icon->ref_count++; return icon; } PurpleBuddyIcon * purple_buddy_icon_unref(PurpleBuddyIcon *icon) { g_return_val_if_fail(icon != NULL, NULL); g_return_val_if_fail(icon->ref_count > 0, NULL); icon->ref_count--; if (icon->ref_count == 0) { purple_buddy_icon_destroy(icon); return NULL; } return icon; } void purple_buddy_icon_update(PurpleBuddyIcon *icon) { PurpleConversation *conv; PurpleAccount *account; const char *username; GSList *sl, *list; g_return_if_fail(icon != NULL); account = purple_buddy_icon_get_account(icon); username = purple_buddy_icon_get_username(icon); for (list = sl = purple_find_buddies(account, username); sl != NULL; sl = sl->next) { PurpleBuddy *buddy = (PurpleBuddy *)sl->data; purple_buddy_set_icon(buddy, icon); } g_slist_free(list); conv = purple_find_conversation_with_account(PURPLE_CONV_TYPE_IM, username, account); if (conv != NULL) purple_conv_im_set_icon(PURPLE_CONV_IM(conv), icon); } static void delete_icon_cache_file(const char *dirname, const char *old_icon) { struct stat st; g_return_if_fail(dirname != NULL); g_return_if_fail(old_icon != NULL); if (g_stat(old_icon, &st) == 0) g_unlink(old_icon); else { char *filename = g_build_filename(dirname, old_icon, NULL); if (g_stat(filename, &st) == 0) g_unlink(filename); g_free(filename); } purple_debug_info("buddyicon", "Uncached file %s\n", old_icon); } void purple_buddy_icon_cache(PurpleBuddyIcon *icon, PurpleBuddy *buddy) { const guchar *data; const char *dirname; char *random; char *filename; const char *old_icon; size_t len = 0; FILE *file = NULL; g_return_if_fail(icon != NULL); g_return_if_fail(buddy != NULL); if (!purple_buddy_icons_is_caching()) return; data = purple_buddy_icon_get_data(icon, &len); random = g_strdup_printf("%x", g_random_int()); dirname = purple_buddy_icons_get_cache_dir(); filename = g_build_filename(dirname, random, NULL); old_icon = purple_blist_node_get_string((PurpleBlistNode*)buddy, "buddy_icon"); if (!g_file_test(dirname, G_FILE_TEST_IS_DIR)) { purple_debug_info("buddyicon", "Creating icon cache directory.\n"); if (g_mkdir(dirname, S_IRUSR | S_IWUSR | S_IXUSR) < 0) { purple_debug_error("buddyicon", "Unable to create directory %s: %s\n", dirname, strerror(errno)); } } if ((file = g_fopen(filename, "wb")) != NULL) { fwrite(data, 1, len, file); fclose(file); purple_debug_info("buddyicon", "Wrote file %s\n", filename); } else { purple_debug_error("buddyicon", "Unable to create file %s: %s\n", filename, strerror(errno)); g_free(filename); g_free(random); return; } g_free(filename); if (old_icon != NULL) delete_icon_cache_file(dirname, old_icon); purple_blist_node_set_string((PurpleBlistNode *)buddy, "buddy_icon", random); g_free(random); } void purple_buddy_icon_uncache(PurpleBuddy *buddy) { const char *old_icon; g_return_if_fail(buddy != NULL); old_icon = purple_blist_node_get_string((PurpleBlistNode *)buddy, "buddy_icon"); if (old_icon != NULL) delete_icon_cache_file(purple_buddy_icons_get_cache_dir(), old_icon); purple_blist_node_remove_setting((PurpleBlistNode *)buddy, "buddy_icon"); /* Unset the icon in case this function is called from * something other than purple_buddy_set_icon(). */ if (buddy->icon != NULL) { purple_buddy_icon_unref(buddy->icon); buddy->icon = NULL; } } void purple_buddy_icon_set_account(PurpleBuddyIcon *icon, PurpleAccount *account) { g_return_if_fail(icon != NULL); g_return_if_fail(account != NULL); icon->account = account; } void purple_buddy_icon_set_username(PurpleBuddyIcon *icon, const char *username) { g_return_if_fail(icon != NULL); g_return_if_fail(username != NULL); g_free(icon->username); icon->username = g_strdup(username); } void purple_buddy_icon_set_data(PurpleBuddyIcon *icon, void *data, size_t len) { g_return_if_fail(icon != NULL); g_free(icon->data); if (data != NULL && len > 0) { icon->data = g_memdup(data, len); icon->len = len; } else { icon->data = NULL; icon->len = 0; } purple_buddy_icon_update(icon); } void purple_buddy_icon_set_path(PurpleBuddyIcon *icon, const gchar *path) { g_return_if_fail(icon != NULL); g_free(icon->path); icon->path = g_strdup(path); } PurpleAccount * purple_buddy_icon_get_account(const PurpleBuddyIcon *icon) { g_return_val_if_fail(icon != NULL, NULL); return icon->account; } const char * purple_buddy_icon_get_username(const PurpleBuddyIcon *icon) { g_return_val_if_fail(icon != NULL, NULL); return icon->username; } const guchar * purple_buddy_icon_get_data(const PurpleBuddyIcon *icon, size_t *len) { g_return_val_if_fail(icon != NULL, NULL); if (len != NULL) *len = icon->len; return icon->data; } const char * purple_buddy_icon_get_path(PurpleBuddyIcon *icon) { g_return_val_if_fail(icon != NULL, NULL); return icon->path; } const char * purple_buddy_icon_get_type(const PurpleBuddyIcon *icon) { const void *data; size_t len; g_return_val_if_fail(icon != NULL, NULL); data = purple_buddy_icon_get_data(icon, &len); /* TODO: Find a way to do this with GDK */ if (len >= 4) { if (!strncmp(data, "BM", 2)) return "bmp"; else if (!strncmp(data, "GIF8", 4)) return "gif"; else if (!strncmp(data, "\xff\xd8\xff\xe0", 4)) return "jpg"; else if (!strncmp(data, "\x89PNG", 4)) return "png"; } return NULL; } void purple_buddy_icons_set_for_user(PurpleAccount *account, const char *username, void *icon_data, size_t icon_len) { g_return_if_fail(account != NULL); g_return_if_fail(username != NULL); if (icon_data == NULL || icon_len == 0) { PurpleBuddyIcon *buddy_icon; buddy_icon = purple_buddy_icons_find(account, username); if (buddy_icon != NULL) purple_buddy_icon_destroy(buddy_icon); } else { PurpleBuddyIcon *icon = purple_buddy_icon_new(account, username, icon_data, icon_len); purple_buddy_icon_unref(icon); } } PurpleBuddyIcon * purple_buddy_icons_find(PurpleAccount *account, const char *username) { GHashTable *icon_cache; PurpleBuddyIcon *ret = NULL; char *filename = NULL; g_return_val_if_fail(account != NULL, NULL); g_return_val_if_fail(username != NULL, NULL); icon_cache = g_hash_table_lookup(account_cache, account); if ((icon_cache == NULL) || ((ret = g_hash_table_lookup(icon_cache, username)) == NULL)) { const char *file; struct stat st; PurpleBuddy *b = purple_find_buddy(account, username); if (!b) return NULL; if ((file = purple_blist_node_get_string((PurpleBlistNode*)b, "buddy_icon")) == NULL) return NULL; if (!g_stat(file, &st)) filename = g_strdup(file); else filename = g_build_filename(purple_buddy_icons_get_cache_dir(), file, NULL); if (!g_stat(filename, &st)) { FILE *f = g_fopen(filename, "rb"); if (f) { char *data = g_malloc(st.st_size); fread(data, 1, st.st_size, f); fclose(f); ret = purple_buddy_icon_create(account, username); purple_buddy_icon_ref(ret); purple_buddy_icon_set_data(ret, data, st.st_size); purple_buddy_icon_unref(ret); g_free(data); g_free(filename); return ret; } } g_free(filename); } return ret; } void purple_buddy_icons_set_caching(gboolean caching) { icon_caching = caching; } gboolean purple_buddy_icons_is_caching(void) { return icon_caching; } void purple_buddy_icons_set_cache_dir(const char *dir) { g_return_if_fail(dir != NULL); g_free(cache_dir); cache_dir = g_strdup(dir); } const char * purple_buddy_icons_get_cache_dir(void) { return cache_dir; } char *purple_buddy_icons_get_full_path(const char *icon) { if (icon == NULL) return NULL; if (g_file_test(icon, G_FILE_TEST_IS_REGULAR)) return g_strdup(icon); else return g_build_filename(purple_buddy_icons_get_cache_dir(), icon, NULL); } void * purple_buddy_icons_get_handle() { static int handle; return &handle; } void purple_buddy_icons_init() { account_cache = g_hash_table_new_full( g_direct_hash, g_direct_equal, NULL, (GFreeFunc)g_hash_table_destroy); cache_dir = g_build_filename(purple_user_dir(), "icons", NULL); } void purple_buddy_icons_uninit() { g_hash_table_destroy(account_cache); } void purple_buddy_icon_get_scale_size(PurpleBuddyIconSpec *spec, int *width, int *height) { int new_width, new_height; new_width = *width; new_height = *height; if (*width < spec->min_width) new_width = spec->min_width; else if (*width > spec->max_width) new_width = spec->max_width; if (*height < spec->min_height) new_height = spec->min_height; else if (*height > spec->max_height) new_height = spec->max_height; /* preserve aspect ratio */ if ((double)*height * (double)new_width > (double)*width * (double)new_height) { new_width = 0.5 + (double)*width * (double)new_height / (double)*height; } else { new_height = 0.5 + (double)*height * (double)new_width / (double)*width; } *width = new_width; *height = new_height; }