Thu, 24 Oct 2013 03:37:02 +0530
Merged default branch
/** * @file tcl_signals.c Purple Tcl signal API * * purple * * Copyright (C) 2003 Ethan Blanton <eblanton@cs.purdue.edu> * * 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 <tcl.h> #include <stdarg.h> #include "tcl_purple.h" #include "internal.h" #include "connection.h" #include "conversation.h" #include "signals.h" #include "debug.h" #include "core.h" static GList *tcl_callbacks; static void *tcl_signal_callback(va_list args, struct tcl_signal_handler *handler); static Tcl_Obj *new_cb_namespace (void); void tcl_signal_init() { tcl_callbacks = NULL; } void tcl_signal_handler_free(struct tcl_signal_handler *handler) { if (handler == NULL) return; Tcl_DecrRefCount(handler->signal); if (handler->namespace) { Tcl_DecrRefCount(handler->namespace); } g_free(handler); } void tcl_signal_cleanup(Tcl_Interp *interp) { GList *cur; struct tcl_signal_handler *handler; for (cur = tcl_callbacks; cur != NULL; cur = g_list_next(cur)) { handler = cur->data; if (handler->interp == interp) { tcl_signal_handler_free(handler); cur->data = NULL; } } tcl_callbacks = g_list_remove_all(tcl_callbacks, NULL); } gboolean tcl_signal_connect(struct tcl_signal_handler *handler) { GString *proc; purple_signal_get_types(handler->instance, Tcl_GetString(handler->signal), &handler->returntype, &handler->nargs, &handler->argtypes); tcl_signal_disconnect(handler->interp, Tcl_GetString(handler->signal), handler->interp); if (!purple_signal_connect_vargs(handler->instance, Tcl_GetString(handler->signal), (void *)handler->interp, PURPLE_CALLBACK(tcl_signal_callback), (void *)handler)) return FALSE; handler->namespace = new_cb_namespace (); Tcl_IncrRefCount(handler->namespace); proc = g_string_new(""); g_string_append_printf(proc, "namespace eval %s { proc cb { %s } { %s } }", Tcl_GetString(handler->namespace), Tcl_GetString(handler->args), Tcl_GetString(handler->proc)); if (Tcl_Eval(handler->interp, proc->str) != TCL_OK) { Tcl_DecrRefCount(handler->namespace); g_string_free(proc, TRUE); return FALSE; } g_string_free(proc, TRUE); tcl_callbacks = g_list_append(tcl_callbacks, (gpointer)handler); return TRUE; } void tcl_signal_disconnect(void *instance, const char *signal, Tcl_Interp *interp) { GList *cur; struct tcl_signal_handler *handler; gboolean found = FALSE; GString *cmd; for (cur = tcl_callbacks; cur != NULL; cur = g_list_next(cur)) { handler = cur->data; if (handler->interp == interp && handler->instance == instance && !strcmp(signal, Tcl_GetString(handler->signal))) { purple_signal_disconnect(instance, signal, handler->interp, PURPLE_CALLBACK(tcl_signal_callback)); cmd = g_string_sized_new(64); g_string_printf(cmd, "namespace delete %s", Tcl_GetString(handler->namespace)); Tcl_EvalEx(interp, cmd->str, -1, TCL_EVAL_GLOBAL); tcl_signal_handler_free(handler); g_string_free(cmd, TRUE); cur->data = NULL; found = TRUE; break; } } if (found) tcl_callbacks = g_list_remove_all(tcl_callbacks, NULL); } static PurpleStringref *ref_purple_type(GType type) { if (type == PURPLE_TYPE_ACCOUNT) return PurpleTclRefAccount; else if (type == PURPLE_TYPE_CONNECTION) return PurpleTclRefConnection; else if (type == PURPLE_TYPE_CONVERSATION) return PurpleTclRefConversation; else if (type == PURPLE_TYPE_PLUGIN) return PurpleTclRefPlugin; else if (type == PURPLE_TYPE_STATUS) return PurpleTclRefStatus; else if (type == PURPLE_TYPE_XFER) return PurpleTclRefXfer; else return NULL; } static void *tcl_signal_callback(va_list args, struct tcl_signal_handler *handler) { GString *name, *val; PurpleBlistNode *node; int i; void *retval = NULL; Tcl_Obj *cmd, *arg, *result; void **vals; /* Used for inout parameters */ char ***strs; vals = g_new0(void *, handler->nargs); strs = g_new0(char **, handler->nargs); name = g_string_sized_new(32); val = g_string_sized_new(32); cmd = Tcl_NewListObj(0, NULL); Tcl_IncrRefCount(cmd); arg = Tcl_DuplicateObj(handler->namespace); Tcl_AppendStringsToObj(arg, "::cb", NULL); Tcl_ListObjAppendElement(handler->interp, cmd, arg); for (i = 0; i < handler->nargs; i++) { #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) g_string_printf(name, "%s::arg%d", Tcl_GetString(handler->namespace), i); #endif switch(handler->argtypes[i]) { case G_TYPE_POINTER: #if 0 case G_TYPE_OBJECT: case G_TYPE_BOXED: /* These are all "pointer" types to us */ if (purple_value_is_outgoing(handler->argtypes[i])) purple_debug_error("tcl", "pointer types do not currently support outgoing arguments\n"); #endif arg = purple_tcl_ref_new(PurpleTclRefPointer, va_arg(args, void *)); break; case G_TYPE_BOOLEAN: #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) { vals[i] = va_arg(args, gboolean *); Tcl_LinkVar(handler->interp, name->str, (char *)&vals[i], TCL_LINK_BOOLEAN); arg = Tcl_NewStringObj(name->str, -1); } else #endif arg = Tcl_NewBooleanObj(va_arg(args, gboolean)); break; case G_TYPE_CHAR: case G_TYPE_UCHAR: case G_TYPE_INT: case G_TYPE_UINT: case G_TYPE_LONG: case G_TYPE_ULONG: /* I should really cast these individually to * preserve as much information as possible ... * but heh */ #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) { vals[i] = va_arg(args, int *); Tcl_LinkVar(handler->interp, name->str, vals[i], TCL_LINK_INT); arg = Tcl_NewStringObj(name->str, -1); } else #endif arg = Tcl_NewIntObj(va_arg(args, int)); break; case G_TYPE_INT64: case G_TYPE_UINT64: /* Tcl < 8.4 doesn't have wide ints, so we have ugly * ifdefs in here */ #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) { vals[i] = (void *)va_arg(args, gint64 *); #if (TCL_MAJOR_VERSION >= 8 && TCL_MINOR_VERSION >= 4) Tcl_LinkVar(handler->interp, name->str, vals[i], TCL_LINK_WIDE_INT); #else /* This is going to cause weirdness at best, * but what do you want ... we're losing * precision */ Tcl_LinkVar(handler->interp, name->str, vals[i], TCL_LINK_INT); #endif /* Tcl >= 8.4 */ arg = Tcl_NewStringObj(name->str, -1); } else { #endif #if (TCL_MAJOR_VERSION >= 8 && TCL_MINOR_VERSION >= 4) arg = Tcl_NewWideIntObj(va_arg(args, gint64)); #else arg = Tcl_NewIntObj((int)va_arg(args, int)); #endif /* Tcl >= 8.4 */ break; case G_TYPE_STRING: #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) { strs[i] = va_arg(args, char **); if (strs[i] == NULL || *strs[i] == NULL) { vals[i] = ckalloc(1); *(char *)vals[i] = '\0'; } else { size_t len = strlen(*strs[i]) + 1; vals[i] = ckalloc(len); g_strlcpy(vals[i], *strs[i], len); } Tcl_LinkVar(handler->interp, name->str, (char *)&vals[i], TCL_LINK_STRING); arg = Tcl_NewStringObj(name->str, -1); } else #endif arg = Tcl_NewStringObj(va_arg(args, char *), -1); break; default: if (handler->argtypes[i] == PURPLE_TYPE_ACCOUNT || handler->argtypes[i] == PURPLE_TYPE_CONNECTION || handler->argtypes[i] == PURPLE_TYPE_CONVERSATION || handler->argtypes[i] == PURPLE_TYPE_STATUS || handler->argtypes[i] == PURPLE_TYPE_PLUGIN || handler->argtypes[i] == PURPLE_TYPE_XFER ) { #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) purple_debug_error("tcl", "pointer subtypes do not currently support outgoing arguments\n"); #endif arg = purple_tcl_ref_new(ref_purple_type(handler->argtypes[i]), va_arg(args, void *)); } else if (handler->argtypes[i] == PURPLE_TYPE_CONTACT || handler->argtypes[i] == PURPLE_TYPE_BUDDY || handler->argtypes[i] == PURPLE_TYPE_GROUP || handler->argtypes[i] == PURPLE_TYPE_CHAT ) { /* We're going to switch again for code-deduping */ #if 0 if (purple_value_is_outgoing(handler->argtypes[i])) node = *va_arg(args, PurpleBlistNode **); else #endif node = va_arg(args, PurpleBlistNode *); if (PURPLE_IS_GROUP(node)) { arg = Tcl_NewListObj(0, NULL); Tcl_ListObjAppendElement(handler->interp, arg, Tcl_NewStringObj("group", -1)); Tcl_ListObjAppendElement(handler->interp, arg, Tcl_NewStringObj(purple_group_get_name(PURPLE_GROUP(node)), -1)); } else if (PURPLE_IS_CONTACT(node)) { /* g_string_printf(val, "contact {%s}", Contact Name? ); */ arg = Tcl_NewStringObj("contact", -1); } else if (PURPLE_IS_BUDDY(node)) { arg = Tcl_NewListObj(0, NULL); Tcl_ListObjAppendElement(handler->interp, arg, Tcl_NewStringObj("buddy", -1)); Tcl_ListObjAppendElement(handler->interp, arg, Tcl_NewStringObj(purple_buddy_get_name(PURPLE_BUDDY(node)), -1)); Tcl_ListObjAppendElement(handler->interp, arg, purple_tcl_ref_new(PurpleTclRefAccount, purple_buddy_get_account(PURPLE_BUDDY(node)))); } else if (PURPLE_IS_CHAT(node)) { arg = Tcl_NewListObj(0, NULL); Tcl_ListObjAppendElement(handler->interp, arg, Tcl_NewStringObj("chat", -1)); Tcl_ListObjAppendElement(handler->interp, arg, Tcl_NewStringObj(purple_chat_get_name(PURPLE_CHAT(node)), -1)); Tcl_ListObjAppendElement(handler->interp, arg, purple_tcl_ref_new(PurpleTclRefAccount, purple_chat_get_account(PURPLE_CHAT(node)))); } } else if (G_TYPE_IS_ENUM(handler->argtypes[i])) { arg = Tcl_NewIntObj(va_arg(args, int)); } else { /* What? I guess just pass the word ... */ /* treat this as a pointer, but complain first */ purple_debug(PURPLE_DEBUG_ERROR, "tcl", "unknown type %s\n", g_type_name(handler->argtypes[i])); } } Tcl_ListObjAppendElement(handler->interp, cmd, arg); } /* Call the friggin' procedure already */ if (Tcl_EvalObjEx(handler->interp, cmd, TCL_EVAL_GLOBAL) != TCL_OK) { purple_debug(PURPLE_DEBUG_ERROR, "tcl", "error evaluating callback: %s\n", Tcl_GetString(Tcl_GetObjResult(handler->interp))); } else { result = Tcl_GetObjResult(handler->interp); /* handle return values -- strings and words only */ if (handler->returntype) { if (handler->returntype == G_TYPE_STRING) { retval = (void *)g_strdup(Tcl_GetString(result)); } else { if (Tcl_GetIntFromObj(handler->interp, result, (int *)&retval) != TCL_OK) { purple_debug(PURPLE_DEBUG_ERROR, "tcl", "Error retrieving procedure result: %s\n", Tcl_GetString(Tcl_GetObjResult(handler->interp))); retval = NULL; } } } } /* And finally clean up */ for (i = 0; i < handler->nargs; i++) { g_string_printf(name, "%s::arg%d", Tcl_GetString(handler->namespace), i); #if 0 if (purple_value_is_outgoing(handler->argtypes[i]) && purple_value_get_type(handler->argtypes[i]) != G_TYPE_SUBTYPE) Tcl_UnlinkVar(handler->interp, name->str); /* We basically only have to deal with strings on the * way out */ switch (handler->argtypes[i]) { case G_TYPE_STRING: if (purple_value_is_outgoing(handler->argtypes[i])) { if (vals[i] != NULL && *(char **)vals[i] != NULL) { g_free(*strs[i]); *strs[i] = g_strdup(vals[i]); } ckfree(vals[i]); } break; default: /* nothing */ ; } #endif } g_string_free(name, TRUE); g_string_free(val, TRUE); g_free(vals); g_free(strs); return retval; } static Tcl_Obj *new_cb_namespace () { static int cbnum; char name[32]; g_snprintf (name, sizeof(name), "::purple::_callback::cb_%d", cbnum++); return Tcl_NewStringObj (name, -1); }