Mercurial > pidgin
view libpurple/protocols/novell/nmrequest.c @ 28844:ee1ed5d16235
Save the event source ID for search_find_cb so that it's removed when the
log dialog is closed. This fixes a double free which might be caused by
having a large logset (and maybe pressing Close multiple times).
It looks something like this, with a few variations:
== Invalid read of size 8
== at 0x483001: search_find_cb (gtklog.c:425)
== by 0xBB91E50: g_main_context_dispatch (gmain.c:1824)
== by 0xBB953E7: g_main_context_iterate (gmain.c:2455)
== by 0xBB958BC: g_main_loop_run (gmain.c:2663)
== by 0x63AF606: gtk_main (gtkmain.c:1205)
== by 0x485ADF: main (gtkmain.c:978)
== Address 0x1b469860 is 56 bytes inside a block of size 72 free'd
== at 0x4C239EF: free (vg_replace_malloc.c:323)
== by 0x483A6C: destroy_cb (gtklog.c:223)
== by 0xAB9B12C: g_closure_invoke (gclosure.c:767)
== by 0xABB02B1: signal_emit_unlocked_R (gsignal.c:3247)
== by 0xABB1806: g_signal_emit_valist (gsignal.c:2980)
== by 0xABB1CD2: g_signal_emit (gsignal.c:3037)
== by 0xAB9B12C: g_closure_invoke (gclosure.c:767)
== by 0xABB02B1: signal_emit_unlocked_R (gsignal.c:3247)
== by 0xABB1806: g_signal_emit_valist (gsignal.c:2980)
== by 0xABB1CD2: g_signal_emit (gsignal.c:3037)
== by 0x630ECB4: gtk_real_button_released (gtkbutton.c:1707)
== by 0xAB9B12C: g_closure_invoke (gclosure.c:767)
==
Fixes #11071.
author | Elliott Sales de Andrade <qulogic@pidgin.im> |
---|---|
date | Tue, 05 Jan 2010 06:43:18 +0000 |
parents | 44b4e8bd759b |
children |
line wrap: on
line source
/* * nmrequest.c * * Copyright (c) 2004 Novell, Inc. All Rights Reserved. * * 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; version 2 of the License. * * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA * */ #include "nmrequest.h" static int count = 0; struct _NMRequest { int trans_id; char *cmd; int gmt; gpointer data; gpointer user_define; nm_response_cb callback; int ref_count; NMERR_T ret_code; }; NMRequest *nm_create_request(const char *cmd, int trans_id, int gmt, nm_response_cb cb, gpointer resp_data, gpointer user_define) { NMRequest *req; if (cmd == NULL) return NULL; req = g_new0(NMRequest, 1); req->cmd = g_strdup(cmd); req->trans_id = trans_id; req->gmt = gmt; req->callback = cb; req->data = resp_data; req->user_define = user_define; req->ref_count = 1; purple_debug_info("novell", "Creating NMRequest instance, total=%d\n", ++count); return req; } void nm_release_request(NMRequest * req) { if (req && (--req->ref_count == 0)) { if (req->cmd) g_free(req->cmd); g_free(req); purple_debug_info("novell", "Releasing NMRequest instance, total=%d\n", --count); } } void nm_request_add_ref(NMRequest * req) { if (req) req->ref_count++; } void nm_request_set_callback(NMRequest * req, nm_response_cb callback) { if (req) req->callback = callback; } void nm_request_set_data(NMRequest * req, gpointer data) { if (req) req->data = data; } void nm_request_set_user_define(NMRequest * req, gpointer user_define) { if (req) req->user_define = user_define; } int nm_request_get_trans_id(NMRequest * req) { if (req) return req->trans_id; else return -1; } const char * nm_request_get_cmd(NMRequest * req) { if (req == NULL) return NULL; return req->cmd; } gpointer nm_request_get_data(NMRequest * req) { if (req == NULL) return NULL; return req->data; } gpointer nm_request_get_user_define(NMRequest * req) { if (req == NULL) return NULL; return req->user_define; } nm_response_cb nm_request_get_callback(NMRequest * req) { if (req == NULL) return NULL; return req->callback; } void nm_request_set_ret_code(NMRequest * req, NMERR_T rc) { if (req) req->ret_code = rc; } NMERR_T nm_request_get_ret_code(NMRequest * req) { if (req) return req->ret_code; else return (NMERR_T) - 1; }