Hello, today as a first step, I've tried to port the libpurple example "nullclient.c" to make it use the ecore_loop instead of the GMainLoop.
To get this, I had mostly to write the wrapper functions needed to populate the PurpleEventLoopUiOps struct. Thanks to the ecore_timer* and the ecore_main_fd* functions all this was quite easy, and you can see my result looking at the nullclient-ecore.c attached file. Unfortunately testing it with the msn-purple plugin (I've tried also with Facebook, but I've some issues with the certs) the test client attached segfaults. As you can easily try (but I could attach also some logs, if you need) the client runs correctly and initializes the connection, however after getting and sending some data (wireshark confirms it too :P), it crashes. :( Using the standard connection mode with gdb I get this: > (01:42:42) dns: DNS query for 'login.live.com' queued > (01:42:42) dns: Successfully sent DNS request to child 2317 > (01:42:43) dns: Got response for '(null)' > > Program received signal SIGSEGV, Segmentation fault. > [Switching to Thread -1214175024 (LWP 2314)] > 0xb7ef469d in host_resolved (data=0x80ade50, source=12, > cond=PURPLE_INPUT_READ) > at dnsquery.c:558 > 558 purple_input_remove(query_data->resolver->inpa); > (gdb) dns[2317]: nobody needs me... =( While using the Http mode (in the case of msn) I get this: > (01:44:38) certificate: Successfully verified certificate for login.live.com > (01:44:39) soap: Sending secure request. > > Program received signal SIGSEGV, Segmentation fault. > [Switching to Thread -1213843248 (LWP 2330)] > 0xb7685386 in msn_soap_write_cb_internal (data=0x80adf10, fd=<value optimized > out>, > cond=<value optimized out>, initial=0) at soap.c:546 > 546 written = purple_ssl_write(conn->ssl, conn->buf->str + > conn->handled_len, However I figure that they depends on the same issue, but from my knowledges I can't really understand where I'm wrong :(. Do you have any idea why is this not working as expected? Thanks. -- Treviño's World - Life and Linux http://www.3v1n0.net/
/* * nullclient-ecore.c - an ecore libpurple wrapper test. * Copyright (C) 2009, Marco Trevisan (Treviño) <m...@3v1n0.net> * * Pidgin 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., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA * */ #include "purple.h" #include <Ecore.h> #include <eina_list.h> #include <signal.h> #include <string.h> #include <unistd.h> // Client related settings #define CUSTOM_USER_DIRECTORY "/dev/null" #define CUSTOM_PLUGIN_PATH "" #define PLUGIN_SAVE_PREF "/purple/nullclient/plugins/saved" #define UI_ID "nullclient" /*** Event uiops ***/ static guint loopID = 0; Eina_List *loopsList = NULL; typedef struct _timerRef { Ecore_Timer *timer; guint id; } timerRef; typedef struct _fdRef { PurpleInputFunction function; Ecore_Fd_Handler_Flags condition; Ecore_Fd_Handler *fdh; guint id; void *data; } fdRef; static guint ecore_purple_timeout_add(guint interval, GSourceFunc function, gpointer data) { timerRef *ref = calloc(1, sizeof(timerRef)); double timer_interval = ((double) interval)/1000; ref->timer = ecore_timer_add(timer_interval, function, data); ref->id = ++loopID; loopsList = eina_list_append(loopsList, ref); return ref->id; } static gboolean ecore_purple_timeout_remove(guint handle) { timerRef *ref; Eina_List *tlist; EINA_LIST_FOREACH(loopsList, tlist, ref) { if (ref->id == handle) { ecore_timer_del(ref->timer); loopsList = eina_list_remove(loopsList, ref); free(ref); return TRUE; } } return FALSE; } static int ecore_purple_input_cb(void *data, Ecore_Fd_Handler *fd_handler) { fdRef *ref = data; PurpleInputCondition purple_cond = 0; if (ref->condition & ECORE_FD_READ) purple_cond |= PURPLE_INPUT_READ; if (ref->condition & ECORE_FD_WRITE) purple_cond |= PURPLE_INPUT_WRITE; if (ref->function && ecore_main_fd_handler_active_get(fd_handler, ref->condition)) { ref->function(ref->data, ecore_main_fd_handler_fd_get(fd_handler), purple_cond); return 1; } else return 0; } static guint ecore_purple_input_add(int fd, PurpleInputCondition condition, PurpleInputFunction func, gpointer user_data) { fdRef *ref = calloc(1, sizeof(fdRef)); if (fd < 0) return ++loopID; ref->condition = 0; if (condition & PURPLE_INPUT_READ) ref->condition |= ECORE_FD_READ; if (condition & PURPLE_INPUT_WRITE) ref->condition |= ECORE_FD_WRITE; ref->id = ++loopID; ref->data = user_data; ref->function = func; ref->fdh = ecore_main_fd_handler_add(fd, ref->condition, ecore_purple_input_cb, ref, NULL, NULL); loopsList = eina_list_append(loopsList, ref); return ref->id; } static gboolean ecore_purple_input_remove(guint handle) { fdRef *ref; Eina_List *tlist; EINA_LIST_FOREACH(loopsList, tlist, ref) { if (ref->id == handle) { if (ref->fdh) ecore_main_fd_handler_del(ref->fdh); loopsList = eina_list_remove(loopsList, ref); free(ref); return TRUE; } } return FALSE; } static PurpleEventLoopUiOps ecore_eventloops = { ecore_purple_timeout_add, ecore_purple_timeout_remove, ecore_purple_input_add, ecore_purple_input_remove, NULL, NULL, NULL, NULL, NULL }; /*** End of the eventloop functions. ***/ /*** Conversation uiops ***/ static void null_write_conv(PurpleConversation *conv, const char *who, const char *alias, const char *message, PurpleMessageFlags flags, time_t mtime) { const char *name; if (alias && *alias) name = alias; else if (who && *who) name = who; else name = NULL; printf("(%s) %s %s: %s\n", purple_conversation_get_name(conv), purple_utf8_strftime("(%H:%M:%S)", localtime(&mtime)), name, message); } static PurpleConversationUiOps null_conv_uiops = { NULL, /* create_conversation */ NULL, /* destroy_conversation */ NULL, /* write_chat */ NULL, /* write_im */ null_write_conv, /* write_conv */ NULL, /* chat_add_users */ NULL, /* chat_rename_user */ NULL, /* chat_remove_users */ NULL, /* chat_update_user */ NULL, /* present */ NULL, /* has_focus */ NULL, /* custom_smiley_add */ NULL, /* custom_smiley_write */ NULL, /* custom_smiley_close */ NULL, /* send_confirm */ NULL, NULL, NULL, NULL }; static void null_ui_init(void) { /** * This should initialize the UI components for all the modules. Here we * just initialize the UI for conversations. */ purple_conversations_set_ui_ops(&null_conv_uiops); } static PurpleCoreUiOps null_core_uiops = { NULL, NULL, null_ui_init, NULL, NULL, NULL, NULL, NULL }; static void init_libpurple(void) { /* Set a custom user directory (optional) */ purple_util_set_user_dir(CUSTOM_USER_DIRECTORY); /* We do not want any debugging for now to keep the noise to a minimum. */ purple_debug_set_enabled(TRUE); /* Set the core-uiops, which is used to * - initialize the ui specific preferences. * - initialize the debug ui. * - initialize the ui components for all the modules. * - uninitialize the ui components for all the modules when the core terminates. */ purple_core_set_ui_ops(&null_core_uiops); /* Set the uiops for the eventloop. */ purple_eventloop_set_ui_ops(&ecore_eventloops); /* Set path to search for plugins. The core (libpurple) takes care of loading the * core-plugins, which includes the protocol-plugins. So it is not essential to add * any path here, but it might be desired, especially for ui-specific plugins. */ purple_plugins_add_search_path(CUSTOM_PLUGIN_PATH); /* Now that all the essential stuff has been set, let's try to init the core. It's * necessary to provide a non-NULL name for the current ui to the core. This name * is used by stuff that depends on this ui, for example the ui-specific plugins. */ if (!purple_core_init(UI_ID)) { /* Initializing the core failed. Terminate. */ fprintf(stderr, "libpurple initialization failed. Dumping core.\n" "Please report this!\n"); abort(); } /* Create and load the buddylist. */ purple_set_blist(purple_blist_new()); purple_blist_load(); /* Load the preferences. */ purple_prefs_load(); /* Load the desired plugins. The client should save the list of loaded plugins in * the preferences using purple_plugins_save_loaded(PLUGIN_SAVE_PREF) */ purple_plugins_load_saved(PLUGIN_SAVE_PREF); /* Load the pounces. */ purple_pounces_load(); } static void signed_on(PurpleConnection *gc, gpointer null) { PurpleBlistNode *node; PurpleAccount *account = purple_connection_get_account(gc); printf("Account connected: %s %s\n", account->username, account->protocol_id); for (node = purple_blist_get_root(); node; node = purple_blist_node_next(node, FALSE)) { if (node) printf("Contact: %s\n", node); } } static void connect_to_signals_for_demonstration_purposes_only(void) { static int handle; purple_signal_connect(purple_connections_get_handle(), "signed-on", &handle, PURPLE_CALLBACK(signed_on), NULL); } int main(int argc, char *argv[]) { const char *prpl; char name[128]; char *password; PurpleAccount *account; PurpleSavedStatus *status; char *res; /* libpurple's built-in DNS resolution forks processes to perform * blocking lookups without blocking the main process. It does not * handle SIGCHLD itself, so if the UI does not you quickly get an army * of zombie subprocesses marching around. */ signal(SIGCHLD, SIG_IGN); ecore_init(); eina_list_init(); init_libpurple(); printf("libpurple initialized.\n"); int i, num; GList *iter; Eina_List *names = NULL; iter = purple_plugins_get_protocols(); for (i = 0; iter; iter = iter->next) { PurplePlugin *plugin = iter->data; PurplePluginInfo *info = plugin->info; if (info && info->name) { printf("\t%d: %s\n", i++, info->name); names = eina_list_append(names, info->id); } } printf("Select the protocol [0-%d]: ", i-1); res = fgets(name, sizeof(name), stdin); if (!res) { fprintf(stderr, "Failed to gets protocol selection."); abort(); } sscanf(name, "%d", &num); prpl = eina_list_nth(names, num); printf("Username: "); res = fgets(name, sizeof(name), stdin); if (!res) { fprintf(stderr, "Failed to read user name."); abort(); } name[strlen(name) - 1] = 0; /* strip the \n at the end */ /* Create the account */ account = purple_account_new(name, prpl); /* Get the password for the account */ password = getpass("Password: "); purple_account_set_password(account, password); /* It's necessary to enable the account first. */ purple_account_set_enabled(account, UI_ID, TRUE); /* Now, to connect the account(s), create a status and activate it. */ status = purple_savedstatus_new(NULL, PURPLE_STATUS_INVISIBLE); purple_savedstatus_activate(status); connect_to_signals_for_demonstration_purposes_only(); ecore_main_loop_begin(); ecore_main_loop_quit(); eina_list_shutdown(); return 0; }
------------------------------------------------------------------------------
_______________________________________________ enlightenment-devel mailing list enlightenment-devel@lists.sourceforge.net https://lists.sourceforge.net/lists/listinfo/enlightenment-devel